@@ -0,0 +1,11 | |||||
|
1 | Element.update('tab-content-members', '<%= escape_javascript(render :partial => 'projects/settings/members') %>'); | |||
|
2 | hideOnLoad(); | |||
|
3 | ||||
|
4 | <% if @members.present? && @members.all? {|m| m.valid? } %> | |||
|
5 | <% @members.each do |member| %> | |||
|
6 | new Effect.Highlight("member-<%= member.id %>"); | |||
|
7 | <% end %> | |||
|
8 | <% else %> | |||
|
9 | <% errors = @members.collect {|m| m.errors.full_messages}.flatten.uniq.join(', ') %> | |||
|
10 | alert('<%= escape_javascript l(:notice_failed_to_save_members, :errors => errors) %>'); | |||
|
11 | <% end %> |
@@ -0,0 +1,2 | |||||
|
1 | Element.update('tab-content-members', '<%= escape_javascript(render :partial => 'projects/settings/members') %>'); | |||
|
2 | hideOnLoad(); |
@@ -0,0 +1,3 | |||||
|
1 | Element.update('tab-content-members', '<%= escape_javascript(render :partial => 'projects/settings/members') %>'); | |||
|
2 | hideOnLoad(); | |||
|
3 | new Effect.Highlight("member-<%= @member.id %>"); |
@@ -1,144 +1,118 | |||||
1 | # Redmine - project management software |
|
1 | # Redmine - project management software | |
2 | # Copyright (C) 2006-2012 Jean-Philippe Lang |
|
2 | # Copyright (C) 2006-2012 Jean-Philippe Lang | |
3 | # |
|
3 | # | |
4 | # This program is free software; you can redistribute it and/or |
|
4 | # This program is free software; you can redistribute it and/or | |
5 | # modify it under the terms of the GNU General Public License |
|
5 | # modify it under the terms of the GNU General Public License | |
6 | # as published by the Free Software Foundation; either version 2 |
|
6 | # as published by the Free Software Foundation; either version 2 | |
7 | # of the License, or (at your option) any later version. |
|
7 | # of the License, or (at your option) any later version. | |
8 | # |
|
8 | # | |
9 | # This program is distributed in the hope that it will be useful, |
|
9 | # This program is distributed in the hope that it will be useful, | |
10 | # but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
10 | # but WITHOUT ANY WARRANTY; without even the implied warranty of | |
11 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
11 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
12 | # GNU General Public License for more details. |
|
12 | # GNU General Public License for more details. | |
13 | # |
|
13 | # | |
14 | # You should have received a copy of the GNU General Public License |
|
14 | # You should have received a copy of the GNU General Public License | |
15 | # along with this program; if not, write to the Free Software |
|
15 | # along with this program; if not, write to the Free Software | |
16 | # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. |
|
16 | # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | |
17 |
|
17 | |||
18 | class MembersController < ApplicationController |
|
18 | class MembersController < ApplicationController | |
19 | model_object Member |
|
19 | model_object Member | |
20 | before_filter :find_model_object, :except => [:index, :create, :autocomplete] |
|
20 | before_filter :find_model_object, :except => [:index, :create, :autocomplete] | |
21 | before_filter :find_project_from_association, :except => [:index, :create, :autocomplete] |
|
21 | before_filter :find_project_from_association, :except => [:index, :create, :autocomplete] | |
22 | before_filter :find_project_by_project_id, :only => [:index, :create, :autocomplete] |
|
22 | before_filter :find_project_by_project_id, :only => [:index, :create, :autocomplete] | |
23 | before_filter :authorize |
|
23 | before_filter :authorize | |
24 | accept_api_auth :index, :show, :create, :update, :destroy |
|
24 | accept_api_auth :index, :show, :create, :update, :destroy | |
25 |
|
25 | |||
26 | def index |
|
26 | def index | |
27 | @offset, @limit = api_offset_and_limit |
|
27 | @offset, @limit = api_offset_and_limit | |
28 | @member_count = @project.member_principals.count |
|
28 | @member_count = @project.member_principals.count | |
29 | @member_pages = Paginator.new self, @member_count, @limit, params['page'] |
|
29 | @member_pages = Paginator.new self, @member_count, @limit, params['page'] | |
30 | @offset ||= @member_pages.current.offset |
|
30 | @offset ||= @member_pages.current.offset | |
31 | @members = @project.member_principals.all( |
|
31 | @members = @project.member_principals.all( | |
32 | :order => "#{Member.table_name}.id", |
|
32 | :order => "#{Member.table_name}.id", | |
33 | :limit => @limit, |
|
33 | :limit => @limit, | |
34 | :offset => @offset |
|
34 | :offset => @offset | |
35 | ) |
|
35 | ) | |
36 |
|
36 | |||
37 | respond_to do |format| |
|
37 | respond_to do |format| | |
38 | format.html { head 406 } |
|
38 | format.html { head 406 } | |
39 | format.api |
|
39 | format.api | |
40 | end |
|
40 | end | |
41 | end |
|
41 | end | |
42 |
|
42 | |||
43 | def show |
|
43 | def show | |
44 | respond_to do |format| |
|
44 | respond_to do |format| | |
45 | format.html { head 406 } |
|
45 | format.html { head 406 } | |
46 | format.api |
|
46 | format.api | |
47 | end |
|
47 | end | |
48 | end |
|
48 | end | |
49 |
|
49 | |||
50 | def create |
|
50 | def create | |
51 | members = [] |
|
51 | members = [] | |
52 | if params[:membership] |
|
52 | if params[:membership] | |
53 | if params[:membership][:user_ids] |
|
53 | if params[:membership][:user_ids] | |
54 | attrs = params[:membership].dup |
|
54 | attrs = params[:membership].dup | |
55 | user_ids = attrs.delete(:user_ids) |
|
55 | user_ids = attrs.delete(:user_ids) | |
56 | user_ids.each do |user_id| |
|
56 | user_ids.each do |user_id| | |
57 | members << Member.new(:role_ids => params[:membership][:role_ids], :user_id => user_id) |
|
57 | members << Member.new(:role_ids => params[:membership][:role_ids], :user_id => user_id) | |
58 | end |
|
58 | end | |
59 | else |
|
59 | else | |
60 | members << Member.new(:role_ids => params[:membership][:role_ids], :user_id => params[:membership][:user_id]) |
|
60 | members << Member.new(:role_ids => params[:membership][:role_ids], :user_id => params[:membership][:user_id]) | |
61 | end |
|
61 | end | |
62 | @project.members << members |
|
62 | @project.members << members | |
63 | end |
|
63 | end | |
64 |
|
64 | |||
65 | respond_to do |format| |
|
65 | respond_to do |format| | |
66 | if members.present? && members.all? {|m| m.valid? } |
|
66 | format.html { redirect_to :controller => 'projects', :action => 'settings', :tab => 'members', :id => @project } | |
67 | format.html { redirect_to :controller => 'projects', :action => 'settings', :tab => 'members', :id => @project } |
|
67 | format.js { @members = members } | |
68 |
|
|
68 | format.api { | |
69 | render(:update) {|page| |
|
69 | @member = members.first | |
70 | page.replace_html "tab-content-members", :partial => 'projects/settings/members' |
|
70 | if @member.valid? | |
71 | page << 'hideOnLoad()' |
|
|||
72 | members.each {|member| page.visual_effect(:highlight, "member-#{member.id}") } |
|
|||
73 | } |
|
|||
74 | } |
|
|||
75 | format.api { |
|
|||
76 | @member = members.first |
|
|||
77 | render :action => 'show', :status => :created, :location => membership_url(@member) |
|
71 | render :action => 'show', :status => :created, :location => membership_url(@member) | |
78 |
|
|
72 | else | |
79 | else |
|
73 | render_validation_errors(@member) | |
80 | format.js { |
|
74 | end | |
81 | render(:update) {|page| |
|
75 | } | |
82 | errors = members.collect {|m| |
|
|||
83 | m.errors.full_messages |
|
|||
84 | }.flatten.uniq |
|
|||
85 |
|
||||
86 | page.alert(l(:notice_failed_to_save_members, :errors => errors.join(', '))) |
|
|||
87 | } |
|
|||
88 | } |
|
|||
89 | format.api { render_validation_errors(members.first) } |
|
|||
90 | end |
|
|||
91 | end |
|
76 | end | |
92 | end |
|
77 | end | |
93 |
|
78 | |||
94 | def update |
|
79 | def update | |
95 | if params[:membership] |
|
80 | if params[:membership] | |
96 | @member.role_ids = params[:membership][:role_ids] |
|
81 | @member.role_ids = params[:membership][:role_ids] | |
97 | end |
|
82 | end | |
98 | saved = @member.save |
|
83 | saved = @member.save | |
99 | respond_to do |format| |
|
84 | respond_to do |format| | |
100 | format.html { redirect_to :controller => 'projects', :action => 'settings', :tab => 'members', :id => @project } |
|
85 | format.html { redirect_to :controller => 'projects', :action => 'settings', :tab => 'members', :id => @project } | |
101 |
format.js |
|
86 | format.js | |
102 | render(:update) {|page| |
|
|||
103 | page.replace_html "tab-content-members", :partial => 'projects/settings/members' |
|
|||
104 | page << 'hideOnLoad()' |
|
|||
105 | page.visual_effect(:highlight, "member-#{@member.id}") |
|
|||
106 | } |
|
|||
107 | } |
|
|||
108 | format.api { |
|
87 | format.api { | |
109 | if saved |
|
88 | if saved | |
110 | render_api_ok |
|
89 | render_api_ok | |
111 | else |
|
90 | else | |
112 | render_validation_errors(@member) |
|
91 | render_validation_errors(@member) | |
113 | end |
|
92 | end | |
114 | } |
|
93 | } | |
115 | end |
|
94 | end | |
116 | end |
|
95 | end | |
117 |
|
96 | |||
118 | def destroy |
|
97 | def destroy | |
119 | if request.delete? && @member.deletable? |
|
98 | if request.delete? && @member.deletable? | |
120 | @member.destroy |
|
99 | @member.destroy | |
121 | end |
|
100 | end | |
122 | respond_to do |format| |
|
101 | respond_to do |format| | |
123 | format.html { redirect_to :controller => 'projects', :action => 'settings', :tab => 'members', :id => @project } |
|
102 | format.html { redirect_to :controller => 'projects', :action => 'settings', :tab => 'members', :id => @project } | |
124 | format.js { render(:update) {|page| |
|
103 | format.js | |
125 | page.replace_html "tab-content-members", :partial => 'projects/settings/members' |
|
|||
126 | page << 'hideOnLoad()' |
|
|||
127 | } |
|
|||
128 | } |
|
|||
129 | format.api { |
|
104 | format.api { | |
130 | if @member.destroyed? |
|
105 | if @member.destroyed? | |
131 | render_api_ok |
|
106 | render_api_ok | |
132 | else |
|
107 | else | |
133 | head :unprocessable_entity |
|
108 | head :unprocessable_entity | |
134 | end |
|
109 | end | |
135 | } |
|
110 | } | |
136 | end |
|
111 | end | |
137 | end |
|
112 | end | |
138 |
|
113 | |||
139 | def autocomplete |
|
114 | def autocomplete | |
140 | @principals = Principal.active.not_member_of(@project).like(params[:q]).all(:limit => 100) |
|
115 | @principals = Principal.active.not_member_of(@project).like(params[:q]).all(:limit => 100) | |
141 | render :layout => false |
|
116 | render :layout => false | |
142 | end |
|
117 | end | |
143 |
|
||||
144 | end |
|
118 | end |
@@ -1,110 +1,122 | |||||
1 | # Redmine - project management software |
|
1 | # Redmine - project management software | |
2 | # Copyright (C) 2006-2012 Jean-Philippe Lang |
|
2 | # Copyright (C) 2006-2012 Jean-Philippe Lang | |
3 | # |
|
3 | # | |
4 | # This program is free software; you can redistribute it and/or |
|
4 | # This program is free software; you can redistribute it and/or | |
5 | # modify it under the terms of the GNU General Public License |
|
5 | # modify it under the terms of the GNU General Public License | |
6 | # as published by the Free Software Foundation; either version 2 |
|
6 | # as published by the Free Software Foundation; either version 2 | |
7 | # of the License, or (at your option) any later version. |
|
7 | # of the License, or (at your option) any later version. | |
8 | # |
|
8 | # | |
9 | # This program is distributed in the hope that it will be useful, |
|
9 | # This program is distributed in the hope that it will be useful, | |
10 | # but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
10 | # but WITHOUT ANY WARRANTY; without even the implied warranty of | |
11 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
11 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
12 | # GNU General Public License for more details. |
|
12 | # GNU General Public License for more details. | |
13 | # |
|
13 | # | |
14 | # You should have received a copy of the GNU General Public License |
|
14 | # You should have received a copy of the GNU General Public License | |
15 | # along with this program; if not, write to the Free Software |
|
15 | # along with this program; if not, write to the Free Software | |
16 | # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. |
|
16 | # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | |
17 |
|
17 | |||
18 | require File.expand_path('../../test_helper', __FILE__) |
|
18 | require File.expand_path('../../test_helper', __FILE__) | |
19 | require 'members_controller' |
|
19 | require 'members_controller' | |
20 |
|
20 | |||
21 | # Re-raise errors caught by the controller. |
|
21 | # Re-raise errors caught by the controller. | |
22 | class MembersController; def rescue_action(e) raise e end; end |
|
22 | class MembersController; def rescue_action(e) raise e end; end | |
23 |
|
23 | |||
24 |
|
24 | |||
25 | class MembersControllerTest < ActionController::TestCase |
|
25 | class MembersControllerTest < ActionController::TestCase | |
26 | fixtures :projects, :members, :member_roles, :roles, :users |
|
26 | fixtures :projects, :members, :member_roles, :roles, :users | |
27 |
|
27 | |||
28 | def setup |
|
28 | def setup | |
29 | @controller = MembersController.new |
|
29 | @controller = MembersController.new | |
30 | @request = ActionController::TestRequest.new |
|
30 | @request = ActionController::TestRequest.new | |
31 | @response = ActionController::TestResponse.new |
|
31 | @response = ActionController::TestResponse.new | |
32 | User.current = nil |
|
32 | User.current = nil | |
33 | @request.session[:user_id] = 2 |
|
33 | @request.session[:user_id] = 2 | |
34 | end |
|
34 | end | |
35 |
|
35 | |||
36 | def test_create |
|
36 | def test_create | |
37 | assert_difference 'Member.count' do |
|
37 | assert_difference 'Member.count' do | |
38 | post :create, :project_id => 1, :membership => {:role_ids => [1], :user_id => 7} |
|
38 | post :create, :project_id => 1, :membership => {:role_ids => [1], :user_id => 7} | |
39 | end |
|
39 | end | |
40 | assert_redirected_to '/projects/ecookbook/settings/members' |
|
40 | assert_redirected_to '/projects/ecookbook/settings/members' | |
41 | assert User.find(7).member_of?(Project.find(1)) |
|
41 | assert User.find(7).member_of?(Project.find(1)) | |
42 | end |
|
42 | end | |
43 |
|
43 | |||
44 | def test_create_multiple |
|
44 | def test_create_multiple | |
45 | assert_difference 'Member.count', 3 do |
|
45 | assert_difference 'Member.count', 3 do | |
46 | post :create, :project_id => 1, :membership => {:role_ids => [1], :user_ids => [7, 8, 9]} |
|
46 | post :create, :project_id => 1, :membership => {:role_ids => [1], :user_ids => [7, 8, 9]} | |
47 | end |
|
47 | end | |
48 | assert_redirected_to '/projects/ecookbook/settings/members' |
|
48 | assert_redirected_to '/projects/ecookbook/settings/members' | |
49 | assert User.find(7).member_of?(Project.find(1)) |
|
49 | assert User.find(7).member_of?(Project.find(1)) | |
50 | end |
|
50 | end | |
51 |
|
51 | |||
52 | def test_xhr_create |
|
52 | def test_xhr_create | |
53 | assert_difference 'Member.count', 3 do |
|
53 | assert_difference 'Member.count', 3 do | |
54 |
post :create, :project_id => 1, :membership => {:role_ids => [1], :user_ids => [7, 8, 9]} |
|
54 | xhr :post, :create, :project_id => 1, :membership => {:role_ids => [1], :user_ids => [7, 8, 9]} | |
|
55 | assert_response :success | |||
|
56 | assert_template 'create' | |||
|
57 | assert_equal 'text/javascript', response.content_type | |||
55 | end |
|
58 | end | |
56 | assert_select_rjs :replace_html, 'tab-content-members' |
|
|||
57 | assert User.find(7).member_of?(Project.find(1)) |
|
59 | assert User.find(7).member_of?(Project.find(1)) | |
58 | assert User.find(8).member_of?(Project.find(1)) |
|
60 | assert User.find(8).member_of?(Project.find(1)) | |
59 | assert User.find(9).member_of?(Project.find(1)) |
|
61 | assert User.find(9).member_of?(Project.find(1)) | |
|
62 | assert_include 'tab-content-members', response.body | |||
60 | end |
|
63 | end | |
61 |
|
64 | |||
62 | def test_xhr_create_with_failure |
|
65 | def test_xhr_create_with_failure | |
63 | assert_no_difference 'Member.count' do |
|
66 | assert_no_difference 'Member.count' do | |
64 |
post :create, :project_id => 1, :membership => {:role_ids => [], :user_ids => [7, 8, 9]} |
|
67 | xhr :post, :create, :project_id => 1, :membership => {:role_ids => [], :user_ids => [7, 8, 9]} | |
|
68 | assert_response :success | |||
|
69 | assert_template 'create' | |||
|
70 | assert_equal 'text/javascript', response.content_type | |||
65 | end |
|
71 | end | |
66 | assert_select '#tab-content-members', 0 |
|
72 | assert_match /alert/, response.body, "Alert message not sent" | |
67 | assert @response.body.match(/alert/i), "Alert message not sent" |
|
|||
68 | end |
|
73 | end | |
69 |
|
74 | |||
70 | def test_edit |
|
75 | def test_edit | |
71 | assert_no_difference 'Member.count' do |
|
76 | assert_no_difference 'Member.count' do | |
72 | put :update, :id => 2, :membership => {:role_ids => [1], :user_id => 3} |
|
77 | put :update, :id => 2, :membership => {:role_ids => [1], :user_id => 3} | |
73 | end |
|
78 | end | |
74 | assert_redirected_to '/projects/ecookbook/settings/members' |
|
79 | assert_redirected_to '/projects/ecookbook/settings/members' | |
75 | end |
|
80 | end | |
76 |
|
81 | |||
77 | def test_xhr_edit |
|
82 | def test_xhr_edit | |
78 | assert_no_difference 'Member.count' do |
|
83 | assert_no_difference 'Member.count' do | |
79 | xhr :put, :update, :id => 2, :membership => {:role_ids => [1], :user_id => 3} |
|
84 | xhr :put, :update, :id => 2, :membership => {:role_ids => [1], :user_id => 3} | |
|
85 | assert_response :success | |||
|
86 | assert_template 'update' | |||
|
87 | assert_equal 'text/javascript', response.content_type | |||
80 | end |
|
88 | end | |
81 | assert_select_rjs :replace_html, 'tab-content-members' |
|
|||
82 | member = Member.find(2) |
|
89 | member = Member.find(2) | |
83 | assert_equal [1], member.role_ids |
|
90 | assert_equal [1], member.role_ids | |
84 | assert_equal 3, member.user_id |
|
91 | assert_equal 3, member.user_id | |
|
92 | assert_include 'tab-content-members', response.body | |||
85 | end |
|
93 | end | |
86 |
|
94 | |||
87 | def test_destroy |
|
95 | def test_destroy | |
88 | assert_difference 'Member.count', -1 do |
|
96 | assert_difference 'Member.count', -1 do | |
89 | delete :destroy, :id => 2 |
|
97 | delete :destroy, :id => 2 | |
90 | end |
|
98 | end | |
91 | assert_redirected_to '/projects/ecookbook/settings/members' |
|
99 | assert_redirected_to '/projects/ecookbook/settings/members' | |
92 | assert !User.find(3).member_of?(Project.find(1)) |
|
100 | assert !User.find(3).member_of?(Project.find(1)) | |
93 | end |
|
101 | end | |
94 |
|
102 | |||
95 | def test_xhr_destroy |
|
103 | def test_xhr_destroy | |
96 | assert_difference 'Member.count', -1 do |
|
104 | assert_difference 'Member.count', -1 do | |
97 | xhr :delete, :destroy, :id => 2 |
|
105 | xhr :delete, :destroy, :id => 2 | |
|
106 | assert_response :success | |||
|
107 | assert_template 'destroy' | |||
|
108 | assert_equal 'text/javascript', response.content_type | |||
98 | end |
|
109 | end | |
99 | assert_select_rjs :replace_html, 'tab-content-members' |
|
110 | assert_nil Member.find_by_id(2) | |
|
111 | assert_include 'tab-content-members', response.body | |||
100 | end |
|
112 | end | |
101 |
|
113 | |||
102 | def test_autocomplete |
|
114 | def test_autocomplete | |
103 | get :autocomplete, :project_id => 1, :q => 'mis' |
|
115 | get :autocomplete, :project_id => 1, :q => 'mis' | |
104 | assert_response :success |
|
116 | assert_response :success | |
105 | assert_template 'autocomplete' |
|
117 | assert_template 'autocomplete' | |
106 |
|
118 | |||
107 | assert_tag :label, :content => /User Misc/, |
|
119 | assert_tag :label, :content => /User Misc/, | |
108 | :child => { :tag => 'input', :attributes => { :name => 'membership[user_ids][]', :value => '8' } } |
|
120 | :child => { :tag => 'input', :attributes => { :name => 'membership[user_ids][]', :value => '8' } } | |
109 | end |
|
121 | end | |
110 | end |
|
122 | end |
General Comments 0
You need to be logged in to leave comments.
Login now