users_select.js.coffee 9.16 KB
Newer Older
1
class @UsersSelect
2
  constructor: (currentUser) ->
3 4
    @usersPath = "/autocomplete/users.json"
    @userPath = "/autocomplete/users/:id.json"
5 6
    if currentUser?
      @currentUser = JSON.parse(currentUser)
7

8
    $('.js-user-search').each (i, dropdown) =>
Phil Hughes's avatar
Phil Hughes committed
9 10 11 12 13 14 15
      $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')
      selectedId = $dropdown.data('selected')
16
      defaultLabel = $dropdown.data('default-label')
17 18 19
      issueURL = $dropdown.data('issueUpdate')
      $selectbox = $dropdown.closest('.selectbox')
      $block = $selectbox.closest('.block')
20
      abilityName = $dropdown.data('ability-name')
21 22
      $value = $block.find('.value')
      $loading = $block.find('.block-loading').fadeOut()
23

24
      $block.on('click', '.js-assign-yourself', (e) =>
25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
        e.preventDefault()
        assignTo(@currentUser.id)
      )

      assignTo = (selected) ->
        data = {}
        data[abilityName] = {}
        data[abilityName].assignee_id = selected
        $loading
          .fadeIn()
        $.ajax(
          type: 'PUT'
          dataType: 'json'
          url: issueURL
          data: data
        ).done (data) ->
          $loading.fadeOut()
          $selectbox.hide()
43

44 45 46 47
          if data.assignee
            user =
              name: data.assignee.name
              username: data.assignee.username
48
              avatar: data.assignee.avatar_url
49 50 51 52 53 54 55 56 57
          else
            user =
              name: 'Unassigned'
              username: ''
              avatar: ''

          $value.html(noAssigneeTemplate(user))
          $value.find('a').attr('href')

58 59 60 61 62 63 64 65 66 67 68 69 70
      noAssigneeTemplate = _.template(
        '<% if (username) { %>
        <a class="author_link " 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="assign-yourself">
71
          No assignee -
72 73 74 75 76 77 78
          <a href="#" class="js-assign-yourself">
            assign yourself
          </a>
        </span>
          <% } %>'
      )

Phil Hughes's avatar
Phil Hughes committed
79
      $dropdown.glDropdown(
Phil Hughes's avatar
Phil Hughes committed
80 81 82
        data: (term, callback) =>
          @users term, (users) =>
            if term.length is 0
83 84
              showDivider = 0

Phil Hughes's avatar
Phil Hughes committed
85 86 87 88 89 90 91 92 93
              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
94
                showDivider += 1
Phil Hughes's avatar
Phil Hughes committed
95
                users.unshift(
96
                  beforeDivider: true
Phil Hughes's avatar
Phil Hughes committed
97 98 99 100 101
                  name: 'Unassigned',
                  id: 0
                )

              if showAnyUser
102
                showDivider += 1
Phil Hughes's avatar
Phil Hughes committed
103 104 105
                name = showAnyUser
                name = 'Any User' if name == true
                anyUser = {
106
                  beforeDivider: true
Phil Hughes's avatar
Phil Hughes committed
107 108 109 110
                  name: name,
                  id: null
                }
                users.unshift(anyUser)
111

112 113 114
            if showDivider
              users.splice(showDivider, 0, "divider")

115 116 117
            # Send the data back
            callback users
        filterable: true
Phil Hughes's avatar
Phil Hughes committed
118
        filterRemote: true
119 120 121
        search:
          fields: ['name', 'username']
        selectable: true
Phil Hughes's avatar
Phil Hughes committed
122
        fieldName: $dropdown.data('field-name')
123

124
        toggleLabel: (selected) ->
Phil Hughes's avatar
Phil Hughes committed
125
          if selected && 'id' of selected
126 127 128
            selected.name
          else
            defaultLabel
129

130 131
        inputId: 'issue_assignee_id'

132
        hidden: (e) ->
133 134 135
          $selectbox.hide()
          $value.show()

136
        clicked: ->
Phil Hughes's avatar
Phil Hughes committed
137 138 139
          page = $('body').data 'page'
          isIssueIndex = page is 'projects:issues:index'
          isMRIndex = page is page is 'projects:merge_requests:index'
140 141
          if $dropdown.hasClass('js-filter-bulk-update')
            return
142

Phil Hughes's avatar
Phil Hughes committed
143 144 145 146
          if $dropdown.hasClass('js-filter-submit') and (isIssueIndex or isMRIndex)
            Issues.filterResults $dropdown.closest('form')
          else if $dropdown.hasClass 'js-filter-submit'
            $dropdown.closest('form').submit()
Jacob Schatz's avatar
Jacob Schatz committed
147
          else
148 149 150
            selected = $dropdown
              .closest('.selectbox')
              .find("input[name='#{$dropdown.data('field-name')}']").val()
151
            assignTo(selected)
152

153 154
        renderRow: (user) ->
          username = if user.username then "@#{user.username}" else ""
155
          avatar = if user.avatar_url then user.avatar_url else false
156
          selected = if user.id is selectedId then "is-active" else ""
157 158
          img = ""

159 160 161
          if user.beforeDivider?
            "<li>
              <a href='#' class='#{selected}'>
162
                #{user.name}
163 164 165 166 167 168
              </a>
            </li>"
          else
            if avatar
              img = "<img src='#{avatar}' class='avatar avatar-inline' width='30' />"

169 170 171 172 173 174 175 176 177 178 179 180 181 182
          # 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>"

183

184 185 186 187
          if username is ''
            listWithUserName = ''

          listWithName + listWithUserName + listClosingTags
188 189
      )

