ported timestamper to a web-component

also implemented legacy methods activateTab() and setActiveTab() for new tabs widget
This commit is contained in:
ralf 2022-08-03 15:48:23 +02:00
parent 839f5f9284
commit c3673b1126
7 changed files with 137 additions and 96 deletions

View File

@ -269,15 +269,21 @@ function send_template()
$str = preg_replace('#<passwd ([^>]+)(/|></passwd)>#', '<et2-password $1></et2-password>', $str); $str = preg_replace('#<passwd ([^>]+)(/|></passwd)>#', '<et2-password $1></et2-password>', $str);
// fix <button(only)?.../> --> <et2-(button|image) (noSubmit="true")?.../> // fix <(button|buttononly|timestamper).../> --> <et2-(button|image|button-timestamp) (noSubmit="true")?.../>
$str = preg_replace_callback('#<button(only)?\s(.*?)(/|></button)>#s', function ($matches) use ($name) $str = preg_replace_callback('#<(button|buttononly|timestamper|button-timestamp)\s(.*?)(/|></(button|buttononly|timestamper|button-timestamp))>#s', function ($matches) use ($name)
{ {
$tag = 'et2-button'; $tag = 'et2-button';
$attrs = parseAttrs($matches[2]); $attrs = parseAttrs($matches[2]);
// replace buttononly tag with noSubmit="true" attribute switch ($matches[1])
if (!empty($matches[1]))
{ {
case 'buttononly': // replace buttononly tag with noSubmit="true" attribute
$attrs['noSubmit'] = 'true'; $attrs['noSubmit'] = 'true';
break;
case 'timestamper':
case 'button-timestamp':
$tag .= '-timestamp';
$attrs['background_image'] = 'true';
break;
} }
// novalidation --> noValidation // novalidation --> noValidation
if (!empty($attrs['novalidation']) && in_array($attrs['novalidation'], ['true', '1'], true)) if (!empty($attrs['novalidation']) && in_array($attrs['novalidation'], ['true', '1'], true))

View File

@ -136,9 +136,9 @@ export class Et2Button extends Et2InputWidget(SlButton)
} }
} }
constructor() constructor(...args : any[])
{ {
super(); super(...args);
// Property default values // Property default values
this.__image = ''; this.__image = '';
@ -259,7 +259,7 @@ export class Et2Button extends Et2InputWidget(SlButton)
if(!this.image) if(!this.image)
{ {
let image = this._get_default_image(this._widget_id); let image = this._get_default_image(this._widget_id);
if(image && image != this._image) if(image && image != this.__image)
{ {
this.image = image; this.image = image;
} }

View File

@ -9,63 +9,59 @@
* @copyright Nathan Gray 2017 * @copyright Nathan Gray 2017
*/ */
/*egw:uses import {et2_IInput} from "../et2_core_interfaces";
et2_button; import {date} from "../lib/date.js";
*/ import {Et2Button} from "./Et2Button";
import {Et2Tabs} from "../Layout/Et2Tabs/Et2Tabs";
import {et2_register_widget, WidgetConfig} from "./et2_core_widget";
import {et2_button} from "./et2_widget_button";
import {ClassWithAttributes} from "./et2_core_inheritance";
import {et2_no_init} from "./et2_core_common";
import {egw} from "../jsapi/egw_global";
import {et2_IInput} from "./et2_core_interfaces";
import {date} from "./lib/date.js";
/** /**
* Class which implements the "button-timestamper" XET-Tag * Class which implements the "et2-button-timestamp" tag
* *
* Clicking the button puts the current time and current user at the end of * Clicking the button puts the current time and current user at the end of
* the provided field. * the provided field.
*
* @augments et2_button
*/ */
export class et2_timestamper extends et2_button export class Et2ButtonTimestamper extends Et2Button
{ {
static readonly _attributes : any = { static get properties()
{
return {
...super.properties,
/**
* Which field to place the timestamp in
*/
target: { target: {
name: "Target field", type: String
type: "string",
default: et2_no_init,
description: "Which field to place the timestamp in"
}, },
/**
* Format for the timestamp. User is always after.
*/
format: { format: {
name: "Time format", type: String
type: "string",
default: et2_no_init,
description: "Format for the timestamp. User is always after."
}, },
/**
* Timezone. Default is user time.
*/
timezone: { timezone: {
name: "Timezone", type: String
type: "string",
default: et2_no_init,
description: "Timezone. Default is user time."
},
statustext: {
default: "Insert timestamp into description field"
}, },
/**
* Icon to use, default "timestamp"
*/
image: { image: {
default: "timestamp" type: String
}, }
background_image: { }
default: true
} }
};
target : string;
constructor(_parent, _attrs? : WidgetConfig, _child? : object) constructor(...args : any[])
{ {
// Call the inherited constructor super(...args);
super(_parent, _attrs, ClassWithAttributes.extendAttributes(et2_timestamper._attributes, _child || {}));
jQuery(this.getDOMNode()).addClass('et2_timestamper'); // Property default values
this.image = 'timestamp';
this.noSubmit = true;
this.onclick = this.stamp.bind(this);
} }
/** /**
@ -74,40 +70,21 @@ export class et2_timestamper extends et2_button
* @param _ev * @param _ev
* @returns {Boolean} * @returns {Boolean}
*/ */
click(_ev) { stamp(event: MouseEvent): boolean
// ignore click on readonly button {
if (this.options.readonly) return false; const now = new Date(new Date().toLocaleString('en-US', {
timeZone: this.timezone || egw.preference('tz')
this._insert_text();
return false;
}
private _insert_text() {
let text = "";
let now = new Date(new Date().toLocaleString('en-US', {
timeZone: this.options.timezone ? this.options.timezone : egw.preference('tz')
})); }));
let format = (this.options.format ? const format = this.format || egw.preference('dateformat') + ' ' + (egw.preference("timeformat") === "12" ? "h:ia" : "H:i")+' ';
this.options.format :
egw.preference('dateformat') + ' ' + (egw.preference("timeformat") === "12" ? "h:ia" : "H:i"))+' ';
text += date(format, now); let text = date(format, now);
// Get properly formatted user name // Get properly formatted user name
let user = parseInt(egw.user('account_id')); const user = parseInt(egw.user('account_id'));
let accounts = egw.accounts('accounts'); const account = egw.accounts('accounts').filter(option => option.value == user)[0];
for(let j = 0; j < accounts.length; j++) text += account.label + ': ';
{
if(accounts[j]["value"] === user)
{
text += accounts[j]["label"];
break;
}
}
text += ': ';
let widget = this._get_input(this.target); const widget = this._get_input(this.target);
let input = widget.input ? widget.input : widget.getDOMNode(); let input = widget.input ? widget.input : widget.getDOMNode();
if(input.context) if(input.context)
{ {
@ -134,6 +111,14 @@ export class et2_timestamper extends et2_button
pos = input.selectionStart; pos = input.selectionStart;
} }
// If on a tab, switch to that tab so user can see it
let tabbox = widget;
while(tabbox._parent && tabbox.nodeName !== 'ET2-TABBOX')
{
tabbox = tabbox._parent;
}
if (tabbox.nodeName === 'ET2-TABBOX') (<Et2Tabs>tabbox).activateTab(widget);
// If tinymce, update it // If tinymce, update it
if(tinymce) if(tinymce)
{ {
@ -144,6 +129,11 @@ export class et2_timestamper extends et2_button
// Insert the text // Insert the text
let front = (input.value).substring(0, pos); let front = (input.value).substring(0, pos);
let back = (input.value).substring(pos, input.value.length); let back = (input.value).substring(pos, input.value.length);
// for webComponent, we need to set the component value too, otherwise the change is lost!
if (typeof widget.tagName !== 'undefined')
{
widget.value = front+text+back;
}
input.value = front+text+back; input.value = front+text+back;
// Clean up a little // Clean up a little
@ -164,13 +154,6 @@ export class et2_timestamper extends et2_button
input.scrollTop = scrollPos; input.scrollTop = scrollPos;
input.focus(); input.focus();
} }
// If on a tab, switch to that tab so user can see it
let tab = widget;
while(tab._parent && tab._type != 'tabbox')
{
tab = tab._parent;
}
if (tab._type == 'tabbox') tab.activateTab(widget);
} }
private _get_input(target) private _get_input(target)
@ -201,4 +184,6 @@ export class et2_timestamper extends et2_button
return input; return input;
} }
} }
et2_register_widget(et2_timestamper, ["button-timestamp", "timestamper"]);
// @ts-ignore TypeScript is not recognizing that Et2Button is a LitElement
customElements.define("et2-button-timestamp", Et2ButtonTimestamper);

View File

@ -393,6 +393,51 @@ export class Et2Tabs extends Et2Widget(SlTabGroup)
{ {
this.setActiveTab(this._selectedIndex); this.setActiveTab(this._selectedIndex);
} }
/**
* Activate the tab containing the given widget
*
* @param {et2_widget} widget
* @return {bool} widget was found in a tab
*/
activateTab(widget)
{
let tab = widget;
while(tab._parent && tab._parent.nodeName !== 'ET2-TABBOX')
{
tab = tab._parent;
}
if (tab.nodeName === 'ET2-TAB-PANEL')
{
this.show(tab.name);
return true;
}
return false;
}
/**
* Reimplement to allow our existing function signatures too
*
* @deprecated use this.show(name : string)
* @param tab number or name of tab (Sl uses that internally with a SlTab!)
* @param options
*/
setActiveTab(tab: SlTab|String|Number, options?: {
emitEvents?: boolean;
scrollBehavior?: 'auto' | 'smooth';
})
{
if (typeof tab === 'number')
{
tab = this.getAllTabs()[tab];
return this.show(tab.panel);
}
if (typeof tab === 'string')
{
return this.show(tab);
}
return super.setActiveTab(<SlTab>tab, options);
}
} }
customElements.define("et2-tabbox", Et2Tabs); customElements.define("et2-tabbox", Et2Tabs);

