326 lines
10 KiB
CoffeeScript
326 lines
10 KiB
CoffeeScript
class @UsersSelect
|
|
constructor: (currentUser) ->
|
|
@usersPath = "/autocomplete/users.json"
|
|
@userPath = "/autocomplete/users/:id.json"
|
|
if currentUser?
|
|
@currentUser = JSON.parse(currentUser)
|
|
|
|
$('.js-user-search').each (i, dropdown) =>
|
|
$dropdown = $(dropdown)
|
|
@projectId = $dropdown.data('project-id')
|
|
@showCurrentUser = $dropdown.data('current-user')
|
|
showNullUser = $dropdown.data('null-user')
|
|
showAnyUser = $dropdown.data('any-user')
|
|
firstUser = $dropdown.data('first-user')
|
|
@authorId = $dropdown.data('author-id')
|
|
selectedId = $dropdown.data('selected')
|
|
defaultLabel = $dropdown.data('default-label')
|
|
issueURL = $dropdown.data('issueUpdate')
|
|
$selectbox = $dropdown.closest('.selectbox')
|
|
$block = $selectbox.closest('.block')
|
|
abilityName = $dropdown.data('ability-name')
|
|
$value = $block.find('.value')
|
|
$collapsedSidebar = $block.find('.sidebar-collapsed-user')
|
|
$loading = $block.find('.block-loading').fadeOut()
|
|
|
|
$block.on('click', '.js-assign-yourself', (e) =>
|
|
e.preventDefault()
|
|
assignTo(@currentUser.id)
|
|
)
|
|
|
|
assignTo = (selected) ->
|
|
data = {}
|
|
data[abilityName] = {}
|
|
data[abilityName].assignee_id = if selected? then selected else null
|
|
$loading
|
|
.fadeIn()
|
|
$dropdown.trigger('loading.gl.dropdown')
|
|
$.ajax(
|
|
type: 'PUT'
|
|
dataType: 'json'
|
|
url: issueURL
|
|
data: data
|
|
).done (data) ->
|
|
$dropdown.trigger('loaded.gl.dropdown')
|
|
$loading.fadeOut()
|
|
$selectbox.hide()
|
|
|
|
if data.assignee
|
|
user =
|
|
name: data.assignee.name
|
|
username: data.assignee.username
|
|
avatar: data.assignee.avatar_url
|
|
else
|
|
user =
|
|
name: 'Unassigned'
|
|
username: ''
|
|
avatar: ''
|
|
$value.html(assigneeTemplate(user))
|
|
$collapsedSidebar.html(collapsedAssigneeTemplate(user))
|
|
|
|
|
|
collapsedAssigneeTemplate = _.template(
|
|
'<% if( avatar ) { %>
|
|
<a class="author_link" href="/u/<%= username %>">
|
|
<img width="24" class="avatar avatar-inline s24" alt="" src="<%= avatar %>">
|
|
<span class="author">Toni Boehm</span>
|
|
</a>
|
|
<% } else { %>
|
|
<i class="fa fa-user"></i>
|
|
<% } %>'
|
|
)
|
|
|
|
assigneeTemplate = _.template(
|
|
'<% if (username) { %>
|
|
<a class="author_link bold" href="/u/<%= username %>">
|
|
<% if( avatar ) { %>
|
|
<img width="32" class="avatar avatar-inline s32" alt="" src="<%= avatar %>">
|
|
<% } %>
|
|
<span class="author"><%= name %></span>
|
|
<span class="username">
|
|
@<%= username %>
|
|
</span>
|
|
</a>
|
|
<% } else { %>
|
|
<span class="no-value assign-yourself">
|
|
No assignee -
|
|
<a href="#" class="js-assign-yourself">
|
|
assign yourself
|
|
</a>
|
|
</span>
|
|
<% } %>'
|
|
)
|
|
|
|
$dropdown.glDropdown(
|
|
data: (term, callback) =>
|
|
isAuthorFilter = $('.js-author-search')
|
|
|
|
@users term, (users) =>
|
|
if term.length is 0
|
|
showDivider = 0
|
|
|
|
if firstUser
|
|
# Move current user to the front of the list
|
|
for obj, index in users
|
|
if obj.username == firstUser
|
|
users.splice(index, 1)
|
|
users.unshift(obj)
|
|
break
|
|
|
|
if showNullUser
|
|
showDivider += 1
|
|
users.unshift(
|
|
beforeDivider: true
|
|
name: 'Unassigned',
|
|
id: 0
|
|
)
|
|
|
|
if showAnyUser
|
|
showDivider += 1
|
|
name = showAnyUser
|
|
name = 'Any User' if name == true
|
|
anyUser = {
|
|
beforeDivider: true
|
|
name: name,
|
|
id: null
|
|
}
|
|
users.unshift(anyUser)
|
|
|
|
if showDivider
|
|
users.splice(showDivider, 0, "divider")
|
|
|
|
# Send the data back
|
|
callback users
|
|
filterable: true
|
|
filterRemote: true
|
|
search:
|
|
fields: ['name', 'username']
|
|
selectable: true
|
|
fieldName: $dropdown.data('field-name')
|
|
|
|
toggleLabel: (selected) ->
|
|
if selected && 'id' of selected
|
|
if selected.text then selected.text else selected.name
|
|
else
|
|
defaultLabel
|
|
|
|
inputId: 'issue_assignee_id'
|
|
|
|
hidden: (e) ->
|
|
$selectbox.hide()
|
|
# display:block overrides the hide-collapse rule
|
|
$value.css('display', '')
|
|
|
|
clicked: (user) ->
|
|
page = $('body').data 'page'
|
|
isIssueIndex = page is 'projects:issues:index'
|
|
isMRIndex = page is page is 'projects:merge_requests:index'
|
|
if $dropdown.hasClass('js-filter-bulk-update')
|
|
return
|
|
|
|
if $dropdown.hasClass('js-filter-submit') and (isIssueIndex or isMRIndex)
|
|
selectedId = user.id
|
|
Issuable.filterResults $dropdown.closest('form')
|
|
else if $dropdown.hasClass 'js-filter-submit'
|
|
$dropdown.closest('form').submit()
|
|
else
|
|
selected = $dropdown
|
|
.closest('.selectbox')
|
|
.find("input[name='#{$dropdown.data('field-name')}']").val()
|
|
assignTo(selected)
|
|
|
|
renderRow: (user) ->
|
|
username = if user.username then "@#{user.username}" else ""
|
|
avatar = if user.avatar_url then user.avatar_url else false
|
|
selected = if user.id is selectedId then "is-active" else ""
|
|
img = ""
|
|
|
|
if user.beforeDivider?
|
|
"<li>
|
|
<a href='#' class='#{selected}'>
|
|
#{user.name}
|
|
</a>
|
|
</li>"
|
|
else
|
|
if avatar
|
|
img = "<img src='#{avatar}' class='avatar avatar-inline' width='30' />"
|
|
|
|
# split into three parts so we can remove the username section if nessesary
|
|
listWithName = "<li>
|
|
<a href='#' class='dropdown-menu-user-link #{selected}'>
|
|
#{img}
|
|
<strong class='dropdown-menu-user-full-name'>
|
|
#{user.name}
|
|
</strong>"
|
|
|
|
listWithUserName = "<span class='dropdown-menu-user-username'>
|
|
#{username}
|
|
</span>"
|
|
listClosingTags = "</a>
|
|
</li>"
|
|
|
|
|
|
if username is ''
|
|
listWithUserName = ''
|
|
|
|
listWithName + listWithUserName + listClosingTags
|
|
)
|
|
|
|
$('.ajax-users-select').each (i, select) =>
|
|
@projectId = $(select).data('project-id')
|
|
@groupId = $(select).data('group-id')
|
|
@showCurrentUser = $(select).data('current-user')
|
|
@authorId = $(select).data('author-id')
|
|
showNullUser = $(select).data('null-user')
|
|
showAnyUser = $(select).data('any-user')
|
|
showEmailUser = $(select).data('email-user')
|
|
firstUser = $(select).data('first-user')
|
|
|
|
$(select).select2
|
|
placeholder: "Search for a user"
|
|
multiple: $(select).hasClass('multiselect')
|
|
minimumInputLength: 0
|
|
query: (query) =>
|
|
@users query.term, (users) =>
|
|
data = { results: users }
|
|
|
|
if query.term.length == 0
|
|
if firstUser
|
|
# Move current user to the front of the list
|
|
for obj, index in data.results
|
|
if obj.username == firstUser
|
|
data.results.splice(index, 1)
|
|
data.results.unshift(obj)
|
|
break
|
|
|
|
if showNullUser
|
|
nullUser = {
|
|
name: 'Unassigned',
|
|
id: 0
|
|
}
|
|
data.results.unshift(nullUser)
|
|
|
|
if showAnyUser
|
|
name = showAnyUser
|
|
name = 'Any User' if name == true
|
|
anyUser = {
|
|
name: name,
|
|
id: null
|
|
}
|
|
data.results.unshift(anyUser)
|
|
|
|
if showEmailUser && data.results.length == 0 && query.term.match(/^[^@]+@[^@]+$/)
|
|
emailUser = {
|
|
name: "Invite \"#{query.term}\"",
|
|
username: query.term,
|
|
id: query.term
|
|
}
|
|
data.results.unshift(emailUser)
|
|
|
|
query.callback(data)
|
|
|
|
initSelection: (args...) =>
|
|
@initSelection(args...)
|
|
formatResult: (args...) =>
|
|
@formatResult(args...)
|
|
formatSelection: (args...) =>
|
|
@formatSelection(args...)
|
|
dropdownCssClass: "ajax-users-dropdown"
|
|
escapeMarkup: (m) -> # we do not want to escape markup since we are displaying html in results
|
|
m
|
|
|
|
initSelection: (element, callback) ->
|
|
id = $(element).val()
|
|
if id == "0"
|
|
nullUser = { name: 'Unassigned' }
|
|
callback(nullUser)
|
|
else if id != ""
|
|
@user(id, callback)
|
|
|
|
formatResult: (user) ->
|
|
if user.avatar_url
|
|
avatar = user.avatar_url
|
|
else
|
|
avatar = gon.default_avatar_url
|
|
|
|
"<div class='user-result #{'no-username' unless user.username}'>
|
|
<div class='user-image'><img class='avatar s24' src='#{avatar}'></div>
|
|
<div class='user-name'>#{user.name}</div>
|
|
<div class='user-username'>#{user.username || ""}</div>
|
|
</div>"
|
|
|
|
formatSelection: (user) ->
|
|
user.name
|
|
|
|
user: (user_id, callback) =>
|
|
url = @buildUrl(@userPath)
|
|
url = url.replace(':id', user_id)
|
|
|
|
$.ajax(
|
|
url: url
|
|
dataType: "json"
|
|
).done (user) ->
|
|
callback(user)
|
|
|
|
# Return users list. Filtered by query
|
|
# Only active users retrieved
|
|
users: (query, callback) =>
|
|
url = @buildUrl(@usersPath)
|
|
|
|
$.ajax(
|
|
url: url
|
|
data:
|
|
search: query
|
|
per_page: 20
|
|
active: true
|
|
project_id: @projectId
|
|
group_id: @groupId
|
|
current_user: @showCurrentUser
|
|
author_id: @authorId
|
|
dataType: "json"
|
|
).done (users) ->
|
|
callback(users)
|
|
|
|
buildUrl: (url) ->
|
|
url = gon.relative_url_root.replace(/\/$/, '') + url if gon.relative_url_root?
|
|
return url
|