Skip to content
Snippets Groups Projects

Compare revisions

Changes are shown as if the source revision was being merged into the target revision. Learn more about comparing revisions.

Source

Select target project
No results found
Select Git revision
Loading items

Target

Select target project
  • to/keycloak-theme-pirati
  • marek.krejpsky/keycloak-theme-pirati
2 results
Select Git revision
Loading items
Show changes
Showing
with 0 additions and 1257 deletions
/**
* Created by Allen Zou on 2016/10/13.
*/
"use strict";
require("./view/tree.less");
var fileIcon = require("./view/imgs/file.png");
var folderIcon = require("./view/imgs/folder.png");
var closedFolderIcon = require("./view/imgs/folder-closed.png");
var plusIcon = require("./view/imgs/plus.png");
var removeIcon = require("./view/imgs/remove.png");
var tree = angular.module("angular.tree", []);
tree
.directive("treeNode", function () {
return {
scope: {
item: "=",
adapter: "=",
icon: "=",
folderOpen: "=",
folderClose: "=",
nodeClick: "=",
childrenLoader: "=",
addItem: "=",
removeItem: "=",
editItem: "="
},
require: [],
restrict: "E",
// templateUrl: "directive/tree/node.html",
template: require("./view/node.html"),
link: function($scope, element, attributes, controllers) {
$scope.open = false;
$scope.add_btn = plusIcon;
$scope.remove_btn = removeIcon;
function load_children() {
if ($scope.childrenLoader) {
$scope.childrenLoader($scope.item)
.then(function(children) {
$scope.subNodes = children;
})
.catch(function(error) {
console.error(error);
$scope.subNodes = [];
})
} else {
$scope.subNodes = [];
}
}
$scope.wrap_node_click = function() {
if ($scope.item) {
var adaptedItem = $scope.adapter($scope.item);
if (adaptedItem.type === "branch") {
if ($scope.open) {
$scope.open = false;
$scope.folderClose && $scope.folderClose($scope.item);
}
else {
$scope.open = true;
$scope.folderOpen && $scope.folderOpen($scope.item);
load_children();
}
}
$scope.nodeClick && $scope.nodeClick($scope.item);
}
return false;
};
$scope.resolve_icon = function() {
var icon = null;
var adaptedItem = $scope.adapter($scope.item);
if (adaptedItem.type === 'branch') {
icon = ($scope.icon && $scope.icon($scope.item, $scope.open))
|| (!$scope.open && closedFolderIcon)
|| ($scope.open && folderIcon);
}
else {
icon = ($scope.icon && $scope.icon($scope.item))
|| fileIcon;
}
return icon;
};
$scope.node_class = function() {
var classes = ["node"];
var adaptedItem = $scope.adapter($scope.item);
if (adaptedItem.type === 'branch') {
classes.push("branch");
if ($scope.open) {
classes.push("open");
}
else {
classes.push("closed");
}
}
else {
classes.push("leaf");
}
return classes;
};
$scope.add_child = function() {
if ($scope.addItem) {
$scope.addItem($scope.item)
.then(function() {
load_children();
})
;
}
return false;
};
$scope.remove_self = function() {
if ($scope.removeItem) {
$scope.removeItem($scope.item)
.then(function() {
load_children();
})
;
}
return false;
};
$scope.edit = function() {
console.log("edit:::");
console.log($scope.editItem);
$scope.editItem && $scope.editItem($scope.item);
return false;
};
}
};
})
.directive("tree", function () {
var link = function($scope, element, attributes, controllers) {
$scope.itemAdapter = $scope.adapter || function(item) {
console.log("in tree .adapter");
return item;
};
$scope.tree_class = function() {
var classes = ["tree"];
return classes;
}
};
return {
scope: {
root: "=root",
adapter: "=",
icon: "=",
folderOpen: "=",
folderClose: "=",
nodeClick: "=",
childrenLoader: "=",
addItem: "=",
removeItem: "=",
editItem: "="
},
require: [],
restrict: "E",
// templateUrl: "directive/tree/tree.html",
template: require("./view/tree.html"),
link: link
}
})
;
module.exports = tree;
<div ng-class="node_class()">
<div class="directory-level" ng-click="wrap_node_click()">
<img class="icon" ng-src="{{ resolve_icon() }}">
<span>{{ adapter(item).text }}</span>
<div class="operation" ng-click="$event.stopPropagation()">
<a href class="add" ng-click="add_child()" ng-if="adapter(item).type==='branch'">
<img ng-src="{{ add_btn }}">
</a>
<a href class="remove" ng-click="remove_self()">
<img ng-src="{{ remove_btn }}">
</a>
<a href class="edit" ng-click="edit()">
<span class="glyphicon glyphicon-edit"></span>
</a>
</div>
</div>
<div class="sub-node" ng-if="open" ng-repeat="node in subNodes">
<tree-node item="node" adapter="adapter" icon="icon"
folder-open="folderOpen" folder-close="folderClose"
node-click="nodeClick" children-loader="childrenLoader"
add-item="addItem" remove-item="removeItem" edit-item="editItem">
</tree-node>
</div>
</div>
\ No newline at end of file
<div ng-class="tree_class()">
<tree-node item="root" adapter="itemAdapter" icon="icon"
folder-open="folderOpen" folder-close="folderClose"
node-click="nodeClick" children-loader="childrenLoader"
add-item="addItem" remove-item="removeItem" edit-item="editItem">
</tree-node>
</div>
\ No newline at end of file
.tree {
@node-height: 16px;
overflow: auto;
.node {
width: 100%;
.directory-level {
position: relative;
padding-right: 4px;
white-space: nowrap;
font-size: @node-height;
line-height: @node-height;
>.icon {
height: @node-height;
}
.operation {
display: inline;
margin-left: 20px;
visibility: hidden;
img {
height: @node-height;
}
}
&:hover {
.operation {
visibility: visible;
}
}
}
.sub-node {
padding-left: 14px;
}
}
}
\ No newline at end of file
/******/ (function(modules) { // webpackBootstrap
/******/ // The module cache
/******/ var installedModules = {};
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/ // Check if module is in cache
/******/ if(installedModules[moduleId])
/******/ return installedModules[moduleId].exports;
/******/ // Create a new module (and put it into the cache)
/******/ var module = installedModules[moduleId] = {
/******/ exports: {},
/******/ id: moduleId,
/******/ loaded: false
/******/ };
/******/ // Execute the module function
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
/******/ // Flag the module as loaded
/******/ module.loaded = true;
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
/******/ // expose the modules object (__webpack_modules__)
/******/ __webpack_require__.m = modules;
/******/ // expose the module cache
/******/ __webpack_require__.c = installedModules;
/******/ // __webpack_public_path__
/******/ __webpack_require__.p = "";
/******/ // Load entry module and return exports
/******/ return __webpack_require__(0);
/******/ })
/************************************************************************/
/******/ ([
/* 0 */
/***/ function(module, exports, __webpack_require__) {
/**
* Created by Allen Zou on 2016/10/13.
*/
"use strict";
__webpack_require__(6);
var fileIcon = __webpack_require__(7);
var folderIcon = __webpack_require__(9);
var closedFolderIcon = __webpack_require__(8);
var plusIcon = __webpack_require__(10);
var removeIcon = __webpack_require__(11);
var tree = angular.module("angular.tree", []);
tree
.directive("treeNode", function () {
return {
scope: {
item: "=",
adapter: "=",
icon: "=",
folderOpen: "=",
folderClose: "=",
nodeClick: "=",
childrenLoader: "=",
addItem: "=",
removeItem: "=",
editItem: "="
},
require: [],
restrict: "E",
// templateUrl: "directive/tree/node.html",
template: __webpack_require__(3),
link: function($scope, element, attributes, controllers) {
$scope.open = false;
$scope.add_btn = plusIcon;
$scope.remove_btn = removeIcon;
function load_children() {
if ($scope.childrenLoader) {
$scope.childrenLoader($scope.item)
.then(function(children) {
$scope.subNodes = children;
})
.catch(function(error) {
console.error(error);
$scope.subNodes = [];
})
} else {
$scope.subNodes = [];
}
}
$scope.wrap_node_click = function() {
if ($scope.item) {
var adaptedItem = $scope.adapter($scope.item);
if (adaptedItem.type === "branch") {
if ($scope.open) {
$scope.open = false;
$scope.folderClose && $scope.folderClose($scope.item);
}
else {
$scope.open = true;
$scope.folderOpen && $scope.folderOpen($scope.item);
load_children();
}
}
$scope.nodeClick && $scope.nodeClick($scope.item);
}
return false;
};
$scope.resolve_icon = function() {
var icon = null;
var adaptedItem = $scope.adapter($scope.item);
if (adaptedItem.type === 'branch') {
icon = ($scope.icon && $scope.icon($scope.item, $scope.open))
|| (!$scope.open && closedFolderIcon)
|| ($scope.open && folderIcon);
}
else {
icon = ($scope.icon && $scope.icon($scope.item))
|| fileIcon;
}
return icon;
};
$scope.node_class = function() {
var classes = ["node"];
var adaptedItem = $scope.adapter($scope.item);
if (adaptedItem.type === 'branch') {
classes.push("branch");
if ($scope.open) {
classes.push("open");
}
else {
classes.push("closed");
}
}
else {
classes.push("leaf");
}
return classes;
};
$scope.add_child = function() {
if ($scope.addItem) {
$scope.addItem($scope.item)
.then(function() {
load_children();
})
;
}
return false;
};
$scope.remove_self = function() {
if ($scope.removeItem) {
$scope.removeItem($scope.item)
.then(function() {
load_children();
})
;
}
return false;
};
$scope.edit = function() {
console.log("edit:::");
console.log($scope.editItem);
$scope.editItem && $scope.editItem($scope.item);
return false;
};
}
};
})
.directive("tree", function () {
var link = function($scope, element, attributes, controllers) {
$scope.itemAdapter = $scope.adapter || function(item) {
console.log("in tree .adapter");
return item;
};
$scope.tree_class = function() {
var classes = ["tree"];
return classes;
}
};
return {
scope: {
root: "=root",
adapter: "=",
icon: "=",
folderOpen: "=",
folderClose: "=",
nodeClick: "=",
childrenLoader: "=",
addItem: "=",
removeItem: "=",
editItem: "="
},
require: [],
restrict: "E",
// templateUrl: "directive/tree/tree.html",
template: __webpack_require__(4),
link: link
}
})
;
module.exports = tree;
/***/ },
/* 1 */
/***/ function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(2)();
// imports
// module
exports.push([module.id, ".tree {\n overflow: auto;\n}\n.tree .node {\n width: 100%;\n}\n.tree .node .directory-level {\n position: relative;\n padding-right: 4px;\n white-space: nowrap;\n font-size: 16px;\n line-height: 16px;\n}\n.tree .node .directory-level > .icon {\n height: 16px;\n}\n.tree .node .directory-level .operation {\n display: inline;\n margin-left: 20px;\n visibility: hidden;\n}\n.tree .node .directory-level .operation img {\n height: 16px;\n}\n.tree .node .directory-level:hover .operation {\n visibility: visible;\n}\n.tree .node .sub-node {\n padding-left: 14px;\n}\n", ""]);
// exports
/***/ },
/* 2 */
/***/ function(module, exports) {
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
// css base code, injected by the css-loader
module.exports = function() {
var list = [];
// return the list of modules as css string
list.toString = function toString() {
var result = [];
for(var i = 0; i < this.length; i++) {
var item = this[i];
if(item[2]) {
result.push("@media " + item[2] + "{" + item[1] + "}");
} else {
result.push(item[1]);
}
}
return result.join("");
};
// import a list of modules into the list
list.i = function(modules, mediaQuery) {
if(typeof modules === "string")
modules = [[null, modules, ""]];
var alreadyImportedModules = {};
for(var i = 0; i < this.length; i++) {
var id = this[i][0];
if(typeof id === "number")
alreadyImportedModules[id] = true;
}
for(i = 0; i < modules.length; i++) {
var item = modules[i];
// skip already imported module
// this implementation is not 100% perfect for weird media query combinations
// when a module is imported multiple times with different media queries.
// I hope this will never occur (Hey this way we have smaller bundles)
if(typeof item[0] !== "number" || !alreadyImportedModules[item[0]]) {
if(mediaQuery && !item[2]) {
item[2] = mediaQuery;
} else if(mediaQuery) {
item[2] = "(" + item[2] + ") and (" + mediaQuery + ")";
}
list.push(item);
}
}
};
return list;
};
/***/ },
/* 3 */
/***/ function(module, exports) {
module.exports = "<div ng-class=\"node_class()\">\n <div class=\"directory-level\" ng-click=\"wrap_node_click()\">\n <img class=\"icon\" ng-src=\"{{ resolve_icon() }}\">\n <span>{{ adapter(item).text }}</span>\n <div class=\"operation\" ng-click=\"$event.stopPropagation()\">\n <a href class=\"add\" ng-click=\"add_child()\" ng-if=\"adapter(item).type==='branch'\">\n <img ng-src=\"{{ add_btn }}\">\n </a>\n <a href class=\"remove\" ng-click=\"remove_self()\">\n <img ng-src=\"{{ remove_btn }}\">\n </a>\n <a href class=\"edit\" ng-click=\"edit()\">\n <span class=\"glyphicon glyphicon-edit\"></span>\n </a>\n </div>\n </div>\n <div class=\"sub-node\" ng-if=\"open\" ng-repeat=\"node in subNodes\">\n <tree-node item=\"node\" adapter=\"adapter\" icon=\"icon\"\n folder-open=\"folderOpen\" folder-close=\"folderClose\"\n node-click=\"nodeClick\" children-loader=\"childrenLoader\"\n add-item=\"addItem\" remove-item=\"removeItem\" edit-item=\"editItem\">\n </tree-node>\n </div>\n</div>";
/***/ },
/* 4 */
/***/ function(module, exports) {
module.exports = "<div ng-class=\"tree_class()\">\n <tree-node item=\"root\" adapter=\"itemAdapter\" icon=\"icon\"\n folder-open=\"folderOpen\" folder-close=\"folderClose\"\n node-click=\"nodeClick\" children-loader=\"childrenLoader\"\n add-item=\"addItem\" remove-item=\"removeItem\" edit-item=\"editItem\">\n </tree-node>\n</div>";
/***/ },
/* 5 */
/***/ function(module, exports, __webpack_require__) {
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
var stylesInDom = {},
memoize = function(fn) {
var memo;
return function () {
if (typeof memo === "undefined") memo = fn.apply(this, arguments);
return memo;
};
},
isOldIE = memoize(function() {
return /msie [6-9]\b/.test(window.navigator.userAgent.toLowerCase());
}),
getHeadElement = memoize(function () {
return document.head || document.getElementsByTagName("head")[0];
}),
singletonElement = null,
singletonCounter = 0,
styleElementsInsertedAtTop = [];
module.exports = function(list, options) {
if(false) {
if(typeof document !== "object") throw new Error("The style-loader cannot be used in a non-browser environment");
}
options = options || {};
// Force single-tag solution on IE6-9, which has a hard limit on the # of <style>
// tags it will allow on a page
if (typeof options.singleton === "undefined") options.singleton = isOldIE();
// By default, add <style> tags to the bottom of <head>.
if (typeof options.insertAt === "undefined") options.insertAt = "bottom";
var styles = listToStyles(list);
addStylesToDom(styles, options);
return function update(newList) {
var mayRemove = [];
for(var i = 0; i < styles.length; i++) {
var item = styles[i];
var domStyle = stylesInDom[item.id];
domStyle.refs--;
mayRemove.push(domStyle);
}
if(newList) {
var newStyles = listToStyles(newList);
addStylesToDom(newStyles, options);
}
for(var i = 0; i < mayRemove.length; i++) {
var domStyle = mayRemove[i];
if(domStyle.refs === 0) {
for(var j = 0; j < domStyle.parts.length; j++)
domStyle.parts[j]();
delete stylesInDom[domStyle.id];
}
}
};
}
function addStylesToDom(styles, options) {
for(var i = 0; i < styles.length; i++) {
var item = styles[i];
var domStyle = stylesInDom[item.id];
if(domStyle) {
domStyle.refs++;
for(var j = 0; j < domStyle.parts.length; j++) {
domStyle.parts[j](item.parts[j]);
}
for(; j < item.parts.length; j++) {
domStyle.parts.push(addStyle(item.parts[j], options));
}
} else {
var parts = [];
for(var j = 0; j < item.parts.length; j++) {
parts.push(addStyle(item.parts[j], options));
}
stylesInDom[item.id] = {id: item.id, refs: 1, parts: parts};
}
}
}
function listToStyles(list) {
var styles = [];
var newStyles = {};
for(var i = 0; i < list.length; i++) {
var item = list[i];
var id = item[0];
var css = item[1];
var media = item[2];
var sourceMap = item[3];
var part = {css: css, media: media, sourceMap: sourceMap};
if(!newStyles[id])
styles.push(newStyles[id] = {id: id, parts: [part]});
else
newStyles[id].parts.push(part);
}
return styles;
}
function insertStyleElement(options, styleElement) {
var head = getHeadElement();
var lastStyleElementInsertedAtTop = styleElementsInsertedAtTop[styleElementsInsertedAtTop.length - 1];
if (options.insertAt === "top") {
if(!lastStyleElementInsertedAtTop) {
head.insertBefore(styleElement, head.firstChild);
} else if(lastStyleElementInsertedAtTop.nextSibling) {
head.insertBefore(styleElement, lastStyleElementInsertedAtTop.nextSibling);
} else {
head.appendChild(styleElement);
}
styleElementsInsertedAtTop.push(styleElement);
} else if (options.insertAt === "bottom") {
head.appendChild(styleElement);
} else {
throw new Error("Invalid value for parameter 'insertAt'. Must be 'top' or 'bottom'.");
}
}
function removeStyleElement(styleElement) {
styleElement.parentNode.removeChild(styleElement);
var idx = styleElementsInsertedAtTop.indexOf(styleElement);
if(idx >= 0) {
styleElementsInsertedAtTop.splice(idx, 1);
}
}
function createStyleElement(options) {
var styleElement = document.createElement("style");
styleElement.type = "text/css";
insertStyleElement(options, styleElement);
return styleElement;
}
function createLinkElement(options) {
var linkElement = document.createElement("link");
linkElement.rel = "stylesheet";
insertStyleElement(options, linkElement);
return linkElement;
}
function addStyle(obj, options) {
var styleElement, update, remove;
if (options.singleton) {
var styleIndex = singletonCounter++;
styleElement = singletonElement || (singletonElement = createStyleElement(options));
update = applyToSingletonTag.bind(null, styleElement, styleIndex, false);
remove = applyToSingletonTag.bind(null, styleElement, styleIndex, true);
} else if(obj.sourceMap &&
typeof URL === "function" &&
typeof URL.createObjectURL === "function" &&
typeof URL.revokeObjectURL === "function" &&
typeof Blob === "function" &&
typeof btoa === "function") {
styleElement = createLinkElement(options);
update = updateLink.bind(null, styleElement);
remove = function() {
removeStyleElement(styleElement);
if(styleElement.href)
URL.revokeObjectURL(styleElement.href);
};
} else {
styleElement = createStyleElement(options);
update = applyToTag.bind(null, styleElement);
remove = function() {
removeStyleElement(styleElement);
};
}
update(obj);
return function updateStyle(newObj) {
if(newObj) {
if(newObj.css === obj.css && newObj.media === obj.media && newObj.sourceMap === obj.sourceMap)
return;
update(obj = newObj);
} else {
remove();
}
};
}
var replaceText = (function () {
var textStore = [];
return function (index, replacement) {
textStore[index] = replacement;
return textStore.filter(Boolean).join('\n');
};
})();
function applyToSingletonTag(styleElement, index, remove, obj) {
var css = remove ? "" : obj.css;
if (styleElement.styleSheet) {
styleElement.styleSheet.cssText = replaceText(index, css);
} else {
var cssNode = document.createTextNode(css);
var childNodes = styleElement.childNodes;
if (childNodes[index]) styleElement.removeChild(childNodes[index]);
if (childNodes.length) {
styleElement.insertBefore(cssNode, childNodes[index]);
} else {
styleElement.appendChild(cssNode);
}
}
}
function applyToTag(styleElement, obj) {
var css = obj.css;
var media = obj.media;
if(media) {
styleElement.setAttribute("media", media)
}
if(styleElement.styleSheet) {
styleElement.styleSheet.cssText = css;
} else {
while(styleElement.firstChild) {
styleElement.removeChild(styleElement.firstChild);
}
styleElement.appendChild(document.createTextNode(css));
}
}
function updateLink(linkElement, obj) {
var css = obj.css;
var sourceMap = obj.sourceMap;
if(sourceMap) {
// http://stackoverflow.com/a/26603875
css += "\n/*# sourceMappingURL=data:application/json;base64," + btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))) + " */";
}
var blob = new Blob([css], { type: "text/css" });
var oldSrc = linkElement.href;
linkElement.href = URL.createObjectURL(blob);
if(oldSrc)
URL.revokeObjectURL(oldSrc);
}
/***/ },
/* 6 */
/***/ function(module, exports, __webpack_require__) {
// style-loader: Adds some css to the DOM by adding a <style> tag
// load the styles
var content = __webpack_require__(1);
if(typeof content === 'string') content = [[module.id, content, '']];
// add the styles to the DOM
var update = __webpack_require__(5)(content, {});
if(content.locals) module.exports = content.locals;
// Hot Module Replacement
if(false) {
// When the styles change, update the <style> tags
if(!content.locals) {
module.hot.accept("!!./../../node_modules/css-loader/index.js!./../../node_modules/less-loader/index.js!./tree.less", function() {
var newContent = require("!!./../../node_modules/css-loader/index.js!./../../node_modules/less-loader/index.js!./tree.less");
if(typeof newContent === 'string') newContent = [[module.id, newContent, '']];
update(newContent);
});
}
// When the module is disposed, remove the <style> tags
module.hot.dispose(function() { update(); });
}
/***/ },
/* 7 */
/***/ function(module, exports) {
module.exports = ""
/***/ },
/* 8 */
/***/ function(module, exports) {
module.exports = ""
/***/ },
/* 9 */
/***/ function(module, exports) {
module.exports = ""
/***/ },
/* 10 */
/***/ function(module, exports) {
module.exports = ""
/***/ },
/* 11 */
/***/ function(module, exports) {
module.exports = ""
/***/ }
/******/ ]);
\ No newline at end of file
{
"directory": "bower_components"
}
\ No newline at end of file
{
"curly": true,
"eqeqeq": true,
"immed": true,
"latedef": true,
"newcap": true,
"noarg": true,
"sub": true,
"boss": true,
"eqnull": true
}
node_modules
bower_components
demo/bower_components
\ No newline at end of file
language: node_js
node_js:
- "0.8"
- "0.10"
before_install:
- export DISPLAY=:99.0
- sh -e /etc/init.d/xvfb start
- npm install -g karma bower grunt-cli
- bower install
- npm install
script: "grunt"
CONTRIBUTING
============
* Open a [Pull Request (PR)](https://github.com/angular-ui/ui-select2/pull/new/master)
* Make sure your PR is on a **new branch** you created off of the latest version of master
* Do **not** open a PR from your master branch
* Open a PR to start a discussion even if the code isn't finished (easier to collect feedback this way)
* Make sure all previous tests pass and add new tests for added behaviors
module.exports = function (grunt) {
'use strict';
var initConfig;
// Loading external tasks
require('load-grunt-tasks')(grunt);
// Project configuration.
initConfig = {
bower: 'bower_components',
pkg: grunt.file.readJSON('package.json'),
watch: {
test: {
// Lint & run unit tests in Karma
// Just running `$ grunt watch` will only lint your code; to run tests
// on watch, use `$ grunt watch:karma` to start a Karma server first
files: ['src/select2.js', 'test/select2Spec.js'],
tasks: ['jshint', 'karma:unit:run']
}
},
karma: {
options: {
configFile: 'test/karma.conf.js',
browsers: ['Firefox', 'PhantomJS']
},
unit: {
singleRun: true
},
watch: {
autoWatch: true
},
server: {
background: true
}
},
jshint: {
all:[
'gruntFile.js',
'src/**/*.js',
'test/**/*Spec.js'
],
options: {
jshintrc: '.jshintrc'
}
},
changelog: {
options: {
dest: 'CHANGELOG.md'
}
}
};
// Register tasks
grunt.registerTask('default', ['jshint', 'karma:unit']);
grunt.registerTask('watch', ['jshint', 'karma:watch']);
grunt.initConfig(initConfig);
};
The MIT License
Copyright (c) 2012 the AngularUI Team, http://angular-ui.github.com
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
ui-select2 [![Build Status](https://travis-ci.org/angular-ui/ui-select2.png)](https://travis-ci.org/angular-ui/ui-select2)
==========
This directive allows you to enhance your select elements with behaviour from the [select2](http://ivaynberg.github.io/select2/) library.
# Requirements
- [AngularJS](http://angularjs.org/)
- [JQuery](http://jquery.com/)
- [Select2](http://ivaynberg.github.io/select2/)
## Setup
1. Install **Karma**, **Grunt** and **Bower**
`$ npm install -g karma grunt-cli bower`
2. Install development dependencies
`$ npm install`
3. Install components
`$ bower install`
4. ???
5. Profit!
## Testing
We use [Grunt](http://gruntjs.com/) to check for JavaScript syntax errors and execute all unit tests. To run Grunt, simply execute:
`$ grunt`
This will lint and test the code, then exit. To have Grunt stay open and automatically lint and test your files whenever you make a code change, use:
`$ grunt karma:server watch`
This will start a Karma server in the background and run unit tests in Firefox and PhantomJS whenever the source code or spec file is saved.
# Usage
We use [bower](https://github.com/bower/bower) for dependency management. Install AngularUI Select2 into your project by running the command
`$ bower install angular-ui-select2`
If you use a `bower.json` file in your project, you can have Bower save ui-select2 as a dependency by passing the `--save` or `--save-dev` flag with the above command.
This will copy the ui-select2 files into your `bower_components` folder, along with its dependencies. Load the script files in your application:
```html
<link rel="stylesheet" href="bower_components/select2/select2.css">
<script type="text/javascript" src="bower_components/jquery/jquery.js"></script>
<script type="text/javascript" src="bower_components/select2/select2.js"></script>
<script type="text/javascript" src="bower_components/angular/angular.js"></script>
<script type="text/javascript" src="bower_components/angular-ui-select2/src/select2.js"></script>
```
(Note that `jquery` must be loaded before `angular` so that it doesn't use `jqLite` internally)
Add the select2 module as a dependency to your application module:
```javascript
var myAppModule = angular.module('MyApp', ['ui.select2']);
```
Apply the directive to your form elements:
```html
<select ui-select2 ng-model="select2" data-placeholder="Pick a number">
<option value=""></option>
<option value="one">First</option>
<option value="two">Second</option>
<option value="three">Third</option>
</select>
```
## Options
All the select2 options can be passed through the directive. You can read more about the supported list of options and what they do on the [Select2 Documentation Page](http://ivaynberg.github.com/select2/)
```javascript
myAppModule.controller('MyController', function($scope) {
$scope.select2Options = {
allowClear:true
};
});
```
```html
<select ui-select2="select2Options" ng-model="select2">
<option value="one">First</option>
<option value="two">Second</option>
<option value="three">Third</option>
</select>
```
Some times it may make sense to specify the options in the template file.
```html
<select ui-select2="{ allowClear: true}" ng-model="select2">
<option value="one">First</option>
<option value="two">Second</option>
<option value="three">Third</option>
</select>
```
To define global defaults, you can configure the `uiSelect2Config` injectable:
```javascript
myAppModule.run(['uiSelect2Config', function(uiSelect2Config) {
uiSelect2Config.placeholder = "Placeholder text";
}]);
```
## Working with ng-model
The ui-select2 directive plays nicely with ng-model and validation directives such as ng-required.
If you add the ng-model directive to same the element as ui-select2 then the picked option is automatically synchronized with the model value.
## Working with dynamic options
`ui-select2` is incompatible with `<select ng-options>`. For the best results use `<option ng-repeat>` instead.
```html
<select ui-select2 ng-model="select2" data-placeholder="Pick a number">
<option value=""></option>
<option ng-repeat="number in range" value="{{number.value}}">{{number.text}}</option>
</select>
```
## Working with placeholder text
In order to properly support the Select2 placeholder, create an empty `<option>` tag at the top of the `<select>` and either set a `data-placeholder` on the select element or pass a `placeholder` option to Select2.
```html
<select ui-select2 ng-model="number" data-placeholder="Pick a number">
<option value=""></option>
<option value="one">First</option>
<option value="two">Second</option>
<option value="three">Third</option>
</select>
```
## ng-required directive
If you apply the required directive to element then the form element is invalid until an option is selected.
Note: Remember that the ng-required directive must be explicitly set, i.e. to "true". This is especially true on divs:
```html
<select ui-select2 ng-model="number" data-placeholder="Pick a number" ng-required="true">
<option value=""></option>
<option value="one">First</option>
<option value="two">Second</option>
<option value="three">Third</option>
</select>
```
## Using simple tagging mode
When AngularJS View-Model tags are stored as a list of strings, setting
the ui-select2 specific option `simple_tags` will allow to keep the model
as a list of strings, and not convert it into a list of Select2 tag objects.
```html
<input
type="hidden"
ui-select2="select2Options"
ng-model="list_of_string"
>
```
```javascript
myAppModule.controller('MyController', function($scope) {
$scope.list_of_string = ['tag1', 'tag2']
$scope.select2Options = {
'multiple': true,
'simple_tags': true,
'tags': ['tag1', 'tag2', 'tag3', 'tag4'] // Can be empty list.
};
});
```
{
"author": "AngularUI",
"name": "angular-ui-select2",
"version": "0.0.5",
"homepage": "http://angular-ui.github.com",
"keywords": [
"angular",
"angularui",
"select2"
],
"main": "./src/select2.js",
"dependencies": {
"angular": ">=1.2.0",
"select2": "~3.4",
"jquery": ">=1.6.4"
},
"devDependencies": {
"angular-mocks": ">=1.0.2"
}
}
var app = angular.module('angular-ui-select2-demo', ['ui.select2']);
app.controller('MainCtrl', function ($scope, $element) {
var states = [
{ text: 'Alaskan/Hawaiian Time Zone', children: [
{ id: 'AK', text: 'Alaska' },
{ id: 'HI', text: 'Hawaii' }
]},
{ text: 'Pacific Time Zone', children: [
{ id: 'CA', text: 'California' },
{ id: 'NV', text: 'Nevada' },
{ id: 'OR', text: 'Oregon' },
{ id: 'WA', text: 'Washington' }
]},
{ text: 'Mountain Time Zone', children: [
{ id: 'AZ', text: 'Arizona' },
{ id: 'CO', text: 'Colorado' },
{ id: 'ID', text: 'Idaho' },
{ id: 'MT', text: 'Montana' },
{ id: 'NE', text: 'Nebraska' },
{ id: 'NM', text: 'New Mexico' },
{ id: 'ND', text: 'North Dakota' },
{ id: 'UT', text: 'Utah' },
{ id: 'WY', text: 'Wyoming' }
]},
{ text: 'Central Time Zone', children: [
{ id: 'AL', text: 'Alabama' },
{ id: 'AR', text: 'Arkansas' },
{ id: 'IL', text: 'Illinois' },
{ id: 'IA', text: 'Iowa' },
{ id: 'KS', text: 'Kansas' },
{ id: 'KY', text: 'Kentucky' },
{ id: 'LA', text: 'Louisiana' },
{ id: 'MN', text: 'Minnesota' },
{ id: 'MS', text: 'Mississippi' },
{ id: 'MO', text: 'Missouri' },
{ id: 'OK', text: 'Oklahoma' },
{ id: 'SD', text: 'South Dakota' },
{ id: 'TX', text: 'Texas' },
{ id: 'TN', text: 'Tennessee' },
{ id: 'WI', text: 'Wisconsin' }
]},
{ text: 'Eastern Time Zone', children: [
{ id: 'CT', text: 'Connecticut' },
{ id: 'DE', text: 'Delaware' },
{ id: 'FL', text: 'Florida' },
{ id: 'GA', text: 'Georgia' },
{ id: 'IN', text: 'Indiana' },
{ id: 'ME', text: 'Maine' },
{ id: 'MD', text: 'Maryland' },
{ id: 'MA', text: 'Massachusetts' },
{ id: 'MI', text: 'Michigan' },
{ id: 'NH', text: 'New Hampshire' },
{ id: 'NJ', text: 'New Jersey' },
{ id: 'NY', text: 'New York' },
{ id: 'NC', text: 'North Carolina' },
{ id: 'OH', text: 'Ohio' },
{ id: 'PA', text: 'Pennsylvania' },
{ id: 'RI', text: 'Rhode Island' },
{ id: 'SC', text: 'South Carolina' },
{ id: 'VT', text: 'Vermont' },
{ id: 'VA', text: 'Virginia' },
{ id: 'WV', text: 'West Virginia' }
]}
];
function findState(id) {
for (var i=0; i<states.length; i++) {
for (var j=0; j<states[i].children.length; j++) {
if (states[i].children[j].id == id) {
return states[i].children[j];
}
}
}
}
$scope.multi2Value = ['CO', 'WA'];
$scope.multi = {
multiple: true,
query: function (query) {
query.callback({ results: states });
},
initSelection: function(element, callback) {
var val = $(element).select2('val'),
results = [];
for (var i=0; i<val.length; i++) {
results.push(findState(val[i]));
}
callback(results);
}
};
$scope.placeholders = {
placeholder: "Select a State"
};
$scope.array = {
data: [{id:0,text:'enhancement'},{id:1,text:'bug'},{id:2,text:'duplicate'},{id:3,text:'invalid'},{id:4,text:'wontfix'}]
};
$scope.arrayAsync = {
query: function (query) {
query.callback({ results: states });
},
initSelection: function(element, callback) {
var val = $(element).select2('val');
return callback(findState(val));
}
};
});