issuable.js.es6 6.94 KB
Newer Older
1
/* eslint-disable no-param-reassign, func-names, no-var, camelcase, no-unused-vars, object-shorthand, space-before-function-paren, no-return-assign, comma-dangle, consistent-return, one-var, one-var-declaration-per-line, quotes, prefer-template, prefer-arrow-callback, wrap-iife, max-len */
2 3
/* global Issuable */

4
((global) => {
Fatih Acet's avatar
Fatih Acet committed
5 6 7 8
  var issuable_created;

  issuable_created = false;

9
  global.Issuable = {
Fatih Acet's avatar
Fatih Acet committed
10
    init: function() {
Phil Hughes's avatar
Phil Hughes committed
11 12 13
      Issuable.initTemplates();
      Issuable.initSearch();
      Issuable.initChecks();
14
      Issuable.initResetFilters();
15
      Issuable.resetIncomingEmailToken();
Phil Hughes's avatar
Phil Hughes committed
16
      return Issuable.initLabelFilterRemove();
Fatih Acet's avatar
Fatih Acet committed
17 18 19 20 21
    },
    initTemplates: function() {
      return Issuable.labelRow = _.template('<% _.each(labels, function(label){ %> <span class="label-row btn-group" role="group" aria-label="<%- label.title %>" style="color: <%- label.text_color %>;"> <a href="#" class="btn btn-transparent has-tooltip" style="background-color: <%- label.color %>;" title="<%- label.description %>" data-container="body"> <%- label.title %> </a> <button type="button" class="btn btn-transparent label-remove js-label-filter-remove" style="background-color: <%- label.color %>;" data-label="<%- label.title %>"> <i class="fa fa-times"></i> </button> </span> <% }); %>');
    },
    initSearch: function() {
22 23 24 25
      const $searchInput = $('#issuable_search');

      Issuable.initSearchState($searchInput);

26
      // `immediate` param set to false debounces on the `trailing` edge, lets user finish typing
27
      const debouncedExecSearch = _.debounce(Issuable.executeSearch, 1000, false);
28

29
      $searchInput.off('keyup').on('keyup', debouncedExecSearch);
30 31

      // ensures existing filters are preserved when manually submitted
32
      $('#issuable_search_form').on('submit', (e) => {
33 34
        e.preventDefault();
        debouncedExecSearch(e);
Fatih Acet's avatar
Fatih Acet committed
35
      });
36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58
    },
    initSearchState: function($searchInput) {
      const currentSearchVal = $searchInput.val();

      Issuable.searchState = {
        elem: $searchInput,
        current: currentSearchVal
      };

      Issuable.maybeFocusOnSearch();
    },
    accessSearchPristine: function(set) {
      // store reference to previous value to prevent search on non-mutating keyup
      const state = Issuable.searchState;
      const currentSearchVal = state.elem.val();

      if (set) {
        state.current = currentSearchVal;
      } else {
        return state.current === currentSearchVal;
      }
    },
    maybeFocusOnSearch: function() {
59
      const currentSearchVal = Issuable.searchState.current;
60
      if (currentSearchVal && currentSearchVal !== '') {
61 62 63 64 65 66 67 68 69 70 71 72 73 74
        const queryLength = currentSearchVal.length;
        const $searchInput = Issuable.searchState.elem;

      /* The following ensures that the cursor is initially placed at
        * the end of search input when focus is applied. It accounts
        * for differences in browser implementations of `setSelectionRange`
        * and cursor placement for elements in focus.
      */
        $searchInput.focus();
        if ($searchInput.setSelectionRange) {
          $searchInput.setSelectionRange(queryLength, queryLength);
        } else {
          $searchInput.val(currentSearchVal);
        }
75
      }
Fatih Acet's avatar
Fatih Acet committed
76
    },
77 78 79 80 81 82
    executeSearch: function(e) {
      const $search = $('#issuable_search');
      const $searchName = $search.attr('name');
      const $searchValue = $search.val();
      const $filtersForm = $('.js-filter-form');
      const $input = $(`input[name='${$searchName}']`, $filtersForm);
83 84 85 86 87
      const isPristine = Issuable.accessSearchPristine();

      if (isPristine) {
        return;
      }
88 89 90 91 92 93 94 95 96

      if (!$input.length) {
        $filtersForm.append(`<input type='hidden' name='${$searchName}' value='${_.escape($searchValue)}'/>`);
      } else {
        $input.val($searchValue);
      }

      Issuable.filterResults($filtersForm);
    },
Fatih Acet's avatar
Fatih Acet committed
97 98 99 100
    initLabelFilterRemove: function() {
      return $(document).off('click', '.js-label-filter-remove').on('click', '.js-label-filter-remove', function(e) {
        var $button;
        $button = $(this);
101
        // Remove the label input box
Fatih Acet's avatar
Fatih Acet committed
102 103 104
        $('input[name="label_name[]"]').filter(function() {
          return this.value === $button.data('label');
        }).remove();
105
        // Submit the form to get new data
Fatih Acet's avatar
Fatih Acet committed
106 107 108 109 110 111
        Issuable.filterResults($('.filter-form'));
      });
    },
    filterResults: (function(_this) {
      return function(form) {
        var formAction, formData, issuesUrl;
112 113 114 115 116
        formData = form.serializeArray();
        formData = formData.filter(function(data) {
          return data.value !== '';
        });
        formData = $.param(formData);
Fatih Acet's avatar
Fatih Acet committed
117 118 119 120
        formAction = form.attr('action');
        issuesUrl = formAction;
        issuesUrl += "" + (formAction.indexOf('?') < 0 ? '?' : '&');
        issuesUrl += formData;
Bryce Johnson's avatar
Bryce Johnson committed
121
        return gl.utils.visitUrl(issuesUrl);
Fatih Acet's avatar
Fatih Acet committed
122 123
      };
    })(this),
124 125 126 127 128 129 130 131
    initResetFilters: function() {
      $('.reset-filters').on('click', function(e) {
        e.preventDefault();
        const target = e.target;
        const $form = $(target).parents('.js-filter-form');
        const baseIssuesUrl = target.href;

        $form.attr('action', baseIssuesUrl);
Bryce Johnson's avatar
Bryce Johnson committed
132
        gl.utils.visitUrl(baseIssuesUrl);
133 134
      });
    },
Fatih Acet's avatar
Fatih Acet committed
135 136 137 138 139 140 141 142 143
    initChecks: function() {
      this.issuableBulkActions = $('.bulk-update').data('bulkActions');
      $('.check_all_issues').off('click').on('click', function() {
        $('.selected_issue').prop('checked', this.checked);
        return Issuable.checkChanged();
      });
      return $('.selected_issue').off('change').on('change', Issuable.checkChanged.bind(this));
    },
    checkChanged: function() {
144
      const $checkedIssues = $('.selected_issue:checked');
145
      const $updateIssuesIds = $('#update_issuable_ids');
146 147 148
      const $issuesOtherFilters = $('.issues-other-filters');
      const $issuesBulkUpdate = $('.issues_bulk_update');

149 150 151
      this.issuableBulkActions.willUpdateLabels = false;
      this.issuableBulkActions.setOriginalDropdownData();

152
      if ($checkedIssues.length > 0) {
153
        const ids = $.map($checkedIssues, function(value) {
Fatih Acet's avatar
Fatih Acet committed
154 155
          return $(value).data('id');
        });
156 157 158
        $updateIssuesIds.val(ids);
        $issuesOtherFilters.hide();
        $issuesBulkUpdate.show();
Fatih Acet's avatar
Fatih Acet committed
159
      } else {
160 161 162
        $updateIssuesIds.val([]);
        $issuesBulkUpdate.hide();
        $issuesOtherFilters.show();
Fatih Acet's avatar
Fatih Acet committed
163 164
      }
      return true;
165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185
    },

    resetIncomingEmailToken: function() {
      $('.incoming-email-token-reset').on('click', function(e) {
        e.preventDefault();

        $.ajax({
          type: 'PUT',
          url: $('.incoming-email-token-reset').attr('href'),
          dataType: 'json',
          success: function(response) {
            $('#issue_email').val(response.new_issue_address).focus();
          },
          beforeSend: function() {
            $('.incoming-email-token-reset').text('resetting...');
          },
          complete: function() {
            $('.incoming-email-token-reset').text('reset it');
          }
        });
      });
Fatih Acet's avatar
Fatih Acet committed
186 187
    }
  };
188
})(window);