angular-chosen
Version:
Angular bindings for https://github.com/harvesthq/chosen
121 lines (115 loc) • 16.5 kB
JavaScript
(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({"./src/chosen.coffee":[function(require,module,exports){
var indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; };
module.exports = angular.module('angular-chosen', []);
angular.module('angular-chosen').directive('chosen', [
'$timeout', function($timeout) {
var CHOSEN_OPTION_WHITELIST, NG_OPTIONS_REGEXP, isEmpty, snakeCase;
NG_OPTIONS_REGEXP = /^\s*(.*?)(?:\s+as\s+(.*?))?(?:\s+group\s+by\s+(.*))?\s+for\s+(?:([\$\w][\$\w]*)|(?:\(\s*([\$\w][\$\w]*)\s*,\s*([\$\w][\$\w]*)\s*\)))\s+in\s+(.*?)(?:\s+track\s+by\s+(.*?))?$/;
CHOSEN_OPTION_WHITELIST = ['noResultsText', 'allowSingleDeselect', 'disableSearchThreshold', 'disableSearch', 'enableSplitWordSearch', 'inheritSelectClasses', 'maxSelectedOptions', 'placeholderTextMultiple', 'placeholderTextSingle', 'searchContains', 'singleBackstrokeDelete', 'displayDisabledOptions', 'displaySelectedOptions', 'width'];
snakeCase = function(input) {
return input.replace(/[A-Z]/g, function($1) {
return "_" + ($1.toLowerCase());
});
};
isEmpty = function(value) {
var key;
if (angular.isArray(value)) {
return value.length === 0;
} else if (angular.isObject(value)) {
for (key in value) {
if (value.hasOwnProperty(key)) {
return false;
}
}
}
return true;
};
return {
restrict: 'A',
require: '?ngModel',
terminal: true,
link: function(scope, element, attr, ngModel) {
var chosen, defaultText, disableWithMessage, empty, initOrUpdate, match, options, origRender, removeEmptyMessage, startLoading, stopLoading, valuesExpr, viewWatch;
element.addClass('angular-chosen');
options = scope.$eval(attr.chosen) || {};
angular.forEach(attr, function(value, key) {
if (indexOf.call(CHOSEN_OPTION_WHITELIST, key) >= 0) {
return options[snakeCase(key)] = scope.$eval(value);
}
});
startLoading = function() {
return element.addClass('loading').attr('disabled', true).trigger('chosen:updated');
};
stopLoading = function() {
return element.removeClass('loading').attr('disabled', false).trigger('chosen:updated');
};
chosen = null;
defaultText = null;
empty = false;
initOrUpdate = function() {
if (chosen) {
return element.trigger('chosen:updated');
} else {
chosen = element.chosen(options).data('chosen');
if (angular.isObject(chosen)) {
return defaultText = chosen.default_text;
}
}
};
removeEmptyMessage = function() {
empty = false;
return element.attr('data-placeholder', defaultText);
};
disableWithMessage = function() {
empty = true;
return element.attr('data-placeholder', chosen.results_none_found).attr('disabled', true).trigger('chosen:updated');
};
if (ngModel) {
origRender = ngModel.$render;
ngModel.$render = function() {
origRender();
return initOrUpdate();
};
if (attr.multiple) {
viewWatch = function() {
return ngModel.$viewValue;
};
scope.$watch(viewWatch, ngModel.$render, true);
}
} else {
initOrUpdate();
}
attr.$observe('disabled', function() {
return element.trigger('chosen:updated');
});
if (attr.ngOptions && ngModel) {
match = attr.ngOptions.match(NG_OPTIONS_REGEXP);
valuesExpr = match[7];
scope.$watchCollection(valuesExpr, function(newVal, oldVal) {
var timer;
return timer = $timeout(function() {
if (angular.isUndefined(newVal)) {
return startLoading();
} else {
if (empty) {
removeEmptyMessage();
}
stopLoading();
if (isEmpty(newVal)) {
return disableWithMessage();
}
}
});
});
return scope.$on('$destroy', function(event) {
if (typeof timer !== "undefined" && timer !== null) {
return $timeout.cancel(timer);
}
});
}
}
};
}
]);
},{}]},{},["./src/chosen.coffee"])
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/vitaly/projects/angular-chosen/src/chosen.coffee"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","module.exports = angular.module('angular-chosen', [])\n\nangular.module('angular-chosen').directive 'chosen', ['$timeout', ($timeout) ->\n\n  # This is stolen from Angular...\n  NG_OPTIONS_REGEXP = /^\\s*(.*?)(?:\\s+as\\s+(.*?))?(?:\\s+group\\s+by\\s+(.*))?\\s+for\\s+(?:([\\$\\w][\\$\\w]*)|(?:\\(\\s*([\\$\\w][\\$\\w]*)\\s*,\\s*([\\$\\w][\\$\\w]*)\\s*\\)))\\s+in\\s+(.*?)(?:\\s+track\\s+by\\s+(.*?))?$/\n\n  # Whitelist of options that will be parsed from the element's attributes and passed into Chosen\n  CHOSEN_OPTION_WHITELIST = [\n    'noResultsText'\n    'allowSingleDeselect'\n    'disableSearchThreshold'\n    'disableSearch'\n    'enableSplitWordSearch'\n    'inheritSelectClasses'\n    'maxSelectedOptions'\n    'placeholderTextMultiple'\n    'placeholderTextSingle'\n    'searchContains'\n    'singleBackstrokeDelete'\n    'displayDisabledOptions'\n    'displaySelectedOptions'\n    'width'\n  ]\n\n  snakeCase = (input) -> input.replace /[A-Z]/g, ($1) -> \"_#{$1.toLowerCase()}\"\n  isEmpty = (value) ->\n    if angular.isArray(value)\n      return value.length is 0\n    else if angular.isObject(value)\n      return false for key of value when value.hasOwnProperty(key)\n    true\n\n  restrict: 'A'\n  require: '?ngModel'\n  terminal: true\n  link: (scope, element, attr, ngModel) ->\n\n    element.addClass('angular-chosen')\n\n    # Take a hash of options from the chosen directive\n    options = scope.$eval(attr.chosen) or {}\n\n    # Options defined as attributes take precedence\n    angular.forEach attr, (value, key) ->\n      options[snakeCase(key)] = scope.$eval(value) if key in CHOSEN_OPTION_WHITELIST\n\n    startLoading = -> element.addClass('loading').attr('disabled', true).trigger('chosen:updated')\n    stopLoading = -> element.removeClass('loading').attr('disabled', false).trigger('chosen:updated')\n\n    chosen = null\n    defaultText = null\n    empty = false\n\n    initOrUpdate = ->\n      if chosen\n        element.trigger('chosen:updated')\n      else\n        chosen = element.chosen(options).data('chosen')\n        if angular.isObject(chosen)\n          defaultText = chosen.default_text\n\n    # Use Chosen's placeholder or no results found text depending on whether there are options available\n    removeEmptyMessage = ->\n      empty = false\n      element.attr('data-placeholder', defaultText)\n\n    disableWithMessage = ->\n      empty = true\n      element.attr('data-placeholder', chosen.results_none_found).attr('disabled', true).trigger('chosen:updated')\n\n    # Watch the underlying ngModel for updates and trigger an update when they occur.\n    if ngModel\n      origRender = ngModel.$render\n      ngModel.$render = ->\n        origRender()\n        initOrUpdate()\n\n      # This is basically taken from angular ngOptions source.  ngModel watches reference, not value,\n      # so when values are added or removed from array ngModels, $render won't be fired.\n      if attr.multiple\n        viewWatch = -> ngModel.$viewValue\n        scope.$watch viewWatch, ngModel.$render, true\n    # If we're not using ngModel (and therefore also not using ngOptions, which requires ngModel),\n    # just initialize chosen immediately since there's no need to wait for ngOptions to render first\n    else initOrUpdate()\n\n    # Watch the disabled attribute (could be set by ngDisabled)\n    attr.$observe 'disabled', -> element.trigger('chosen:updated')\n\n    # Watch the collection in ngOptions and update chosen when it changes.  This works with promises!\n    # ngOptions doesn't do anything unless there is an ngModel, so neither do we.\n    if attr.ngOptions and ngModel\n      match = attr.ngOptions.match(NG_OPTIONS_REGEXP)\n      valuesExpr = match[7]\n\n      scope.$watchCollection valuesExpr, (newVal, oldVal) ->\n        # Defer execution until DOM is loaded\n        timer = $timeout(->\n          if angular.isUndefined(newVal)\n            startLoading()\n          else\n            removeEmptyMessage() if empty\n            stopLoading()\n            disableWithMessage() if isEmpty(newVal)\n        )\n\n      scope.$on '$destroy', (event) ->\n        $timeout.cancel timer if timer?\n]\n"]}