diff options
author | Petr Vobornik <pvoborni@redhat.com> | 2013-04-08 16:49:56 +0200 |
---|---|---|
committer | Petr Vobornik <pvoborni@redhat.com> | 2013-05-06 16:22:23 +0200 |
commit | fcbf5cc411de2f6195f53fe70c54458386fc588a (patch) | |
tree | eccc5171fd882af4d4b47b9fceb2bdeffddd313a /install/ui/src | |
parent | abdb5455d1954f2fab4d5def42c6b99590067eae (diff) | |
download | freeipa-fcbf5cc411de2f6195f53fe70c54458386fc588a.tar.gz freeipa-fcbf5cc411de2f6195f53fe70c54458386fc588a.tar.xz freeipa-fcbf5cc411de2f6195f53fe70c54458386fc588a.zip |
Spec modification by diff object
https://fedorahosted.org/freeipa/ticket/3235
Diffstat (limited to 'install/ui/src')
-rw-r--r-- | install/ui/src/freeipa/_base/Builder.js | 19 | ||||
-rw-r--r-- | install/ui/src/freeipa/_base/Spec_mod.js | 177 |
2 files changed, 194 insertions, 2 deletions
diff --git a/install/ui/src/freeipa/_base/Builder.js b/install/ui/src/freeipa/_base/Builder.js index 10c544cce..9fc116ba8 100644 --- a/install/ui/src/freeipa/_base/Builder.js +++ b/install/ui/src/freeipa/_base/Builder.js @@ -21,8 +21,9 @@ define(['dojo/_base/declare', 'dojo/_base/array', 'dojo/_base/lang', - './construct' - ], function(declare, array, lang, construct) { + './construct', + './Spec_mod' + ], function(declare, array, lang, construct, Spec_mod) { var Builder = declare(null, { /** @@ -39,6 +40,11 @@ define(['dojo/_base/declare', registry: null, /** + * Specification modifier + */ + spec_mod: null, + + /** * Build object based on spec. * * @param {String|Function|Object} Build spec @@ -152,6 +158,9 @@ define(['dojo/_base/declare', if (preop_t === 'function') { cs.spec = preop(cs.spec || {}); } else if (preop_t === 'object') { + var temp = lang.clone(preop); + this.spec_mod.mod(cs.spec, temp); + this.spec_mod.del_rules(temp); lang.mixin(cs.spec, preop); } } @@ -159,6 +168,10 @@ define(['dojo/_base/declare', cs.spec = cs.spec || {}; + // do we want following?, remove? + this.spec_mod.mod(cs.spec, cs.spec); + this.spec_mod.del_rules(cs.spec); + if (cs.factory && typeof cs.factory === 'function') { obj = cs.factory(cs.spec); } else if (cs.constructor && typeof cs.constructor === 'function') { @@ -194,6 +207,8 @@ define(['dojo/_base/declare', spec = spec || {}; if (spec.registry) this.registry = spec.registry; + if (spec.spec_mod) this.spec_mod = spec.spec_mod; + else this.spec_mod = new Spec_mod(); } }); diff --git a/install/ui/src/freeipa/_base/Spec_mod.js b/install/ui/src/freeipa/_base/Spec_mod.js new file mode 100644 index 000000000..60d4b9229 --- /dev/null +++ b/install/ui/src/freeipa/_base/Spec_mod.js @@ -0,0 +1,177 @@ +/* Authors: + * Petr Vobornik <pvoborni@redhat.com> + * + * Copyright (C) 2013 Red Hat + * see file 'COPYING' for use and warranty information + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. +*/ + +define(['dojo/_base/declare', + 'dojo/_base/lang' + ], function(declare, lang) { + + var Spec_mod = declare(null, { + + /** + * Modifies spec according to rules defined in diff object. + * + * Diff should have following structure: { + * $add: array of add rules + * $del: array of del rules + * $set: array of set rules + * } + * + * The order of modification is del, add, set. + * + * @param {Object} spec + * @param {Object} diff + */ + mod: function(spec, diff) { + + if (!diff) return spec; + + this.del(spec, diff.$del); + this.add(spec, diff.$add); + this.set(spec, diff.$set); + + return spec; + }, + + + /** + * Adds objects according to rules to array. + * + * A rule is a triple of path and a object and position to add: + * ['path.to.spec.array', {}, position] + * + */ + add: function(spec, rules) { + + return this._apply_rules(spec, rules, this._add); + }, + + /** + * Deletes objects according to rules from an array. + * + * A rule is a pair of path and delete conditions: + * ['path.to.spec.array', [ { name: 'foo'}, { name: 'baz'} ]] + * + * Deletes all objects with name 'baz' or 'foo'. + */ + del: function(spec, rules) { + + return this._apply_rules(spec, rules, this._del); + }, + + /** + * A rule is a pair of path and a object to set. + * ['path.to.spec.property', {}] + */ + set: function(spec, rules) { + + return this._apply_rules(spec, rules, this._set); + }, + + /** + * Removes all rule props + */ + del_rules: function(diff) { + delete diff.$add; + delete diff.$del; + delete diff.$set; + }, + + _apply_rules: function(spec, rules, method) { + if (!lang.isArrayLike(rules)) return spec; + + for (var i=0; i<rules.length; i++) { + method.call(this, spec, rules[i]); + } + + return spec; + }, + + _add: function(spec, rule) { + + var path = rule[0]; + var value = rule[1]; + var pos = rule[2]; + var arr = lang.getObject(path, false, spec); + + if (!arr) { + arr = []; + lang.setObject(path, arr, spec); + } + + if (typeof pos !== 'number') pos = arr.length; + else if (pos < 0) pos = 0; + else if (pos > arr.length) pos = arr.length; + + arr.splice(pos, 0, value); + return spec; + }, + + _del: function(spec, rule) { + + var path = rule[0]; + var conds = rule[1]; + var arr = lang.getObject(path, false, spec); + + if (!arr) return spec; + + var del = []; + var i,j; + + for (i=0; i<arr.length; i++) { + for (j=0; j<conds.length; j++) { + if (this._match(arr[i], conds[j])) { + del.push(i); + break; + } + } + } + + del.sort(function(a,b) {return b-a;}); + for (i=0; i<del.length;i++) { + arr.splice(del[i], 1); + } + return spec; + }, + + _match: function(value, cond) { + var match = true; + + if (typeof cond !== 'object') { + match = cond === value; + } else { + for (var prop in cond) { + if (cond.hasOwnProperty(prop) && cond[prop] !== value[prop]) { + match = false; + break; + } + } + } + + return match; + }, + + _set: function(spec, rule) { + lang.setObject(rule[0], rule[1], spec); + return spec; + } + }); + + return Spec_mod; +});
\ No newline at end of file |