From e63bd99c0ac50e51eb0e1828fd62b74c095b89e1 Mon Sep 17 00:00:00 2001 From: Carlos A Date: Mon, 16 Oct 2023 12:12:08 +0200 Subject: [PATCH] remove messages in console --- dist/powerbuttons.compress.js | 2 +- dist/powerbuttons.full.js | 66 +++++++++------- dist/powerbuttons.js | 64 +++++++++------ dist/powerbuttons.min.js | 2 +- dist/powerbuttons.raw.js | 125 ++++++++++++++++-------------- js/plugins/40.actionformset.js | 3 - js/plugins/80.actionformbutton.js | 1 - js/powerbuttons.js | 2 - 8 files changed, 149 insertions(+), 116 deletions(-) diff --git a/dist/powerbuttons.compress.js b/dist/powerbuttons.compress.js index 336744d..67a493d 100644 --- a/dist/powerbuttons.compress.js +++ b/dist/powerbuttons.compress.js @@ -1,2 +1,2 @@ /* Copyright 2021 Carlos A. (https://github.com/dealfonso); License: http://www.apache.org/licenses/LICENSE-2.0 */ -!function(window,document){function pascalToSnake(t){return t.replace(/[A-Z]/g,t=>"_"+t.toLowerCase()).replace(/^_*/,"")}function pascalToKebab(t){return t.replace(/[A-Z]/g,t=>"-"+t.toLowerCase()).replace(/^-*/,"")}function snakeCaseToCamel(t){t.replace(/-([a-z])/g,t=>t[1].toUpperCase())}function pascalToCamel(t){return t.charAt(0).toLowerCase()+t.slice(1)}function CamelToCamel(t){return t.charAt(0).toUpperCase()+t.slice(1)}function isElement(t){t instanceof Element||0 instanceof HTMLDocument}function parseBoolean(t){return"boolean"==typeof t?t:"string"==typeof t?"true"===(t=t.toLowerCase())||"yes"===t||"1"===t:!!t}function createTag(t,e={},n=null){var o=t.split("#");let i=null;var s,o=(t=1==o.length?o[0]:(o[1]=o[1].split("."),i=o[1][0],[o[0],...o[1].slice(1)].join("."))).split("."),l=(""===(t=o[0])&&(t="div"),"string"==typeof e&&(n=e,e={}),null!==n&&(e.innerHTML=n),null!==i&&(e.id=i),e.className=[e.className,...o.slice(1)].filter(function(t){return""!==(""+t).trim()}).join(" "),document.createElement(t));for(s in e)void 0!==l[s]?l[s]=e[s]:l.setAttribute(s,e[s]);return l}function appendToElement(t,...e){e=e.filter(t=>null!=t);return t.append(...e),t}function searchForm(t){let e=null;return null!==t&&void 0===(e=document.forms[t])&&null===(e=document.querySelector(t))&&console.warn(`form ${t} not found`),null!==e&&"form"!==e.tagName.toLowerCase()&&console.warn(`form ${t} is not a form`),e}function getValueWithJavascriptSupport(value,context=null){if("function"==typeof value)return value.bind(context);if("string"==typeof value){let internalValue=value.trim();if(internalValue.startsWith("javascript:"))try{let f=internalValue.substring(11);value=function(){return eval(f)}.bind(context)}catch(e){console.error(`Error executing javascript code ${internalValue.substring(11)}, error: `+e),value=null}}return value}function promiseForEvent(e,n){let o=null;var t=new Promise(t=>{o=t});return e.addEventListener(n,function t(){e.removeEventListener(n,t),o()}),t}function isEmpty(t){return null==t||(t instanceof Array?0===t.length:t instanceof Object?0===Object.keys(t).length:"string"==typeof t&&""===t.trim())}window.powerButtons=function(t,e=[],n={}){let o=e,i=("string"==typeof o?o=document.querySelectorAll(o):void 0===o.length&&(o=[o]),null);for(var s in PowerButtons.actionsRegistered)if(t.toLocaleLowerCase()===s.toLocaleLowerCase()){i=PowerButtons.actionsRegistered[s];break}if(void 0===i)console.error(`The action ${t} is not registered`);else for(var l of o)i.initialize(l,n);return e},window.powerButtons.version="2.0.0",void 0!==window.$&&(window.$.fn.powerButtons=function(t,e={}){return window.powerButtons(t,this,e),this},window.$.fn.powerButtons.version=window.powerButtons.version);class DialogLegacy{DEFAULTS={message:"Main message",buttonCount:2};constructor(t={},e=null,n=null){this.options={...this.DEFAULTS,...t},this.buttonCount=this.options.buttons.length,this.options.buttons=["Accept","Cancel"],this.result=null,this.onButton=e,this.onHidden=n}dispose(){}show(t=null,e=null){switch(null!==t&&(this.onButton=t),null!==e&&(this.onHidden=e),this.buttonCount){case 0:case 1:alert(this.message),this.result=0;break;case 2:this.result=confirm(this.options.message)?0:1;break;default:throw"Unsupported button count "+this.buttonCount}null!==this.onButton&&this.onButton(this.result,{button:this.result,text:this.options.buttons[this.result]},null),null!==this.onHidden&&this.onHidden(this.result,{button:this.result,text:this.options.buttons[this.result]},null)}}class Dialog{static create(t={},e=null,n=null){if(void 0===window.bootstrap||void 0===window.bootstrap.Modal)return new DialogLegacy(t,e,n);if(void 0!==t.selector&&null!==t.selector||void 0!==t.dialogFunction&&null!==t.dialogFunction)throw new Error("not implemented, yet");return new Dialog(t,e,n)}DEFAULTS={title:"Title",message:"Main message",customContent:null,buttons:["Accept"],buttonClasses:["btn-primary","btn-secondary"],buttonPanelClasses:["text-end"],escapeKeyCancels:!0,header:!0,footer:!0,body:!0,close:!0};dialog=null;options=null;modal=null;onButton=null;result=null;onHidden=null;onButton=null;constructor(t={},e=null,n=null){if(void 0===window.bootstrap||void 0===window.bootstrap.Modal)throw new Error("Bootstrap is required to use this class");this.options={...this.DEFAULTS,...t};var o=[];for(let e=0;ethis._handleButton(-1,null,e)),parseBoolean(i.header)&&(t=createTag(".modal-header"),null!==i.title&&t.append(appendToElement(createTag(".modal-title"),appendToElement(createTag("h5",i.title)))),parseBoolean(i.close))&&t.append(e);var s=[];if(null!==i.buttons)for(let o=0;ot.trim()).filter(t=>""!==t).join("."),n=(!1===i.footer&&(e+=".mx-1"),createTag("button.btn"+(e=""!==e?"."+e:e)+".button"+o,{type:"button"},t.text));n.addEventListener("click",function(){this._handleButton(o,t,n),void 0!==t.handler&&null!==t.handler&&t.handler(o,t,n)}.bind(this)),s.push(n)}let n=null,o=(parseBoolean(i.footer)&&(n=appendToElement(createTag(".modal-footer"),...s)),null);if(parseBoolean(i.body)&&(o=createTag(".modal-body"),null===t&&parseBoolean(i.close)&&o.append(appendToElement(createTag(".text-end"),e)),null!==i.message&&o.append(createTag("p.message.text-center",i.message)),null!==i.customContent&&o.append(createTag(".custom-content.mx-auto",i.customContent)),null===n)){let t=i.buttonPanelClasses.map(t=>t.trim()).filter(t=>""!==t).join(".");""!==t&&(t="."+t),appendToElement(o,appendToElement(createTag(".buttons"+t),...s))}return appendToElement(createTag(".modal.fade",{tabindex:"-1",role:"dialog","aria-hidden":"true","data-keyboard":"false"}),appendToElement(createTag(".modal-dialog.modal-dialog-centered",{role:"document"}),appendToElement(createTag(".modal-content"),t,o,n)))}}function confirmDialog(t,e="this action needs confirmation",n=null,o=null,i=!0){e=new Dialog({title:e,message:t,buttons:[{text:"Cancel",class:"btn-secondary",handler:o},{text:"Confirm",class:"btn-primary",handler:n}],escapeKeyCancels:i});return e.show(),e}function alertDialog(t,e="Alert",n=null){e=new Dialog({title:e,message:t,buttons:[{text:"Accept",class:"btn-primary",handler:n}],escapeKeyCancels:!0});return e.show(),e}function loadingDialog(t,e=null,n=null){"function"==typeof e&&(n=e,e=null);t=new Dialog({title:null,message:t,buttons:[{text:"Cancel",class:"btn-primary"}],customContent:e,escapeKeyCancels:!1,close:!1,header:!1,footer:!1},n);return t.show(),t}void 0===window.powerButtons.utils&&(window.powerButtons.utils={}),Object.assign(window.powerButtons.utils,{confirmDialog:confirmDialog,alertDialog:alertDialog,loadingDialog:loadingDialog});class PowerButtons{static actionsRegistered={};static registerAction(t){console.debug("Registering action "+t.NAME.toLowerCase()),this.actionsRegistered[t.NAME.toLowerCase()]=t,void 0===window.powerButtons&&(window.powerButtons={}),void 0===window.powerButtons.defaults&&(window.powerButtons.defaults={}),window.powerButtons.defaults[t.NAME.toLowerCase()]=Object.assign({},t.DEFAULTS)}static getActionSettings(t,e){if(void 0===this.actionsRegistered[t.NAME.toLowerCase()])return console.error(`The action ${t.NAME} is not registered`),{};let n={};return void 0!==window.powerButtons&&void 0!==window.powerButtons.defaults&&void 0!==window.powerButtons.defaults[t.NAME.toLowerCase()]&&(n=window.powerButtons.defaults[t.NAME.toLowerCase()]),Object.assign({},t.DEFAULTS,n,e)}static addAction(t,e={}){PowerButtons.addActionSupport(t).appendAction(e)}static addActionSupport(t){return void 0===t._powerButtons?t._powerButtons=new PowerButtons(t):t._powerButtons.reset(),t._powerButtons}el=null;current_action=0;actions=[];back_onclick=null;constructor(t){(t._powerButtons=this).el=t,this.current_action=0,this.actions=[],this.back_onclick=null,void 0!==t.onclick&&null!==t.onclick&&(this.back_onclick=t.onclick,t.onclick=null),t.addEventListener("click",this.handlerClick.bind(this))}appendAction(t={}){if(void 0===t.type)throw"The type of the action is mandatory";this.actions.push(t)}handlerClick(t){if(this.current_action>=this.actions.length)this.current_action=0,"function"!=typeof this.back_onclick||this.back_onclick()||t.preventDefault();else{var e=this.actions[this.current_action],n=(t.preventDefault(),t.stopImmediatePropagation(),function(){this.current_action++,this.current_action>=this.actions.length&&void 0!==this.el.click?this.el.click():this.el.dispatchEvent(new Event(t.type,t))}.bind(this)),o=this.constructor.actionsRegistered[e.type];if(void 0===o)throw`The action ${e.type} is not registered`;o.execute(e,n,()=>this.reset())}}reset(){this.current_action=0}static initializeAll(){Object.entries(this.actionsRegistered).forEach(([,t])=>{t.initializeAll()})}}function init(){PowerButtons.initializeAll()}void 0!==document.addEventListener&&document.addEventListener("DOMContentLoaded",function(t){init()});class Action{static NAME=null;static register(){PowerButtons.registerAction(this)}static DEFAULTS={};static extractOptions(t,e=null,n=null){null===e&&(e=this.NAME.toLowerCase()),null===n&&((n={})[e]=e);var o,i={};for(o in this.DEFAULTS){var s=void 0!==n[s=o]?n[s]:e+CamelToCamel(s);void 0!==t.dataset[s]&&(i[o]=t.dataset[s])}return i}static initialize(t,e={}){PowerButtons.addAction(t,Object.assign({type:this.NAME.toLowerCase()},e))}static initializeAll(t=null){var e,n=this.NAME.toLowerCase();for(e of document.querySelectorAll("[data-"+n)){var o=null===t?this.extractOptions(e,n):Object.assign({},t);this.initialize(e,o)}}static execute(t,e,n){throw new Error("The execute method must be implemented by the derived class")}}class ActionVerify extends Action{static NAME="Verify";static DEFAULTS={verify:null,form:null,verified:null,notVerified:"The condition for this action is not met",customContentVerified:null,customContentNotVerified:null,titleNotVerified:"The action requires verification",titleVerified:null,buttonAccept:"Accept",buttonClose:!1,escapeKey:!0,header:!0,footer:!0};static execute(options,onNextAction,onCancelActions){let settings=PowerButtons.getActionSettings(this,options),result=null,bindObject=searchForm(settings.form);null===bindObject&&(bindObject=document);try{result="function"==typeof settings.verify?settings.verify.bind(bindObject)():"string"==typeof settings.verify?function(){return eval(settings.verify)}.bind(bindObject)():parseBoolean(settings.verify)}catch(e){console.error("Error executing verification function",e),result=!1}let dialog=null,onVerificationSuccess=onNextAction,onVerificationFailure=onCancelActions;result?null===settings.verified&&null===settings.customContentVerified&&null===settings.titleVerified||(dialog=Dialog.create({title:settings.titleVerified,message:settings.verified,customContent:settings.customContentVerified,buttons:[settings.buttonAccept],escapeKeyCancels:settings.escapeKey,close:settings.buttonClose},null,function(t){null!==onVerificationSuccess&&onVerificationSuccess()})):null===settings.notVerified&&null===settings.customContentNotVerified&&null===settings.titleNotVerified||(dialog=Dialog.create({title:settings.titleNotVerified,message:settings.notVerified,customContent:settings.customContentNotVerified,buttons:[settings.buttonAccept],escapeKeyCancels:settings.escapeKey,close:settings.buttonClose},null,function(t){null!==onVerificationFailure&&onVerificationFailure()})),null!==dialog?dialog.show():result?null!==onVerificationSuccess&&onVerificationSuccess():null!==onVerificationFailure&&onVerificationFailure()}}ActionVerify.register();class ActionConfirm extends Action{static NAME="Confirm";static DEFAULTS={confirm:"Please confirm this action",customContent:null,title:"The action requires confirmation",buttonConfirm:"Confirm",buttonCancel:"Cancel",buttonClose:!0,escapeKey:!0};static extractOptions(t,e=null,n=null){t=super.extractOptions(t,e,n);return""==t.confirm.trim()&&delete t.confirm,t}static execute(t,e,n){t=PowerButtons.getActionSettings(this,t);Dialog.create({title:t.title,message:t.confirm,customContent:t.customContent,buttons:[t.buttonConfirm,t.buttonCancel],escapeKeyCancels:t.escapeKey,close:t.buttonClose},null,function(t){0===t?null!==e&&e():null!==n&&n()}).show()}}ActionConfirm.register();class ActionAsyncTask extends Action{static NAME="AsyncTask";static DEFAULTS={task:null,message:"Please wait...",customContent:null,title:null,buttonCancel:"Cancel",cancel:null,header:!0,footer:!0};static extractOptions(t,e=null,n){return super.extractOptions(t,e,{task:"asynctask"})}static execute(options,onNextAction,onCancelActions){let settings=PowerButtons.getActionSettings(this,options);if(null===settings.task)console.error("The task to execute cannot be null");else{let task=null;if("string"==typeof settings.task)task=async function(){return eval(settings.task)};else{if("function"!=typeof settings.task)return void console.error("The task to execute must be either a string or a function");task=settings.task}let buttons=[],cancelHandler=null,dialog=(null!==settings.cancel&&(buttons=[settings.buttonCancel],"string"==typeof settings.cancel?cancelHandler=function(){eval(settings.cancel)}:"function"==typeof settings.cancel?cancelHandler=settings.cancel:console.error("The cancel handler must be either a string or a function")),Dialog.create({title:settings.title,message:settings.message,customContent:settings.customContent,buttons:buttons,escapeKeyCancels:!1,close:!1,header:void 0!==options.header?settings.header:null!==settings.title&&""!=settings.title,footer:void 0!==options.footer?settings.footer:null!==cancelHandler},function(){cancelHandler(),onCancelActions()},function(t){null!==onNextAction&&onNextAction()}));dialog.show().then(function(){task().finally(function(){dialog.hide()})})}}}ActionAsyncTask.register();class ActionShowMessage extends Action{static NAME="ShowMessage";static DEFAULTS={showmessage:"This is a message",customContent:null,title:null,buttonAccept:"Accept",escapeKey:!0,buttonClose:!0,header:!0,footer:!0};static execute(t,e,n){var o=PowerButtons.getActionSettings(this,t);Dialog.create({title:o.title,message:o.showmessage,customContent:o.customContent,buttons:[o.buttonAccept],escapeKeyCancels:o.escapeKey,close:o.buttonClose,header:void 0!==t.header?o.header:null!==o.title&&""!=o.title,footer:void 0!==t.footer?o.footer:null!==o.buttonAccept&&""!=o.buttonAccept},null,function(t){null!==e&&e()}).show()}}ActionShowMessage.register();class ActionFormset extends Action{static NAME="Formset";static DEFAULTS={form:null,fields:{}};static extractOptions(e,n=null,t){var o,i=super.extractOptions(e,n,{form:"formset"}),s={};for(o in e.dataset)if(o.startsWith(n)){let t=o.substring(n.length);""!==t&&t[0]===t[0].toUpperCase()&&(s[t=t.toLocaleLowerCase()]=e.dataset[o])}return void 0===i.fields&&(i.fields={}),Object.assign(i.fields,s),i}static execute(t,e,n){var o=PowerButtons.getActionSettings(this,t),i=searchForm(o.form);if(null===i)console.error("Form not found "+o.form);else{for(var s,l={},a=0;a"_"+t.toLowerCase()).replace(/^_*/,"")}function pascalToKebab(t){return t.replace(/[A-Z]/g,t=>"-"+t.toLowerCase()).replace(/^-*/,"")}function snakeCaseToCamel(t){t.replace(/-([a-z])/g,t=>t[1].toUpperCase())}function pascalToCamel(t){return t.charAt(0).toLowerCase()+t.slice(1)}function CamelToCamel(t){return t.charAt(0).toUpperCase()+t.slice(1)}function isElement(t){t instanceof Element||0 instanceof HTMLDocument}function parseBoolean(t){return"boolean"==typeof t?t:"string"==typeof t?"true"===(t=t.toLowerCase())||"yes"===t||"1"===t:!!t}function createTag(t,e={},n=null){var o=t.split("#");let i=null;var s,o=(t=1==o.length?o[0]:(o[1]=o[1].split("."),i=o[1][0],[o[0],...o[1].slice(1)].join("."))).split("."),l=(""===(t=o[0])&&(t="div"),"string"==typeof e&&(n=e,e={}),null!==n&&(e.innerHTML=n),null!==i&&(e.id=i),e.className=[e.className,...o.slice(1)].filter(function(t){return""!==(""+t).trim()}).join(" "),document.createElement(t));for(s in e)void 0!==l[s]?l[s]=e[s]:l.setAttribute(s,e[s]);return l}function appendToElement(t,...e){e=e.filter(t=>null!=t);return t.append(...e),t}function searchForm(t){let e=null;return null!==t&&void 0===(e=document.forms[t])&&null===(e=document.querySelector(t))&&console.warn(`form ${t} not found`),null!==e&&"form"!==e.tagName.toLowerCase()&&console.warn(`form ${t} is not a form`),e}function getValueWithJavascriptSupport(value,context=null){if("function"==typeof value)return value.bind(context);if("string"==typeof value){let internalValue=value.trim();if(internalValue.startsWith("javascript:"))try{let f=internalValue.substring(11);value=function(){return eval(f)}.bind(context)}catch(e){console.error(`Error executing javascript code ${internalValue.substring(11)}, error: `+e),value=null}}return value}function promiseForEvent(e,n){let o=null;var t=new Promise(t=>{o=t});return e.addEventListener(n,function t(){e.removeEventListener(n,t),o()}),t}function isEmpty(t){return null==t||(t instanceof Array?0===t.length:t instanceof Object?0===Object.keys(t).length:"string"==typeof t&&""===t.trim())}window.powerButtons=function(t,e=[],n={}){let o=e,i=("string"==typeof o?o=document.querySelectorAll(o):void 0===o.length&&(o=[o]),null);for(var s in PowerButtons.actionsRegistered)if(t.toLocaleLowerCase()===s.toLocaleLowerCase()){i=PowerButtons.actionsRegistered[s];break}if(void 0===i)console.error(`The action ${t} is not registered`);else for(var l of o)i.initialize(l,n);return e},window.powerButtons.version="2.0.1",window.powerButtons.plugins=function(){return Object.keys(PowerButtons.actionsRegistered)},void 0!==window.$&&(window.$.fn.powerButtons=function(t,e={}){return window.powerButtons(t,this,e),this},window.$.fn.powerButtons.version=window.powerButtons.version,window.$.fn.powerButtons.plugins=window.powerButtons.plugins);class DialogLegacy{DEFAULTS={message:"Main message",buttonCount:2};constructor(t={},e=null,n=null){this.options={...this.DEFAULTS,...t},this.buttonCount=this.options.buttons.length,this.options.buttons=["Accept","Cancel"],this.result=null,this.onButton=e,this.onHidden=n}dispose(){}show(t=null,e=null){switch(null!==t&&(this.onButton=t),null!==e&&(this.onHidden=e),this.buttonCount){case 0:case 1:alert(this.message),this.result=0;break;case 2:this.result=confirm(this.options.message)?0:1;break;default:throw"Unsupported button count "+this.buttonCount}null!==this.onButton&&this.onButton(this.result,{button:this.result,text:this.options.buttons[this.result]},null),null!==this.onHidden&&this.onHidden(this.result,{button:this.result,text:this.options.buttons[this.result]},null)}}class Dialog{static create(t={},e=null,n=null){if(void 0===window.bootstrap||void 0===window.bootstrap.Modal)return new DialogLegacy(t,e,n);if(void 0!==t.selector&&null!==t.selector||void 0!==t.dialogFunction&&null!==t.dialogFunction)throw new Error("not implemented, yet");return new Dialog(t,e,n)}DEFAULTS={title:"Title",message:"Main message",customContent:null,buttons:["Accept"],buttonClasses:["btn-primary","btn-secondary"],buttonPanelClasses:["text-end"],escapeKeyCancels:!0,header:!0,footer:!0,body:!0,close:!0};dialog=null;options=null;modal=null;onButton=null;result=null;onHidden=null;onButton=null;constructor(t={},e=null,n=null){if(void 0===window.bootstrap||void 0===window.bootstrap.Modal)throw new Error("Bootstrap is required to use this class");this.options={...this.DEFAULTS,...t};var o=[];for(let e=0;ethis._handleButton(-1,null,e)),parseBoolean(i.header)&&(t=createTag(".modal-header"),null!==i.title&&t.append(appendToElement(createTag(".modal-title"),appendToElement(createTag("h5",i.title)))),parseBoolean(i.close))&&t.append(e);var s=[];if(null!==i.buttons)for(let o=0;ot.trim()).filter(t=>""!==t).join("."),n=(!1===i.footer&&(e+=".mx-1"),createTag("button.btn"+(e=""!==e?"."+e:e)+".button"+o,{type:"button"},t.text));n.addEventListener("click",function(){this._handleButton(o,t,n),void 0!==t.handler&&null!==t.handler&&t.handler(o,t,n)}.bind(this)),s.push(n)}let n=null,o=(parseBoolean(i.footer)&&(n=appendToElement(createTag(".modal-footer"),...s)),null);if(parseBoolean(i.body)&&(o=createTag(".modal-body"),null===t&&parseBoolean(i.close)&&o.append(appendToElement(createTag(".text-end"),e)),null!==i.message&&o.append(createTag("p.message.text-center",i.message)),null!==i.customContent&&o.append(createTag(".custom-content.mx-auto",i.customContent)),null===n)){let t=i.buttonPanelClasses.map(t=>t.trim()).filter(t=>""!==t).join(".");""!==t&&(t="."+t),appendToElement(o,appendToElement(createTag(".buttons"+t),...s))}return appendToElement(createTag(".modal.fade",{tabindex:"-1",role:"dialog","aria-hidden":"true","data-keyboard":"false"}),appendToElement(createTag(".modal-dialog.modal-dialog-centered",{role:"document"}),appendToElement(createTag(".modal-content"),t,o,n)))}}function confirmDialog(t,e="this action needs confirmation",n=null,o=null,i=!0){e=new Dialog({title:e,message:t,buttons:[{text:"Cancel",class:"btn-secondary",handler:o},{text:"Confirm",class:"btn-primary",handler:n}],escapeKeyCancels:i});return e.show(),e}function alertDialog(t,e="Alert",n=null){e=new Dialog({title:e,message:t,buttons:[{text:"Accept",class:"btn-primary",handler:n}],escapeKeyCancels:!0});return e.show(),e}function loadingDialog(t,e=null,n=null){"function"==typeof e&&(n=e,e=null);t=new Dialog({title:null,message:t,buttons:[{text:"Cancel",class:"btn-primary"}],customContent:e,escapeKeyCancels:!1,close:!1,header:!1,footer:!1},n);return t.show(),t}void 0===window.powerButtons.utils&&(window.powerButtons.utils={}),Object.assign(window.powerButtons.utils,{confirmDialog:confirmDialog,alertDialog:alertDialog,loadingDialog:loadingDialog});class PowerButtons{static actionsRegistered={};static registerAction(t){this.actionsRegistered[t.NAME.toLowerCase()]=t,void 0===window.powerButtons&&(window.powerButtons={}),void 0===window.powerButtons.defaults&&(window.powerButtons.defaults={}),window.powerButtons.defaults[t.NAME.toLowerCase()]=Object.assign({},t.DEFAULTS)}static getActionSettings(t,e){if(void 0===this.actionsRegistered[t.NAME.toLowerCase()])return console.error(`The action ${t.NAME} is not registered`),{};let n={};return void 0!==window.powerButtons&&void 0!==window.powerButtons.defaults&&void 0!==window.powerButtons.defaults[t.NAME.toLowerCase()]&&(n=window.powerButtons.defaults[t.NAME.toLowerCase()]),Object.assign({},t.DEFAULTS,n,e)}static addAction(t,e={}){PowerButtons.addActionSupport(t).appendAction(e)}static addActionSupport(t){return void 0===t._powerButtons?t._powerButtons=new PowerButtons(t):t._powerButtons.reset(),t._powerButtons}el=null;current_action=0;actions=[];back_onclick=null;constructor(t){(t._powerButtons=this).el=t,this.current_action=0,this.actions=[],this.back_onclick=null,void 0!==t.onclick&&null!==t.onclick&&(this.back_onclick=t.onclick,t.onclick=null),t.addEventListener("click",this.handlerClick.bind(this))}appendAction(t={}){if(void 0===t.type)throw"The type of the action is mandatory";this.actions.push(t)}handlerClick(t){if(this.current_action>=this.actions.length)this.current_action=0,"function"!=typeof this.back_onclick||this.back_onclick()||t.preventDefault();else{var e=this.actions[this.current_action],n=(t.preventDefault(),t.stopImmediatePropagation(),function(){this.current_action++,this.current_action>=this.actions.length&&void 0!==this.el.click?this.el.click():this.el.dispatchEvent(new Event(t.type,t))}.bind(this)),o=this.constructor.actionsRegistered[e.type];if(void 0===o)throw`The action ${e.type} is not registered`;o.execute(this.el,e,n,()=>this.reset())}}reset(){this.current_action=0}static initializeAll(){Object.entries(this.actionsRegistered).forEach(([,t])=>{t.initializeAll()})}}function init(){PowerButtons.initializeAll()}void 0!==document.addEventListener&&document.addEventListener("DOMContentLoaded",function(t){init()});class Action{static NAME=null;static register(){PowerButtons.registerAction(this)}static DEFAULTS={};static extractOptions(t,e=null,n=null){null===e&&(e=this.NAME.toLowerCase()),null===n&&((n={})[e]=e);var o,i={};for(o in this.DEFAULTS){var s=void 0!==n[s=o]?n[s]:e+CamelToCamel(s);void 0!==t.dataset[s]&&(i[o]=t.dataset[s])}return i}static initialize(t,e={}){PowerButtons.addAction(t,Object.assign({type:this.NAME.toLowerCase()},e))}static initializeAll(t=null){var e,n=this.NAME.toLowerCase();for(e of document.querySelectorAll("[data-"+n)){var o=null===t?this.extractOptions(e,n):Object.assign({},t);this.initialize(e,o)}}static execute(t,e,n,o){throw new Error("The execute method must be implemented by the derived class")}}class ActionVerify extends Action{static NAME="Verify";static DEFAULTS={verify:null,form:null,verified:null,notVerified:"The condition for this action is not met",customContentVerified:null,customContentNotVerified:null,titleNotVerified:"The action requires verification",titleVerified:null,buttonAccept:"Accept",buttonClose:!1,escapeKey:!0,header:!0,footer:!0};static execute(el,options,onNextAction,onCancelActions){let settings=PowerButtons.getActionSettings(this,options),result=null,bindObject=searchForm(settings.form);null===bindObject&&(bindObject=document);try{result="function"==typeof settings.verify?settings.verify.bind(bindObject)():"string"==typeof settings.verify?function(){return eval(settings.verify)}.bind(bindObject)():parseBoolean(settings.verify)}catch(e){console.error("Error executing verification function",e),result=!1}let dialog=null,onVerificationSuccess=onNextAction,onVerificationFailure=onCancelActions;result?null===settings.verified&&null===settings.customContentVerified&&null===settings.titleVerified||(dialog=Dialog.create({title:settings.titleVerified,message:settings.verified,customContent:settings.customContentVerified,buttons:[settings.buttonAccept],escapeKeyCancels:settings.escapeKey,close:settings.buttonClose},null,function(t){null!==onVerificationSuccess&&onVerificationSuccess()})):null===settings.notVerified&&null===settings.customContentNotVerified&&null===settings.titleNotVerified||(dialog=Dialog.create({title:settings.titleNotVerified,message:settings.notVerified,customContent:settings.customContentNotVerified,buttons:[settings.buttonAccept],escapeKeyCancels:settings.escapeKey,close:settings.buttonClose},null,function(t){null!==onVerificationFailure&&onVerificationFailure()})),null!==dialog?dialog.show():result?null!==onVerificationSuccess&&onVerificationSuccess():null!==onVerificationFailure&&onVerificationFailure()}}ActionVerify.register();class ActionConfirm extends Action{static NAME="Confirm";static DEFAULTS={confirm:"Please confirm this action",customContent:null,title:"The action requires confirmation",buttonConfirm:"Confirm",buttonCancel:"Cancel",buttonClose:!0,escapeKey:!0};static extractOptions(t,e=null,n=null){t=super.extractOptions(t,e,n);return""==t.confirm.trim()&&delete t.confirm,t}static execute(t,e,n,o){e=PowerButtons.getActionSettings(this,e);Dialog.create({title:e.title,message:e.confirm,customContent:e.customContent,buttons:[e.buttonConfirm,e.buttonCancel],escapeKeyCancels:e.escapeKey,close:e.buttonClose},null,function(t){0===t?null!==n&&n():null!==o&&o()}).show()}}ActionConfirm.register();class ActionAsyncTask extends Action{static NAME="AsyncTask";static DEFAULTS={task:null,message:"Please wait...",customContent:null,title:null,buttonCancel:"Cancel",cancel:null,header:!0,footer:!0};static extractOptions(t,e=null,n){return super.extractOptions(t,e,{task:"asynctask"})}static execute(el,options,onNextAction,onCancelActions){let settings=PowerButtons.getActionSettings(this,options);if(null===settings.task)console.error("The task to execute cannot be null");else{let task=null;if("string"==typeof settings.task)task=async function(){return eval(settings.task)};else{if("function"!=typeof settings.task)return void console.error("The task to execute must be either a string or a function");task=settings.task}let buttons=[],cancelHandler=null,dialog=(null!==settings.cancel&&(buttons=[settings.buttonCancel],"string"==typeof settings.cancel?cancelHandler=function(){eval(settings.cancel)}:"function"==typeof settings.cancel?cancelHandler=settings.cancel:console.error("The cancel handler must be either a string or a function")),Dialog.create({title:settings.title,message:settings.message,customContent:settings.customContent,buttons:buttons,escapeKeyCancels:!1,close:!1,header:void 0!==options.header?settings.header:null!==settings.title&&""!=settings.title,footer:void 0!==options.footer?settings.footer:null!==cancelHandler},function(){cancelHandler(),onCancelActions()},function(t){null!==onNextAction&&onNextAction()}));dialog.show().then(function(){task().finally(function(){dialog.hide()})})}}}ActionAsyncTask.register();class ActionShowMessage extends Action{static NAME="ShowMessage";static DEFAULTS={showmessage:"This is a message",customContent:null,title:null,buttonAccept:"Accept",escapeKey:!0,buttonClose:!0,header:!0,footer:!0};static execute(t,e,n,o){var i=PowerButtons.getActionSettings(this,e);Dialog.create({title:i.title,message:i.showmessage,customContent:i.customContent,buttons:[i.buttonAccept],escapeKeyCancels:i.escapeKey,close:i.buttonClose,header:void 0!==e.header?i.header:null!==i.title&&""!=i.title,footer:void 0!==e.footer?i.footer:null!==i.buttonAccept&&""!=i.buttonAccept},null,function(t){null!==n&&n()}).show()}}ActionShowMessage.register();class ActionFormset extends Action{static NAME="Formset";static DEFAULTS={form:null,fields:{}};static extractOptions(e,n=null,t){var o,i=super.extractOptions(e,n,{form:"formset"}),s={};for(o in e.dataset)if(o.startsWith(n)){let t=o.substring(n.length);""!==t&&t[0]===t[0].toUpperCase()&&(s[t=t.toLocaleLowerCase()]=e.dataset[o])}return void 0===i.fields&&(i.fields={}),Object.assign(i.fields,s),i}static execute(t,e,n,o){var i,s=PowerButtons.getActionSettings(this,e);let l=null,a=[],r=[];if(""==s.form)null!==t.form?l=t.form:r=Array.from(document.querySelectorAll("input")).filter(t=>null===t.form);else if(null===(l=searchForm(s.form)))return void console.error("Form not found "+s.form);for(i in(r=null!==l?Array.from(l.elements):r).forEach(t=>{""!==t.name&&(a[t.name.toLocaleLowerCase()]=t),""!==t.id&&(a[t.id.toLocaleLowerCase()]=t)}),s.fields)if(void 0!==a[i]){var c=s.fields[i];let t=getValueWithJavascriptSupport(c,null!==l?l:a);if("function"==typeof t)try{t=t()}catch(t){console.error("Error executing "+c,t);continue}a[i].value=t}n()}}ActionFormset.register();class ActionFormButton extends Action{static NAME="FormButton";static DEFAULTS={formbutton:null,method:"post",formClass:"formbutton",convertCase:"none",formId:null,fields:{}};static extractOptions(e,n=null,t=null){var o,i=super.extractOptions(e,n,t),s={};for(o in n+="Field",e.dataset)if(o.startsWith(n)){let t=o.substring(n.length);if(""!==t&&t[0]===t[0].toUpperCase()){switch(i.convertCase){case"kebab":t=pascalToKebab(t);break;case"snake":t=pascalToSnake(t);break;case"camel":t=pascalToCamel(t)}s[t]=e.dataset[o]}}return void 0===i.fields&&(i.fields={}),Object.assign(i.fields,s),i}static initialize(t,e={}){for(var n=PowerButtons.getActionSettings(this,e),o=document.createElement("form"),i=(o.method=n.method,isEmpty(n.formbutton)||(o.action=n.formbutton),null!==n.formId&&(o.id=n.formId),n.formClass.split(" ")),s=0;s this.reset()) + action.execute(this.el, currentActionSettings, onNextAction, () => this.reset()) } reset() { this.current_action = 0 @@ -665,7 +668,7 @@ this.initialize(el, options) } } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { throw new Error("The execute method must be implemented by the derived class") } } @@ -686,7 +689,7 @@ header: true, footer: true }; - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); let result = null; let bindObject = searchForm(settings.form); @@ -775,7 +778,7 @@ } return options } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); let dialog = Dialog.create({ title: settings.title, @@ -816,7 +819,7 @@ task: "asynctask" }) } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); if (settings.task === null) { console.error("The task to execute cannot be null"); @@ -884,7 +887,7 @@ header: true, footer: true }; - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); let dialog = Dialog.create({ title: settings.title, @@ -934,27 +937,39 @@ Object.assign(options.fields, fields); return options } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); - let formToSet = searchForm(settings.form); - if (formToSet === null) { - console.error(`Form not found ${settings.form}`); - return + let formToSet = null; + let inputFields = []; + let elements = []; + if (settings.form == "") { + if (el.form !== null) { + formToSet = el.form + } else { + elements = Array.from(document.querySelectorAll("input")).filter(input => input.form === null) + } + } else { + formToSet = searchForm(settings.form); + if (formToSet === null) { + console.error(`Form not found ${settings.form}`); + return + } + } + if (formToSet !== null) { + elements = Array.from(formToSet.elements) } - let nameMap = {}; - for (var i = 0; i < formToSet.elements.length; i++) { - let element = formToSet.elements[i]; + elements.forEach(element => { if (element.name !== "") { - nameMap[element.name.toLocaleLowerCase()] = element.name + inputFields[element.name.toLocaleLowerCase()] = element } if (element.id !== "") { - nameMap[element.id.toLocaleLowerCase()] = element.id + inputFields[element.id.toLocaleLowerCase()] = element } - } - for (var field in settings.fields) { - if (nameMap[field] !== undefined) { + }); + for (let field in settings.fields) { + if (inputFields[field] !== undefined) { let value = settings.fields[field]; - let result = getValueWithJavascriptSupport(value, formToSet); + let result = getValueWithJavascriptSupport(value, formToSet !== null ? formToSet : inputFields); if (typeof result === "function") { try { result = result() @@ -963,7 +978,7 @@ continue } } - formToSet[nameMap[field]].value = result + inputFields[field].value = result } } onNextAction() @@ -1001,7 +1016,6 @@ fieldname = pascalToSnake(fieldname); break; case "camel": - console.log(fieldname, pascalToCamel(fieldname)); fieldname = pascalToCamel(fieldname); break; case "pascal": @@ -1058,7 +1072,7 @@ super.initialize(el, settings) } } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); let error = false; for (let fieldName in settings.fields) { diff --git a/dist/powerbuttons.js b/dist/powerbuttons.js index a4d47b5..a5b8cd2 100644 --- a/dist/powerbuttons.js +++ b/dist/powerbuttons.js @@ -41,13 +41,17 @@ } return els; }; - window.powerButtons.version = "2.0.0"; + window.powerButtons.version = "2.0.1"; + window.powerButtons.plugins = function () { + return Object.keys(PowerButtons.actionsRegistered); + }; if (window.$ !== undefined) { window.$.fn.powerButtons = function (pluginName, options = {}) { window.powerButtons(pluginName, this, options); return this; }; window.$.fn.powerButtons.version = window.powerButtons.version; + window.$.fn.powerButtons.plugins = window.powerButtons.plugins; } function pascalToSnake(str) { @@ -516,7 +520,6 @@ class PowerButtons { static actionsRegistered = {}; static registerAction(action) { - console.debug(`Registering action ${action.NAME.toLowerCase()}`); this.actionsRegistered[action.NAME.toLowerCase()] = action; if (window.powerButtons === undefined) { window.powerButtons = {}; @@ -600,7 +603,7 @@ if (action === undefined) { throw `The action ${currentActionSettings.type} is not registered`; } - action.execute(currentActionSettings, onNextAction, () => this.reset()); + action.execute(this.el, currentActionSettings, onNextAction, () => this.reset()); } reset() { this.current_action = 0; @@ -665,7 +668,7 @@ this.initialize(el, options); } } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { throw new Error("The execute method must be implemented by the derived class"); } } @@ -686,7 +689,7 @@ header: true, footer: true }; - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); let result = null; let bindObject = searchForm(settings.form); @@ -775,7 +778,7 @@ } return options; } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); let dialog = Dialog.create({ title: settings.title, @@ -816,7 +819,7 @@ task: "asynctask" }); } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); if (settings.task === null) { console.error("The task to execute cannot be null"); @@ -884,7 +887,7 @@ header: true, footer: true }; - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); let dialog = Dialog.create({ title: settings.title, @@ -934,27 +937,39 @@ Object.assign(options.fields, fields); return options; } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); - let formToSet = searchForm(settings.form); - if (formToSet === null) { - console.error(`Form not found ${settings.form}`); - return; + let formToSet = null; + let inputFields = []; + let elements = []; + if (settings.form == "") { + if (el.form !== null) { + formToSet = el.form; + } else { + elements = Array.from(document.querySelectorAll("input")).filter(input => input.form === null); + } + } else { + formToSet = searchForm(settings.form); + if (formToSet === null) { + console.error(`Form not found ${settings.form}`); + return; + } + } + if (formToSet !== null) { + elements = Array.from(formToSet.elements); } - let nameMap = {}; - for (var i = 0; i < formToSet.elements.length; i++) { - let element = formToSet.elements[i]; + elements.forEach(element => { if (element.name !== "") { - nameMap[element.name.toLocaleLowerCase()] = element.name; + inputFields[element.name.toLocaleLowerCase()] = element; } if (element.id !== "") { - nameMap[element.id.toLocaleLowerCase()] = element.id; + inputFields[element.id.toLocaleLowerCase()] = element; } - } - for (var field in settings.fields) { - if (nameMap[field] !== undefined) { + }); + for (let field in settings.fields) { + if (inputFields[field] !== undefined) { let value = settings.fields[field]; - let result = getValueWithJavascriptSupport(value, formToSet); + let result = getValueWithJavascriptSupport(value, formToSet !== null ? formToSet : inputFields); if (typeof result === "function") { try { result = result(); @@ -963,7 +978,7 @@ continue; } } - formToSet[nameMap[field]].value = result; + inputFields[field].value = result; } } onNextAction(); @@ -1001,7 +1016,6 @@ fieldname = pascalToSnake(fieldname); break; case "camel": - console.log(fieldname, pascalToCamel(fieldname)); fieldname = pascalToCamel(fieldname); break; case "pascal": @@ -1058,7 +1072,7 @@ super.initialize(el, settings); } } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { let settings = PowerButtons.getActionSettings(this, options); let error = false; for (let fieldName in settings.fields) { diff --git a/dist/powerbuttons.min.js b/dist/powerbuttons.min.js index d78ba3f..870e045 100644 --- a/dist/powerbuttons.min.js +++ b/dist/powerbuttons.min.js @@ -1,2 +1,2 @@ /* Copyright 2021 Carlos A. (https://github.com/dealfonso); License: http://www.apache.org/licenses/LICENSE-2.0 */ -(function(window,document){"use strict";window.powerButtons=function(e,t=[],n={}){let i=t;if(typeof i==="string"){i=document.querySelectorAll(i)}else{if(i.length===undefined){i=[i]}}let o=null;for(let t in PowerButtons.actionsRegistered){if(e.toLocaleLowerCase()===t.toLocaleLowerCase()){o=PowerButtons.actionsRegistered[t];break}}if(o===undefined){console.error(`The action ${e} is not registered`)}else{for(let t of i){o.initialize(t,n)}}return t};window.powerButtons.version="2.0.0";if(window.$!==undefined){window.$.fn.powerButtons=function(t,e={}){window.powerButtons(t,this,e);return this};window.$.fn.powerButtons.version=window.powerButtons.version}function pascalToSnake(t){return t.replace(/[A-Z]/g,t=>`_${t.toLowerCase()}`).replace(/^_*/,"")}function pascalToKebab(t){return t.replace(/[A-Z]/g,t=>`-${t.toLowerCase()}`).replace(/^-*/,"")}function snakeCaseToCamel(t){return t.replace(/-([a-z])/g,t=>t[1].toUpperCase())}function pascalToCamel(t){return t.charAt(0).toLowerCase()+t.slice(1)}function CamelToCamel(t){return t.charAt(0).toUpperCase()+t.slice(1)}function isElement(t){return t instanceof Element||t instanceof HTMLDocument}function parseBoolean(t){if(typeof t==="boolean"){return t}if(typeof t==="string"){t=t.toLowerCase();if(t==="true"||t==="yes"||t==="1"){return true}return false}return!!t}function createTag(t,e={},n=null){let i=t.split("#");let o=null;if(i.length==1){t=i[0]}else{i[1]=i[1].split(".");o=i[1][0];t=[i[0],...i[1].slice(1)].join(".")}let s=t.split(".");t=s[0];if(t===""){t="div"}if(typeof e==="string"){n=e;e={}}if(n!==null){e.innerHTML=n}if(o!==null){e.id=o}e.className=[e.className,...s.slice(1)].filter(function(t){return`${t}`.trim()!==""}).join(" ");let l=document.createElement(t);for(let t in e){if(l[t]!==undefined){l[t]=e[t]}else{l.setAttribute(t,e[t])}}return l}function appendToElement(t,...e){let n=e.filter(t=>t!==null&&t!==undefined);t.append(...n);return t}function searchForm(t){let e=null;if(t!==null){e=document.forms[t];if(e===undefined){e=document.querySelector(t);if(e===null){console.warn(`form ${t} not found`)}}}if(e!==null){if(e.tagName.toLowerCase()!=="form"){console.warn(`form ${t} is not a form`)}}return e}function getValueWithJavascriptSupport(value,context=null){if(typeof value==="function"){return value.bind(context)}if(typeof value==="string"){let internalValue=value.trim();if(internalValue.startsWith("javascript:")){try{let f=internalValue.substring(11);value=function(){return eval(f)}.bind(context)}catch(e){console.error(`Error executing javascript code ${internalValue.substring(11)}, error: ${e}`);value=null}}}return value}function promiseForEvent(t,e){let n=null;let i=new Promise(t=>{n=t});let o=function(){t.removeEventListener(e,o);n()};t.addEventListener(e,o);return i}function isEmpty(t){if(t===null||t===undefined){return true}if(t instanceof Array){return t.length===0}if(t instanceof Object){return Object.keys(t).length===0}if(typeof t==="string"){return t.trim()===""}return false}class DialogLegacy{DEFAULTS={message:"Main message",buttonCount:2};constructor(t={},e=null,n=null){this.options={...this.DEFAULTS,...t};this.buttonCount=this.options.buttons.length;this.options.buttons=["Accept","Cancel"];this.result=null;this.onButton=e;this.onHidden=n}dispose(){}show(t=null,e=null){if(t!==null){this.onButton=t}if(e!==null){this.onHidden=e}switch(this.buttonCount){case 0:case 1:alert(this.message);this.result=0;break;case 2:this.result=confirm(this.options.message)?0:1;break;default:throw`Unsupported button count ${this.buttonCount}`}if(this.onButton!==null){this.onButton(this.result,{button:this.result,text:this.options.buttons[this.result]},null)}if(this.onHidden!==null){this.onHidden(this.result,{button:this.result,text:this.options.buttons[this.result]},null)}}}class Dialog{static create(t={},e=null,n=null){if(window.bootstrap===undefined||window.bootstrap.Modal===undefined){return new DialogLegacy(t,e,n)}if(t.selector!==undefined&&t.selector!==null||t.dialogFunction!==undefined&&t.dialogFunction!==null){throw new Error("not implemented, yet")}return new Dialog(t,e,n)}DEFAULTS={title:"Title",message:"Main message",customContent:null,buttons:["Accept"],buttonClasses:["btn-primary","btn-secondary"],buttonPanelClasses:["text-end"],escapeKeyCancels:true,header:true,footer:true,body:true,close:true};dialog=null;options=null;modal=null;onButton=null;result=null;onHidden=null;onButton=null;constructor(t={},e=null,n=null){if(window.bootstrap===undefined||window.bootstrap.Modal===undefined){throw new Error("Bootstrap is required to use this class")}this.options={...this.DEFAULTS,...t};let i=[];for(let e=0;e=0){this.onHidden(this.result,{button:this.result,text:this.options.buttons[this.result]})}else{this.onHidden(this.result,null)}}}dispose(){if(this.modal!==null){this.modal.dispose();this.modal=null}this.dialog.remove();this.dialog=null}show(t=null,e=null){if(this.dialog===null){this.dialog=this._build_dialog(this.options)}if(this.modal===null){this.modal=new bootstrap.Modal(this.dialog,{backdrop:this.options.escapeKeyCancels?true:"static",keyboard:this.options.escapeKeyCancels})}this.dialog.addEventListener("hidden.bs.modal",this._hiddenHandler);this.result=null;if(t!==null){this.onButton=t}if(e!==null){this.onHidden=e}this.modal.show();return promiseForEvent(this.dialog,"shown.bs.modal")}hide(){this.modal.hide();return promiseForEvent(this.dialog,"hidden.bs.modal")}_handleButton(t,e,n){this.result=t;let i=true;if(this.onButton!==null){i=!(this.onButton(t,e,n)===false)}if(i){this.hide()}}_build_dialog(o={}){let t=null;let e=null;if(parseBoolean(o.close)){e=createTag("button.close.btn-close",{type:"button","aria-label":"Close"});e.addEventListener("click",()=>this._handleButton(-1,null,e))}if(parseBoolean(o.header)){t=createTag(".modal-header");if(o.title!==null){t.append(appendToElement(createTag(".modal-title"),appendToElement(createTag("h5",o.title))))}if(parseBoolean(o.close)){t.append(e)}}let s=[];if(o.buttons!==null){for(let i=0;it.trim()).filter(t=>t!=="").join(".");if(o.footer===false){e+=".mx-1"}if(e!==""){e="."+e}let n=createTag("button.btn"+e+".button"+i,{type:"button"},t.text);n.addEventListener("click",function(){this._handleButton(i,t,n);if(t.handler!==undefined&&t.handler!==null){t.handler(i,t,n)}}.bind(this));s.push(n)}}let n=null;if(parseBoolean(o.footer)){n=appendToElement(createTag(".modal-footer"),...s)}let i=null;if(parseBoolean(o.body)){i=createTag(".modal-body");if(t===null){if(parseBoolean(o.close)){i.append(appendToElement(createTag(".text-end"),e))}}if(o.message!==null){i.append(createTag("p.message.text-center",o.message))}if(o.customContent!==null){i.append(createTag(".custom-content.mx-auto",o.customContent))}if(n===null){let t=o.buttonPanelClasses.map(t=>t.trim()).filter(t=>t!=="").join(".");if(t!==""){t="."+t}appendToElement(i,appendToElement(createTag(".buttons"+t),...s))}}let l=appendToElement(createTag(".modal.fade",{tabindex:"-1",role:"dialog","aria-hidden":"true","data-keyboard":"false"}),appendToElement(createTag(".modal-dialog.modal-dialog-centered",{role:"document"}),appendToElement(createTag(".modal-content"),t,i,n)));return l}}function confirmDialog(t,e="this action needs confirmation",n=null,i=null,o=true){let s=new Dialog({title:e,message:t,buttons:[{text:"Cancel",class:"btn-secondary",handler:i},{text:"Confirm",class:"btn-primary",handler:n}],escapeKeyCancels:o});s.show();return s}function alertDialog(t,e="Alert",n=null){let i=new Dialog({title:e,message:t,buttons:[{text:"Accept",class:"btn-primary",handler:n}],escapeKeyCancels:true});i.show();return i}function loadingDialog(t,e=null,n=null){if(typeof e==="function"){n=e;e=null}let i=new Dialog({title:null,message:t,buttons:[{text:"Cancel",class:"btn-primary"}],customContent:e,escapeKeyCancels:false,close:false,header:false,footer:false},n);i.show();return i}if(window.powerButtons.utils===undefined){window.powerButtons.utils={}}Object.assign(window.powerButtons.utils,{confirmDialog:confirmDialog,alertDialog:alertDialog,loadingDialog:loadingDialog});class PowerButtons{static actionsRegistered={};static registerAction(t){console.debug(`Registering action ${t.NAME.toLowerCase()}`);this.actionsRegistered[t.NAME.toLowerCase()]=t;if(window.powerButtons===undefined){window.powerButtons={}}if(window.powerButtons.defaults===undefined){window.powerButtons.defaults={}}window.powerButtons.defaults[t.NAME.toLowerCase()]=Object.assign({},t.DEFAULTS)}static getActionSettings(t,e){if(this.actionsRegistered[t.NAME.toLowerCase()]===undefined){console.error(`The action ${t.NAME} is not registered`);return{}}let n={};if(window.powerButtons!==undefined&&window.powerButtons.defaults!==undefined&&window.powerButtons.defaults[t.NAME.toLowerCase()]!==undefined){n=window.powerButtons.defaults[t.NAME.toLowerCase()]}return Object.assign({},t.DEFAULTS,n,e)}static addAction(t,e={}){let n=PowerButtons.addActionSupport(t);n.appendAction(e)}static addActionSupport(t){if(t._powerButtons===undefined){t._powerButtons=new PowerButtons(t)}else{t._powerButtons.reset()}return t._powerButtons}el=null;current_action=0;actions=[];back_onclick=null;constructor(t){t._powerButtons=this;this.el=t;this.current_action=0;this.actions=[];this.back_onclick=null;if(t.onclick!==undefined&&t.onclick!==null){this.back_onclick=t.onclick;t.onclick=null}t.addEventListener("click",this.handlerClick.bind(this))}appendAction(t={}){if(t.type===undefined){throw"The type of the action is mandatory"}this.actions.push(t)}handlerClick(t){if(this.current_action>=this.actions.length){this.current_action=0;if(typeof this.back_onclick==="function"){if(!this.back_onclick()){t.preventDefault()}}return}let e=this.actions[this.current_action];t.preventDefault();t.stopImmediatePropagation();let n=function(){this.current_action++;if(this.current_action>=this.actions.length){if(this.el.click!==undefined){this.el.click()}else{this.el.dispatchEvent(new Event(t.type,t))}}else{this.el.dispatchEvent(new Event(t.type,t))}}.bind(this);let i=this.constructor.actionsRegistered[e.type];if(i===undefined){throw`The action ${e.type} is not registered`}i.execute(e,n,()=>this.reset())}reset(){this.current_action=0}static initializeAll(){Object.entries(this.actionsRegistered).forEach(([t,e])=>{e.initializeAll()})}}function init(){PowerButtons.initializeAll()}if(document.addEventListener!==undefined){document.addEventListener("DOMContentLoaded",function(t){init()})}class Action{static NAME=null;static register(){PowerButtons.registerAction(this)}static DEFAULTS={};static extractOptions(n,i=null,o=null){if(i===null){i=this.NAME.toLowerCase()}if(o===null){o={};o[i]=i}let s={};for(let e in this.DEFAULTS){let t=e;if(o[t]!==undefined){t=o[t]}else{t=i+CamelToCamel(t)}if(n.dataset[t]!==undefined){s[e]=n.dataset[t]}}return s}static initialize(t,e={}){PowerButtons.addAction(t,Object.assign({type:this.NAME.toLowerCase()},e))}static initializeAll(n=null){let i=this.NAME.toLowerCase();for(let e of document.querySelectorAll(`[data-${i}`)){let t=null;if(n===null){t=this.extractOptions(e,i)}else{t=Object.assign({},n)}this.initialize(e,t)}}static execute(t,e,n){throw new Error("The execute method must be implemented by the derived class")}}class ActionVerify extends Action{static NAME="Verify";static DEFAULTS={verify:null,form:null,verified:null,notVerified:"The condition for this action is not met",customContentVerified:null,customContentNotVerified:null,titleNotVerified:"The action requires verification",titleVerified:null,buttonAccept:"Accept",buttonClose:false,escapeKey:true,header:true,footer:true};static execute(options,onNextAction,onCancelActions){let settings=PowerButtons.getActionSettings(this,options);let result=null;let bindObject=searchForm(settings.form);if(bindObject===null){bindObject=document}try{if(typeof settings.verify==="function"){result=settings.verify.bind(bindObject)()}else if(typeof settings.verify==="string"){result=function(){return eval(settings.verify)}.bind(bindObject)()}else{result=parseBoolean(settings.verify)}}catch(e){console.error("Error executing verification function",e);result=false}let dialog=null;let onVerificationSuccess=onNextAction;let onVerificationFailure=onCancelActions;if(result){if(settings.verified!==null||settings.customContentVerified!==null||settings.titleVerified!==null){dialog=Dialog.create({title:settings.titleVerified,message:settings.verified,customContent:settings.customContentVerified,buttons:[settings.buttonAccept],escapeKeyCancels:settings.escapeKey,close:settings.buttonClose},null,function(t){if(onVerificationSuccess!==null){onVerificationSuccess()}})}}else{if(settings.notVerified!==null||settings.customContentNotVerified!==null||settings.titleNotVerified!==null){dialog=Dialog.create({title:settings.titleNotVerified,message:settings.notVerified,customContent:settings.customContentNotVerified,buttons:[settings.buttonAccept],escapeKeyCancels:settings.escapeKey,close:settings.buttonClose},null,function(t){if(onVerificationFailure!==null){onVerificationFailure()}})}}if(dialog!==null){dialog.show()}else{if(result){if(onVerificationSuccess!==null){onVerificationSuccess()}}else{if(onVerificationFailure!==null){onVerificationFailure()}}}}}ActionVerify.register();class ActionConfirm extends Action{static NAME="Confirm";static DEFAULTS={confirm:"Please confirm this action",customContent:null,title:"The action requires confirmation",buttonConfirm:"Confirm",buttonCancel:"Cancel",buttonClose:true,escapeKey:true};static extractOptions(t,e=null,n=null){let i=super.extractOptions(t,e,n);if(i.confirm.trim()==""){delete i.confirm}return i}static execute(t,e,n){let i=PowerButtons.getActionSettings(this,t);let o=Dialog.create({title:i.title,message:i.confirm,customContent:i.customContent,buttons:[i.buttonConfirm,i.buttonCancel],escapeKeyCancels:i.escapeKey,close:i.buttonClose},null,function(t){if(t===0){if(e!==null){e()}}else{if(n!==null){n()}}});o.show()}}ActionConfirm.register();class ActionAsyncTask extends Action{static NAME="AsyncTask";static DEFAULTS={task:null,message:"Please wait...",customContent:null,title:null,buttonCancel:"Cancel",cancel:null,header:true,footer:true};static extractOptions(t,e=null,n=null){return super.extractOptions(t,e,{task:"asynctask"})}static execute(options,onNextAction,onCancelActions){let settings=PowerButtons.getActionSettings(this,options);if(settings.task===null){console.error("The task to execute cannot be null");return}let task=null;if(typeof settings.task==="string"){task=async function(){return await eval(settings.task)}}else if(typeof settings.task==="function"){task=settings.task}else{console.error("The task to execute must be either a string or a function");return}let buttons=[];let cancelHandler=null;if(settings.cancel!==null){buttons=[settings.buttonCancel];if(typeof settings.cancel==="string"){cancelHandler=function(){eval(settings.cancel)}}else if(typeof settings.cancel==="function"){cancelHandler=settings.cancel}else{console.error("The cancel handler must be either a string or a function")}}let dialog=Dialog.create({title:settings.title,message:settings.message,customContent:settings.customContent,buttons:buttons,escapeKeyCancels:false,close:false,header:options.header!==undefined?settings.header:settings.title!==null&&settings.title!="",footer:options.footer!==undefined?settings.footer:cancelHandler!==null},function(){cancelHandler();onCancelActions()},function(t){if(onNextAction!==null){onNextAction()}});dialog.show().then(function(){task().finally(function(){dialog.hide()})})}}ActionAsyncTask.register();class ActionShowMessage extends Action{static NAME="ShowMessage";static DEFAULTS={showmessage:"This is a message",customContent:null,title:null,buttonAccept:"Accept",escapeKey:true,buttonClose:true,header:true,footer:true};static execute(t,e,n){let i=PowerButtons.getActionSettings(this,t);let o=Dialog.create({title:i.title,message:i.showmessage,customContent:i.customContent,buttons:[i.buttonAccept],escapeKeyCancels:i.escapeKey,close:i.buttonClose,header:t.header!==undefined?i.header:i.title!==null&&i.title!="",footer:t.footer!==undefined?i.footer:i.buttonAccept!==null&&i.buttonAccept!=""},null,function(t){if(e!==null){e()}});o.show()}}ActionShowMessage.register();class ActionFormset extends Action{static NAME="Formset";static DEFAULTS={form:null,fields:{}};static extractOptions(n,i=null,t=null){let e=super.extractOptions(n,i,{form:"formset"});let o={};for(let e in n.dataset){if(e.startsWith(i)){let t=e.substring(i.length);if(t===""){continue}if(t[0]!==t[0].toUpperCase()){continue}t=t.toLocaleLowerCase();o[t]=n.dataset[e]}}if(e.fields===undefined){e.fields={}}Object.assign(e.fields,o);return e}static execute(t,e,n){let i=PowerButtons.getActionSettings(this,t);let o=searchForm(i.form);if(o===null){console.error(`Form not found ${i.form}`);return}let s={};for(var l=0;l0){n.fields=a;n._formObject=i;super.initialize(t,n)}}static execute(t,e,n){let i=PowerButtons.getActionSettings(this,t);let o=false;for(let e in i.fields){try{let t=i.fields[e]();i._formObject[e].value=t}catch(t){console.error(`Error obtaining value for field ${e}: ${t}`);o=true}}if(o){n()}else{e()}}}ActionFormButton.register()})(window,document); +(function(window,document){"use strict";window.powerButtons=function(e,t=[],n={}){let i=t;if(typeof i==="string"){i=document.querySelectorAll(i)}else{if(i.length===undefined){i=[i]}}let o=null;for(let t in PowerButtons.actionsRegistered){if(e.toLocaleLowerCase()===t.toLocaleLowerCase()){o=PowerButtons.actionsRegistered[t];break}}if(o===undefined){console.error(`The action ${e} is not registered`)}else{for(let t of i){o.initialize(t,n)}}return t};window.powerButtons.version="2.0.1";window.powerButtons.plugins=function(){return Object.keys(PowerButtons.actionsRegistered)};if(window.$!==undefined){window.$.fn.powerButtons=function(t,e={}){window.powerButtons(t,this,e);return this};window.$.fn.powerButtons.version=window.powerButtons.version;window.$.fn.powerButtons.plugins=window.powerButtons.plugins}function pascalToSnake(t){return t.replace(/[A-Z]/g,t=>`_${t.toLowerCase()}`).replace(/^_*/,"")}function pascalToKebab(t){return t.replace(/[A-Z]/g,t=>`-${t.toLowerCase()}`).replace(/^-*/,"")}function snakeCaseToCamel(t){return t.replace(/-([a-z])/g,t=>t[1].toUpperCase())}function pascalToCamel(t){return t.charAt(0).toLowerCase()+t.slice(1)}function CamelToCamel(t){return t.charAt(0).toUpperCase()+t.slice(1)}function isElement(t){return t instanceof Element||t instanceof HTMLDocument}function parseBoolean(t){if(typeof t==="boolean"){return t}if(typeof t==="string"){t=t.toLowerCase();if(t==="true"||t==="yes"||t==="1"){return true}return false}return!!t}function createTag(t,e={},n=null){let i=t.split("#");let o=null;if(i.length==1){t=i[0]}else{i[1]=i[1].split(".");o=i[1][0];t=[i[0],...i[1].slice(1)].join(".")}let s=t.split(".");t=s[0];if(t===""){t="div"}if(typeof e==="string"){n=e;e={}}if(n!==null){e.innerHTML=n}if(o!==null){e.id=o}e.className=[e.className,...s.slice(1)].filter(function(t){return`${t}`.trim()!==""}).join(" ");let l=document.createElement(t);for(let t in e){if(l[t]!==undefined){l[t]=e[t]}else{l.setAttribute(t,e[t])}}return l}function appendToElement(t,...e){let n=e.filter(t=>t!==null&&t!==undefined);t.append(...n);return t}function searchForm(t){let e=null;if(t!==null){e=document.forms[t];if(e===undefined){e=document.querySelector(t);if(e===null){console.warn(`form ${t} not found`)}}}if(e!==null){if(e.tagName.toLowerCase()!=="form"){console.warn(`form ${t} is not a form`)}}return e}function getValueWithJavascriptSupport(value,context=null){if(typeof value==="function"){return value.bind(context)}if(typeof value==="string"){let internalValue=value.trim();if(internalValue.startsWith("javascript:")){try{let f=internalValue.substring(11);value=function(){return eval(f)}.bind(context)}catch(e){console.error(`Error executing javascript code ${internalValue.substring(11)}, error: ${e}`);value=null}}}return value}function promiseForEvent(t,e){let n=null;let i=new Promise(t=>{n=t});let o=function(){t.removeEventListener(e,o);n()};t.addEventListener(e,o);return i}function isEmpty(t){if(t===null||t===undefined){return true}if(t instanceof Array){return t.length===0}if(t instanceof Object){return Object.keys(t).length===0}if(typeof t==="string"){return t.trim()===""}return false}class DialogLegacy{DEFAULTS={message:"Main message",buttonCount:2};constructor(t={},e=null,n=null){this.options={...this.DEFAULTS,...t};this.buttonCount=this.options.buttons.length;this.options.buttons=["Accept","Cancel"];this.result=null;this.onButton=e;this.onHidden=n}dispose(){}show(t=null,e=null){if(t!==null){this.onButton=t}if(e!==null){this.onHidden=e}switch(this.buttonCount){case 0:case 1:alert(this.message);this.result=0;break;case 2:this.result=confirm(this.options.message)?0:1;break;default:throw`Unsupported button count ${this.buttonCount}`}if(this.onButton!==null){this.onButton(this.result,{button:this.result,text:this.options.buttons[this.result]},null)}if(this.onHidden!==null){this.onHidden(this.result,{button:this.result,text:this.options.buttons[this.result]},null)}}}class Dialog{static create(t={},e=null,n=null){if(window.bootstrap===undefined||window.bootstrap.Modal===undefined){return new DialogLegacy(t,e,n)}if(t.selector!==undefined&&t.selector!==null||t.dialogFunction!==undefined&&t.dialogFunction!==null){throw new Error("not implemented, yet")}return new Dialog(t,e,n)}DEFAULTS={title:"Title",message:"Main message",customContent:null,buttons:["Accept"],buttonClasses:["btn-primary","btn-secondary"],buttonPanelClasses:["text-end"],escapeKeyCancels:true,header:true,footer:true,body:true,close:true};dialog=null;options=null;modal=null;onButton=null;result=null;onHidden=null;onButton=null;constructor(t={},e=null,n=null){if(window.bootstrap===undefined||window.bootstrap.Modal===undefined){throw new Error("Bootstrap is required to use this class")}this.options={...this.DEFAULTS,...t};let i=[];for(let e=0;e=0){this.onHidden(this.result,{button:this.result,text:this.options.buttons[this.result]})}else{this.onHidden(this.result,null)}}}dispose(){if(this.modal!==null){this.modal.dispose();this.modal=null}this.dialog.remove();this.dialog=null}show(t=null,e=null){if(this.dialog===null){this.dialog=this._build_dialog(this.options)}if(this.modal===null){this.modal=new bootstrap.Modal(this.dialog,{backdrop:this.options.escapeKeyCancels?true:"static",keyboard:this.options.escapeKeyCancels})}this.dialog.addEventListener("hidden.bs.modal",this._hiddenHandler);this.result=null;if(t!==null){this.onButton=t}if(e!==null){this.onHidden=e}this.modal.show();return promiseForEvent(this.dialog,"shown.bs.modal")}hide(){this.modal.hide();return promiseForEvent(this.dialog,"hidden.bs.modal")}_handleButton(t,e,n){this.result=t;let i=true;if(this.onButton!==null){i=!(this.onButton(t,e,n)===false)}if(i){this.hide()}}_build_dialog(o={}){let t=null;let e=null;if(parseBoolean(o.close)){e=createTag("button.close.btn-close",{type:"button","aria-label":"Close"});e.addEventListener("click",()=>this._handleButton(-1,null,e))}if(parseBoolean(o.header)){t=createTag(".modal-header");if(o.title!==null){t.append(appendToElement(createTag(".modal-title"),appendToElement(createTag("h5",o.title))))}if(parseBoolean(o.close)){t.append(e)}}let s=[];if(o.buttons!==null){for(let i=0;it.trim()).filter(t=>t!=="").join(".");if(o.footer===false){e+=".mx-1"}if(e!==""){e="."+e}let n=createTag("button.btn"+e+".button"+i,{type:"button"},t.text);n.addEventListener("click",function(){this._handleButton(i,t,n);if(t.handler!==undefined&&t.handler!==null){t.handler(i,t,n)}}.bind(this));s.push(n)}}let n=null;if(parseBoolean(o.footer)){n=appendToElement(createTag(".modal-footer"),...s)}let i=null;if(parseBoolean(o.body)){i=createTag(".modal-body");if(t===null){if(parseBoolean(o.close)){i.append(appendToElement(createTag(".text-end"),e))}}if(o.message!==null){i.append(createTag("p.message.text-center",o.message))}if(o.customContent!==null){i.append(createTag(".custom-content.mx-auto",o.customContent))}if(n===null){let t=o.buttonPanelClasses.map(t=>t.trim()).filter(t=>t!=="").join(".");if(t!==""){t="."+t}appendToElement(i,appendToElement(createTag(".buttons"+t),...s))}}let l=appendToElement(createTag(".modal.fade",{tabindex:"-1",role:"dialog","aria-hidden":"true","data-keyboard":"false"}),appendToElement(createTag(".modal-dialog.modal-dialog-centered",{role:"document"}),appendToElement(createTag(".modal-content"),t,i,n)));return l}}function confirmDialog(t,e="this action needs confirmation",n=null,i=null,o=true){let s=new Dialog({title:e,message:t,buttons:[{text:"Cancel",class:"btn-secondary",handler:i},{text:"Confirm",class:"btn-primary",handler:n}],escapeKeyCancels:o});s.show();return s}function alertDialog(t,e="Alert",n=null){let i=new Dialog({title:e,message:t,buttons:[{text:"Accept",class:"btn-primary",handler:n}],escapeKeyCancels:true});i.show();return i}function loadingDialog(t,e=null,n=null){if(typeof e==="function"){n=e;e=null}let i=new Dialog({title:null,message:t,buttons:[{text:"Cancel",class:"btn-primary"}],customContent:e,escapeKeyCancels:false,close:false,header:false,footer:false},n);i.show();return i}if(window.powerButtons.utils===undefined){window.powerButtons.utils={}}Object.assign(window.powerButtons.utils,{confirmDialog:confirmDialog,alertDialog:alertDialog,loadingDialog:loadingDialog});class PowerButtons{static actionsRegistered={};static registerAction(t){this.actionsRegistered[t.NAME.toLowerCase()]=t;if(window.powerButtons===undefined){window.powerButtons={}}if(window.powerButtons.defaults===undefined){window.powerButtons.defaults={}}window.powerButtons.defaults[t.NAME.toLowerCase()]=Object.assign({},t.DEFAULTS)}static getActionSettings(t,e){if(this.actionsRegistered[t.NAME.toLowerCase()]===undefined){console.error(`The action ${t.NAME} is not registered`);return{}}let n={};if(window.powerButtons!==undefined&&window.powerButtons.defaults!==undefined&&window.powerButtons.defaults[t.NAME.toLowerCase()]!==undefined){n=window.powerButtons.defaults[t.NAME.toLowerCase()]}return Object.assign({},t.DEFAULTS,n,e)}static addAction(t,e={}){let n=PowerButtons.addActionSupport(t);n.appendAction(e)}static addActionSupport(t){if(t._powerButtons===undefined){t._powerButtons=new PowerButtons(t)}else{t._powerButtons.reset()}return t._powerButtons}el=null;current_action=0;actions=[];back_onclick=null;constructor(t){t._powerButtons=this;this.el=t;this.current_action=0;this.actions=[];this.back_onclick=null;if(t.onclick!==undefined&&t.onclick!==null){this.back_onclick=t.onclick;t.onclick=null}t.addEventListener("click",this.handlerClick.bind(this))}appendAction(t={}){if(t.type===undefined){throw"The type of the action is mandatory"}this.actions.push(t)}handlerClick(t){if(this.current_action>=this.actions.length){this.current_action=0;if(typeof this.back_onclick==="function"){if(!this.back_onclick()){t.preventDefault()}}return}let e=this.actions[this.current_action];t.preventDefault();t.stopImmediatePropagation();let n=function(){this.current_action++;if(this.current_action>=this.actions.length){if(this.el.click!==undefined){this.el.click()}else{this.el.dispatchEvent(new Event(t.type,t))}}else{this.el.dispatchEvent(new Event(t.type,t))}}.bind(this);let i=this.constructor.actionsRegistered[e.type];if(i===undefined){throw`The action ${e.type} is not registered`}i.execute(this.el,e,n,()=>this.reset())}reset(){this.current_action=0}static initializeAll(){Object.entries(this.actionsRegistered).forEach(([t,e])=>{e.initializeAll()})}}function init(){PowerButtons.initializeAll()}if(document.addEventListener!==undefined){document.addEventListener("DOMContentLoaded",function(t){init()})}class Action{static NAME=null;static register(){PowerButtons.registerAction(this)}static DEFAULTS={};static extractOptions(n,i=null,o=null){if(i===null){i=this.NAME.toLowerCase()}if(o===null){o={};o[i]=i}let s={};for(let e in this.DEFAULTS){let t=e;if(o[t]!==undefined){t=o[t]}else{t=i+CamelToCamel(t)}if(n.dataset[t]!==undefined){s[e]=n.dataset[t]}}return s}static initialize(t,e={}){PowerButtons.addAction(t,Object.assign({type:this.NAME.toLowerCase()},e))}static initializeAll(n=null){let i=this.NAME.toLowerCase();for(let e of document.querySelectorAll(`[data-${i}`)){let t=null;if(n===null){t=this.extractOptions(e,i)}else{t=Object.assign({},n)}this.initialize(e,t)}}static execute(t,e,n,i){throw new Error("The execute method must be implemented by the derived class")}}class ActionVerify extends Action{static NAME="Verify";static DEFAULTS={verify:null,form:null,verified:null,notVerified:"The condition for this action is not met",customContentVerified:null,customContentNotVerified:null,titleNotVerified:"The action requires verification",titleVerified:null,buttonAccept:"Accept",buttonClose:false,escapeKey:true,header:true,footer:true};static execute(el,options,onNextAction,onCancelActions){let settings=PowerButtons.getActionSettings(this,options);let result=null;let bindObject=searchForm(settings.form);if(bindObject===null){bindObject=document}try{if(typeof settings.verify==="function"){result=settings.verify.bind(bindObject)()}else if(typeof settings.verify==="string"){result=function(){return eval(settings.verify)}.bind(bindObject)()}else{result=parseBoolean(settings.verify)}}catch(e){console.error("Error executing verification function",e);result=false}let dialog=null;let onVerificationSuccess=onNextAction;let onVerificationFailure=onCancelActions;if(result){if(settings.verified!==null||settings.customContentVerified!==null||settings.titleVerified!==null){dialog=Dialog.create({title:settings.titleVerified,message:settings.verified,customContent:settings.customContentVerified,buttons:[settings.buttonAccept],escapeKeyCancels:settings.escapeKey,close:settings.buttonClose},null,function(t){if(onVerificationSuccess!==null){onVerificationSuccess()}})}}else{if(settings.notVerified!==null||settings.customContentNotVerified!==null||settings.titleNotVerified!==null){dialog=Dialog.create({title:settings.titleNotVerified,message:settings.notVerified,customContent:settings.customContentNotVerified,buttons:[settings.buttonAccept],escapeKeyCancels:settings.escapeKey,close:settings.buttonClose},null,function(t){if(onVerificationFailure!==null){onVerificationFailure()}})}}if(dialog!==null){dialog.show()}else{if(result){if(onVerificationSuccess!==null){onVerificationSuccess()}}else{if(onVerificationFailure!==null){onVerificationFailure()}}}}}ActionVerify.register();class ActionConfirm extends Action{static NAME="Confirm";static DEFAULTS={confirm:"Please confirm this action",customContent:null,title:"The action requires confirmation",buttonConfirm:"Confirm",buttonCancel:"Cancel",buttonClose:true,escapeKey:true};static extractOptions(t,e=null,n=null){let i=super.extractOptions(t,e,n);if(i.confirm.trim()==""){delete i.confirm}return i}static execute(t,e,n,i){let o=PowerButtons.getActionSettings(this,e);let s=Dialog.create({title:o.title,message:o.confirm,customContent:o.customContent,buttons:[o.buttonConfirm,o.buttonCancel],escapeKeyCancels:o.escapeKey,close:o.buttonClose},null,function(t){if(t===0){if(n!==null){n()}}else{if(i!==null){i()}}});s.show()}}ActionConfirm.register();class ActionAsyncTask extends Action{static NAME="AsyncTask";static DEFAULTS={task:null,message:"Please wait...",customContent:null,title:null,buttonCancel:"Cancel",cancel:null,header:true,footer:true};static extractOptions(t,e=null,n=null){return super.extractOptions(t,e,{task:"asynctask"})}static execute(el,options,onNextAction,onCancelActions){let settings=PowerButtons.getActionSettings(this,options);if(settings.task===null){console.error("The task to execute cannot be null");return}let task=null;if(typeof settings.task==="string"){task=async function(){return await eval(settings.task)}}else if(typeof settings.task==="function"){task=settings.task}else{console.error("The task to execute must be either a string or a function");return}let buttons=[];let cancelHandler=null;if(settings.cancel!==null){buttons=[settings.buttonCancel];if(typeof settings.cancel==="string"){cancelHandler=function(){eval(settings.cancel)}}else if(typeof settings.cancel==="function"){cancelHandler=settings.cancel}else{console.error("The cancel handler must be either a string or a function")}}let dialog=Dialog.create({title:settings.title,message:settings.message,customContent:settings.customContent,buttons:buttons,escapeKeyCancels:false,close:false,header:options.header!==undefined?settings.header:settings.title!==null&&settings.title!="",footer:options.footer!==undefined?settings.footer:cancelHandler!==null},function(){cancelHandler();onCancelActions()},function(t){if(onNextAction!==null){onNextAction()}});dialog.show().then(function(){task().finally(function(){dialog.hide()})})}}ActionAsyncTask.register();class ActionShowMessage extends Action{static NAME="ShowMessage";static DEFAULTS={showmessage:"This is a message",customContent:null,title:null,buttonAccept:"Accept",escapeKey:true,buttonClose:true,header:true,footer:true};static execute(t,e,n,i){let o=PowerButtons.getActionSettings(this,e);let s=Dialog.create({title:o.title,message:o.showmessage,customContent:o.customContent,buttons:[o.buttonAccept],escapeKeyCancels:o.escapeKey,close:o.buttonClose,header:e.header!==undefined?o.header:o.title!==null&&o.title!="",footer:e.footer!==undefined?o.footer:o.buttonAccept!==null&&o.buttonAccept!=""},null,function(t){if(n!==null){n()}});s.show()}}ActionShowMessage.register();class ActionFormset extends Action{static NAME="Formset";static DEFAULTS={form:null,fields:{}};static extractOptions(n,i=null,t=null){let e=super.extractOptions(n,i,{form:"formset"});let o={};for(let e in n.dataset){if(e.startsWith(i)){let t=e.substring(i.length);if(t===""){continue}if(t[0]!==t[0].toUpperCase()){continue}t=t.toLocaleLowerCase();o[t]=n.dataset[e]}}if(e.fields===undefined){e.fields={}}Object.assign(e.fields,o);return e}static execute(t,e,n,i){let o=PowerButtons.getActionSettings(this,e);let s=null;let l=[];let r=[];if(o.form==""){if(t.form!==null){s=t.form}else{r=Array.from(document.querySelectorAll("input")).filter(t=>t.form===null)}}else{s=searchForm(o.form);if(s===null){console.error(`Form not found ${o.form}`);return}}if(s!==null){r=Array.from(s.elements)}r.forEach(t=>{if(t.name!==""){l[t.name.toLocaleLowerCase()]=t}if(t.id!==""){l[t.id.toLocaleLowerCase()]=t}});for(let n in o.fields){if(l[n]!==undefined){let e=o.fields[n];let t=getValueWithJavascriptSupport(e,s!==null?s:l);if(typeof t==="function"){try{t=t()}catch(t){console.error(`Error executing ${e}`,t);continue}}l[n].value=t}}n()}}ActionFormset.register();class ActionFormButton extends Action{static NAME="FormButton";static DEFAULTS={formbutton:null,method:"post",formClass:"formbutton",convertCase:"none",formId:null,fields:{}};static extractOptions(n,i=null,t=null){let o=super.extractOptions(n,i,t);let s={};i=i+"Field";for(let e in n.dataset){if(e.startsWith(i)){let t=e.substring(i.length);if(t===""){continue}if(t[0]!==t[0].toUpperCase()){continue}switch(o.convertCase){case"kebab":t=pascalToKebab(t);break;case"snake":t=pascalToSnake(t);break;case"camel":t=pascalToCamel(t);break;case"pascal":break}s[t]=n.dataset[e]}}if(o.fields===undefined){o.fields={}}Object.assign(o.fields,s);return o}static initialize(t,e={}){let n=PowerButtons.getActionSettings(this,e);let i=document.createElement("form");i.method=n.method;if(!isEmpty(n.formbutton)){i.action=n.formbutton}if(n.formId!==null){i.id=n.formId}let o=n.formClass.split(" ");for(var s=0;s0){n.fields=r;n._formObject=i;super.initialize(t,n)}}static execute(t,e,n,i){let o=PowerButtons.getActionSettings(this,e);let s=false;for(let e in o.fields){try{let t=o.fields[e]();o._formObject[e].value=t}catch(t){console.error(`Error obtaining value for field ${e}: ${t}`);s=true}}if(s){i()}else{n()}}}ActionFormButton.register()})(window,document); diff --git a/dist/powerbuttons.raw.js b/dist/powerbuttons.raw.js index bd6fa97..685869d 100644 --- a/dist/powerbuttons.raw.js +++ b/dist/powerbuttons.raw.js @@ -28,7 +28,10 @@ window.powerButtons = function(pluginName, els = [], options = {}) { return els; }; -window.powerButtons.version = '2.0.0'; +window.powerButtons.version = '2.0.1'; +window.powerButtons.plugins = function() { + return Object.keys(PowerButtons.actionsRegistered); +} // Now we add the plugin to jQuery, if it has been loaded if (window.$ !== undefined) { @@ -37,6 +40,7 @@ if (window.$ !== undefined) { return this; } window.$.fn.powerButtons.version = window.powerButtons.version; + window.$.fn.powerButtons.plugins = window.powerButtons.plugins; }function pascalToSnake(str) { return str.replace(/[A-Z]/g, letter => `_${letter.toLowerCase()}`).replace(/^_*/,'') }; function pascalToKebab(str) { return str.replace(/[A-Z]/g, letter => `-${letter.toLowerCase()}`).replace(/^-*/,''); } function snakeCaseToCamel(str) { return str.replace(/-([a-z])/g, g => g[1].toUpperCase()); } @@ -635,8 +639,6 @@ Object.assign(window.powerButtons.utils, { * @param action, the action to be registered */ static registerAction(action) { - console.debug(`Registering action ${action.NAME.toLowerCase()}`); - this.actionsRegistered[action.NAME.toLowerCase()] = action; if (window.powerButtons === undefined) { @@ -794,7 +796,7 @@ Object.assign(window.powerButtons.utils, { if (action === undefined) { throw `The action ${currentActionSettings.type} is not registered`; } - action.execute(currentActionSettings, onNextAction, () => this.reset()); + action.execute(this.el, currentActionSettings, onNextAction, () => this.reset()); } /** @@ -836,6 +838,7 @@ class Action { /** * Extract the values of the options in the DEFAULTS of the class from the data attributes of the element + * * - The data attributes are assumed to be named like data-{prefix}-{option} in the HTML element, where * {prefix} is the prefix provided (or the name of the action) and {option} is the name of the option * in the DEFAULTS of the class. @@ -856,9 +859,10 @@ class Action { * This also works for renaming the data attribute, so if we set a map { 'form': 'formset' }, to fill the * key 'form' in the options object, the data attribute to use will be data-formset. * - * @param {*} el - * @param {*} prefix - * @param {*} map + * @param {HTMLElement} el, the element to extract the options from + * @param {string} prefix, the prefix to use for the data attributes (if not provided, the name of the action will be used) + * @param {object} map, the map to use to map the data attribute names to the options names (if not provided, the map will be empty) + * * @returns an object with the extracted options, with (at most) the same keys as the DEFAULTS of the class, but * only those that are defined in the data attributes of the element. */ @@ -889,8 +893,7 @@ class Action { } /** - * Searches the element for attributes in the dataset estructure that correspond to this action, and initializes - * the element, by adding the action using the class PowerButtons. + * Initializes the element, by registering the action to the PowerButtons library. * * - The data attributes are assumed to be named like data-{prefix}-{option} in the HTML element, where * {prefix} is the prefix provided (or the name of the action) and {option} is the name of the option @@ -901,10 +904,8 @@ class Action { * the keys should be those defined in the DEFAULTS of the class. * - The map is used to map the data attribute names to the options names (see `extractOptions` for more info). * - * @param {*} values, the specific values to use for the initialization appart from the default ones - * @param {*} prefix, the prefix to use for the data attribute (defaults to the name of the action in lowercase) - * @param {*} map, a map of the data attribute names to the options names; if not provided, it will be assumed - * that the data attribute names are the same as the options names using the camelCase convention + * @param {HTMLElement} el, the element to initialize + * @param {object} values, the specific values to use for the initialization appart from the default ones */ static initialize(el, values = {}) { PowerButtons.addAction(el, Object.assign({type: this.NAME.toLowerCase()}, @@ -914,7 +915,8 @@ class Action { /** * Searches for any element with a data-{name} attribute, extract the values from the dataset (if any) and initializes * it using the `initialize` method. (see `initialize` for more info). - * @param {*} values, the specific values to use for the initialization appart from the default ones; if not provided, + * + * @param {object} values, the specific values to use for the initialization appart from the default ones; if not provided, * the values will be extracted from the data attributes */ static initializeAll(values = null) { @@ -933,13 +935,12 @@ class Action { /** * Executes the action - * @param {*} options, the options to use for the execution of the action (those extracted from the data attributes and the - * user-provided ones) - * @param {*} onNextAction, the action to execute after the current one has finished (i.e. to be executed to get to the next - * action in the process) - * @param {*} onCancelActions, the actions to execute if the user cancels the current action (i.e. to stop executing actions) + * @param {HTMLElement} el, the element that triggered the action (i.e. the element that has the data-{name} attribute) + * @param {object} options, the options to use for the execution of the action (those extracted from the data attributes and the user-provided ones) + * @param {function} onNextAction, the action to execute after the current one has finished (i.e. to be executed to get to the next action in the process) + * @param {function} onCancelActions, the actions to execute if the user cancels the current action (i.e. to stop executing actions) */ - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { throw new Error("The execute method must be implemented by the derived class"); } } @@ -975,7 +976,7 @@ class ActionVerify extends Action { footer: true } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { // We merge the options with the defaults to get a valid settings object let settings = PowerButtons.getActionSettings(this, options); @@ -1079,16 +1080,7 @@ ActionVerify.register();class ActionConfirm extends Action { return options; } - - /** - * Executes the action - * @param {*} options, the options to use for the execution of the action (those extracted from the data attributes and the - * user-provided ones) - * @param {*} onNextAction, the action to execute after the current one has finished (i.e. to be executed to get to the next - * action in the process) - * @param {*} onCancelActions, the actions to execute if the user cancels the current action (i.e. to stop executing actions) - */ - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { // We merge the options with the defaults to get a valid settings object let settings = PowerButtons.getActionSettings(this, options); @@ -1146,7 +1138,7 @@ ActionConfirm.register();class ActionAsyncTask extends Action { return super.extractOptions(el, prefix, { task: "asynctask" }); } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { // We merge the options with the defaults to get a valid settings object let settings = PowerButtons.getActionSettings(this, options); @@ -1236,7 +1228,7 @@ ActionAsyncTask.register();class ActionShowMessage extends Action { footer: true } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { // We merge the options with the defaults to get a valid settings object let settings = PowerButtons.getActionSettings(this, options); @@ -1302,43 +1294,63 @@ ActionShowMessage.register();class ActionFormset extends Action { return options; } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { // We merge the options with the defaults to get a valid settings object let settings = PowerButtons.getActionSettings(this, options); - // Get the form to set using the provided selector (or name) - let formToSet = searchForm(settings.form); - if (formToSet === null) { - console.error(`Form not found ${settings.form}`); - return; + let formToSet = null; + let inputFields = []; + let elements = []; + + if (settings.form == "") { + // This is a special shortcut to + // 1. set the form to the form that contains the button + // 2. set the value of an input that is not in any form + if (el.form !== null) { + // The element is in a form, so we'll set the form to the form that contains the button + formToSet = el.form; + } else { + // The element is not in a form, so we are dealing with the inputs that are not in a form + elements = Array.from(document.querySelectorAll('input')).filter(input => input.form === null); + } + } else { + // We'll get the form to set using the provided selector (or name) + formToSet = searchForm(settings.form); + if (formToSet === null) { + console.error(`Form not found ${settings.form}`); + return; + } + } + + // If there is a form, we'll get the input fields + if (formToSet !== null) { + elements = Array.from(formToSet.elements); } + // Now use the lowercase name and id of the input fields to create a map + // // As the dataset attributes are always in camelCase, the name of the fields to set is case insensitive. So - // in the form we may have the field and in the button the attribute data-setform-name="John" - // We'll do this by creating a map of the field name in lowercase and the field name in the form, and then - // we'll use the map to set the values. + // in the form we may have the field but in the button the attribute should be data-setform-name="John" // // This mechanism restricts the existence of the same field name for different fields in the form, with different // case. For example, if we have the fields and , we'll set the value for // the last one, as the map will have the reference to it. - let nameMap = {}; - for (var i = 0; i < formToSet.elements.length; i++) { - let element = formToSet.elements[i]; - + elements.forEach(element => { if (element.name !== '') { - nameMap[element.name.toLocaleLowerCase()] = element.name; + inputFields[element.name.toLocaleLowerCase()] = element; } - // In a form, the same input is indexed both by name or by id, but assigning a value is done in the same way if (element.id !== '') { - nameMap[element.id.toLocaleLowerCase()] = element.id; + inputFields[element.id.toLocaleLowerCase()] = element; } - } - for (var field in settings.fields) { - if (nameMap[field] !== undefined) { + }); + + // Now we'll set the values of the fields + for (let field in settings.fields) { + if (inputFields[field] !== undefined) { let value = settings.fields[field]; - // Get the value with the support for javascript expressions - let result = getValueWithJavascriptSupport(value, formToSet); + // Get the value with the support for javascript expressions; if the form is null, we'll use the inputFields + let result = getValueWithJavascriptSupport(value, formToSet !== null ? formToSet : inputFields); if (typeof(result) === 'function') { try { result = result(); @@ -1347,7 +1359,7 @@ ActionShowMessage.register();class ActionFormset extends Action { continue; } } - formToSet[nameMap[field]].value = result; + inputFields[field].value = result; } } @@ -1401,7 +1413,6 @@ ActionFormset.register();class ActionFormButton extends Action { fieldname = pascalToSnake(fieldname); break; case 'camel': - console.log(fieldname, pascalToCamel(fieldname)); fieldname = pascalToCamel(fieldname); break; case 'pascal': @@ -1489,7 +1500,7 @@ ActionFormset.register();class ActionFormButton extends Action { } } - static execute(options, onNextAction, onCancelActions) { + static execute(el, options, onNextAction, onCancelActions) { // We merge the options with the defaults to get a valid settings object let settings = PowerButtons.getActionSettings(this, options); diff --git a/js/plugins/40.actionformset.js b/js/plugins/40.actionformset.js index 3aa5475..5d4a028 100644 --- a/js/plugins/40.actionformset.js +++ b/js/plugins/40.actionformset.js @@ -51,9 +51,6 @@ class ActionFormset extends Action { // This is a special shortcut to // 1. set the form to the form that contains the button // 2. set the value of an input that is not in any form - - console.debug("form is empty"); - if (el.form !== null) { // The element is in a form, so we'll set the form to the form that contains the button formToSet = el.form; diff --git a/js/plugins/80.actionformbutton.js b/js/plugins/80.actionformbutton.js index 07d4f48..c7d0e99 100644 --- a/js/plugins/80.actionformbutton.js +++ b/js/plugins/80.actionformbutton.js @@ -43,7 +43,6 @@ class ActionFormButton extends Action { fieldname = pascalToSnake(fieldname); break; case 'camel': - console.log(fieldname, pascalToCamel(fieldname)); fieldname = pascalToCamel(fieldname); break; case 'pascal': diff --git a/js/powerbuttons.js b/js/powerbuttons.js index d3b7f33..ff33650 100644 --- a/js/powerbuttons.js +++ b/js/powerbuttons.js @@ -7,8 +7,6 @@ class PowerButtons { * @param action, the action to be registered */ static registerAction(action) { - console.debug(`Registering action ${action.NAME.toLowerCase()}`); - this.actionsRegistered[action.NAME.toLowerCase()] = action; if (window.powerButtons === undefined) {