Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
0a63d613
Commit
0a63d613
authored
Apr 05, 2017
by
Simon Knox
Committed by
Jose Ivan Vargas
Apr 07, 2017
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
refactor dropdown
clear hidden inputs after posting update
parent
0cf79217
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
207 additions
and
179 deletions
+207
-179
app/assets/javascripts/api.js
app/assets/javascripts/api.js
+2
-2
app/assets/javascripts/approvers_select.js
app/assets/javascripts/approvers_select.js
+179
-0
app/assets/javascripts/project_new.js
app/assets/javascripts/project_new.js
+15
-171
app/views/projects/ee/_merge_request_settings.html.haml
app/views/projects/ee/_merge_request_settings.html.haml
+2
-1
spec/javascripts/approvers_select_spec.js
spec/javascripts/approvers_select_spec.js
+9
-5
No files found.
app/assets/javascripts/api.js
View file @
0a63d613
...
@@ -23,7 +23,7 @@ var Api = {
...
@@ -23,7 +23,7 @@ var Api = {
return
callback
(
group
);
return
callback
(
group
);
});
});
},
},
users
:
function
(
search
,
options
,
callback
)
{
users
:
function
(
search
,
options
,
callback
=
$
.
noop
)
{
var
url
=
Api
.
buildUrl
(
"
/autocomplete/users.json
"
);
var
url
=
Api
.
buildUrl
(
"
/autocomplete/users.json
"
);
return
$
.
ajax
({
return
$
.
ajax
({
url
,
url
,
...
@@ -35,7 +35,7 @@ var Api = {
...
@@ -35,7 +35,7 @@ var Api = {
}).
done
(
callback
);
}).
done
(
callback
);
},
},
// Return groups list. Filtered by query
// Return groups list. Filtered by query
groups
:
function
(
query
,
options
,
callback
)
{
groups
:
function
(
query
,
options
,
callback
=
$
.
noop
)
{
var
url
=
Api
.
buildUrl
(
Api
.
groupsPath
);
var
url
=
Api
.
buildUrl
(
Api
.
groupsPath
);
return
$
.
ajax
({
return
$
.
ajax
({
url
:
url
,
url
:
url
,
...
...
app/assets/javascripts/approvers_select.js
0 → 100644
View file @
0a63d613
/* global Api */
export
default
class
ApproversSelect
{
constructor
()
{
const
approverSelect
=
document
.
querySelector
(
'
.js-select-user-and-group
'
);
const
name
=
approverSelect
.
dataset
.
name
;
this
.
fieldNames
=
[
`
${
name
}
[approver_ids]`
,
`
${
name
}
[approver_group_ids]`
];
this
.
bindEvents
();
this
.
addEvents
();
$
(
'
.js-select-user-and-group
'
).
select2
({
placeholder
:
'
Search for users or groups
'
,
multiple
:
true
,
minimumInputLength
:
0
,
query
:
(
query
)
=>
{
const
fetchGroups
=
this
.
fetchGroups
(
query
.
term
);
const
fetchUsers
=
this
.
fetchUsers
(
query
.
term
);
return
$
.
when
(
fetchGroups
,
fetchUsers
).
then
((
groups
,
users
)
=>
{
const
data
=
{
results
:
groups
[
0
].
concat
(
users
[
0
]),
};
return
query
.
callback
(
data
);
});
},
formatResult
:
ApproversSelect
.
formatResult
,
formatSelection
:
ApproversSelect
.
formatSelection
,
dropdownCssClass
:
'
ajax-groups-dropdown
'
,
})
.
on
(
'
change
'
,
this
.
handleSelectChange
);
}
bindEvents
()
{
this
.
addApprover
=
this
.
addApprover
.
bind
(
this
);
this
.
handleSelectChange
=
this
.
handleSelectChange
.
bind
(
this
);
this
.
fetchGroups
=
this
.
fetchGroups
.
bind
(
this
);
this
.
fetchUsers
=
this
.
fetchUsers
.
bind
(
this
);
}
addEvents
()
{
$
(
document
).
on
(
'
click
'
,
'
.js-approvers
'
,
this
.
addApprover
);
$
(
document
).
on
(
'
click
'
,
'
.js-approver-remove
'
,
ApproversSelect
.
removeApprover
);
}
static
getOptions
(
fieldName
,
selector
,
key
)
{
const
input
=
$
(
`[name="
${
fieldName
}
"]`
);
const
existingApprovers
=
[].
map
.
call
(
document
.
querySelectorAll
(
selector
),
item
=>
parseInt
(
item
.
getAttribute
(
'
data-id
'
),
10
),
);
const
selectedApprovers
=
input
.
val
()
.
split
(
'
,
'
)
.
filter
(
val
=>
val
!==
''
);
const
options
=
{
[
key
]:
[...
existingApprovers
,
...
selectedApprovers
],
};
return
options
;
}
fetchGroups
(
term
)
{
const
options
=
ApproversSelect
.
getOptions
(
this
.
fieldNames
[
1
],
'
.js-approver-group
'
,
'
skip_groups
'
);
return
Api
.
groups
(
term
,
options
);
}
fetchUsers
(
term
)
{
const
options
=
ApproversSelect
.
getOptions
(
this
.
fieldNames
[
0
],
'
.js-approver
'
,
'
skip_users
'
);
return
Api
.
users
(
term
,
options
);
}
handleSelectChange
(
evt
)
{
const
{
added
,
removed
}
=
evt
;
const
userInput
=
$
(
`[name="
${
this
.
fieldNames
[
0
]}
"]`
);
const
groupInput
=
$
(
`[name="
${
this
.
fieldNames
[
1
]}
"]`
);
if
(
added
)
{
if
(
added
.
full_name
)
{
groupInput
.
val
(
`
${
groupInput
.
val
()}
,
${
added
.
id
}
`
.
replace
(
/^,/
,
''
));
}
else
{
userInput
.
val
(
`
${
userInput
.
val
()}
,
${
added
.
id
}
`
.
replace
(
/^,/
,
''
));
}
}
if
(
removed
)
{
if
(
removed
.
full_name
)
{
groupInput
.
val
(
groupInput
.
val
().
replace
(
new
RegExp
(
`,?
${
removed
.
id
}
`
),
''
));
}
else
{
userInput
.
val
(
userInput
.
val
().
replace
(
new
RegExp
(
`,?
${
removed
.
id
}
`
),
''
));
}
}
}
static
formatSelection
(
group
)
{
return
group
.
full_name
||
group
.
name
;
}
static
formatResult
({
avatar_url
:
avatarUrl
,
full_name
:
fullName
,
full_path
:
fullPath
,
name
,
username
,
})
{
if
(
username
)
{
const
avatar
=
avatarUrl
||
gon
.
default_avatar_url
;
return
`
<div class="user-result">
<div class="user-image">
<img class="avatar s24" src="
${
avatar
}
">
</div>
<div class="user-name">
${
name
}
</div>
<div class="user-username">@
${
username
}
</div>
</div>
`
;
}
return
`
<div class="group-result">
<div class="group-name">
${
fullName
}
</div>
<div class="group-path">
${
fullPath
}
</div>
</div>
`
;
}
addApprover
()
{
this
.
fieldNames
.
forEach
(
ApproversSelect
.
saveApprovers
);
}
static
saveApprovers
(
fieldName
)
{
const
$input
=
$
(
`[name="
${
fieldName
}
"]`
);
const
newValue
=
$input
.
val
();
if
(
!
newValue
)
{
return
;
}
const
$form
=
$
(
'
.js-approvers
'
).
closest
(
'
form
'
);
$
(
'
.load-wrapper
'
).
removeClass
(
'
hidden
'
);
$
.
ajax
({
url
:
$form
.
attr
(
'
action
'
),
type
:
'
POST
'
,
data
:
{
_method
:
'
PATCH
'
,
[
fieldName
]:
newValue
,
},
success
:
ApproversSelect
.
updateApproverList
,
complete
()
{
$input
.
val
(
'
val
'
,
''
);
$
(
'
.js-select-user-and-group
'
).
select2
(
'
val
'
,
''
);
$
(
'
.load-wrapper
'
).
addClass
(
'
hidden
'
);
},
error
()
{
// TODO: scroll into view or toast
window
.
Flash
(
'
Failed to add Approver
'
,
'
alert
'
);
},
});
}
static
removeApprover
(
evt
)
{
evt
.
preventDefault
();
const
target
=
evt
.
currentTarget
;
$
(
'
.load-wrapper
'
).
removeClass
(
'
hidden
'
);
$
.
ajax
({
url
:
target
.
getAttribute
(
'
href
'
),
type
:
'
POST
'
,
data
:
{
_method
:
'
DELETE
'
,
},
success
:
ApproversSelect
.
updateApproverList
,
complete
:
()
=>
$
(
'
.load-wrapper
'
).
addClass
(
'
hidden
'
),
error
()
{
window
.
Flash
(
'
Failed to remove Approver
'
,
'
alert
'
);
},
});
}
static
updateApproverList
(
html
)
{
$
(
'
.approver-list
'
).
html
(
$
(
html
).
find
(
'
.approver-list
'
).
html
());
}
}
app/assets/javascripts/project_new.js
View file @
0a63d613
/* eslint-disable func-names, space-before-function-paren, no-var, prefer-rest-params, wrap-iife, no-unused-vars, one-var, no-underscore-dangle, prefer-template, no-else-return, prefer-arrow-callback, max-len */
/* eslint-disable func-names, space-before-function-paren, no-var, prefer-rest-params, wrap-iife, no-unused-vars, one-var, no-underscore-dangle, prefer-template, no-else-return, prefer-arrow-callback, max-len */
/* global Api */
/* global Api */
(
function
()
{
import
ApproversSelect
from
'
./approvers_select
'
;
var
bind
=
function
(
fn
,
me
)
{
return
function
()
{
return
fn
.
apply
(
me
,
arguments
);
};
};
(
function
()
{
this
.
ProjectNew
=
(
function
()
{
this
.
ProjectNew
=
(
function
()
{
function
ProjectNew
()
{
function
ProjectNew
()
{
this
.
toggleSettings
=
bind
(
this
.
toggleSettings
,
this
);
this
.
addApprover
=
this
.
addApprover
.
bind
(
this
);
this
.
removeApprover
=
this
.
removeApprover
.
bind
(
this
);
this
.
$selects
=
$
(
'
.project-feature select
'
);
this
.
$selects
=
$
(
'
.project-feature select
'
);
this
.
$repoSelects
=
this
.
$selects
.
filter
(
'
.js-repo-select
'
);
this
.
$repoSelects
=
this
.
$selects
.
filter
(
'
.js-repo-select
'
);
this
.
$enableApprovers
=
$
(
'
.js-require-approvals-toggle
'
);
this
.
$enableApprovers
=
$
(
'
.js-require-approvals-toggle
'
);
...
@@ -27,173 +23,17 @@
...
@@ -27,173 +23,17 @@
this
.
addEvents
();
this
.
addEvents
();
this
.
toggleRepoVisibility
();
this
.
toggleRepoVisibility
();
this
.
initApproverSelect
();
this
.
approversSelect
=
new
ApproversSelect
();
}
ProjectNew
.
prototype
.
addEvents
=
function
()
{
this
.
$selects
.
on
(
'
change
'
,
this
.
toggleSettings
);
$
(
'
.js-approvers
'
).
on
(
'
click
'
,
this
.
addApprover
);
$
(
document
).
on
(
'
click
'
,
'
.js-approver-remove
'
,
this
.
removeApprover
);
$
(
'
#require_approvals
'
).
on
(
'
change
'
,
function
()
{
const
enabled
=
$
(
this
).
prop
(
'
checked
'
);
const
val
=
enabled
?
1
:
0
;
$
(
'
#project_approvals_before_merge
'
).
val
(
val
);
$
(
'
#project_approvals_before_merge
'
).
prop
(
'
min
'
,
val
);
$
(
'
.nested-settings
'
).
toggleClass
(
'
hidden
'
,
!
enabled
);
});
};
ProjectNew
.
prototype
.
initApproverSelect
=
function
()
{
$
(
'
.js-select-user-and-group
'
).
select2
({
placeholder
:
'
Search for users or groups
'
,
multiple
:
true
,
minimumInputLength
:
0
,
query
(
query
)
{
const
existingGroupApprovers
=
[].
map
.
call
(
document
.
querySelectorAll
(
'
.js-approver-group
'
),
item
=>
parseInt
(
item
.
getAttribute
(
'
data-id
'
),
10
),
);
const
selectedGroupApprovers
=
$
(
'
[name="project[approver_group_ids]"]
'
).
val
()
.
split
(
'
,
'
)
.
filter
(
val
=>
val
!==
''
);
const
groupOptions
=
{
skip_groups
:
[...
existingGroupApprovers
,
...
selectedGroupApprovers
],
};
const
groupsApi
=
Api
.
groups
(
query
.
term
,
groupOptions
,
function
(
groups
)
{
return
groups
;
});
const
existingApprovers
=
[].
map
.
call
(
document
.
querySelectorAll
(
'
.js-approver
'
),
item
=>
parseInt
(
item
.
getAttribute
(
'
data-id
'
),
10
),
);
const
selectedApprovers
=
$
(
'
[name="project[approver_ids]"]
'
).
val
()
.
split
(
'
,
'
)
.
filter
(
id
=>
id
!==
''
);
const
userOptions
=
{
skip_users
:
[...
existingApprovers
,
...
selectedApprovers
],
};
const
usersApi
=
Api
.
users
(
query
.
term
,
userOptions
,
function
(
groups
)
{
return
groups
;
});
return
$
.
when
(
groupsApi
,
usersApi
).
then
((
groups
,
users
)
=>
{
const
data
=
{
results
:
groups
[
0
].
concat
(
users
[
0
]),
};
return
query
.
callback
(
data
);
});
},
formatResult
:
this
.
formatResult
,
formatSelection
:
this
.
formatSelection
,
dropdownCssClass
:
'
ajax-groups-dropdown
'
,
})
.
on
(
'
change
'
,
(
evt
)
=>
{
const
{
added
,
removed
}
=
evt
;
const
groupInput
=
$
(
'
[name="project[approver_group_ids]"]
'
);
const
userInput
=
$
(
'
[name="project[approver_ids]"]
'
);
if
(
added
)
{
if
(
added
.
full_name
)
{
groupInput
.
val
(
`
${
groupInput
.
val
()}
,
${
added
.
id
}
`
.
replace
(
/^,/
,
''
));
}
else
{
userInput
.
val
(
`
${
userInput
.
val
()}
,
${
added
.
id
}
`
.
replace
(
/^,/
,
''
));
}
}
if
(
removed
)
{
if
(
removed
.
full_name
)
{
groupInput
.
val
(
groupInput
.
val
().
replace
(
new
RegExp
(
`,?
${
removed
.
id
}
`
),
''
));
}
else
{
userInput
.
val
(
userInput
.
val
().
replace
(
new
RegExp
(
`,?
${
removed
.
id
}
`
),
''
));
}
}
});
};
ProjectNew
.
prototype
.
formatResult
=
function
(
group
)
{
if
(
group
.
username
)
{
return
"
<div class='group-result'> <div class='group-name'>
"
+
group
.
name
+
"
</div> <div class='group-path'></div> </div>
"
;
}
}
let
avatar
;
ProjectNew
.
prototype
.
bindEvents
=
function
()
{
if
(
group
.
avatar_url
)
{
this
.
toggleSettings
=
this
.
toggleSettings
.
bind
(
this
);
avatar
=
group
.
avatar_url
;
this
.
toggleApproverSettingsVisibility
=
this
.
toggleApproverSettingsVisibility
.
bind
(
this
);
}
else
{
avatar
=
gon
.
default_avatar_url
;
}
return
`
<div class='group-result'>
<div class='group-name'>
${
group
.
full_name
}
</div>
<div class='group-path'>
${
group
.
full_path
}
</div>
</div>
`
;
};
ProjectNew
.
prototype
.
formatSelection
=
function
(
group
)
{
return
group
.
full_name
||
group
.
name
;
};
};
ProjectNew
.
prototype
.
addApprover
=
function
(
evt
)
{
ProjectNew
.
prototype
.
addEvents
=
function
()
{
const
fieldNames
=
[
'
project[approver_ids]
'
,
'
project[approver_group_ids]
'
];
this
.
$selects
.
on
(
'
change
'
,
this
.
toggleSettings
);
fieldNames
.
forEach
((
fieldName
)
=>
{
$
(
'
#require_approvals
'
).
on
(
'
change
'
,
this
.
toggleApproverSettingsVisibility
);
const
$select
=
$
(
`[name="
${
fieldName
}
"]`
);
const
newValue
=
$select
.
val
();
if
(
!
newValue
)
{
return
;
}
const
$form
=
$
(
'
.js-approvers
'
).
closest
(
'
form
'
);
$
(
'
.load-wrapper
'
).
removeClass
(
'
hidden
'
);
$
.
ajax
({
url
:
$form
.
attr
(
'
action
'
),
type
:
'
POST
'
,
data
:
{
_method
:
'
PATCH
'
,
[
fieldName
]:
newValue
,
},
success
:
this
.
updateApproverList
,
complete
()
{
$select
.
select2
(
'
val
'
,
''
);
$
(
'
.js-select-user-and-group
'
).
select2
(
'
val
'
,
''
);
$
(
'
.load-wrapper
'
).
addClass
(
'
hidden
'
);
},
error
(
err
)
{
// TODO: scroll into view or toast
window
.
Flash
(
'
Failed to add Approver
'
,
'
alert
'
);
},
});
});
};
ProjectNew
.
prototype
.
removeApprover
=
function
(
evt
)
{
evt
.
preventDefault
();
const
target
=
evt
.
currentTarget
;
$
(
'
.load-wrapper
'
).
removeClass
(
'
hidden
'
);
$
.
ajax
({
url
:
target
.
getAttribute
(
'
href
'
),
type
:
'
POST
'
,
data
:
{
_method
:
'
DELETE
'
,
},
success
:
this
.
updateApproverList
,
complete
:
()
=>
$
(
'
.load-wrapper
'
).
addClass
(
'
hidden
'
),
error
(
err
)
{
window
.
Flash
(
'
Failed to remove Approver
'
,
'
alert
'
);
},
});
};
ProjectNew
.
prototype
.
updateApproverList
=
function
(
html
)
{
const
fakeEl
=
document
.
createElement
(
'
template
'
);
fakeEl
.
innerHTML
=
html
;
document
.
querySelector
(
'
.well-list.approver-list
'
).
innerHTML
=
fakeEl
.
content
.
querySelector
(
'
.well-list.approver-list
'
).
innerHTML
;
};
};
ProjectNew
.
prototype
.
initVisibilitySelect
=
function
()
{
ProjectNew
.
prototype
.
initVisibilitySelect
=
function
()
{
...
@@ -204,8 +44,12 @@
...
@@ -204,8 +44,12 @@
};
};
ProjectNew
.
prototype
.
toggleApproverSettingsVisibility
=
function
(
evt
)
{
ProjectNew
.
prototype
.
toggleApproverSettingsVisibility
=
function
(
evt
)
{
const
enabled
=
evt
.
value
;
this
.
$requiredApprovals
=
$
(
'
#project_approvals_before_merge
'
);
$
(
'
.nested-settings
'
).
toggleClass
(
'
hidden
'
,
enabled
);
const
enabled
=
$
(
evt
.
target
).
prop
(
'
checked
'
);
const
val
=
enabled
?
1
:
0
;
this
.
$requiredApprovals
.
val
(
val
);
this
.
$requiredApprovals
.
prop
(
'
min
'
,
val
);
$
(
'
.nested-settings
'
).
toggleClass
(
'
hidden
'
,
!
enabled
);
};
};
ProjectNew
.
prototype
.
toggleSettings
=
function
()
{
ProjectNew
.
prototype
.
toggleSettings
=
function
()
{
...
...
app/views/projects/ee/_merge_request_settings.html.haml
View file @
0a63d613
...
@@ -58,7 +58,7 @@
...
@@ -58,7 +58,7 @@
=
hidden_field_tag
"project[approver_ids]"
=
hidden_field_tag
"project[approver_ids]"
=
hidden_field_tag
"project[approver_group_ids]"
=
hidden_field_tag
"project[approver_group_ids]"
.input-group.input-btn-group
.input-group.input-btn-group
=
hidden_field_tag
:approver_user_and_group_ids
,
''
,
{
class:
'js-select-user-and-group input-large'
,
tabindex:
1
}
=
hidden_field_tag
:approver_user_and_group_ids
,
''
,
{
class:
'js-select-user-and-group input-large'
,
tabindex:
1
,
data:
{
name:
'project'
}
}
%button
.btn.btn-success.js-approvers
{
type:
'button'
,
title:
'Add approvers(s)'
}
%button
.btn.btn-success.js-approvers
{
type:
'button'
,
title:
'Add approvers(s)'
}
Add
Add
.help-block
.help-block
...
@@ -67,6 +67,7 @@
...
@@ -67,6 +67,7 @@
.panel.panel-default.prepend-top-10
.panel.panel-default.prepend-top-10
.panel-heading
.panel-heading
Approvers
Approvers
%span
.badge
-# TODO: badge with project.approver_group_ids.count + project.approver_ids.count
-# TODO: badge with project.approver_group_ids.count + project.approver_ids.count
%ul
.well-list.approver-list
%ul
.well-list.approver-list
.load-wrapper.hidden
.load-wrapper.hidden
...
...
spec/javascripts/
project_new
_spec.js
→
spec/javascripts/
approvers_select
_spec.js
View file @
0a63d613
/* global ProjectNew */
import
ApproversSelect
from
'
~/approvers_select
'
;
require
(
'
~/project_new
'
);
describe
(
'
ApproversSelect
'
,
function
()
{
describe
(
'
Project settings
'
,
function
()
{
const
projectSettingsTemplate
=
'
projects/edit.html.raw
'
;
const
projectSettingsTemplate
=
'
projects/edit.html.raw
'
;
preloadFixtures
(
projectSettingsTemplate
);
preloadFixtures
(
projectSettingsTemplate
);
beforeEach
(()
=>
{
beforeEach
(()
=>
{
loadFixtures
(
projectSettingsTemplate
);
loadFixtures
(
projectSettingsTemplate
);
this
.
$requireApprovalsToggle
=
$
(
'
.js-require-approvals-toggle
'
);
this
.
$requireApprovalsToggle
=
$
(
'
.js-require-approvals-toggle
'
);
this
.
project
=
new
ProjectNew
();
this
.
project
=
new
ApproversSelect
();
});
});
it
(
'
shows approver settings if enabled
'
,
()
=>
{
it
(
'
shows approver settings if enabled
'
,
()
=>
{
...
@@ -37,4 +35,10 @@ describe('Project settings', function () {
...
@@ -37,4 +35,10 @@ describe('Project settings', function () {
this
.
$requireApprovalsToggle
.
click
();
this
.
$requireApprovalsToggle
.
click
();
expect
(
$
(
'
[name="project[approvals_before_merge]"]
'
).
val
()).
toBe
(
'
1
'
);
expect
(
$
(
'
[name="project[approvals_before_merge]"]
'
).
val
()).
toBe
(
'
1
'
);
});
});
it
(
'
sets minimum for approvers field if enabled
'
,
()
=>
{
expect
(
$
(
'
[name="project[approvals_before_merge]"]
'
).
attr
(
'
min
'
)).
toBe
(
'
0
'
);
this
.
$requireApprovalsToggle
.
click
();
expect
(
$
(
'
[name="project[approvals_before_merge]"]
'
).
attr
(
'
min
'
)).
toBe
(
'
1
'
);
});
});
});
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment