123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428 |
- /*
- This file is part of RelaxNGui.
- RelaxNGui 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.
- RelaxNGui 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
- RelaxNGui (file COPYING in the main directory). If not, see
- <http://www.gnu.org/licenses/>.
- */
- var RelaxNGui = function(rng,target,ceval,ignore=false) {
- if (!(rng instanceof XMLDocument)) { // rng has to be XMLDocument //{{{
- rng = $XR($(rng).serializeXML());
- } //}}}
- var lenextract = function(tag, lencount) { //{{{
- $.each(tag.attributes,function(k,v){
- if ((v.localName == 'label') && (v.namespaceURI == 'http://rngui.org')) { lencount = v.nodeValue.length > lencount ? v.nodeValue.length : lencount; }
- });
- return lencount;
- }; //}}}
- var labextract = function(type,tag) { //{{{
- var ret = { 'type': type, 'wrap': false, 'readonly': false, 'label': '', default: '' };
- $.each(tag.attributes,function(k,v){
- if ((v.localName == 'label') && (v.namespaceURI == 'http://rngui.org')) { ret['label'] = v.nodeValue; }
- if ((v.localName == 'readonly') && (v.namespaceURI == 'http://rngui.org')) { ret['readonly'] = v.nodeValue == 'true' ? true : false; }
- if ((v.localName == 'wrap') && (v.namespaceURI == 'http://rngui.org')) { ret['wrap'] = v.nodeValue == 'true' ? true : false; }
- });
- $.each(tag.children,function(k,v){
- if ((v.localName == 'param') && (v.namespaceURI == 'http://relaxng.org/ns/structure/1.0')) {
- $.each(v.attributes,function(l,w){
- if ((w.localName == 'name') && (w.nodeValue == 'minInclusive')) {
- ret['min'] = v.lastChild.nodeValue;
- }
- if ((w.localName == 'name') && (w.nodeValue == 'maxInclusive')) {
- ret['max'] = v.lastChild.nodeValue;
- }
- });
- }
- });
- return ret;
- }; //}}}
- var addelements = function(target) { //{{{
- var template = target.parent().find('> .relaxngui_template').clone(true,true);
- template.removeClass('relaxngui_template');
- template.find('[data-relaxngui-template]').each(function(j,t){
- $(t).attr('data-relaxngui-template',false);
- });
- template.find('.relaxngui_template [data-relaxngui-template]').each(function(j,t){
- $(t).attr('data-relaxngui-template',true);
- });
- template.addClass('relaxngui_dyn');
- template.on('click', '> *:first-child', function(ev){ delelements($(ev.target)); });
- if (target.parent().find('> .relaxngui_dyn').length % 2 == 1) {
- var evens = template.find('.even');
- var odds = template.find('.odd');
- evens.each(function(k,v){
- $(v).removeClass('even');
- $(v).addClass('odd');
- });
- odds.each(function(k,v){
- $(v).removeClass('odd');
- $(v).addClass('even');
- });
- }
- target.parent().find('> .relaxngui_control').before(template);
- }; //}}}
- var delelements = function(target) { //{{{
- var it = target.parent();
- var par = target.parent().parent();
- it.remove();
- par.trigger('relaxngui_remove');
- }; //}}}
- var recshow_header = function(tag,ret){ //{{{
- var header;
- $.each(tag.attributes,function(k,v){
- if ((v.localName == 'header') && (v.namespaceURI == 'http://rngui.org')) { header = v.nodeValue; }
- });
- if (header) {
- ret.append($("<div class='relaxngui_header'>" + header + "</div>"));
- }
- } //}}}
- var recshow_single = function(tag,ret,template,path,lencount,optional){ //{{{
- var node = $('<div class="relaxngui_row"/>');
- var name;
- var label;
- var second = {};
- var datalist = [];
- var defaul = '';
- var hint = '';
- var retcount = 0;
- $.each(tag.attributes,function(k,v){
- if ((v.localName == 'label') && (v.namespaceURI == 'http://rngui.org')) { label = v.nodeValue; }
- if ((v.localName == 'date') && (v.namespaceURI == 'http://rngui.org')) { label = v.nodeValue; }
- if ((v.localName == 'default') && (v.namespaceURI == 'http://rngui.org')) { defaul = v.nodeValue; }
- if ((v.localName == 'hint') && (v.namespaceURI == 'http://rngui.org')) { hint = v.nodeValue; }
- if (v.localName == 'name') { name = v.nodeValue; }
- });
- $.each($(tag).children('data[type=string]'), function(k,v) { second = labextract('string',v); });
- $.each($(tag).children('data[type=integer]'), function(k,v) { second = labextract('integer',v); });
- $.each($(tag).children('data[type=nonNegativeInteger]'), function(k,v) { second = labextract('nonNegativeInteger',v); });
- $.each($(tag).children('data[type=positiveInteger]'), function(k,v) { second = labextract('positiveInteger',v); });
- $.each($(tag).children('data[type=float]'), function(k,v) { second = labextract('float',v); });
- $.each($(tag).children('data[type=date]'), function(k,v) { second = labextract('date',v); });
- $.each($(tag).children('text'), function(k,v) { second = labextract('text',v); });
- $.each($(tag).find('choice > value'), function(k,v) {
- second = labextract('datalist',$(v).parent()[0]);
- datalist.push(v.textContent);
- });
- if (name && label) {
- node.append($("<label class='relaxngui_cell" + (optional && defaul == '' ? " optional": "") + "' style='min-width: " + (lencount+1) + "ex' for=''>" + label + "</label><span class='relaxngui_cell'>⇒</span>"));
- } else if (name) {
- // a tag without information is ignored
- } else if (label) {
- node.append($("<input data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + path + " > *[data-name]' class='relaxngui_cell' type='text' pattern='^[a-z][a-zA-Z0-9_]*$' id='' placeholder='" + label + "'></input><span class='relaxngui_cell'>⇒</span>"));
- }
- var tpath = ((typeof name === 'undefined') ? path + ' > *' : (tag.localName == 'element' ? path + ' > ' + $(tag).attr('name') : path + '[' + $(tag).attr('name') + ']'));
- if (label) {
- if (defaul && typeof defaul == 'string' && defaul.match(/^javascript:/)) {
- defaul = defaul.replace(/^javascript:/,'');
- defaul = ceval ? ceval(defaul) : eval(defaul);
- }
- var os = (optional ? " onkeyup='var sl = $(this).siblings(\"label\"); if ($(this).get_val() == \"\") { if (!sl.hasClass(\"optional\")) { sl.addClass(\"optional\") } } else { sl.removeClass(\"optional\") }' data-optional='true'" : " data-optional='false'");
- if (second.readonly)
- node.append($("<input " + (defaul && typeof defaul == 'string' ? 'value="' + defaul + '"' : '') + " data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + tpath + "' class='relaxngui_cell' type='text' id='' readonly='readonly'" + os + "></input>"));
- else {
- if (second.type == 'string') {
- node.append($("<input " + (defaul && typeof defaul == 'string' ? 'value="' + defaul + '"' : '') + " data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + tpath + "' class='relaxngui_cell' type='text' id='' placeholder='" + second.label + "'" + os + "></input>"));
- } else if (second.type == 'integer') {
- node.append($("<input " + (defaul && typeof defaul == 'string' ? 'value="' + defaul + '"' : '') + " data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + tpath + "' class='relaxngui_cell' type='number' id='' placeholder='" + second.label + "'" + (second.min != undefined ? (" min='" + second.min + "'") : '') + (second.max != undefined ? (" max='" + second.max + "'") : '') + os + "></input>"));
- } else if (second.type == 'positiveInteger') {
- if (second.min == undefined) second.min = 1;
- node.append($("<input " + (defaul && typeof defaul == 'string' ? 'value="' + defaul + '"' : '') + " data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + tpath + "' class='relaxngui_cell' type='number' id='' placeholder='" + second.label + "'" + (second.min != undefined ? (" min='" + second.min + "'") : '') + (second.max != undefined ? (" max='" + second.max + "'") : '') + os + "></input>"));
- } else if (second.type == 'nonNegativeInteger') {
- if (second.min == undefined) second.min = 0;
- node.append($("<input " + (defaul && typeof defaul == 'string' ? 'value="' + defaul + '"' : '') + " data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + tpath + "' class='relaxngui_cell' type='number' id='' placeholder='" + second.label + "'" + (second.min != undefined ? (" min='" + second.min + "'") : '') + (second.max != undefined ? (" max='" + second.max + "'") : '') + os + "></input>"));
- } else if (second.type == 'date') {
- node.append($("<input " + (defaul && typeof defaul == 'string' ? 'value="' + defaul + '"' : '') + " data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + tpath + "' class='relaxngui_cell' type='date' id='' placeholder='" + second.label + "'" + os + "></input>"));
- } else if (second.type == 'float') {
- node.append($("<input " + (defaul && typeof defaul == 'string' ? 'value="' + defaul + '"' : '') + " data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + tpath + "' class='relaxngui_cell' type='number' step='any' id='' placeholder='" + second.label + "'" + (second.min != undefined ? (" min='" + second.min + "'") : '') + (second.max != undefined ? (" max='" + second.max + "'") : '') + os + "></input>"));
- } else if (second.type == 'text') {
- node.append($("<div contenteditable='true' data-relaxngui-wrap='" + second.wrap + "' " + (defaul && typeof defaul == 'string' ? 'value="' + defaul + '"' : '') + " data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + tpath + "' class='relaxngui_cell' id='' placeholder='" + second.label + "'" + os + "></div>"));
- } else if (second.type == 'datalist') {
- var tnode = $("<select " + (defaul && typeof defaul == 'string' ? 'value="' + defaul + '"' : '') + " data-relaxngui-template='" + template + "' data-relaxngui-parent='" + path + "' data-relaxngui-path='" + tpath + "' class='relaxngui_cell' id='' size='1'" + os + "></select>");
- $.each(datalist,function(didx,dname){
- if (dname == defaul)
- tnode.append('<option value="' + dname + '" selected="selected">' + dname + '</value>');
- else
- tnode.append('<option value="' + dname + '">' + dname + '</value>');
- });
- node.append(tnode);
- }
- }
- ret.append(node);
- retcount += 1;
- } else {
- if (tag.localName != 'element') { // its an attribute, simulate its empty-ness
- node.attr('data-relaxngui-template',template);
- node.attr('data-relaxngui-parent',path);
- node.attr('data-relaxngui-path',tpath);
- ret.append(node);
- retcount += 1;
- }
- }
- if (hint) {
- var n = $('<div class="relaxngui_hint"/>');
- var s1 = $('<em>Hint: </em>');
- var s2 = $('<span/>');
- s2.text(hint);
- n.append(s1);
- n.append(s2);
- ret.append(n);
- retcount += 1;
- }
- return retcount;
- } //}}}
- var recshow = function(elements,template,path,attr) { //{{{
- // delete all elements with relaxngui:ignore
- if (attr.ignore) {
- var tele = $.grep(elements,function(tagv){
- var include = true;
- $.each(tagv.attributes,function(k,v){
- if ((v.localName == 'ignore') && (v.namespaceURI == 'http://rngui.org')) {
- include = false;
- }
- });
- return include;
- });
- elements = $(tele);
- }
- var ret = $('<div/>');
- var lencount = 0;
- $.each(elements,function(k,v){
- var tag = $(v)[0];
- if ((tag.localName == 'element') && (tag.namespaceURI == 'http://relaxng.org/ns/structure/1.0')) {
- $(tag).children('attribute').each(function(l,w){
- lencount = lenextract($(w)[0],lencount);
- });
- lencount = lenextract(tag,lencount);
- }
- });
- $.each(elements,function(k,v){
- if (attr.mode == 'even') { attr.mode = 'odd' }
- else { attr.mode = 'even'; }
- var tag = $(v)[0];
- if ((tag.localName == 'element') && (tag.namespaceURI == 'http://relaxng.org/ns/structure/1.0')) {
- var xxx;
- if (template) {
- var yyy = $('<div class="relaxngui_table ' + attr.mode + '" data-relaxngui-template="true" data-relaxngui-parent="' + path + '" data-relaxngui-path="' + (path == '' ? ' > ' + (typeof elements.attr('name') === 'undefined' ? '*' : elements.attr('name')) : path + ' > ' + (typeof $(tag).attr('name') === 'undefined' ? '*' : $(tag).attr('name'))) + '[data-main]">');
- xxx = $('<div class="relaxngui_template"><span>✖</span></div>');
- xxx.append(yyy);
- ret.append(xxx);
- xxx = yyy;
- } else {
- xxx = $('<div class="relaxngui_table ' + attr.mode + '" data-relaxngui-template="false" data-relaxngui-parent="' + path + '" data-relaxngui-path="' + (path == '' ? ' > ' + elements.attr('name') : path + ' > ' + $(tag).attr('name')) + '[data-main]">');
- ret.append(xxx);
- }
- recshow_header(tag,xxx);
- var rcount = 0;
- $(tag).children('attribute').each(function(l,w){
- var ttag = $(w)[0];
- rcount += recshow_single(ttag,xxx,template,path + ' > ' + $(tag).attr('name'),lencount,attr.optional ? true : false);
- });
- rcount += recshow_single(tag,xxx,template,path,lencount,attr.optional ? true : false);
- var sub;
- if (sub = recshow($(tag).children('element, zeroOrMore, optional'),false,path + ' > ' + $(tag).attr('name'),{ ignore: attr.ignore, mode: (attr.mode == 'even' && rcount % 2 == 0 ? 'odd' : 'even' ) })) {
- var inode = xxx.append(sub);
- if (template) {
- inode.find('[data-relaxngui-template]').each(function(j,t){
- $(t).attr('data-relaxngui-template',true);
- });
- }
- }
- } else if ((tag.localName == 'zeroOrMore') && (tag.namespaceURI == 'http://relaxng.org/ns/structure/1.0')) {
- var label;
- $.each(tag.attributes,function(k,v){
- if ((v.localName == 'label') && (v.namespaceURI == 'http://rngui.org')) { label = v.nodeValue; }
- });
- var but = $('<button class="relaxngui_control">' + label + '</button>');
- but.on('click',function(ev){ addelements($(ev.target)); });
- ret.append(recshow($(tag).children(),true,path,{ ignore: attr.ignore, mode: attr.mode }));
- ret.append(but);
- } else if ((tag.localName == 'optional') && (tag.namespaceURI == 'http://relaxng.org/ns/structure/1.0')) {
- ret.append(recshow($(tag).children('element, zeroOrMore'),false,path,{ ignore: attr.ignore, mode: (attr.mode == 'even' ? 'odd' : 'even'), optional: true }));
- }
- });
- return ret.children().length > 0 ? ret.children() : undefined;
- }; //}}}
- var serializeXML = function (xml) { //{{{
- var out = '';
- if (typeof XMLSerializer == 'function') {
- var xs = new XMLSerializer();
- $(xml).each(function() {
- out += xs.serializeToString(this);
- });
- } else if (xml && xml.xml != 'undefined') {
- $(xml).each(function() {
- out += this.xml;
- });
- }
- return out;
- }; //}}}
- this.save = function() { //{{{
- var xml;
- var curr;
- var tar = target.find('[data-relaxngui-path]:not([data-relaxngui-template=true])');
- for (var i = 0; i<tar.length;) {
- var path = $(tar[i]).attr('data-relaxngui-path');
- var parent_path = $(tar[i]).attr('data-relaxngui-parent');
- if (i == 0) {
- var par = path.replace(/\[data-main\]/,'').replace(/ > /,'');
- xml = $XR('<' + par + '/>');
- } else {
- var ma = path.match(/([^\s]+)$/)[1];
- var att;
- if (ma.match(/\*\[data-main\]/)) {
- // do nothing. seriously. explicitly.
- } else if (ma.match(/\[data-main\]/)) {
- var par = ma.replace(/\[data-main\]/,'');
- var curr = $($XR('<' + par + '/>').documentElement);
- $(parent_path,xml).last().append(curr);
- } else if (ma.match(/\[data-name\]/)) {
- if ($(tar[i]).get_val()) {
- var nn = $($XR('<' + $(tar[i]).get_val() + '/>').documentElement).text($(tar[i+1]).get_val());
- $(parent_path,xml).append(nn);
- }
- i+=1;
- } else if (att = ma.match(/\[([^\]]+)\]$/)) {
- $(parent_path + ':last-child',xml).last().attr(att[1],$(tar[i]).get_val());
- } else {
- if ($(tar[i]).attr('data-optional') == 'true' && $(tar[i]).get_val() == '') {
- $(path + ':last-child',xml).last().remove();
- } else {
- $(path + ':last-child',xml).last().text($(tar[i]).get_val())
- }
- }
- }
- i+=1;
- }
- return xml;
- }; //}}}
- this.save_text = function() { //{{{
- return serializeXML(self.save());
- } //}}}
- this.content = function(data) { //{{{
- if (!(data instanceof XMLDocument)) { // data has to be XMLDocument //{{{
- data = $XR($(data).serializeXML());
- } //}}}
- if (data) {
- var x = $(data).serializePrettyXML();
- x = x.replace(/\s+xmlns(:[a-zA-Z0-9]+)?=\"[^\"]+\"/g, "");
- x = x.replace(/<\?[^>]+>/g, "");
- x = x.trim();
- y = $(self.save()).serializePrettyXML();
- if (x != y) {
- target.find('.relaxngui_dyn').remove();
- target.find('[data-relaxngui-path]').each(function(k,pa){
- var path = $(pa).attr('data-relaxngui-path');
- if (!path.match(/data-\w+\]$/)) {
- if ($(pa).attr('data-relaxngui-template') == 'true' && path.match(/\*$/)) {
- $(data).find(path).each(function(index,ele){
- $(target.find('[data-relaxngui-path="' + path + '[data-name]"][data-relaxngui-template="false"]').get(index)).set_val(ele.localName);
- $(target.find('[data-relaxngui-path="' + path + '"][data-relaxngui-template="false"]').get(index)).set_val($(ele).text());
- });
- } else if ($(pa).attr('data-relaxngui-template') == 'true' && !path.match(/\*$/)) {
- $(data).find(path).each(function(index,ele){
- var att;
- var val;
- if (att = path.match(/(.*)\[([^\]]+)\]$/)) {
- val = $(ele).attr(att[2]);
- } else {
- val = $(ele).text();
- }
- if (val && val != '') {
- $(target.find('[data-relaxngui-path="' + path + '"][data-relaxngui-template="false"]').get(index)).set_val(val);
- }
- });
- } else {
- var att;
- var val;
- if (att = path.match(/(.*)\[([^\]]+)\]$/)) {
- val = $(data).find(att[1]).attr(att[2]);
- } else {
- val = $(data).find(path).text();
- }
- if (val && val != '') {
- var t = target.find('[data-relaxngui-path="' + path + '"]');
- t.set_val(val);
- if (t.attr('data-optional') == 'true') {
- t.siblings('label').removeClass('optional');
- }
- }
- }
- } else {
- if ($(pa).attr('data-relaxngui-template') == 'true') {
- var but = target.find('.relaxngui_table[data-relaxngui-template="false"] > .relaxngui_template > [data-relaxngui-path="' + path + '"][data-relaxngui-template="true"]').parent().parent().find('> button');
- if (but.length > 0) {
- var dpath = path.replace(/\[data-main\]$/,'');
- var par = undefined;
- var ind = -1;
- $(data).find(dpath).each(function(ke,ele){
- if (par != $(ele).parent()[0]) {
- ind += 1;
- par = $(ele).parent()[0];
- }
- if ($(but.get(ind)).attr('disabled')) {
- $(but.get(ind)).removeAttr('disabled');
- but.get(ind).click();
- $(but.get(ind)).attr('disabled','disabled');
- } else {
- but.get(ind).click();
- }
- });
- }
- }
- }
- });
- self.set_checkpoint();
- }
- }
- }; //}}}
- // stuff to determine if user changed something
- var orig = '';
- this.set_checkpoint = function() { //{{{
- orig = self.save_text();
- } //}}}
- this.has_changed = function() { //{{{
- if (orig != self.save_text()) {
- return true;
- } else {
- return false;
- }
- } //}}}
- target.append(recshow($(rng.documentElement),false,'',{ ignore: ignore, mode: 'even'}));
- var self = this;
- };
|