190
    $('.ajax-users-select').each (i, select) =>
191 192
      @projectId = $(select).data('project-id')
      @groupId = $(select).data('group-id')
193
      @showCurrentUser = $(select).data('current-user')
194 195
      showNullUser = $(select).data('null-user')
      showAnyUser = $(select).data('any-user')
196
      showEmailUser = $(select).data('email-user')
197
      firstUser = $(select).data('first-user')
198

199 200 201 202
      $(select).select2
        placeholder: "Search for a user"
        multiple: $(select).hasClass('multiselect')
        minimumInputLength: 0
203 204
        query: (query) =>
          @users query.term, (users) =>
205
            data = { results: users }
206 207

            if query.term.length == 0
208 209 210 211 212 213 214
              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
215

216
              if showNullUser
217 218 219 220
                nullUser = {
                  name: 'Unassigned',
                  id: 0
                }
221
                data.results.unshift(nullUser)
222

223
              if showAnyUser
224 225
                name = showAnyUser
                name = 'Any User' if name == true
226
                anyUser = {
227
                  name: name,
228 229
                  id: null
                }
230 231
                data.results.unshift(anyUser)

232 233 234 235 236 237 238 239
            if showEmailUser && data.results.length == 0 && query.term.match(/^[^@]+@[^@]+$/)
              emailUser = {
                name: "Invite \"#{query.term}\"",
                username: query.term,
                id: query.term
              }
              data.results.unshift(emailUser)

240 241
            query.callback(data)

242 243
        initSelection: (args...) =>
          @initSelection(args...)
244 245 246 247 248 249 250 251
        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

252 253 254 255 256 257 258 259
  initSelection: (element, callback) ->
    id = $(element).val()
    if id == "0"
      nullUser = { name: 'Unassigned' }
      callback(nullUser)
    else if id != ""
      @user(id, callback)

260
  formatResult: (user) ->
261
    if user.avatar_url
262
      avatar = user.avatar_url
263
    else
264
      avatar = gon.default_avatar_url
265

266
    "<div class='user-result #{'no-username' unless user.username}'>
267 268
       <div class='user-image'><img class='avatar s24' src='#{avatar}'></div>
       <div class='user-name'>#{user.name}</div>
269
       <div class='user-username'>#{user.username || ""}</div>
270
     </div>"
271

272
  formatSelection: (user) ->
273
    user.name
274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297

  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
298
        current_user: @showCurrentUser
299 300 301 302 303
      dataType: "json"
    ).done (users) ->
      callback(users)

  buildUrl: (url) ->
Igor Matsko's avatar
Igor Matsko committed
304
    url = gon.relative_url_root.replace(/\/$/, '') + url if gon.relative_url_root?
305
    return url