##// END OF EJS Templates
Removes RJS from MembersController....
Jean-Philippe Lang -
r9875:d3bfbb800c2b
parent child
Show More
@@ -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 format.js {
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]}, :format => "js"
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]}, :format => "js"
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