From 0c1bf25af8a63f522d5adf192a59448fe9539281 Mon Sep 17 00:00:00 2001 From: David Jensen Date: Tue, 19 Aug 2014 13:44:12 +0200 Subject: [PATCH] Shut jscs up Poked and prodded everything until jscs stopped complaining. So now we follow a code standard. Yay! --- .jscs.json | 4 + README.md | 9 +- dist/bootstrap-datepicker.min.js | 2 +- dist/schema-form.js | 944 +++++++++--------- dist/schema-form.min.js | 2 +- gulpfile.js | 76 +- package.json | 1 + src/directives/array.js | 356 ++++--- src/directives/changed.js | 14 +- .../bootstrap/bootstrap-decorator.js | 84 +- .../bootstrap/datepicker/angular-pickadate.js | 22 +- .../datepicker/bootstrap-datepicker.js | 41 +- src/directives/schema-form.js | 210 ++-- src/directives/schema-validate.js | 13 +- src/module.js | 2 +- src/services/Select.js | 30 +- src/services/decorators.js | 111 +- src/services/schema-form.js | 177 ++-- src/services/validator.js | 8 +- src/sfPath.js | 42 +- 20 files changed, 1087 insertions(+), 1061 deletions(-) create mode 100644 .jscs.json diff --git a/.jscs.json b/.jscs.json new file mode 100644 index 000000000..5d5a43647 --- /dev/null +++ b/.jscs.json @@ -0,0 +1,4 @@ +{ + "preset": "google", + "maximumLineLength": 100 +} diff --git a/README.md b/README.md index d07d74067..0e65b9f67 100644 --- a/README.md +++ b/README.md @@ -158,7 +158,10 @@ $ karma start karma.conf.js Contributing ------------ -**Heads up!** Sometime soon we will go over and change the code style to follow -whatever [jscs](https://github.com/mdevils/node-jscs) says with preset set to 'google'. +All contributions are welcome! We're trying to use +[git flow](http://danielkummer.github.io/git-flow-cheatsheet/), so please base any merge request +on the **development** branch instead of **master**. -All contributions are welcome! We're trying to use [git flow](http://danielkummer.github.io/git-flow-cheatsheet/), so please base any merge request on the **development** branch instead of **master**. +Also run any code through the code style checker [jscs](https://github.com/mdevils/node-jscs) +(or even better use it in your editor) with preset set to `google`. You can also us `gulp jscs` to +check your code. diff --git a/dist/bootstrap-datepicker.min.js b/dist/bootstrap-datepicker.min.js index 9baa01ca1..95eae69cd 100644 --- a/dist/bootstrap-datepicker.min.js +++ b/dist/bootstrap-datepicker.min.js @@ -1 +1 @@ -angular.module("schemaForm").run(["$templateCache",function(e){e.put("directives/decorators/bootstrap/datepicker/datepicker.html",'
{{ (hasError() && errorMessage(schemaError())) || form.description}}
')}]),angular.module("schemaForm").directive("pickADate",function(){var e=function(e){return angular.isString(e)||angular.isNumber(e)?new Date(e):e};return{restrict:"A",require:"ngModel",scope:{ngModel:"=",minDate:"=",maxDate:"="},link:function(r,a,t,o){if(a.pickadate){a.pickadate({onClose:function(){a.blur()},formatSubmit:null});var i="yyyy-mm-dd",n=$.fn.pickadate.defaults.format,s=a.pickadate("picker");if(o.$formatters.push(function(e){return angular.isUndefined(e)||null===e?e:(s.set("view",e,{format:t.format||i}),s.set("highlight",e,{format:t.format||i}),s.get("highlight",n))}),o.$parsers.push(function(){return s.get("select",t.format||i)}),angular.isDefined(t.minDate))var c=r.$watch("minDate",function(r){r&&(s.set("min",e(r)),c())},!0);if(angular.isDefined(t.maxDate))var m=r.$watch("maxDate",function(r){r&&(s.set("max",e(r)),m())},!0)}}}}),angular.module("schemaForm").config(["schemaFormProvider","schemaFormDecoratorsProvider","sfPathProvider",function(e,r,a){var t=function(r,t,o){if("string"===t.type&&"date"==t.format){var i=e.stdFormObj(r,t,o);return i.key=o.path,i.type="datepicker",o.lookup[a.stringify(o.path)]=i,i}};e.defaults.string.unshift(t),r.addMapping("bootstrapDecorator","datepicker","directives/decorators/bootstrap/datepicker/datepicker.html"),r.createDirective("datepicker","directives/decorators/bootstrap/datepicker/datepicker.html")}]); \ No newline at end of file +angular.module("schemaForm").run(["$templateCache",function(e){e.put("directives/decorators/bootstrap/datepicker/datepicker.html",'
{{ (hasError() && errorMessage(schemaError())) || form.description}}
')}]),angular.module("schemaForm").directive("pickADate",function(){var e=function(e){return angular.isString(e)||angular.isNumber(e)?new Date(e):e};return{restrict:"A",require:"ngModel",scope:{ngModel:"=",minDate:"=",maxDate:"="},link:function(r,a,t,o){if(a.pickadate){a.pickadate({onClose:function(){a.blur()},formatSubmit:null});var i="yyyy-mm-dd",n=$.fn.pickadate.defaults.format,s=a.pickadate("picker");if(o.$formatters.push(function(e){return angular.isUndefined(e)||null===e?e:(s.set("view",e,{format:t.format||i}),s.set("highlight",e,{format:t.format||i}),s.get("highlight",n))}),o.$parsers.push(function(){return s.get("select",t.format||i)}),angular.isDefined(t.minDate))var c=r.$watch("minDate",function(r){r&&(s.set("min",e(r)),c())},!0);if(angular.isDefined(t.maxDate))var m=r.$watch("maxDate",function(r){r&&(s.set("max",e(r)),m())},!0)}}}}),angular.module("schemaForm").config(["schemaFormProvider","schemaFormDecoratorsProvider","sfPathProvider",function(e,r,a){var t=function(r,t,o){if("string"===t.type&&"date"===t.format){var i=e.stdFormObj(r,t,o);return i.key=o.path,i.type="datepicker",o.lookup[a.stringify(o.path)]=i,i}};e.defaults.string.unshift(t),r.addMapping("bootstrapDecorator","datepicker","directives/decorators/bootstrap/datepicker/datepicker.html"),r.createDirective("datepicker","directives/decorators/bootstrap/datepicker/datepicker.html")}]); \ No newline at end of file diff --git a/dist/schema-form.js b/dist/schema-form.js index 43f168448..e589451b0 100644 --- a/dist/schema-form.js +++ b/dist/schema-form.js @@ -11,53 +11,60 @@ try { deps.push('ui.sortable'); } catch (e) {} -angular.module('schemaForm',deps); - -angular.module('schemaForm').provider('sfPath',['ObjectPathProvider',function(ObjectPathProvider){ - var ObjectPath = { parse: ObjectPathProvider.parse }; - - // if we're on Angular 1.2.x, we need to continue using dot notation - if(angular.version.major === 1 && angular.version.minor < 3) { - ObjectPath.stringify = function(arr) { - return Array.isArray(arr) ? arr.join('.') : arr.toString(); - }; - } else { - ObjectPath.stringify = ObjectPathProvider.stringify; - } - - // we want this to use whichever stringify method is defined above, so we have to copy the code here - ObjectPath.normalize = function(data, quote){ return ObjectPath.stringify(Array.isArray(data) ? data : ObjectPath.parse(data), quote); } - - this.parse = ObjectPath.parse; - this.stringify = ObjectPath.stringify; - this.normalize = ObjectPath.normalize; - this.$get = function(){ - return ObjectPath; - }; +angular.module('schemaForm', deps); + +angular.module('schemaForm').provider('sfPath', +['ObjectPathProvider', function(ObjectPathProvider) { + var ObjectPath = {parse: ObjectPathProvider.parse}; + + // if we're on Angular 1.2.x, we need to continue using dot notation + if (angular.version.major === 1 && angular.version.minor < 3) { + ObjectPath.stringify = function(arr) { + return Array.isArray(arr) ? arr.join('.') : arr.toString(); + }; + } else { + ObjectPath.stringify = ObjectPathProvider.stringify; + } + + // We want this to use whichever stringify method is defined above, + // so we have to copy the code here. + ObjectPath.normalize = function(data, quote) { + return ObjectPath.stringify(Array.isArray(data) ? data : ObjectPath.parse(data), quote); + }; + + this.parse = ObjectPath.parse; + this.stringify = ObjectPath.stringify; + this.normalize = ObjectPath.normalize; + this.$get = function () { + return ObjectPath; + }; }]); + /** * @ngdoc service * @name sfSelect * @kind function * - * @description - * Utility method to access deep properties without - * throwing errors when things are not defined. - * Can also set a value in a deep structure, creating objects when missing - * ex. - * var foo = Select('address.contact.name',obj) - * Select('address.contact.name',obj,'Leeroy') - * - * @param {string} projection A dot path to the property you want to get/set - * @param {object} obj (optional) The object to project on, defaults to 'this' - * @param {Any} value (opional) The value to set, if parts of the path of - * the projection is missing empty objects will be created. - * @returns {Any|undefined} returns the value at the end of the projection path - * or undefined if there is none. */ angular.module('schemaForm').factory('sfSelect', ['sfPath', function (sfPath) { var numRe = /^\d+$/; + /** + * @description + * Utility method to access deep properties without + * throwing errors when things are not defined. + * Can also set a value in a deep structure, creating objects when missing + * ex. + * var foo = Select('address.contact.name',obj) + * Select('address.contact.name',obj,'Leeroy') + * + * @param {string} projection A dot path to the property you want to get/set + * @param {object} obj (optional) The object to project on, defaults to 'this' + * @param {Any} valueToSet (opional) The value to set, if parts of the path of + * the projection is missing empty objects will be created. + * @returns {Any|undefined} returns the value at the end of the projection path + * or undefined if there is none. + */ return function(projection, obj, valueToSet) { if (!obj) { obj = this; @@ -108,11 +115,12 @@ angular.module('schemaForm').factory('sfSelect', ['sfPath', function (sfPath) { }; }]); -angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider','sfPathProvider',function($compileProvider, sfPathProvider){ +angular.module('schemaForm').provider('schemaFormDecorators', +['$compileProvider', 'sfPathProvider', function($compileProvider, sfPathProvider) { var defaultDecorator = ''; var directives = {}; - var templateUrl = function(name,form) { + var templateUrl = function(name, form) { //schemaDecorator is alias for whatever is set as default if (name === 'sfDecorator') { name = defaultDecorator; @@ -122,7 +130,7 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' //rules first var rules = directive.rules; - for (var i = 0; i< rules.length; i++) { + for (var i = 0; i < rules.length; i++) { var res = rules[i](form); if (res) { return res; @@ -138,10 +146,9 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' return directive.mappings['default']; }; - - var createDirective = function(name){ - $compileProvider.directive(name,['$parse','$compile','$http','$templateCache', - function($parse, $compile, $http, $templateCache){ + var createDirective = function(name) { + $compileProvider.directive(name, ['$parse', '$compile', '$http', '$templateCache', + function($parse, $compile, $http, $templateCache) { return { restrict: 'AE', @@ -149,9 +156,9 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' transclude: false, scope: true, require: '?^sfSchema', - link: function(scope,element,attrs,sfSchema) { + link: function(scope, element, attrs, sfSchema) { //rebind our part of the form to the scope. - var once = scope.$watch(attrs.form,function(form){ + var once = scope.$watch(attrs.form, function(form) { if (form) { scope.form = form; @@ -159,10 +166,14 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' //ok let's replace that template! //We do this manually since we need to bind ng-model properly and also //for fieldsets to recurse properly. - var url = templateUrl(name,form); - $http.get(url,{ cache: $templateCache }).then(function(res){ - var key = form.key ? sfPathProvider.stringify(form.key).replace(/"/g, '"') : ''; - var template = res.data.replace(/\$\$value\$\$/g,'model'+(key[0] !== '['?'.':'')+key); + var url = templateUrl(name, form); + $http.get(url, {cache: $templateCache}).then(function(res) { + var key = form.key ? + sfPathProvider.stringify(form.key).replace(/"/g, '"') : ''; + var template = res.data.replace( + /\$\$value\$\$/g, + 'model' + (key[0] !== '[' ? '.' : '') + key + ); element.html(template); $compile(element.contents())(scope); }); @@ -175,17 +186,17 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' return scope.form && scope.form.notitle !== true && scope.form.title; }; - scope.listToCheckboxValues = function(list){ + scope.listToCheckboxValues = function(list) { var values = {}; - angular.forEach(list,function(v){ + angular.forEach(list, function(v) { values[v] = true; }); return values; }; - scope.checkboxValuesToList = function(values){ + scope.checkboxValuesToList = function(values) { var lst = []; - angular.forEach(values,function(v,k){ + angular.forEach(values, function(v, k) { if (v) { lst.push(k); } @@ -193,15 +204,15 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' return lst; }; - scope.buttonClick = function($event,form) { + scope.buttonClick = function($event, form) { if (angular.isFunction(form.onClick)) { - form.onClick($event,form); + form.onClick($event, form); } else if (angular.isString(form.onClick)) { if (sfSchema) { //evaluating in scope outside of sfSchemas isolated scope - sfSchema.evalInParentScope(form.onClick,{'$event':$event,form:form}); + sfSchema.evalInParentScope(form.onClick, {'$event': $event, form: form}); } else { - scope.$eval(form.onClick,{'$event':$event,form:form}); + scope.$eval(form.onClick, {'$event': $event, form: form}); } } }; @@ -213,13 +224,13 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' * @param {Object} locals (optional) * @return {Any} the result of the expression */ - scope.evalExpr = function(expression,locals) { + scope.evalExpr = function(expression, locals) { if (sfSchema) { //evaluating in scope outside of sfSchemas isolated scope - return sfSchema.evalInParentScope(expression,locals); + return sfSchema.evalInParentScope(expression, locals); } - return scope.$eval(expression,locals); + return scope.$eval(expression, locals); }; /** @@ -229,10 +240,10 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' * @param {Object} locals (optional) * @return {Any} the result of the expression */ - scope.evalInScope = function(expression,locals) { - if (expression) { - return scope.$eval(expression,locals); - } + scope.evalInScope = function(expression, locals) { + if (expression) { + return scope.$eval(expression, locals); + } }; /** @@ -248,9 +259,12 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' return scope.form.validationMessage; } - return scope.form.validationMessage[schemaError.code] || scope.form.validationMessage['default']; + return scope.form.validationMessage[schemaError.code] || + scope.form.validationMessage['default']; } else { - return scope.form.validationMessage.required || scope.form.validationMessage['default'] || scope.form.validationMessage; + return scope.form.validationMessage.required || + scope.form.validationMessage['default'] || + scope.form.validationMessage; } } @@ -260,35 +274,36 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' } //Otherwise we only use required so it must be it. - return "Required"; + return 'Required'; }; } }; - }]); + } + ]); }; - var createManualDirective = function(type,templateUrl,transclude) { - transclude = angular.isDefined(transclude)? transclude : false; - $compileProvider.directive('sf'+angular.uppercase(type[0])+type.substr(1), function(){ + var createManualDirective = function(type, templateUrl, transclude) { + transclude = angular.isDefined(transclude) ? transclude : false; + $compileProvider.directive('sf' + angular.uppercase(type[0]) + type.substr(1), function() { return { - restrict: "EAC", + restrict: 'EAC', scope: true, replace: true, transclude: transclude, template: '', - link: function(scope,element,attrs) { + link: function(scope, element, attrs) { var watchThis = { 'items': 'c', 'titleMap': 'c', 'schema': 'c' }; - var form = { type: type }; + var form = {type: type}; var once = true; - angular.forEach(attrs,function(value,name){ + angular.forEach(attrs, function(value, name) { if (name[0] !== '$' && name.indexOf('ng') !== 0 && name !== 'sfField') { - var updateForm = function(val){ + var updateForm = function(val) { if (angular.isDefined(val) && val !== form[name]) { form[name] = val; @@ -303,17 +318,17 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' if (name === 'model') { //"model" is bound to scope under the name "model" since this is what the decorators //know and love. - scope.$watch(value,function(val){ + scope.$watch(value, function(val) { if (val && scope.model !== val) { scope.model = val; } }); } else if (watchThis[name] === 'c') { //watch collection - scope.$watchCollection(value,updateForm); + scope.$watchCollection(value, updateForm); } else { //$observe - attrs.$observe(name,updateForm); + attrs.$observe(name, updateForm); } } }); @@ -322,8 +337,6 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' }); }; - - /** * Create a decorator directive and its sibling "manual" use directives. * The directive can be used to create form fields or other form entities. @@ -335,11 +348,12 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' ** * @param {string} name directive name (CamelCased) * @param {Object} mappings, an object that maps "type" => "templateUrl" - * @param {Array} rules (optional) a list of functions, function(form){}, that are each tried in turn, + * @param {Array} rules (optional) a list of functions, function(form) {}, that are each tried in + * turn, * if they return a string then that is used as the templateUrl. Rules come before * mappings. */ - this.createDecorator = function(name,mappings,rules){ + this.createDecorator = function(name, mappings, rules) { directives[name] = { mappings: mappings || {}, rules: rules || [] @@ -371,8 +385,8 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' * @param {Object} mappings */ this.createDirectives = function(mappings) { - angular.forEach(mappings,function(url,type){ - createManualDirective(type,url); + angular.forEach(mappings, function(url, type) { + createManualDirective(type, url); }); }; @@ -393,15 +407,14 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' * @param {String} type Form type for the mapping * @param {String} url The template url */ - this.addMapping = function(name,type,url) { + this.addMapping = function(name, type, url) { if (directives[name]) { directives[name].mappings[type] = url; } }; - //Service is just a getter for directive mappings and rules - this.$get = function(){ + this.$get = function() { return { directive: function(name) { return directives[name]; @@ -410,7 +423,6 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' }; }; - //Create a default directive createDirective('sfDecorator'); @@ -421,13 +433,14 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' * This service is not that useful outside of schema form directive * but makes the code more testable. */ -angular.module('schemaForm').provider('schemaForm',['sfPathProvider', function(sfPathProvider){ +angular.module('schemaForm').provider('schemaForm', +['sfPathProvider', function(sfPathProvider) { //Creates an default titleMap list from an enum, i.e. a list of strings. var enumToTitleMap = function(enm) { var titleMap = []; //canonical titleMap format is a list. - enm.forEach(function(name){ - titleMap.push({ name: name, value: name}); + enm.forEach(function(name) { + titleMap.push({name: name, value: name}); }); return titleMap; }; @@ -437,20 +450,20 @@ angular.module('schemaForm').provider('schemaForm',['sfPathProvider', function(s var canonicalTitleMap = function(titleMap) { if (!angular.isArray(titleMap)) { var canonical = []; - angular.forEach(titleMap, function(name,value) { - canonical.push({ name: name, value: value }); + angular.forEach(titleMap, function(name, value) { + canonical.push({name: name, value: value}); }); return canonical; } return titleMap; }; - var defaultFormDefinition = function(name,schema,options){ + var defaultFormDefinition = function(name, schema, options) { var rules = defaults[schema.type]; if (rules) { var def; - for (var i=0;i 1) { - subForm = { type: 'section', items: form.items }; + return { + restrict: 'A', + scope: true, + require: '?ngModel', + link: function(scope, element, attrs, ngModel) { + var formDefCache = {}; + + // Watch for the form definition and then rewrite it. + // It's the (first) array part of the key, '[]' that needs a number + // corresponding to an index of the form. + var once = scope.$watch(attrs.sfArray, function(form) { + + // An array model always needs a key so we know what part of the model + // to look at. This makes us a bit incompatible with JSON Form, on the + // other hand it enables two way binding. + var list = sfSelect(form.key, scope.model); + + // Since ng-model happily creates objects in a deep path when setting a + // a value but not arrays we need to create the array. + if (angular.isUndefined(list)) { + list = []; + sfSelect(form.key, scope.model, list); } - } - - // We ceate copies of the form on demand, caching them for - // later requests - scope.copyWithIndex = function(index) { - if (!formDefCache[index]) { - if (subForm) { - var copy = angular.copy(subForm); - copy.arrayIndex= index; - schemaForm.traverseForm(copy, setIndex(index)); - formDefCache[index] = copy; + scope.modelArray = list; + + // Arrays with titleMaps, i.e. checkboxes doesn't have items. + if (form.items) { + + // To be more compatible with JSON Form we support an array of items + // in the form definition of "array" (the schema just a value). + // for the subforms code to work this means we wrap everything in a + // section. Unless there is just one. + var subForm = form.items[0]; + if (form.items.length > 1) { + subForm = {type: 'section', items: form.items}; } } - return formDefCache[index]; - }; + // We ceate copies of the form on demand, caching them for + // later requests + scope.copyWithIndex = function(index) { + if (!formDefCache[index]) { + if (subForm) { + var copy = angular.copy(subForm); + copy.arrayIndex = index; + schemaForm.traverseForm(copy, setIndex(index)); + formDefCache[index] = copy; + } + } + return formDefCache[index]; + }; + + scope.appendToArray = function() { + var len = list.length; + var copy = scope.copyWithIndex(len); + schemaForm.traverseForm(copy, function(part) { + if (part.key && angular.isDefined(part.default)) { + sfSelect(part.key, scope.model, part.default); + } + }); - scope.appendToArray = function() { - var len = list.length; - var copy = scope.copyWithIndex(len); - schemaForm.traverseForm(copy, function(part){ - if (part.key && angular.isDefined(part.default)) { - sfSelect(part.key, scope.model, part.default); + // If there are no defaults nothing is added so we need to initialize + // the array. undefined for basic values, {} or [] for the others. + if (len === list.length) { + var type = sfSelect('schema.items.type', form); + var dflt; + if (type === 'object') { + dflt = {}; + } else if (type === 'array') { + dflt = []; + } + list.push(dflt); } - }); - // If there are no defaults nothing is added so we need to initialize - // the array. undefined for basic values, {} or [] for the others. - if (len === list.length) { - var type = sfSelect('schema.items.type',form); - var dflt; - if (type === 'object') { - dflt = {}; - } else if (type === 'array') { - dflt = []; + // Trigger validation. + if (scope.validateArray) { + scope.validateArray(); } - list.push(dflt); - } + }; - // Trigger validation. - if (scope.validateArray) { - scope.validateArray(); - } - }; + scope.deleteFromArray = function(index) { + list.splice(index, 1); - scope.deleteFromArray = function(index) { - list.splice(index,1); + // Trigger validation. + if (scope.validateArray) { + scope.validateArray(); + } + }; - // Trigger validation. - if (scope.validateArray) { - scope.validateArray(); + // Always start with one empty form unless configured otherwise. + // Special case: don't do it if form has a titleMap + if (!form.titleMap && form.startEmpty !== true && list.length === 0) { + scope.appendToArray(); } - }; - - // Always start with one empty form unless configured otherwise. - // Special case: don't do it if form has a titleMap - if (!form.titleMap && form.startEmpty !== true && list.length === 0) { - scope.appendToArray(); - } - // Title Map handling - // If form has a titleMap configured we'd like to enable looping over - // titleMap instead of modelArray, this is used for intance in - // checkboxes. So instead of variable number of things we like to create - // a array value from a subset of values in the titleMap. - // The problem here is that ng-model on a checkbox doesn't really map to - // a list of values. This is here to fix that. - if (form.titleMap && form.titleMap.length > 0) { - scope.titleMapValues = []; - - // We watch the model for changes and the titleMapValues to reflect - // the modelArray - var updateTitleMapValues = function(arr) { + // Title Map handling + // If form has a titleMap configured we'd like to enable looping over + // titleMap instead of modelArray, this is used for intance in + // checkboxes. So instead of variable number of things we like to create + // a array value from a subset of values in the titleMap. + // The problem here is that ng-model on a checkbox doesn't really map to + // a list of values. This is here to fix that. + if (form.titleMap && form.titleMap.length > 0) { scope.titleMapValues = []; - arr = arr || []; - form.titleMap.forEach(function(item) { - scope.titleMapValues.push( arr.indexOf(item.value) !== -1 ); - }); + // We watch the model for changes and the titleMapValues to reflect + // the modelArray + var updateTitleMapValues = function(arr) { + scope.titleMapValues = []; + arr = arr || []; - }; - //Catch default values - updateTitleMapValues(scope.modelArray); - scope.$watchCollection('modelArray',updateTitleMapValues); - - //To get two way binding we also watch our titleMapValues - scope.$watchCollection('titleMapValues', function(vals) { - if (vals) { - var arr = scope.modelArray; - - // Apparently the fastest way to clear an array, readable too. - // http://jsperf.com/array-destroy/32 - while (arr.length > 0) { - arr.shift(); - } + form.titleMap.forEach(function(item) { + scope.titleMapValues.push(arr.indexOf(item.value) !== -1); + }); - form.titleMap.forEach(function(item,index) { - if (vals[index]) { - arr.push(item.value); + }; + //Catch default values + updateTitleMapValues(scope.modelArray); + scope.$watchCollection('modelArray', updateTitleMapValues); + + //To get two way binding we also watch our titleMapValues + scope.$watchCollection('titleMapValues', function(vals) { + if (vals) { + var arr = scope.modelArray; + + // Apparently the fastest way to clear an array, readable too. + // http://jsperf.com/array-destroy/32 + while (arr.length > 0) { + arr.shift(); } - }); - } - }); - } + form.titleMap.forEach(function(item, index) { + if (vals[index]) { + arr.push(item.value); + } + }); + } + }); + } - // If there is a ngModel present we need to validate when asked. - if (ngModel) { - var error; - - scope.validateArray = function() { - // The actual content of the array is validated by each field - // so we settle for checking validations specific to arrays - - // Since we prefill with empty arrays we can get the funny situation - // where the array is required but empty in the gui but still validates. - // Thats why we check the length. - var result = sfValidator.validate( - form, - scope.modelArray.length > 0 ? scope.modelArray : undefined - ); - if (result.valid === false && - result.error && - (result.error.dataPath === '' || - result.error.dataPath === '/'+form.key[form.key.length - 1])) { - - // Set viewValue to trigger $dirty on field. If someone knows a - // a better way to do it please tell. - ngModel.$setViewValue(scope.modelArray); - error = result.error; - ngModel.$setValidity('schema', false); - - } else { - ngModel.$setValidity('schema', true); - } - }; + // If there is a ngModel present we need to validate when asked. + if (ngModel) { + var error; + + scope.validateArray = function() { + // The actual content of the array is validated by each field + // so we settle for checking validations specific to arrays + + // Since we prefill with empty arrays we can get the funny situation + // where the array is required but empty in the gui but still validates. + // Thats why we check the length. + var result = sfValidator.validate( + form, + scope.modelArray.length > 0 ? scope.modelArray : undefined + ); + if (result.valid === false && + result.error && + (result.error.dataPath === '' || + result.error.dataPath === '/' + form.key[form.key.length - 1])) { + + // Set viewValue to trigger $dirty on field. If someone knows a + // a better way to do it please tell. + ngModel.$setViewValue(scope.modelArray); + error = result.error; + ngModel.$setValidity('schema', false); - scope.$on('schemaFormValidate',scope.validateArray); + } else { + ngModel.$setValidity('schema', true); + } + }; + scope.$on('schemaFormValidate', scope.validateArray); - scope.hasSuccess = function(){ - return ngModel.$valid && !ngModel.$pristine; - }; + scope.hasSuccess = function() { + return ngModel.$valid && !ngModel.$pristine; + }; - scope.hasError = function(){ - return ngModel.$invalid; - }; + scope.hasError = function() { + return ngModel.$invalid; + }; - scope.schemaError = function() { - return error; - }; + scope.schemaError = function() { + return error; + }; - } + } - once(); - }); - } - }; -}]); + once(); + }); + } + }; + } +]); /** * A version of ng-changed that only listens if @@ -1123,22 +1124,22 @@ function(sfSelect, schemaForm, sfValidator) { * Takes the form definition as argument. * If the form definition has a "onChange" defined as either a function or */ -angular.module('schemaForm').directive('sfChanged',function(){ +angular.module('schemaForm').directive('sfChanged', function() { return { require: 'ngModel', restrict: 'AC', scope: false, - link: function(scope,element,attrs,ctrl) { + link: function(scope, element, attrs, ctrl) { var form = scope.$eval(attrs.sfChanged); //"form" is really guaranteed to be here since the decorator directive //waits for it. But best be sure. if (form && form.onChange) { ctrl.$viewChangeListeners.push(function() { - if (angular.isFunction(form.onChange)) { - form.onChange(ctrl.$modelValue,form); - } else { - scope.evalExpr(form.onChange,{ 'modelValue': ctrl.$modelValue, form: form }); - } + if (angular.isFunction(form.onChange)) { + form.onChange(ctrl.$modelValue, form); + } else { + scope.evalExpr(form.onChange, {'modelValue': ctrl.$modelValue, form: form}); + } }); } } @@ -1147,125 +1148,126 @@ angular.module('schemaForm').directive('sfChanged',function(){ /* FIXME: real documentation -
+
*/ angular.module('schemaForm') .directive('sfSchema', - ['$compile','schemaForm','schemaFormDecorators','sfSelect', -function($compile, schemaForm, schemaFormDecorators, sfSelect){ - - var SNAKE_CASE_REGEXP = /[A-Z]/g; - function snake_case(name, separator){ - separator = separator || '_'; - return name.replace(SNAKE_CASE_REGEXP, function(letter, pos) { - return (pos ? separator : '') + letter.toLowerCase(); - }); - } +['$compile', 'schemaForm', 'schemaFormDecorators', 'sfSelect', + function($compile, schemaForm, schemaFormDecorators, sfSelect) { + + var SNAKE_CASE_REGEXP = /[A-Z]/g; + var snakeCase = function(name, separator) { + separator = separator || '_'; + return name.replace(SNAKE_CASE_REGEXP, function(letter, pos) { + return (pos ? separator : '') + letter.toLowerCase(); + }); + }; - return { - scope: { - schema: '=sfSchema', - initialForm: '=sfForm', - model: '=sfModel' - }, - controller: ['$scope',function($scope){ - this.evalInParentScope = function(expr,locals){ - return $scope.$parent.$eval(expr,locals); - }; - }], - replace: false, - restrict: "A", - transclude: true, - require: '?form', - link: function(scope,element,attrs,formCtrl,transclude) { - - //expose form controller on scope so that we don't force authors to use name on form - scope.formCtrl = formCtrl; - - //We'd like to handle existing markup, - //besides using it in our template we also - //check for ng-model and add that to an ignore list - //i.e. even if form has a definition for it or form is ["*"] - //we don't generate it. - var ignore = {}; - transclude(scope,function(clone){ - clone.addClass('schema-form-ignore'); - element.prepend(clone); - - if (element[0].querySelectorAll) { - var models = element[0].querySelectorAll('[ng-model]'); - if (models){ - for (var i=0; i < models.length; i++){ - var key = models[i].getAttribute('ng-model'); - //skip first part before . - ignore[key.substring(key.indexOf('.')+1)] = true; + return { + scope: { + schema: '=sfSchema', + initialForm: '=sfForm', + model: '=sfModel' + }, + controller: ['$scope', function($scope) { + this.evalInParentScope = function(expr, locals) { + return $scope.$parent.$eval(expr, locals); + }; + }], + replace: false, + restrict: 'A', + transclude: true, + require: '?form', + link: function(scope, element, attrs, formCtrl, transclude) { + + //expose form controller on scope so that we don't force authors to use name on form + scope.formCtrl = formCtrl; + + //We'd like to handle existing markup, + //besides using it in our template we also + //check for ng-model and add that to an ignore list + //i.e. even if form has a definition for it or form is ["*"] + //we don't generate it. + var ignore = {}; + transclude(scope, function(clone) { + clone.addClass('schema-form-ignore'); + element.prepend(clone); + + if (element[0].querySelectorAll) { + var models = element[0].querySelectorAll('[ng-model]'); + if (models) { + for (var i = 0; i < models.length; i++) { + var key = models[i].getAttribute('ng-model'); + //skip first part before . + ignore[key.substring(key.indexOf('.') + 1)] = true; + } } } - } - }); - //Since we are dependant on up to three - //attributes we'll do a common watch - var lastDigest = {}; - - scope.$watch(function(){ + }); + //Since we are dependant on up to three + //attributes we'll do a common watch + var lastDigest = {}; - var schema = scope.schema; - var form = scope.initialForm || ['*']; + scope.$watch(function() { - //The check for schema.type is to ensure that schema is not {} - if (form && schema && schema.type && (lastDigest.form !== form || lastDigest.schema !== schema) && Object.keys(schema.properties).length > 0) { - lastDigest.schema = schema; - lastDigest.form = form; + var schema = scope.schema; + var form = scope.initialForm || ['*']; - // Check for options - var options = scope.$eval(attrs.sfOptions); + //The check for schema.type is to ensure that schema is not {} + if (form && schema && schema.type && + (lastDigest.form !== form || lastDigest.schema !== schema) && + Object.keys(schema.properties).length > 0) { + lastDigest.schema = schema; + lastDigest.form = form; - var merged = schemaForm.merge(schema,form,ignore,options); - var frag = document.createDocumentFragment(); + // Check for options + var options = scope.$eval(attrs.sfOptions); - //make the form available to decorators - scope.schemaForm = { form: merged, schema: schema }; + var merged = schemaForm.merge(schema, form, ignore, options); + var frag = document.createDocumentFragment(); - //Create directives from the form definition - angular.forEach(merged,function(obj,i){ - var n = document.createElement(attrs.sfDecoratorName || snake_case(schemaFormDecorators.defaultDecorator,'-')); - n.setAttribute('form','schemaForm.form['+i+']'); - frag.appendChild(n); - }); + //make the form available to decorators + scope.schemaForm = {form: merged, schema: schema}; - //clean all but pre existing html. - element.children(':not(.schema-form-ignore)').remove(); + //Create directives from the form definition + angular.forEach(merged, function(obj, i) { + var n = document.createElement(attrs.sfDecoratorName || + snakeCase(schemaFormDecorators.defaultDecorator, '-')); + n.setAttribute('form', 'schemaForm.form[' + i + ']'); + frag.appendChild(n); + }); - element[0].appendChild(frag); + //clean all but pre existing html. + element.children(':not(.schema-form-ignore)').remove(); - //compile only children - $compile(element.children())(scope); + element[0].appendChild(frag); - //ok, now that that is done let's set any defaults - schemaForm.traverseSchema(schema,function(prop,path){ + //compile only children + $compile(element.children())(scope); - if (angular.isDefined(prop['default'])) { - var val = sfSelect(path, scope.model); - if (angular.isUndefined(val)) { - sfSelect(path, scope.model, prop['default']); + //ok, now that that is done let's set any defaults + schemaForm.traverseSchema(schema, function(prop, path) { + if (angular.isDefined(prop['default'])) { + var val = sfSelect(path, scope.model); + if (angular.isUndefined(val)) { + sfSelect(path, scope.model, prop['default']); + } } - } - }); - - } - }); - } - }; -}]); + }); + } + }); + } + }; + } +]); -/* global tv4 */ -angular.module('schemaForm').directive('schemaValidate',['sfValidator',function(sfValidator){ +angular.module('schemaForm').directive('schemaValidate', ['sfValidator', function(sfValidator) { return { restrict: 'A', scope: false, require: 'ngModel', - link: function(scope,element,attrs,ngModel) { + link: function(scope, element, attrs, ngModel) { //Since we have scope false this is the same scope //as the decorator scope.ngModel = ngModel; @@ -1291,7 +1293,7 @@ angular.module('schemaForm').directive('schemaValidate',['sfValidator',function( // An empty field gives us the an empty string, which JSON schema // happily accepts as a proper defined string, but an empty field // for the user should trigger "required". So we set it to undefined. - if (viewValue === "") { + if (viewValue === '') { viewValue = undefined; } @@ -1313,7 +1315,7 @@ angular.module('schemaForm').directive('schemaValidate',['sfValidator',function( ngModel.$parsers.unshift(validate); // Listen to an event so we can validate the input on request - scope.$on('schemaFormValidate',function() { + scope.$on('schemaFormValidate', function() { if (ngModel.$commitViewValue) { ngModel.$commitViewValue(true); @@ -1324,11 +1326,11 @@ angular.module('schemaForm').directive('schemaValidate',['sfValidator',function( //This works since we now we're inside a decorator and that this is the decorators scope. //If $pristine and empty don't show success (even if it's valid) - scope.hasSuccess = function(){ + scope.hasSuccess = function() { return ngModel.$valid && (!ngModel.$pristine || !ngModel.$isEmpty(ngModel.$modelValue)); }; - scope.hasError = function(){ + scope.hasError = function() { return ngModel.$invalid && !ngModel.$pristine; }; diff --git a/dist/schema-form.min.js b/dist/schema-form.min.js index 8b9b02ad1..70183587d 100644 --- a/dist/schema-form.min.js +++ b/dist/schema-form.min.js @@ -1 +1 @@ -var deps=["ObjectPath"];try{angular.module("ngSanitize"),deps.push("ngSanitize")}catch(e){}try{angular.module("ui.sortable"),deps.push("ui.sortable")}catch(e){}angular.module("schemaForm",deps),angular.module("schemaForm").provider("sfPath",["ObjectPathProvider",function(e){var r={parse:e.parse};r.stringify=1===angular.version.major&&angular.version.minor<3?function(e){return Array.isArray(e)?e.join("."):e.toString()}:e.stringify,r.normalize=function(e,t){return r.stringify(Array.isArray(e)?e:r.parse(e),t)},this.parse=r.parse,this.stringify=r.stringify,this.normalize=r.normalize,this.$get=function(){return r}}]),angular.module("schemaForm").factory("sfSelect",["sfPath",function(e){var r=/^\d+$/;return function(t,a,n){a||(a=this);var i="string"==typeof t?e.parse(t):t;if("undefined"!=typeof n&&1===i.length)return a[i[0]]=n,a;"undefined"!=typeof n&&"undefined"==typeof a[i[0]]&&(a[i[0]]=i.length>2&&r.test(i[1])?[]:{});for(var o=a[i[0]],u=1;u',link:function(e,t,a){var n={items:"c",titleMap:"c",schema:"c"},i={type:r},o=!0;angular.forEach(a,function(r,t){if("$"!==t[0]&&0!==t.indexOf("ng")&&"sfField"!==t){var u=function(r){angular.isDefined(r)&&r!==i[t]&&(i[t]=r,o&&i.type&&(i.key||angular.isUndefined(a.key))&&(e.form=i,o=!1))};"model"===t?e.$watch(r,function(r){r&&e.model!==r&&(e.model=r)}):"c"===n[t]?e.$watchCollection(r,u):a.$observe(t,u)}})}}})};this.createDecorator=function(e,r,n){a[e]={mappings:r||{},rules:n||[]},a[t]||(t=e),i(e)},this.createDirective=o,this.createDirectives=function(e){angular.forEach(e,function(e,r){o(r,e)})},this.directive=function(e){return e=e||t,a[e]},this.addMapping=function(e,r,t){a[e]&&(a[e].mappings[r]=t)},this.$get=function(){return{directive:function(e){return a[e]},defaultDecorator:t}},i("sfDecorator")}]),angular.module("schemaForm").provider("schemaForm",["sfPathProvider",function(e){var r=function(e){var r=[];return e.forEach(function(e){r.push({name:e,value:e})}),r},t=function(e){if(!angular.isArray(e)){var r=[];return angular.forEach(e,function(e,t){r.push({name:e,value:t})}),r}return e},a=function(e,r,t){var a=d[r.type];if(a)for(var n,i=0;i1&&(c={type:"section",items:i.items})}if(n.copyWithIndex=function(e){if(!l[e]&&c){var t=angular.copy(c);t.arrayIndex=e,r.traverseForm(t,a(e)),l[e]=t}return l[e]},n.appendToArray=function(){var t=o.length,a=n.copyWithIndex(t);if(r.traverseForm(a,function(r){r.key&&angular.isDefined(r.default)&&e(r.key,n.model,r.default)}),t===o.length){var u,l=e("schema.items.type",i);"object"===l?u={}:"array"===l&&(u=[]),o.push(u)}n.validateArray&&n.validateArray()},n.deleteFromArray=function(e){o.splice(e,1),n.validateArray&&n.validateArray()},i.titleMap||i.startEmpty===!0||0!==o.length||n.appendToArray(),i.titleMap&&i.titleMap.length>0){n.titleMapValues=[];var f=function(e){n.titleMapValues=[],e=e||[],i.titleMap.forEach(function(r){n.titleMapValues.push(-1!==e.indexOf(r.value))})};f(n.modelArray),n.$watchCollection("modelArray",f),n.$watchCollection("titleMapValues",function(e){if(e){for(var r=n.modelArray;r.length>0;)r.shift();i.titleMap.forEach(function(t,a){e[a]&&r.push(t.value)})}})}if(u){var m;n.validateArray=function(){var e=t.validate(i,n.modelArray.length>0?n.modelArray:void 0);e.valid!==!1||!e.error||""!==e.error.dataPath&&e.error.dataPath!=="/"+i.key[i.key.length-1]?u.$setValidity("schema",!0):(u.$setViewValue(n.modelArray),m=e.error,u.$setValidity("schema",!1))},n.$on("schemaFormValidate",n.validateArray),n.hasSuccess=function(){return u.$valid&&!u.$pristine},n.hasError=function(){return u.$invalid},n.schemaError=function(){return m}}s()})}}}]),angular.module("schemaForm").directive("sfChanged",function(){return{require:"ngModel",restrict:"AC",scope:!1,link:function(e,r,t,a){var n=e.$eval(t.sfChanged);n&&n.onChange&&a.$viewChangeListeners.push(function(){angular.isFunction(n.onChange)?n.onChange(a.$modelValue,n):e.evalExpr(n.onChange,{modelValue:a.$modelValue,form:n})})}}}),angular.module("schemaForm").directive("sfSchema",["$compile","schemaForm","schemaFormDecorators","sfSelect",function(e,r,t,a){function n(e,r){return r=r||"_",e.replace(i,function(e,t){return(t?r:"")+e.toLowerCase()})}var i=/[A-Z]/g;return{scope:{schema:"=sfSchema",initialForm:"=sfForm",model:"=sfModel"},controller:["$scope",function(e){this.evalInParentScope=function(r,t){return e.$parent.$eval(r,t)}}],replace:!1,restrict:"A",transclude:!0,require:"?form",link:function(i,o,u,l,s){i.formCtrl=l;var c={};s(i,function(e){if(e.addClass("schema-form-ignore"),o.prepend(e),o[0].querySelectorAll){var r=o[0].querySelectorAll("[ng-model]");if(r)for(var t=0;t0){f.schema=l,f.form=s;var m=i.$eval(u.sfOptions),d=r.merge(l,s,c,m),p=document.createDocumentFragment();i.schemaForm={form:d,schema:l},angular.forEach(d,function(e,r){var a=document.createElement(u.sfDecoratorName||n(t.defaultDecorator,"-"));a.setAttribute("form","schemaForm.form["+r+"]"),p.appendChild(a)}),o.children(":not(.schema-form-ignore)").remove(),o[0].appendChild(p),e(o.children())(i),r.traverseSchema(l,function(e,r){if(angular.isDefined(e["default"])){var t=a(r,i.model);angular.isUndefined(t)&&a(r,i.model,e["default"])}})}})}}}]),angular.module("schemaForm").directive("schemaValidate",["sfValidator",function(e){return{restrict:"A",scope:!1,require:"ngModel",link:function(r,t,a,n){r.ngModel=n;var i=null,o=r.$eval(a.schemaValidate),u=function(t){if(o||(o=r.$eval(a.schemaValidate)),!o)return t;if(angular.isDefined(a.ngRequired)&&angular.isUndefined(t))return void 0;""===t&&(t=void 0);var u=e.validate(o,t);return u.valid?(n.$setValidity("schema",!0),t):(n.$setValidity("schema",!1),void(i=u.error))};n.$parsers.unshift(u),r.$on("schemaFormValidate",function(){n.$commitViewValue?n.$commitViewValue(!0):n.$setViewValue(n.$viewValue)}),r.hasSuccess=function(){return n.$valid&&(!n.$pristine||!n.$isEmpty(n.$modelValue))},r.hasError=function(){return n.$invalid&&!n.$pristine},r.schemaError=function(){return i}}}}]); \ No newline at end of file +var deps=["ObjectPath"];try{angular.module("ngSanitize"),deps.push("ngSanitize")}catch(e){}try{angular.module("ui.sortable"),deps.push("ui.sortable")}catch(e){}angular.module("schemaForm",deps),angular.module("schemaForm").provider("sfPath",["ObjectPathProvider",function(e){var r={parse:e.parse};r.stringify=1===angular.version.major&&angular.version.minor<3?function(e){return Array.isArray(e)?e.join("."):e.toString()}:e.stringify,r.normalize=function(e,t){return r.stringify(Array.isArray(e)?e:r.parse(e),t)},this.parse=r.parse,this.stringify=r.stringify,this.normalize=r.normalize,this.$get=function(){return r}}]),angular.module("schemaForm").factory("sfSelect",["sfPath",function(e){var r=/^\d+$/;return function(t,a,n){a||(a=this);var i="string"==typeof t?e.parse(t):t;if("undefined"!=typeof n&&1===i.length)return a[i[0]]=n,a;"undefined"!=typeof n&&"undefined"==typeof a[i[0]]&&(a[i[0]]=i.length>2&&r.test(i[1])?[]:{});for(var o=a[i[0]],u=1;u',link:function(e,t,a){var n={items:"c",titleMap:"c",schema:"c"},i={type:r},o=!0;angular.forEach(a,function(r,t){if("$"!==t[0]&&0!==t.indexOf("ng")&&"sfField"!==t){var u=function(r){angular.isDefined(r)&&r!==i[t]&&(i[t]=r,o&&i.type&&(i.key||angular.isUndefined(a.key))&&(e.form=i,o=!1))};"model"===t?e.$watch(r,function(r){r&&e.model!==r&&(e.model=r)}):"c"===n[t]?e.$watchCollection(r,u):a.$observe(t,u)}})}}})};this.createDecorator=function(e,r,n){a[e]={mappings:r||{},rules:n||[]},a[t]||(t=e),i(e)},this.createDirective=o,this.createDirectives=function(e){angular.forEach(e,function(e,r){o(r,e)})},this.directive=function(e){return e=e||t,a[e]},this.addMapping=function(e,r,t){a[e]&&(a[e].mappings[r]=t)},this.$get=function(){return{directive:function(e){return a[e]},defaultDecorator:t}},i("sfDecorator")}]),angular.module("schemaForm").provider("schemaForm",["sfPathProvider",function(e){var r=function(e){var r=[];return e.forEach(function(e){r.push({name:e,value:e})}),r},t=function(e){if(!angular.isArray(e)){var r=[];return angular.forEach(e,function(e,t){r.push({name:e,value:t})}),r}return e},a=function(e,r,t){var a=d[r.type];if(a)for(var n,i=0;i1&&(c={type:"section",items:i.items})}if(n.copyWithIndex=function(e){if(!l[e]&&c){var t=angular.copy(c);t.arrayIndex=e,r.traverseForm(t,a(e)),l[e]=t}return l[e]},n.appendToArray=function(){var t=o.length,a=n.copyWithIndex(t);if(r.traverseForm(a,function(r){r.key&&angular.isDefined(r.default)&&e(r.key,n.model,r.default)}),t===o.length){var u,l=e("schema.items.type",i);"object"===l?u={}:"array"===l&&(u=[]),o.push(u)}n.validateArray&&n.validateArray()},n.deleteFromArray=function(e){o.splice(e,1),n.validateArray&&n.validateArray()},i.titleMap||i.startEmpty===!0||0!==o.length||n.appendToArray(),i.titleMap&&i.titleMap.length>0){n.titleMapValues=[];var f=function(e){n.titleMapValues=[],e=e||[],i.titleMap.forEach(function(r){n.titleMapValues.push(-1!==e.indexOf(r.value))})};f(n.modelArray),n.$watchCollection("modelArray",f),n.$watchCollection("titleMapValues",function(e){if(e){for(var r=n.modelArray;r.length>0;)r.shift();i.titleMap.forEach(function(t,a){e[a]&&r.push(t.value)})}})}if(u){var m;n.validateArray=function(){var e=t.validate(i,n.modelArray.length>0?n.modelArray:void 0);e.valid!==!1||!e.error||""!==e.error.dataPath&&e.error.dataPath!=="/"+i.key[i.key.length-1]?u.$setValidity("schema",!0):(u.$setViewValue(n.modelArray),m=e.error,u.$setValidity("schema",!1))},n.$on("schemaFormValidate",n.validateArray),n.hasSuccess=function(){return u.$valid&&!u.$pristine},n.hasError=function(){return u.$invalid},n.schemaError=function(){return m}}s()})}}}]),angular.module("schemaForm").directive("sfChanged",function(){return{require:"ngModel",restrict:"AC",scope:!1,link:function(e,r,t,a){var n=e.$eval(t.sfChanged);n&&n.onChange&&a.$viewChangeListeners.push(function(){angular.isFunction(n.onChange)?n.onChange(a.$modelValue,n):e.evalExpr(n.onChange,{modelValue:a.$modelValue,form:n})})}}}),angular.module("schemaForm").directive("sfSchema",["$compile","schemaForm","schemaFormDecorators","sfSelect",function(e,r,t,a){var n=/[A-Z]/g,i=function(e,r){return r=r||"_",e.replace(n,function(e,t){return(t?r:"")+e.toLowerCase()})};return{scope:{schema:"=sfSchema",initialForm:"=sfForm",model:"=sfModel"},controller:["$scope",function(e){this.evalInParentScope=function(r,t){return e.$parent.$eval(r,t)}}],replace:!1,restrict:"A",transclude:!0,require:"?form",link:function(n,o,u,l,s){n.formCtrl=l;var c={};s(n,function(e){if(e.addClass("schema-form-ignore"),o.prepend(e),o[0].querySelectorAll){var r=o[0].querySelectorAll("[ng-model]");if(r)for(var t=0;t0){f.schema=l,f.form=s;var m=n.$eval(u.sfOptions),d=r.merge(l,s,c,m),p=document.createDocumentFragment();n.schemaForm={form:d,schema:l},angular.forEach(d,function(e,r){var a=document.createElement(u.sfDecoratorName||i(t.defaultDecorator,"-"));a.setAttribute("form","schemaForm.form["+r+"]"),p.appendChild(a)}),o.children(":not(.schema-form-ignore)").remove(),o[0].appendChild(p),e(o.children())(n),r.traverseSchema(l,function(e,r){if(angular.isDefined(e["default"])){var t=a(r,n.model);angular.isUndefined(t)&&a(r,n.model,e["default"])}})}})}}}]),angular.module("schemaForm").directive("schemaValidate",["sfValidator",function(e){return{restrict:"A",scope:!1,require:"ngModel",link:function(r,t,a,n){r.ngModel=n;var i=null,o=r.$eval(a.schemaValidate),u=function(t){if(o||(o=r.$eval(a.schemaValidate)),!o)return t;if(angular.isDefined(a.ngRequired)&&angular.isUndefined(t))return void 0;""===t&&(t=void 0);var u=e.validate(o,t);return u.valid?(n.$setValidity("schema",!0),t):(n.$setValidity("schema",!1),void(i=u.error))};n.$parsers.unshift(u),r.$on("schemaFormValidate",function(){n.$commitViewValue?n.$commitViewValue(!0):n.$setViewValue(n.$viewValue)}),r.hasSuccess=function(){return n.$valid&&(!n.$pristine||!n.$isEmpty(n.$modelValue))},r.hasError=function(){return n.$invalid&&!n.$pristine},r.schemaError=function(){return i}}}}]); \ No newline at end of file diff --git a/gulpfile.js b/gulpfile.js index 2a6a1bbda..4c58da9e5 100644 --- a/gulpfile.js +++ b/gulpfile.js @@ -3,62 +3,61 @@ var gulp = require('gulp'); var templateCache = require('gulp-angular-templatecache'); -var minifyHtml = require("gulp-minify-html"); -var concat = require("gulp-concat"); -var uglify = require("gulp-uglify"); +var minifyHtml = require('gulp-minify-html'); +var concat = require('gulp-concat'); +var uglify = require('gulp-uglify'); var streamqueue = require('streamqueue'); +var jscs = require('gulp-jscs'); + gulp.task('bootstrap', function() { - var stream = streamqueue({ objectMode: true }); + var stream = streamqueue({objectMode: true}); stream.queue( - gulp.src("./src/directives/decorators/bootstrap/*.html") + gulp.src('./src/directives/decorators/bootstrap/*.html') .pipe(minifyHtml({ - empty: true, - spare: true, - quotes: true + empty: true, + spare: true, + quotes: true })) .pipe(templateCache({ - module: "schemaForm", - root: "directives/decorators/bootstrap/" + module: 'schemaForm', + root: 'directives/decorators/bootstrap/' })) ); - stream.queue(gulp.src('./src/directives/decorators/bootstrap/*.js')); + stream.queue(gulp.src('./src/directives/decorators/bootstrap/*.js')); - stream.done() - .pipe(concat('bootstrap-decorator.min.js')) - .pipe(uglify()) - .pipe(gulp.dest("./dist/")); + stream.done() + .pipe(concat('bootstrap-decorator.min.js')) + .pipe(uglify()) + .pipe(gulp.dest('./dist/')); }); gulp.task('bootstrap-datepicker', function() { - var stream = streamqueue({ objectMode: true }); + var stream = streamqueue({objectMode: true}); stream.queue( - gulp.src("./src/directives/decorators/bootstrap/datepicker/*.html") + gulp.src('./src/directives/decorators/bootstrap/datepicker/*.html') .pipe(minifyHtml({ - empty: true, - spare: true, - quotes: true + empty: true, + spare: true, + quotes: true })) .pipe(templateCache({ - module: "schemaForm", - root: "directives/decorators/bootstrap/datepicker/" + module: 'schemaForm', + root: 'directives/decorators/bootstrap/datepicker/' })) ); - stream.queue(gulp.src('./src/directives/decorators/bootstrap/datepicker/*.js')); + stream.queue(gulp.src('./src/directives/decorators/bootstrap/datepicker/*.js')); - stream.done() - .pipe(concat('bootstrap-datepicker.min.js')) - .pipe(uglify()) - .pipe(gulp.dest("./dist/")); - + stream.done() + .pipe(concat('bootstrap-datepicker.min.js')) + .pipe(uglify()) + .pipe(gulp.dest('./dist/')); }); - - -gulp.task('minify',function(){ +gulp.task('minify', function() { gulp.src([ './src/module.js', './src/sfPath.js', @@ -70,8 +69,7 @@ gulp.task('minify',function(){ .pipe(gulp.dest('./dist/')); }); - -gulp.task('non-minified-dist',function(){ +gulp.task('non-minified-dist', function() { gulp.src([ './src/module.js', './src/sfPath.js', @@ -82,9 +80,17 @@ gulp.task('non-minified-dist',function(){ .pipe(gulp.dest('./dist/')); }); +gulp.task('jscs', function() { + gulp.src('./src/**/*.js') + .pipe(jscs()); +}); -gulp.task('default',['minify','bootstrap','bootstrap-datepicker','non-minified-dist']); - +gulp.task('default', [ + 'minify', + 'bootstrap', + 'bootstrap-datepicker', + 'non-minified-dist' +]); gulp.task('watch', function() { gulp.watch('./src/**/*', ['default']); diff --git a/package.json b/package.json index 83445e48a..6a99a42cd 100644 --- a/package.json +++ b/package.json @@ -13,6 +13,7 @@ "gulp": "^3.5.6", "gulp-angular-templatecache": "^1.2.1", "gulp-concat": "^2.2.0", + "gulp-jscs": "^1.1.0", "gulp-minify-html": "^0.1.1", "gulp-uglify": "^0.2.1", "karma": "^0.12.0", diff --git a/src/directives/array.js b/src/directives/array.js index f35bb14a3..8b9f047a5 100644 --- a/src/directives/array.js +++ b/src/directives/array.js @@ -1,210 +1,208 @@ /** * Directive that handles the model arrays */ -angular.module('schemaForm').directive('sfArray', ['sfSelect','schemaForm','sfValidator', -function(sfSelect, schemaForm, sfValidator) { +angular.module('schemaForm').directive('sfArray', ['sfSelect', 'schemaForm', 'sfValidator', + function(sfSelect, schemaForm, sfValidator) { - var setIndex = function(index) { - return function(form) { - if (form.key) { - form.key[form.key.indexOf('')] = index; - } - }; - }; - - return { - restrict: 'A', - scope: true, - require: '?ngModel', - link: function(scope, element, attrs, ngModel) { - var formDefCache = {}; - - // Watch for the form definition and then rewrite it. - // It's the (first) array part of the key, '[]' that needs a number - // corresponding to an index of the form. - var once = scope.$watch(attrs.sfArray, function(form) { - - // An array model always needs a key so we know what part of the model - // to look at. This makes us a bit incompatible with JSON Form, on the - // other hand it enables two way binding. - var list = sfSelect(form.key,scope.model); - - // Since ng-model happily creates objects in a deep path when setting a - // a value but not arrays we need to create the array. - if (angular.isUndefined(list)) { - list = []; - sfSelect(form.key, scope.model, list); - } - scope.modelArray = list; - - // Arrays with titleMaps, i.e. checkboxes doesn't have items. - if (form.items) { - - // To be more compatible with JSON Form we support an array of items - // in the form definition of "array" (the schema just a value). - // for the subforms code to work this means we wrap everything in a - // section. Unless there is just one. - var subForm = form.items[0]; - if (form.items.length > 1) { - subForm = { type: 'section', items: form.items }; - } + var setIndex = function(index) { + return function(form) { + if (form.key) { + form.key[form.key.indexOf('')] = index; } + }; + }; - // We ceate copies of the form on demand, caching them for - // later requests - scope.copyWithIndex = function(index) { - if (!formDefCache[index]) { - if (subForm) { - var copy = angular.copy(subForm); - copy.arrayIndex= index; - schemaForm.traverseForm(copy, setIndex(index)); - formDefCache[index] = copy; + return { + restrict: 'A', + scope: true, + require: '?ngModel', + link: function(scope, element, attrs, ngModel) { + var formDefCache = {}; + + // Watch for the form definition and then rewrite it. + // It's the (first) array part of the key, '[]' that needs a number + // corresponding to an index of the form. + var once = scope.$watch(attrs.sfArray, function(form) { + + // An array model always needs a key so we know what part of the model + // to look at. This makes us a bit incompatible with JSON Form, on the + // other hand it enables two way binding. + var list = sfSelect(form.key, scope.model); + + // Since ng-model happily creates objects in a deep path when setting a + // a value but not arrays we need to create the array. + if (angular.isUndefined(list)) { + list = []; + sfSelect(form.key, scope.model, list); + } + scope.modelArray = list; + + // Arrays with titleMaps, i.e. checkboxes doesn't have items. + if (form.items) { + + // To be more compatible with JSON Form we support an array of items + // in the form definition of "array" (the schema just a value). + // for the subforms code to work this means we wrap everything in a + // section. Unless there is just one. + var subForm = form.items[0]; + if (form.items.length > 1) { + subForm = {type: 'section', items: form.items}; } } - return formDefCache[index]; - }; + // We ceate copies of the form on demand, caching them for + // later requests + scope.copyWithIndex = function(index) { + if (!formDefCache[index]) { + if (subForm) { + var copy = angular.copy(subForm); + copy.arrayIndex = index; + schemaForm.traverseForm(copy, setIndex(index)); + formDefCache[index] = copy; + } + } + return formDefCache[index]; + }; + + scope.appendToArray = function() { + var len = list.length; + var copy = scope.copyWithIndex(len); + schemaForm.traverseForm(copy, function(part) { + if (part.key && angular.isDefined(part.default)) { + sfSelect(part.key, scope.model, part.default); + } + }); - scope.appendToArray = function() { - var len = list.length; - var copy = scope.copyWithIndex(len); - schemaForm.traverseForm(copy, function(part){ - if (part.key && angular.isDefined(part.default)) { - sfSelect(part.key, scope.model, part.default); + // If there are no defaults nothing is added so we need to initialize + // the array. undefined for basic values, {} or [] for the others. + if (len === list.length) { + var type = sfSelect('schema.items.type', form); + var dflt; + if (type === 'object') { + dflt = {}; + } else if (type === 'array') { + dflt = []; + } + list.push(dflt); } - }); - - // If there are no defaults nothing is added so we need to initialize - // the array. undefined for basic values, {} or [] for the others. - if (len === list.length) { - var type = sfSelect('schema.items.type',form); - var dflt; - if (type === 'object') { - dflt = {}; - } else if (type === 'array') { - dflt = []; + + // Trigger validation. + if (scope.validateArray) { + scope.validateArray(); } - list.push(dflt); - } + }; - // Trigger validation. - if (scope.validateArray) { - scope.validateArray(); - } - }; + scope.deleteFromArray = function(index) { + list.splice(index, 1); - scope.deleteFromArray = function(index) { - list.splice(index,1); + // Trigger validation. + if (scope.validateArray) { + scope.validateArray(); + } + }; - // Trigger validation. - if (scope.validateArray) { - scope.validateArray(); + // Always start with one empty form unless configured otherwise. + // Special case: don't do it if form has a titleMap + if (!form.titleMap && form.startEmpty !== true && list.length === 0) { + scope.appendToArray(); } - }; - // Always start with one empty form unless configured otherwise. - // Special case: don't do it if form has a titleMap - if (!form.titleMap && form.startEmpty !== true && list.length === 0) { - scope.appendToArray(); - } - - // Title Map handling - // If form has a titleMap configured we'd like to enable looping over - // titleMap instead of modelArray, this is used for intance in - // checkboxes. So instead of variable number of things we like to create - // a array value from a subset of values in the titleMap. - // The problem here is that ng-model on a checkbox doesn't really map to - // a list of values. This is here to fix that. - if (form.titleMap && form.titleMap.length > 0) { - scope.titleMapValues = []; - - // We watch the model for changes and the titleMapValues to reflect - // the modelArray - var updateTitleMapValues = function(arr) { + // Title Map handling + // If form has a titleMap configured we'd like to enable looping over + // titleMap instead of modelArray, this is used for intance in + // checkboxes. So instead of variable number of things we like to create + // a array value from a subset of values in the titleMap. + // The problem here is that ng-model on a checkbox doesn't really map to + // a list of values. This is here to fix that. + if (form.titleMap && form.titleMap.length > 0) { scope.titleMapValues = []; - arr = arr || []; - - form.titleMap.forEach(function(item) { - scope.titleMapValues.push( arr.indexOf(item.value) !== -1 ); - }); - }; - //Catch default values - updateTitleMapValues(scope.modelArray); - scope.$watchCollection('modelArray',updateTitleMapValues); - - //To get two way binding we also watch our titleMapValues - scope.$watchCollection('titleMapValues', function(vals) { - if (vals) { - var arr = scope.modelArray; - - // Apparently the fastest way to clear an array, readable too. - // http://jsperf.com/array-destroy/32 - while (arr.length > 0) { - arr.shift(); - } + // We watch the model for changes and the titleMapValues to reflect + // the modelArray + var updateTitleMapValues = function(arr) { + scope.titleMapValues = []; + arr = arr || []; - form.titleMap.forEach(function(item,index) { - if (vals[index]) { - arr.push(item.value); - } + form.titleMap.forEach(function(item) { + scope.titleMapValues.push(arr.indexOf(item.value) !== -1); }); - } - }); - } + }; + //Catch default values + updateTitleMapValues(scope.modelArray); + scope.$watchCollection('modelArray', updateTitleMapValues); + //To get two way binding we also watch our titleMapValues + scope.$watchCollection('titleMapValues', function(vals) { + if (vals) { + var arr = scope.modelArray; - // If there is a ngModel present we need to validate when asked. - if (ngModel) { - var error; - - scope.validateArray = function() { - // The actual content of the array is validated by each field - // so we settle for checking validations specific to arrays - - // Since we prefill with empty arrays we can get the funny situation - // where the array is required but empty in the gui but still validates. - // Thats why we check the length. - var result = sfValidator.validate( - form, - scope.modelArray.length > 0 ? scope.modelArray : undefined - ); - if (result.valid === false && - result.error && - (result.error.dataPath === '' || - result.error.dataPath === '/'+form.key[form.key.length - 1])) { - - // Set viewValue to trigger $dirty on field. If someone knows a - // a better way to do it please tell. - ngModel.$setViewValue(scope.modelArray); - error = result.error; - ngModel.$setValidity('schema', false); - - } else { - ngModel.$setValidity('schema', true); - } - }; + // Apparently the fastest way to clear an array, readable too. + // http://jsperf.com/array-destroy/32 + while (arr.length > 0) { + arr.shift(); + } + + form.titleMap.forEach(function(item, index) { + if (vals[index]) { + arr.push(item.value); + } + }); + + } + }); + } - scope.$on('schemaFormValidate',scope.validateArray); + // If there is a ngModel present we need to validate when asked. + if (ngModel) { + var error; + + scope.validateArray = function() { + // The actual content of the array is validated by each field + // so we settle for checking validations specific to arrays + + // Since we prefill with empty arrays we can get the funny situation + // where the array is required but empty in the gui but still validates. + // Thats why we check the length. + var result = sfValidator.validate( + form, + scope.modelArray.length > 0 ? scope.modelArray : undefined + ); + if (result.valid === false && + result.error && + (result.error.dataPath === '' || + result.error.dataPath === '/' + form.key[form.key.length - 1])) { + + // Set viewValue to trigger $dirty on field. If someone knows a + // a better way to do it please tell. + ngModel.$setViewValue(scope.modelArray); + error = result.error; + ngModel.$setValidity('schema', false); + + } else { + ngModel.$setValidity('schema', true); + } + }; + scope.$on('schemaFormValidate', scope.validateArray); - scope.hasSuccess = function(){ - return ngModel.$valid && !ngModel.$pristine; - }; + scope.hasSuccess = function() { + return ngModel.$valid && !ngModel.$pristine; + }; - scope.hasError = function(){ - return ngModel.$invalid; - }; + scope.hasError = function() { + return ngModel.$invalid; + }; - scope.schemaError = function() { - return error; - }; + scope.schemaError = function() { + return error; + }; - } + } - once(); - }); - } - }; -}]); + once(); + }); + } + }; + } +]); diff --git a/src/directives/changed.js b/src/directives/changed.js index 1d4534df0..8fc948c3f 100644 --- a/src/directives/changed.js +++ b/src/directives/changed.js @@ -5,22 +5,22 @@ * Takes the form definition as argument. * If the form definition has a "onChange" defined as either a function or */ -angular.module('schemaForm').directive('sfChanged',function(){ +angular.module('schemaForm').directive('sfChanged', function() { return { require: 'ngModel', restrict: 'AC', scope: false, - link: function(scope,element,attrs,ctrl) { + link: function(scope, element, attrs, ctrl) { var form = scope.$eval(attrs.sfChanged); //"form" is really guaranteed to be here since the decorator directive //waits for it. But best be sure. if (form && form.onChange) { ctrl.$viewChangeListeners.push(function() { - if (angular.isFunction(form.onChange)) { - form.onChange(ctrl.$modelValue,form); - } else { - scope.evalExpr(form.onChange,{ 'modelValue': ctrl.$modelValue, form: form }); - } + if (angular.isFunction(form.onChange)) { + form.onChange(ctrl.$modelValue, form); + } else { + scope.evalExpr(form.onChange, {'modelValue': ctrl.$modelValue, form: form}); + } }); } } diff --git a/src/directives/decorators/bootstrap/bootstrap-decorator.js b/src/directives/decorators/bootstrap/bootstrap-decorator.js index b45beaf2d..ccd3af788 100644 --- a/src/directives/decorators/bootstrap/bootstrap-decorator.js +++ b/src/directives/decorators/bootstrap/bootstrap-decorator.js @@ -1,60 +1,60 @@ -angular.module('schemaForm').config(['schemaFormDecoratorsProvider',function(decoratorsProvider){ +angular.module('schemaForm').config(['schemaFormDecoratorsProvider', function(decoratorsProvider) { var base = 'directives/decorators/bootstrap/'; - decoratorsProvider.createDecorator('bootstrapDecorator',{ - textarea: base+'textarea.html', - fieldset: base+'fieldset.html', - array: base+'array.html', - tabarray: base+'tabarray.html', - tabs: base+'tabs.html', - section: base+'section.html', - conditional: base+'section.html', - actions: base+'actions.html', - select: base+'select.html', - checkbox: base+'checkbox.html', - checkboxes: base+'checkboxes.html', - number: base+'default.html', - password: base+'default.html', - submit: base+'submit.html', - button: base+'submit.html', - radios: base+'radios.html', - 'radios-inline': base+'radios-inline.html', - radiobuttons: base+'radio-buttons.html', - help: base+'help.html', - 'default': base+'default.html' - },[ - function(form){ + decoratorsProvider.createDecorator('bootstrapDecorator', { + textarea: base + 'textarea.html', + fieldset: base + 'fieldset.html', + array: base + 'array.html', + tabarray: base + 'tabarray.html', + tabs: base + 'tabs.html', + section: base + 'section.html', + conditional: base + 'section.html', + actions: base + 'actions.html', + select: base + 'select.html', + checkbox: base + 'checkbox.html', + checkboxes: base + 'checkboxes.html', + number: base + 'default.html', + password: base + 'default.html', + submit: base + 'submit.html', + button: base + 'submit.html', + radios: base + 'radios.html', + 'radios-inline': base + 'radios-inline.html', + radiobuttons: base + 'radio-buttons.html', + help: base + 'help.html', + 'default': base + 'default.html' + }, [ + function(form) { if (form.readonly && form.key && form.type !== 'fieldset') { - return base+'readonly.html'; + return base + 'readonly.html'; } } ]); //manual use directives decoratorsProvider.createDirectives({ - textarea: base+'textarea.html', - select: base+'select.html', - checkbox: base+'checkbox.html', - checkboxes: base+'checkboxes.html', - number: base+'default.html', - submit: base+'submit.html', - button: base+'submit.html', - text: base+'default.html', - date: base+'default.html', - password: base+'default.html', - datepicker: base+'datepicker.html', - input: base+'default.html', - radios: base+'radios.html', - 'radios-inline': base+'radios-inline.html', - radiobuttons: base+'radio-buttons.html', + textarea: base + 'textarea.html', + select: base + 'select.html', + checkbox: base + 'checkbox.html', + checkboxes: base + 'checkboxes.html', + number: base + 'default.html', + submit: base + 'submit.html', + button: base + 'submit.html', + text: base + 'default.html', + date: base + 'default.html', + password: base + 'default.html', + datepicker: base + 'datepicker.html', + input: base + 'default.html', + radios: base + 'radios.html', + 'radios-inline': base + 'radios-inline.html', + radiobuttons: base + 'radio-buttons.html', }); -}]).directive('sfFieldset',function(){ +}]).directive('sfFieldset', function() { return { transclude: true, scope: true, templateUrl: 'directives/decorators/bootstrap/fieldset-trcl.html', - link: function(scope,element,attrs) { + link: function(scope, element, attrs) { scope.title = scope.$eval(attrs.title); } }; diff --git a/src/directives/decorators/bootstrap/datepicker/angular-pickadate.js b/src/directives/decorators/bootstrap/datepicker/angular-pickadate.js index 7398b99ae..dd67a585d 100644 --- a/src/directives/decorators/bootstrap/datepicker/angular-pickadate.js +++ b/src/directives/decorators/bootstrap/datepicker/angular-pickadate.js @@ -12,12 +12,12 @@ angular.module('schemaForm').directive('pickADate', function () { }; return { - restrict: "A", + restrict: 'A', require: 'ngModel', scope: { - ngModel: '=', - minDate: '=', - maxDate: '=' + ngModel: '=', + minDate: '=', + maxDate: '=' }, link: function (scope, element, attrs, ngModel) { //Bail out gracefully if pickadate is not loaded. @@ -37,7 +37,7 @@ angular.module('schemaForm').directive('pickADate', function () { //Defaultformat is for json schema date-time is ISO8601 //i.e. "yyyy-mm-dd" - var defaultFormat = "yyyy-mm-dd"; + var defaultFormat = 'yyyy-mm-dd'; //View format on the other hand we get from the pickadate translation file var viewFormat = $.fn.pickadate.defaults.format; @@ -45,22 +45,22 @@ angular.module('schemaForm').directive('pickADate', function () { var picker = element.pickadate('picker'); //The view value - ngModel.$formatters.push(function(value){ + ngModel.$formatters.push(function(value) { if (angular.isUndefined(value) || value === null) { return value; } //We set 'view' and 'highlight' instead of 'select' //since the latter also changes the input, which we do not want. - picker.set('view',value,{ format: attrs.format || defaultFormat }); - picker.set('highlight',value,{ format: attrs.format || defaultFormat }); + picker.set('view', value, {format: attrs.format || defaultFormat}); + picker.set('highlight', value, {format: attrs.format || defaultFormat}); //piggy back on highlight to and let pickadate do the transformation. - return picker.get('highlight',viewFormat); + return picker.get('highlight', viewFormat); }); - ngModel.$parsers.push(function(){ - return picker.get('select',attrs.format || defaultFormat); + ngModel.$parsers.push(function() { + return picker.get('select', attrs.format || defaultFormat); }); //bind once. diff --git a/src/directives/decorators/bootstrap/datepicker/bootstrap-datepicker.js b/src/directives/decorators/bootstrap/datepicker/bootstrap-datepicker.js index 9c66533db..2358a948c 100644 --- a/src/directives/decorators/bootstrap/datepicker/bootstrap-datepicker.js +++ b/src/directives/decorators/bootstrap/datepicker/bootstrap-datepicker.js @@ -1,21 +1,28 @@ angular.module('schemaForm').config( - ['schemaFormProvider','schemaFormDecoratorsProvider','sfPathProvider', -function(schemaFormProvider, schemaFormDecoratorsProvider, sfPathProvider){ +['schemaFormProvider', 'schemaFormDecoratorsProvider', 'sfPathProvider', + function(schemaFormProvider, schemaFormDecoratorsProvider, sfPathProvider) { - var datepicker = function(name,schema,options) { - if (schema.type === 'string' && schema.format == "date") { - var f = schemaFormProvider.stdFormObj(name,schema,options); - f.key = options.path; - f.type = 'datepicker'; - options.lookup[sfPathProvider.stringify(options.path)] = f; - return f; - } - }; + var datepicker = function(name, schema, options) { + if (schema.type === 'string' && schema.format === 'date') { + var f = schemaFormProvider.stdFormObj(name, schema, options); + f.key = options.path; + f.type = 'datepicker'; + options.lookup[sfPathProvider.stringify(options.path)] = f; + return f; + } + }; - schemaFormProvider.defaults.string.unshift(datepicker); + schemaFormProvider.defaults.string.unshift(datepicker); - //Add to the bootstrap directive - schemaFormDecoratorsProvider.addMapping('bootstrapDecorator','datepicker','directives/decorators/bootstrap/datepicker/datepicker.html'); - schemaFormDecoratorsProvider.createDirective('datepicker','directives/decorators/bootstrap/datepicker/datepicker.html'); - -}]); + //Add to the bootstrap directive + schemaFormDecoratorsProvider.addMapping( + 'bootstrapDecorator', + 'datepicker', + 'directives/decorators/bootstrap/datepicker/datepicker.html' + ); + schemaFormDecoratorsProvider.createDirective( + 'datepicker', + 'directives/decorators/bootstrap/datepicker/datepicker.html' + ); + } +]); diff --git a/src/directives/schema-form.js b/src/directives/schema-form.js index 03909829f..de304fa0e 100644 --- a/src/directives/schema-form.js +++ b/src/directives/schema-form.js @@ -1,113 +1,115 @@ /* FIXME: real documentation -
+
*/ angular.module('schemaForm') .directive('sfSchema', - ['$compile','schemaForm','schemaFormDecorators','sfSelect', -function($compile, schemaForm, schemaFormDecorators, sfSelect){ - - var SNAKE_CASE_REGEXP = /[A-Z]/g; - function snake_case(name, separator){ - separator = separator || '_'; - return name.replace(SNAKE_CASE_REGEXP, function(letter, pos) { - return (pos ? separator : '') + letter.toLowerCase(); - }); - } - - return { - scope: { - schema: '=sfSchema', - initialForm: '=sfForm', - model: '=sfModel' - }, - controller: ['$scope',function($scope){ - this.evalInParentScope = function(expr,locals){ - return $scope.$parent.$eval(expr,locals); - }; - }], - replace: false, - restrict: "A", - transclude: true, - require: '?form', - link: function(scope,element,attrs,formCtrl,transclude) { - - //expose form controller on scope so that we don't force authors to use name on form - scope.formCtrl = formCtrl; - - //We'd like to handle existing markup, - //besides using it in our template we also - //check for ng-model and add that to an ignore list - //i.e. even if form has a definition for it or form is ["*"] - //we don't generate it. - var ignore = {}; - transclude(scope,function(clone){ - clone.addClass('schema-form-ignore'); - element.prepend(clone); - - if (element[0].querySelectorAll) { - var models = element[0].querySelectorAll('[ng-model]'); - if (models){ - for (var i=0; i < models.length; i++){ - var key = models[i].getAttribute('ng-model'); - //skip first part before . - ignore[key.substring(key.indexOf('.')+1)] = true; - } - } - } +['$compile', 'schemaForm', 'schemaFormDecorators', 'sfSelect', + function($compile, schemaForm, schemaFormDecorators, sfSelect) { + + var SNAKE_CASE_REGEXP = /[A-Z]/g; + var snakeCase = function(name, separator) { + separator = separator || '_'; + return name.replace(SNAKE_CASE_REGEXP, function(letter, pos) { + return (pos ? separator : '') + letter.toLowerCase(); }); - //Since we are dependant on up to three - //attributes we'll do a common watch - var lastDigest = {}; - - scope.$watch(function(){ - - var schema = scope.schema; - var form = scope.initialForm || ['*']; - - //The check for schema.type is to ensure that schema is not {} - if (form && schema && schema.type && (lastDigest.form !== form || lastDigest.schema !== schema) && Object.keys(schema.properties).length > 0) { - lastDigest.schema = schema; - lastDigest.form = form; - - // Check for options - var options = scope.$eval(attrs.sfOptions); - - var merged = schemaForm.merge(schema,form,ignore,options); - var frag = document.createDocumentFragment(); - - //make the form available to decorators - scope.schemaForm = { form: merged, schema: schema }; - - //Create directives from the form definition - angular.forEach(merged,function(obj,i){ - var n = document.createElement(attrs.sfDecoratorName || snake_case(schemaFormDecorators.defaultDecorator,'-')); - n.setAttribute('form','schemaForm.form['+i+']'); - frag.appendChild(n); - }); - - //clean all but pre existing html. - element.children(':not(.schema-form-ignore)').remove(); - - element[0].appendChild(frag); - - //compile only children - $compile(element.children())(scope); - - //ok, now that that is done let's set any defaults - schemaForm.traverseSchema(schema,function(prop,path){ - - if (angular.isDefined(prop['default'])) { - var val = sfSelect(path, scope.model); - if (angular.isUndefined(val)) { - sfSelect(path, scope.model, prop['default']); + }; + + return { + scope: { + schema: '=sfSchema', + initialForm: '=sfForm', + model: '=sfModel' + }, + controller: ['$scope', function($scope) { + this.evalInParentScope = function(expr, locals) { + return $scope.$parent.$eval(expr, locals); + }; + }], + replace: false, + restrict: 'A', + transclude: true, + require: '?form', + link: function(scope, element, attrs, formCtrl, transclude) { + + //expose form controller on scope so that we don't force authors to use name on form + scope.formCtrl = formCtrl; + + //We'd like to handle existing markup, + //besides using it in our template we also + //check for ng-model and add that to an ignore list + //i.e. even if form has a definition for it or form is ["*"] + //we don't generate it. + var ignore = {}; + transclude(scope, function(clone) { + clone.addClass('schema-form-ignore'); + element.prepend(clone); + + if (element[0].querySelectorAll) { + var models = element[0].querySelectorAll('[ng-model]'); + if (models) { + for (var i = 0; i < models.length; i++) { + var key = models[i].getAttribute('ng-model'); + //skip first part before . + ignore[key.substring(key.indexOf('.') + 1)] = true; } } - }); - - } - }); - } - }; -}]); + } + }); + //Since we are dependant on up to three + //attributes we'll do a common watch + var lastDigest = {}; + + scope.$watch(function() { + + var schema = scope.schema; + var form = scope.initialForm || ['*']; + + //The check for schema.type is to ensure that schema is not {} + if (form && schema && schema.type && + (lastDigest.form !== form || lastDigest.schema !== schema) && + Object.keys(schema.properties).length > 0) { + lastDigest.schema = schema; + lastDigest.form = form; + + // Check for options + var options = scope.$eval(attrs.sfOptions); + + var merged = schemaForm.merge(schema, form, ignore, options); + var frag = document.createDocumentFragment(); + + //make the form available to decorators + scope.schemaForm = {form: merged, schema: schema}; + + //Create directives from the form definition + angular.forEach(merged, function(obj, i) { + var n = document.createElement(attrs.sfDecoratorName || + snakeCase(schemaFormDecorators.defaultDecorator, '-')); + n.setAttribute('form', 'schemaForm.form[' + i + ']'); + frag.appendChild(n); + }); + + //clean all but pre existing html. + element.children(':not(.schema-form-ignore)').remove(); + + element[0].appendChild(frag); + + //compile only children + $compile(element.children())(scope); + + //ok, now that that is done let's set any defaults + schemaForm.traverseSchema(schema, function(prop, path) { + if (angular.isDefined(prop['default'])) { + var val = sfSelect(path, scope.model); + if (angular.isUndefined(val)) { + sfSelect(path, scope.model, prop['default']); + } + } + }); + } + }); + } + }; + } +]); diff --git a/src/directives/schema-validate.js b/src/directives/schema-validate.js index 178aebf0f..96c20e0c4 100644 --- a/src/directives/schema-validate.js +++ b/src/directives/schema-validate.js @@ -1,10 +1,9 @@ -/* global tv4 */ -angular.module('schemaForm').directive('schemaValidate',['sfValidator',function(sfValidator){ +angular.module('schemaForm').directive('schemaValidate', ['sfValidator', function(sfValidator) { return { restrict: 'A', scope: false, require: 'ngModel', - link: function(scope,element,attrs,ngModel) { + link: function(scope, element, attrs, ngModel) { //Since we have scope false this is the same scope //as the decorator scope.ngModel = ngModel; @@ -30,7 +29,7 @@ angular.module('schemaForm').directive('schemaValidate',['sfValidator',function( // An empty field gives us the an empty string, which JSON schema // happily accepts as a proper defined string, but an empty field // for the user should trigger "required". So we set it to undefined. - if (viewValue === "") { + if (viewValue === '') { viewValue = undefined; } @@ -52,7 +51,7 @@ angular.module('schemaForm').directive('schemaValidate',['sfValidator',function( ngModel.$parsers.unshift(validate); // Listen to an event so we can validate the input on request - scope.$on('schemaFormValidate',function() { + scope.$on('schemaFormValidate', function() { if (ngModel.$commitViewValue) { ngModel.$commitViewValue(true); @@ -63,11 +62,11 @@ angular.module('schemaForm').directive('schemaValidate',['sfValidator',function( //This works since we now we're inside a decorator and that this is the decorators scope. //If $pristine and empty don't show success (even if it's valid) - scope.hasSuccess = function(){ + scope.hasSuccess = function() { return ngModel.$valid && (!ngModel.$pristine || !ngModel.$isEmpty(ngModel.$modelValue)); }; - scope.hasError = function(){ + scope.hasError = function() { return ngModel.$invalid && !ngModel.$pristine; }; diff --git a/src/module.js b/src/module.js index a54f386df..fffbc4a3b 100644 --- a/src/module.js +++ b/src/module.js @@ -11,4 +11,4 @@ try { deps.push('ui.sortable'); } catch (e) {} -angular.module('schemaForm',deps); +angular.module('schemaForm', deps); diff --git a/src/services/Select.js b/src/services/Select.js index a2b23f433..fdb68310c 100644 --- a/src/services/Select.js +++ b/src/services/Select.js @@ -3,24 +3,26 @@ * @name sfSelect * @kind function * - * @description - * Utility method to access deep properties without - * throwing errors when things are not defined. - * Can also set a value in a deep structure, creating objects when missing - * ex. - * var foo = Select('address.contact.name',obj) - * Select('address.contact.name',obj,'Leeroy') - * - * @param {string} projection A dot path to the property you want to get/set - * @param {object} obj (optional) The object to project on, defaults to 'this' - * @param {Any} value (opional) The value to set, if parts of the path of - * the projection is missing empty objects will be created. - * @returns {Any|undefined} returns the value at the end of the projection path - * or undefined if there is none. */ angular.module('schemaForm').factory('sfSelect', ['sfPath', function (sfPath) { var numRe = /^\d+$/; + /** + * @description + * Utility method to access deep properties without + * throwing errors when things are not defined. + * Can also set a value in a deep structure, creating objects when missing + * ex. + * var foo = Select('address.contact.name',obj) + * Select('address.contact.name',obj,'Leeroy') + * + * @param {string} projection A dot path to the property you want to get/set + * @param {object} obj (optional) The object to project on, defaults to 'this' + * @param {Any} valueToSet (opional) The value to set, if parts of the path of + * the projection is missing empty objects will be created. + * @returns {Any|undefined} returns the value at the end of the projection path + * or undefined if there is none. + */ return function(projection, obj, valueToSet) { if (!obj) { obj = this; diff --git a/src/services/decorators.js b/src/services/decorators.js index dca2f6e70..6c2de9294 100644 --- a/src/services/decorators.js +++ b/src/services/decorators.js @@ -1,8 +1,9 @@ -angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider','sfPathProvider',function($compileProvider, sfPathProvider){ +angular.module('schemaForm').provider('schemaFormDecorators', +['$compileProvider', 'sfPathProvider', function($compileProvider, sfPathProvider) { var defaultDecorator = ''; var directives = {}; - var templateUrl = function(name,form) { + var templateUrl = function(name, form) { //schemaDecorator is alias for whatever is set as default if (name === 'sfDecorator') { name = defaultDecorator; @@ -12,7 +13,7 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' //rules first var rules = directive.rules; - for (var i = 0; i< rules.length; i++) { + for (var i = 0; i < rules.length; i++) { var res = rules[i](form); if (res) { return res; @@ -28,10 +29,9 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' return directive.mappings['default']; }; - - var createDirective = function(name){ - $compileProvider.directive(name,['$parse','$compile','$http','$templateCache', - function($parse, $compile, $http, $templateCache){ + var createDirective = function(name) { + $compileProvider.directive(name, ['$parse', '$compile', '$http', '$templateCache', + function($parse, $compile, $http, $templateCache) { return { restrict: 'AE', @@ -39,9 +39,9 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' transclude: false, scope: true, require: '?^sfSchema', - link: function(scope,element,attrs,sfSchema) { + link: function(scope, element, attrs, sfSchema) { //rebind our part of the form to the scope. - var once = scope.$watch(attrs.form,function(form){ + var once = scope.$watch(attrs.form, function(form) { if (form) { scope.form = form; @@ -49,10 +49,14 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' //ok let's replace that template! //We do this manually since we need to bind ng-model properly and also //for fieldsets to recurse properly. - var url = templateUrl(name,form); - $http.get(url,{ cache: $templateCache }).then(function(res){ - var key = form.key ? sfPathProvider.stringify(form.key).replace(/"/g, '"') : ''; - var template = res.data.replace(/\$\$value\$\$/g,'model'+(key[0] !== '['?'.':'')+key); + var url = templateUrl(name, form); + $http.get(url, {cache: $templateCache}).then(function(res) { + var key = form.key ? + sfPathProvider.stringify(form.key).replace(/"/g, '"') : ''; + var template = res.data.replace( + /\$\$value\$\$/g, + 'model' + (key[0] !== '[' ? '.' : '') + key + ); element.html(template); $compile(element.contents())(scope); }); @@ -65,17 +69,17 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' return scope.form && scope.form.notitle !== true && scope.form.title; }; - scope.listToCheckboxValues = function(list){ + scope.listToCheckboxValues = function(list) { var values = {}; - angular.forEach(list,function(v){ + angular.forEach(list, function(v) { values[v] = true; }); return values; }; - scope.checkboxValuesToList = function(values){ + scope.checkboxValuesToList = function(values) { var lst = []; - angular.forEach(values,function(v,k){ + angular.forEach(values, function(v, k) { if (v) { lst.push(k); } @@ -83,15 +87,15 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' return lst; }; - scope.buttonClick = function($event,form) { + scope.buttonClick = function($event, form) { if (angular.isFunction(form.onClick)) { - form.onClick($event,form); + form.onClick($event, form); } else if (angular.isString(form.onClick)) { if (sfSchema) { //evaluating in scope outside of sfSchemas isolated scope - sfSchema.evalInParentScope(form.onClick,{'$event':$event,form:form}); + sfSchema.evalInParentScope(form.onClick, {'$event': $event, form: form}); } else { - scope.$eval(form.onClick,{'$event':$event,form:form}); + scope.$eval(form.onClick, {'$event': $event, form: form}); } } }; @@ -103,13 +107,13 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' * @param {Object} locals (optional) * @return {Any} the result of the expression */ - scope.evalExpr = function(expression,locals) { + scope.evalExpr = function(expression, locals) { if (sfSchema) { //evaluating in scope outside of sfSchemas isolated scope - return sfSchema.evalInParentScope(expression,locals); + return sfSchema.evalInParentScope(expression, locals); } - return scope.$eval(expression,locals); + return scope.$eval(expression, locals); }; /** @@ -119,10 +123,10 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' * @param {Object} locals (optional) * @return {Any} the result of the expression */ - scope.evalInScope = function(expression,locals) { - if (expression) { - return scope.$eval(expression,locals); - } + scope.evalInScope = function(expression, locals) { + if (expression) { + return scope.$eval(expression, locals); + } }; /** @@ -138,9 +142,12 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' return scope.form.validationMessage; } - return scope.form.validationMessage[schemaError.code] || scope.form.validationMessage['default']; + return scope.form.validationMessage[schemaError.code] || + scope.form.validationMessage['default']; } else { - return scope.form.validationMessage.required || scope.form.validationMessage['default'] || scope.form.validationMessage; + return scope.form.validationMessage.required || + scope.form.validationMessage['default'] || + scope.form.validationMessage; } } @@ -150,35 +157,36 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' } //Otherwise we only use required so it must be it. - return "Required"; + return 'Required'; }; } }; - }]); + } + ]); }; - var createManualDirective = function(type,templateUrl,transclude) { - transclude = angular.isDefined(transclude)? transclude : false; - $compileProvider.directive('sf'+angular.uppercase(type[0])+type.substr(1), function(){ + var createManualDirective = function(type, templateUrl, transclude) { + transclude = angular.isDefined(transclude) ? transclude : false; + $compileProvider.directive('sf' + angular.uppercase(type[0]) + type.substr(1), function() { return { - restrict: "EAC", + restrict: 'EAC', scope: true, replace: true, transclude: transclude, template: '', - link: function(scope,element,attrs) { + link: function(scope, element, attrs) { var watchThis = { 'items': 'c', 'titleMap': 'c', 'schema': 'c' }; - var form = { type: type }; + var form = {type: type}; var once = true; - angular.forEach(attrs,function(value,name){ + angular.forEach(attrs, function(value, name) { if (name[0] !== '$' && name.indexOf('ng') !== 0 && name !== 'sfField') { - var updateForm = function(val){ + var updateForm = function(val) { if (angular.isDefined(val) && val !== form[name]) { form[name] = val; @@ -193,17 +201,17 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' if (name === 'model') { //"model" is bound to scope under the name "model" since this is what the decorators //know and love. - scope.$watch(value,function(val){ + scope.$watch(value, function(val) { if (val && scope.model !== val) { scope.model = val; } }); } else if (watchThis[name] === 'c') { //watch collection - scope.$watchCollection(value,updateForm); + scope.$watchCollection(value, updateForm); } else { //$observe - attrs.$observe(name,updateForm); + attrs.$observe(name, updateForm); } } }); @@ -212,8 +220,6 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' }); }; - - /** * Create a decorator directive and its sibling "manual" use directives. * The directive can be used to create form fields or other form entities. @@ -225,11 +231,12 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' ** * @param {string} name directive name (CamelCased) * @param {Object} mappings, an object that maps "type" => "templateUrl" - * @param {Array} rules (optional) a list of functions, function(form){}, that are each tried in turn, + * @param {Array} rules (optional) a list of functions, function(form) {}, that are each tried in + * turn, * if they return a string then that is used as the templateUrl. Rules come before * mappings. */ - this.createDecorator = function(name,mappings,rules){ + this.createDecorator = function(name, mappings, rules) { directives[name] = { mappings: mappings || {}, rules: rules || [] @@ -261,8 +268,8 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' * @param {Object} mappings */ this.createDirectives = function(mappings) { - angular.forEach(mappings,function(url,type){ - createManualDirective(type,url); + angular.forEach(mappings, function(url, type) { + createManualDirective(type, url); }); }; @@ -283,15 +290,14 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' * @param {String} type Form type for the mapping * @param {String} url The template url */ - this.addMapping = function(name,type,url) { + this.addMapping = function(name, type, url) { if (directives[name]) { directives[name].mappings[type] = url; } }; - //Service is just a getter for directive mappings and rules - this.$get = function(){ + this.$get = function() { return { directive: function(name) { return directives[name]; @@ -300,7 +306,6 @@ angular.module('schemaForm').provider('schemaFormDecorators',['$compileProvider' }; }; - //Create a default directive createDirective('sfDecorator'); diff --git a/src/services/schema-form.js b/src/services/schema-form.js index 96d28b6d5..7e5005b3d 100644 --- a/src/services/schema-form.js +++ b/src/services/schema-form.js @@ -3,13 +3,14 @@ * This service is not that useful outside of schema form directive * but makes the code more testable. */ -angular.module('schemaForm').provider('schemaForm',['sfPathProvider', function(sfPathProvider){ +angular.module('schemaForm').provider('schemaForm', +['sfPathProvider', function(sfPathProvider) { //Creates an default titleMap list from an enum, i.e. a list of strings. var enumToTitleMap = function(enm) { var titleMap = []; //canonical titleMap format is a list. - enm.forEach(function(name){ - titleMap.push({ name: name, value: name}); + enm.forEach(function(name) { + titleMap.push({name: name, value: name}); }); return titleMap; }; @@ -19,20 +20,20 @@ angular.module('schemaForm').provider('schemaForm',['sfPathProvider', function(s var canonicalTitleMap = function(titleMap) { if (!angular.isArray(titleMap)) { var canonical = []; - angular.forEach(titleMap, function(name,value) { - canonical.push({ name: name, value: value }); + angular.forEach(titleMap, function(name, value) { + canonical.push({name: name, value: value}); }); return canonical; } return titleMap; }; - var defaultFormDefinition = function(name,schema,options){ + var defaultFormDefinition = function(name, schema, options) { var rules = defaults[schema.type]; if (rules) { var def; - for (var i=0;i