View File

@ -30,6 +30,7 @@ import './Layout/Et2Tabs/Et2Tabs';
import './Layout/Et2Tabs/Et2TabPanel'; import './Layout/Et2Tabs/Et2TabPanel';
import './Et2Avatar/Et2Avatar'; import './Et2Avatar/Et2Avatar';
import './Et2Button/Et2Button'; import './Et2Button/Et2Button';
import './Et2Button/Et2ButtonTimestamper';
import './Et2Checkbox/Et2Checkbox'; import './Et2Checkbox/Et2Checkbox';
import './Et2Checkbox/Et2CheckboxReadonly'; import './Et2Checkbox/Et2CheckboxReadonly';
import './Et2Date/Et2Date'; import './Et2Date/Et2Date';
@ -121,7 +122,6 @@ import './et2_widget_html';
import './et2_widget_htmlarea'; import './et2_widget_htmlarea';
import './et2_widget_tabs'; import './et2_widget_tabs';
import './et2_widget_taglist'; import './et2_widget_taglist';
import './et2_widget_timestamper';
import './et2_widget_toolbar'; import './et2_widget_toolbar';
import './et2_widget_tree'; import './et2_widget_tree';
import './et2_widget_historylog'; import './et2_widget_historylog';

View File

@ -69,7 +69,9 @@ button.infologExtraButton:hover {
.tab_toolbar { .tab_toolbar {
position: absolute; position: absolute;
top: 8px; top: 6px;
right: 0;
z-index: 10000; /* otherwise et2-tabs gets our clicks! */
} }
.tab_toolbar > button, .tab_toolbar > et2-button { .tab_toolbar > button, .tab_toolbar > et2-button {

View File

@ -125,7 +125,10 @@ button.infologExtraButton:hover {
} }
.tab_toolbar { .tab_toolbar {
position: absolute; position: absolute;
top: 8px; top: 6px;
right: 0;
z-index: 10000;
/* otherwise et2-tabs gets our clicks! */
} }
.tab_toolbar > button, .tab_toolbar > button,
.tab_toolbar > et2-button { .tab_toolbar > et2-button {