diff --git a/src/main/resources/META-INF/xnat/spawner/site-admin-elements.yaml b/src/main/resources/META-INF/xnat/spawner/site-admin-elements.yaml index d9ae46d056acf18288a3bdd9b9ce3e46c41fd8d0..a36dea2ddd51796e2f1cfcfca33354f2dd334e0e 100644 --- a/src/main/resources/META-INF/xnat/spawner/site-admin-elements.yaml +++ b/src/main/resources/META-INF/xnat/spawner/site-admin-elements.yaml @@ -38,7 +38,7 @@ siteDescriptionFile: label: " " description: "Specify a velocity template file to display on the login page" siteDescriptionText: - kind: panel.input.text + kind: panel.textarea id: siteDescriptionText name: siteDescriptionText label: " " @@ -48,7 +48,7 @@ siteUrl: id: siteUrl name: siteUrl label: Site Url - validation: required id onblur + validation: required id description: "" adminEmail: kind: panel.input.email @@ -70,29 +70,33 @@ archiveRootPath: id: archiveRootPath name: archiveRootPath label: Archive Root Path - validation: required id onblur + validation: required path description: "" + value: ??XNAT.data.siteConfig.archiveRootPath cachePath: kind: panel.input.text id: cachePath name: cachePath label: Cache Path - validation: required id onblur + validation: required path description: "" + value: ??XNAT.data.siteConfig.cachePath prearchivePath: kind: panel.input.text id: prearchivePath name: prearchivePath label: Prearchive Path - validation: required id onblur + validation: required path description: "" + value: ??XNAT.data.siteConfig.prearchivePath ftpPath: kind: panel.input.text id: ftpPath name: ftpPath label: FTP Path - validation: required id onblur + validation: required path description: "" + value: ??XNAT.data.siteConfig.cachePath buildPath: kind: panel.input.text id: buildPath @@ -100,6 +104,7 @@ buildPath: label: Build Path validation: required id onblur description: "" + value: ??XNAT.data.siteConfig.cachePath pipelinePath: kind: panel.input.text id: pipelinePath @@ -107,6 +112,7 @@ pipelinePath: label: Pipeline Path validation: required id onblur description: "" + value: ??XNAT.data.siteConfig.cachePath dataFolders: kind: panel.input.text id: dataFolders @@ -152,13 +158,10 @@ generalSecuritySettings: label: Security Channel value: https options: - http: - label: http - value: http - https: - label: https - value: https - config: + # value: label # <- simplest setup + http: http + https: https + element: id: security-channel title: Security Channel requireUserLogin: @@ -172,6 +175,11 @@ userLoginsSessionControls: kind: panel.form name: userLoginsSessionControls label: User Logins / Session Controls + method: POST + action: /xapi/siteConfig/batch + load: + lookup: XNAT.data.siteConfig + refresh: /xapi/siteConfig contents: sessionTimeout: kind: panel.input.number diff --git a/src/main/webapp/scripts/globals.js b/src/main/webapp/scripts/globals.js index 42efbd168c7b2dde85c1fc35e946481b5cffd9fa..032b900554c3159b71372375641b0b55db78261e 100644 --- a/src/main/webapp/scripts/globals.js +++ b/src/main/webapp/scripts/globals.js @@ -270,7 +270,7 @@ function extendCopyDeep(){ // return a cloned copy of a single 'obj' function cloneObject(obj){ - return extend(true, {}, obj); + return extend(true, {}, obj || {}); } // add child objects to 'obj' object from string diff --git a/src/main/webapp/scripts/xnat/ui/input.js b/src/main/webapp/scripts/xnat/ui/input.js index 3332fa57f730068d5993d9e0c99ff363aac70cd8..737e31ad144827582630db4b56e70988b03fd4a0 100644 --- a/src/main/webapp/scripts/xnat/ui/input.js +++ b/src/main/webapp/scripts/xnat/ui/input.js @@ -29,16 +29,39 @@ var XNAT = getObject(XNAT); input_ = XNAT.ui.input || {}; function lookupValue(el, lookup){ + var val = ''; try { - el.value = eval(lookup) + val = eval(lookup.trim()) } catch (e) { - el.value = ''; + val = ''; console.log(e); } - return el.value; + el.value = val; + return val; } + function lookupObjectValue(root, objStr){ + var val = ''; + if (!objStr) { + objStr = root; + root = window; + } + root = root || window; + objStr.toString().trim().split('.').forEach(function(part, i){ + part = part.trim(); + // start at the root object + if (i === 0) { + val = root[part] || {}; + } + else { + val = val[part]; + } + }); + return val; + } + + // ======================================== // MAIN FUNCTION input = function(type, config){ @@ -53,12 +76,12 @@ var XNAT = getObject(XNAT); // lookup a value if it starts with '??' var doLookup = '??'; if (config.value && config.value.toString().indexOf(doLookup) === 0) { - lookupValue(config, config.value.split(doLookup)[1]) + config.value = lookupObjectValue(config.value.split(doLookup)[1]) } // lookup a value from a namespaced object // if no value is given if (!config.value && config.data && config.data.lookup) { - lookupValue(config, config.data.lookup) + config.value = lookupObjectValue(config.data.lookup) } var spawned = spawn('input', config); return { @@ -118,7 +141,7 @@ var XNAT = getObject(XNAT); $(window).load(function(){ $(':input[data-lookup]').each(function(){ var $input = $(this); - var val = lookupValue(this, $input.dataAttr('lookup')); + var val = lookupObjectValue($input.dataAttr('lookup')); $input.changeVal(val); }); }); diff --git a/src/main/webapp/scripts/xnat/ui/panel.js b/src/main/webapp/scripts/xnat/ui/panel.js index 561bb6e5ac91bb333f43cc60c1c75151bbc37828..6753bcdaf4e9d8cc65c0bc94e5fb66d0b773a326 100644 --- a/src/main/webapp/scripts/xnat/ui/panel.js +++ b/src/main/webapp/scripts/xnat/ui/panel.js @@ -32,9 +32,48 @@ var XNAT = getObject(XNAT || {}); return obj.data; } + // another way to do this without using eval() + // is to loop over object string using dot notation: + // var myVal = lookupObjectValue(XNAT, 'data.siteConfig.siteId'); + // --> myVal == 'myXnatSiteId' + function lookupObjectValue(root, objStr){ + var val = ''; + if (!objStr) { + objStr = root; + root = window; + } + root = root || window; + objStr.toString().trim().split('.').forEach(function(part, i){ + // start at the root object + if (i === 0) { + val = root[part] || {}; + } + else { + val = val[part]; + } + }); + return val; + } + + // string that indicates to look for a namespaced object value + var doLookupString = '??'; + + function doLookup(input){ + if (!input) return ''; + if (input.toString().trim().indexOf(doLookupString) === 0){ + return lookupObjectValue(window, input.split(doLookupString)[1]); + } + return input; + } + + /** + * Initialize panel. + * @param [opts] {Object} Config object + * @returns {{}} + */ panel.init = function panelInit(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.element = opts.element || opts.config || {}; var _target = spawn('div.panel-body', opts.element), @@ -68,11 +107,10 @@ var XNAT = getObject(XNAT || {}); } }; - // creates a panel that's a form that can be submitted panel.form = function panelForm(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.element = opts.element || opts.config || {}; var _target = spawn('div.panel-body', opts.element), @@ -140,7 +178,7 @@ var XNAT = getObject(XNAT || {}); obj = opts.load || {}; } - obj = extend(true, {}, obj); + obj = cloneObject(obj); obj.form = obj.form || obj.target || obj.element || _formPanel; @@ -321,7 +359,7 @@ var XNAT = getObject(XNAT || {}); // creates a panel that submits all forms contained within panel.multiForm = function(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.element = opts.element || opts.config || {}; var inner = spawn('div.panel-body', opts.element), @@ -407,7 +445,7 @@ var XNAT = getObject(XNAT || {}); panel.element = function(opts){ var _element, _inner = [], _target; - opts = getObject(opts); + opts = cloneObject(opts); opts.element = opts.element || opts.config || {}; if (opts.id || opts.element.id) { opts.element.id = (opts.id || opts.element.id) + '-element'; @@ -443,7 +481,7 @@ var XNAT = getObject(XNAT || {}); }; panel.subhead = function(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.html = opts.html || opts.text || opts.label; return XNAT.ui.template.panelSubhead(opts).spawned; }; @@ -453,7 +491,7 @@ var XNAT = getObject(XNAT || {}); var _section, _inner = [], _body; - opts = getObject(opts); + opts = cloneObject(opts); opts.element = opts.element || opts.config || {}; opts.header = opts.header || opts.label || opts.title || ''; @@ -495,34 +533,34 @@ var XNAT = getObject(XNAT || {}); }; panel.input.number = function panelInputNumber(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.type = 'number'; return XNAT.ui.template.panelInput(opts).spawned; }; panel.input.email = function panelInputEmail(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.type = 'text'; addClassName(opts, 'email'); return XNAT.ui.template.panelInput(opts).spawned; }; panel.input.password = function panelInputPassword(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.type = 'password'; addClassName(opts, 'password'); return XNAT.ui.template.panelInput(opts).spawned; }; panel.input.checkbox = function panelInputCheckbox(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.type = 'checkbox'; //addClassName(opts, 'checkbox'); return XNAT.ui.template.panelInput(opts).spawned; }; panel.input.upload = function panelInputUpload(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.id = (opts.id||randomID('upload-', false)); opts.element = opts.element || opts.config || {}; opts.element.id = opts.id; @@ -560,13 +598,31 @@ var XNAT = getObject(XNAT || {}); } }; + panel.textarea = function(opts){ + opts = cloneObject(opts); + opts.element = opts.element || opts.config || {}; + if (opts.id) opts.element.id = opts.id; + if (opts.name) opts.element.name = opts.name; + opts.element.html = + opts.element.html || + opts.element.value || + opts.value || + opts.text || + opts.html || ''; + + opts.element.html = doLookup(opts.element.html); + + var textarea = spawn('textarea', opts.element); + return XNAT.ui.template.panelDisplay(opts, textarea).spawned; + }; + panel.select = {}; panel.select.menu = function panelSelectSingle(opts, multi){ var _menu; - opts = getObject(opts); + opts = cloneObject(opts); opts.element = opts.element || opts.config || {}; opts.element.name = opts.element.name || opts.name || ''; opts.element.id = opts.element.id || opts.id || toDashed(opts.element.name); @@ -596,7 +652,7 @@ var XNAT = getObject(XNAT || {}); }; panel.selectMenu = function panelSelectMenu(opts){ - opts = getObject(opts); + opts = cloneObject(opts); return XNAT.ui.template.panelSelect(opts).spawned; }; @@ -640,94 +696,6 @@ var XNAT = getObject(XNAT || {}); // !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! - /** - * Initialize panel and optionally render it. - * If 'opts' argument is passed, 'setup' method will run. - * If 'container' argument is passed, the 'render' method will run. - * So if both args are passed, 'setup' and 'render' do NOT need to be called. - * @param [opts] {Object} Config object - * @param [container] {Element} Container for panel - * @returns {{}} - */ - function panel(opts, container){ - - // `this` object - var _panel = {}; - - /** - * Standard panel widget - */ - function newPanel(){ - - var sections = [ - ['div.panel-heading', [ - ['h3.panel-title', _panel.opts.title] - ]], - ['div.panel-body', _panel.opts.body] - ]; - - if (_panel.opts.footer) { - sections.push(['div.panel-footer', _panel.opts.footer]) - } - - return spawn((_panel.opts.tag) + '.panel.panel-default', _panel.opts.attr, sections); - //return $(spawn('div.panel.panel-default')).append(content); - } - - /** - * Sets up elements before rendering to the page - * @param _opts Config object - * @returns {{}} - */ - _panel.setup = function _panelSetup(_opts){ - - _panel.opts = extend(true, {}, _opts); - - _panel.opts.tag = _panel.opts.tag || 'div'; - _panel.opts.title = _panel.opts.title || _panel.opts.header || ''; - _panel.opts.body = _panel.opts.body || _panel.opts.content || ''; - _panel.opts.footer = _panel.opts.footer || ''; - - _panel.opts.attr = _panel.opts.attr || {}; - - if (_panel.opts.id) { - _panel.opts.attr.id = _panel.opts.id - } - - if (_panel.opts.name) { - _panel.opts.attr.data = getObject(_panel.opts.attr.data); - _panel.opts.attr.data.name = _panel.opts.name; - } - - _panel.panel = _panel.element = newPanel(); - - return _panel; - }; - - // if 'opts' arg is passed to .panel(), call .setup() - if (opts) { - _panel.setup(opts); - } - - // render the panel and append to 'container' - _panel.render = function _panelRender(container){ - $$(container).append(_panel.panel); - return _panel; - }; - - // render immediately if 'container' is specified - if (container) { - _panel.render(container); - } - - _panel.get = function _panelGet(){ - return _panel.element; - }; - - return _panel; - - } - /** * Panel widget with default 'Submit' and 'Revert' buttons diff --git a/src/main/webapp/scripts/xnat/ui/templates.js b/src/main/webapp/scripts/xnat/ui/templates.js index ad59c311faf06359daa3ef64dfe5a923e58aade5..e61089a4ceeadf7268e1dd12355d5a4da50f3878 100644 --- a/src/main/webapp/scripts/xnat/ui/templates.js +++ b/src/main/webapp/scripts/xnat/ui/templates.js @@ -42,7 +42,7 @@ var XNAT = getObject(XNAT); function lookupValue(el, lookup){ var val = ''; try { - val = eval(lookup); + val = eval(lookup.trim()); } catch (e) { val = ''; @@ -65,10 +65,10 @@ var XNAT = getObject(XNAT); root = window; } root = root || window; - objStr.toString().split('.').forEach(function(part, i){ + objStr.toString().trim().split('.').forEach(function(part, i){ // start at the root object if (i === 0) { - val = root[part]; + val = root[part] || {}; } else { val = val[part]; @@ -82,7 +82,7 @@ var XNAT = getObject(XNAT); // subhead element to segment panels template.panelSubhead = function(opts){ var _templ, _spawn, _html; - opts = getObject(opts); + opts = cloneObject(opts); _templ = ['h4.panel-subhead', opts]; _spawn = function(){ return spawn.apply(null, _templ); @@ -103,7 +103,7 @@ var XNAT = getObject(XNAT); // generic panel element template.panelElement = function(opts, content){ var _templ, _spawn, _html; - opts = getObject(opts); + opts = cloneObject(opts); addClassName(opts, 'panel-element'); _templ = [ 'div|data-name='+(opts.name||''), @@ -131,18 +131,26 @@ var XNAT = getObject(XNAT); // ======================================== // display only element for form panels template.panelDisplay = function(opts, element){ - opts = getObject(opts); + + opts = cloneObject(opts); opts.id = opts.id||toDashed(opts.name||''); opts.label = opts.label||opts.title||opts.name||''; + + // pass in an element or create a new 'div' element + element = + element || spawn('div', { + id: opts.id, + className: opts.className||'', + title: opts.title||opts.name||opts.id, + html: opts.value||opts.html||opts.text||opts.body||'' + }); + return template.panelElement(opts, [ - ['label.element-label|for='+opts.id, opts.label], + ['label.element-label|for='+element.id||opts.id, opts.label], ['div.element-wrapper', [ - element || ['div', { - id: opts.id, - className: opts.className||'', - title: opts.title||opts.name||opts.id, - html: opts.value||opts.html||opts.text||opts.body||'' - }], + + element , + ['div.description', opts.description] ]] ]); @@ -153,7 +161,7 @@ var XNAT = getObject(XNAT); // ======================================== // input element for form panels template.panelInput = function(opts, element){ - opts = getObject(opts); + opts = cloneObject(opts); opts.name = opts.name || opts.id || randomID('input-', false); opts.id = opts.id||toDashed(opts.name||''); opts.label = opts.label||opts.title||opts.name||''; @@ -192,7 +200,7 @@ var XNAT = getObject(XNAT); // look up a namespaced object value if the value starts with '??' var doLookup = '??'; if (opts.value && opts.value.toString().indexOf(doLookup) === 0) { - lookupValue(element, opts.value.split(doLookup)[1]); + element.value = lookupObjectValue(opts.value.split(doLookup)[1]); } if (opts.load) { @@ -233,7 +241,7 @@ var XNAT = getObject(XNAT); // ======================================== // select element for form panels template.panelSelect = function(opts){ - opts = getObject(opts); + opts = cloneObject(opts); opts.name = opts.name || opts.id || randomID('select-', false); opts.id = opts.id || toDashed(opts.name||''); opts.element = extend({ @@ -265,7 +273,7 @@ var XNAT = getObject(XNAT); template.panelElementGroup = function(opts, elements){ - opts = getObject(opts); + opts = cloneObject(opts); return template.panelElement(opts, [ ['label.element-label|for='+opts.id, opts.label||opts.title||opts.name], ['div.element-wrapper', elements]