Merge web-components branch

This commit is contained in:
nathan 2021-12-06 11:17:43 -07:00
commit e3ca8b8f7f
60 changed files with 13298 additions and 958 deletions

View File

@ -83,9 +83,7 @@
<radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_fax,&amp;hearts;"/> <radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_fax,&amp;hearts;"/>
<description for="adr_one_countryname" value="country"/> <description for="adr_one_countryname" value="country"/>
<vbox class="city_state_postcode" width="100%"> <vbox class="city_state_postcode" width="100%">
<menulist class="et2_fullWidth"> <select type="select-country" tags="true" width="100%" class="countrySelect et2_fullWidth" id="adr_one_countrycode" tabindex="15" onchange="app.addressbook.show_custom_country(this);" options="Select one,0,1" autocomplete="country"/>
<menupopup type="select-country" submit="1" tags="true" width="100%" class="countrySelect et2_fullWidth" id="adr_one_countrycode" tabindex="15" onchange="app.addressbook.show_custom_country(this);" options="Select one,0,1" autocomplete="country"/>
</menulist>
<textbox id="adr_one_countryname" class="custom_country et2_fullWidth" autocomplete="country-name"/> <textbox id="adr_one_countryname" class="custom_country et2_fullWidth" autocomplete="country-name"/>
</vbox> </vbox>
<description/> <description/>
@ -116,9 +114,9 @@
<radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_fax,&amp;hearts;"/> <radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_fax,&amp;hearts;"/>
<description for="adr_one_countryname" value="country"/> <description for="adr_one_countryname" value="country"/>
<vbox width="100%" tabindex="16"> <vbox width="100%" tabindex="16">
<menulist tabindex="16"> <select-country tabindex="16" tags="true" width="100%" class="countrySelect et2_fullWidth"
<menupopup type="select-country" tabindex="16" tags="true" width="100%" class="countrySelect et2_fullWidth" id="adr_one_countrycode" onchange="app.addressbook.show_custom_country(this);" options="Select one,0,1" autocomplete="country"/> id="adr_one_countrycode" onchange="app.addressbook.show_custom_country(this);"
</menulist> options="Select one,0,1" autocomplete="country"/>
<textbox id="adr_one_countryname" class="custom_country et2_fullWidth" tabindex="16" autocomplete="country-name"/> <textbox id="adr_one_countryname" class="custom_country et2_fullWidth" tabindex="16" autocomplete="country-name"/>
</vbox> </vbox>
<description/> <description/>
@ -137,9 +135,7 @@
<radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_car,&amp;hearts;"/> <radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_car,&amp;hearts;"/>
<description disabled="@no_tid" for="tid" value="Type"/> <description disabled="@no_tid" for="tid" value="Type"/>
<hbox width="102%"> <hbox width="102%">
<menulist disabled="@no_tid"> <select id="tid" no_lang="1" class="et2_fullWidth" onchange="1" disabled="@no_tid"/>
<menupopup id="tid" no_lang="1" class="et2_fullWidth" onchange="1"/>
</menulist>
</hbox> </hbox>
<description/> <description/>
</row> </row>
@ -247,9 +243,9 @@
<radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_pager,&amp;hearts;"/> <radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_pager,&amp;hearts;"/>
<description for="adr_two_countryname" value="country"/> <description for="adr_two_countryname" value="country"/>
<vbox width="100%" class="city_state_postcode"> <vbox width="100%" class="city_state_postcode">
<menulist class="et2_fullWidth"> <select-country tabindex="37" tags="true" width="100%" class="countrySelect et2_fullWidth"
<menupopup type="select-country" tabindex="37" tags="true" width="100%" class="countrySelect et2_fullWidth" id="adr_two_countrycode" onchange="app.addressbook.show_custom_country(this);" options="Select one,0,1" autocomplete="section-two country" /> id="adr_two_countrycode" onchange="app.addressbook.show_custom_country(this);"
</menulist> options="Select one,0,1" autocomplete="section-two country"/>
<textbox id="adr_two_countryname" class="custom_country et2_fullWidth" autocomplete="section-two country-name" /> <textbox id="adr_two_countryname" class="custom_country et2_fullWidth" autocomplete="section-two country-name" />
</vbox> </vbox>
<description/> <description/>
@ -280,9 +276,9 @@
<radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_pager,&amp;hearts;"/> <radio statustext="select phone number as prefered way of contact" id="tel_prefer" options="tel_pager,&amp;hearts;"/>
<description for="adr_two_countryname" value="country"/> <description for="adr_two_countryname" value="country"/>
<vbox width="100%"> <vbox width="100%">
<menulist> <select-country class="countrySelect et2_fullWidth" tags="true" width="100%"
<menupopup type="select-country" class="countrySelect et2_fullWidth" tags="true" width="100%" id="adr_two_countrycode" onchange="app.addressbook.show_custom_country(this);" options="Select one,0,1" autocomplete="section-two country" /> id="adr_two_countrycode" onchange="app.addressbook.show_custom_country(this);"
</menulist> options="Select one,0,1" autocomplete="section-two country"/>
<textbox id="adr_two_countryname" class="custom_country et2_fullWidth" autocomplete="section-two country-name" /> <textbox id="adr_two_countryname" class="custom_country et2_fullWidth" autocomplete="section-two country-name" />
</vbox> </vbox>
<description/> <description/>
@ -416,9 +412,9 @@
</row> </row>
<row class="dialogOperators"> <row class="dialogOperators">
<description value="Addressbook"/> <description value="Addressbook"/>
<menulist class="et2_fullWidth"> <select class="owner et2_fullWidth" statustext="Addressbook the contact should be saved to"
<menupopup class="owner" statustext="Addressbook the contact should be saved to" id="owner" no_lang="1" onchange="widget.getInstanceManager().submit(null,false,true); return false;" /> id="owner" no_lang="1"
</menulist> onchange="widget.getInstanceManager().submit(null,false,true); return false;"/>
<description/> <description/>
<description value="own sorting"/> <description value="own sorting"/>
<menulist> <menulist>

View File

@ -232,7 +232,6 @@ select#addressbook-index_col_filter\[tid\] {
#addressbook-edit { #addressbook-edit {
height: auto; height: auto;
min-height: 390px; min-height: 390px;
overflow: auto;
} }
/*##################################################################*/ /*##################################################################*/
/*Infolog*/ /*Infolog*/

View File

@ -84,7 +84,6 @@
#addressbook-edit { #addressbook-edit {
height: auto; height: auto;
min-height: 390px; min-height: 390px;
overflow: auto;
} }
/*##################################################################*/ /*##################################################################*/

View File

@ -175,15 +175,15 @@ class AdminApp extends EgwApp
if(ajax) if(ajax)
{ {
if(this.ajax_target.node.children.length) if(this.ajax_target.getDOMNode().children.length)
{ {
// Node has children already? Check for loading over an // Node has children already? Check for loading over an
// existing etemplate, and remove it first // existing etemplate, and remove it first
jQuery(this.ajax_target.node.children).each(function() { jQuery(this.ajax_target.getDOMNode().children).each(function() {
var old = etemplate2.getById(this.id); var old = etemplate2.getById(this.id);
if(old) old.clear(); if(old) old.clear();
}); });
jQuery(this.ajax_target.node).empty(); jQuery(this.ajax_target.getDOMNode()).empty();
} }
this.egw.json( this.egw.json(
framework.activeApp.getMenuaction('ajax_exec', _url), framework.activeApp.getMenuaction('ajax_exec', _url),
@ -394,7 +394,7 @@ class AdminApp extends EgwApp
if(!_data || _data.type != undefined) return; if(!_data || _data.type != undefined) return;
// Insert the content, etemplate will load into it // Insert the content, etemplate will load into it
jQuery(this.ajax_target.node).append(typeof _data === 'string' ? _data : _data[0]); jQuery(this.ajax_target.getDOMNode()).append(typeof _data === 'string' ? _data : _data[0]);
} }
/** /**

126
api/etemplate.php Normal file
View File

@ -0,0 +1,126 @@
<?php
/**
* API: loading for web-components modified eTemplate from server
*
* Usage: /egroupware/api/etemplate.php/<app>/templates/default/<name>.xet
*
* @link https://www.egroupware.org
* @author Ralf Becker <rb@egroupware-org>
* @package api
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
*/
use EGroupware\Api;
// add et2- prefix to following widgets/tags
const ADD_ET2_PREFIX_REGEXP = '#<((/?)([vh]?box|textbox|textarea|button))(/?|\s[^>]*)>#m';
// switch evtl. set output-compression off, as we cant calculate a Content-Length header with transparent compression
ini_set('zlib.output_compression', 0);
$GLOBALS['egw_info'] = array(
'flags' => array(
'currentapp' => 'api',
'noheader' => true,
// miss-use session creation callback to send the template, in case we have no session
'autocreate_session_callback' => 'send_template',
'nocachecontrol' => true,
)
);
$start = microtime(true);
include '../header.inc.php';
send_template();
function send_template()
{
$header_include = microtime(true);
// release session, as we don't need it and it blocks parallel requests
$GLOBALS['egw']->session->commit_session();
header('Content-Type: application/xml; charset=UTF-8');
//$path = EGW_SERVER_ROOT.$_SERVER['PATH_INFO'];
// check for customized template in VFS
list(, $app, , $template, $name) = explode('/', $_SERVER['PATH_INFO']);
$path = Api\Etemplate::rel2path(Api\Etemplate::relPath($app . '.' . basename($name, '.xet'), $template));
if(empty($path) || !file_exists($path) || !is_readable($path))
{
http_response_code(404);
exit;
}
/* disable caching for now, as you need to delete the cache, once you change ADD_ET2_PREFIX_REGEXP
$cache = $GLOBALS['egw_info']['server']['temp_dir'].'/egw_cache/eT2-Cache-'.$GLOBALS['egw_info']['server']['install_id'].$_SERVER['PATH_INFO'];
if (file_exists($cache) && filemtime($cache) > filemtime($path) &&
($str = file_get_contents($cache)) !== false)
{
$cache_read = microtime(true);
}
else*/
if(($str = file_get_contents($path)) !== false)
{
// fix <menulist...><menupopup type="select-*"/></menulist> --> <select type="select-*" .../>
$str = preg_replace('#<menulist([^>]*)>[\r\n\s]*<menupopup([^>]+>)[\r\n\s]*</menulist>#', '<select$1$2', $str);
// fix <textbox multiline="true" .../> --> <textarea .../> (et2-prefix and self-closing is handled below)
$str = preg_replace('#<textbox(.*?)\smultiline="true"(.*?)/>#u', '<textarea$1$2/>', $str);
// fix <buttononly.../> --> <button type="buttononly".../>
$str = preg_replace('#<buttononly\s(.*?)/>#u', '<button type="buttononly" $1/>', $str);
$str = preg_replace_callback(ADD_ET2_PREFIX_REGEXP, static function (array $matches)
{
return '<' . $matches[2] . 'et2-' . $matches[3] .
// web-components must not be self-closing (no "<et2-button .../>", but "<et2-button ...></et2-button>")
(substr($matches[4], -1) === '/' ? substr($matches[4], 0, -1) . '></et2-' . $matches[3] : $matches[4]) . '>';
}, $str);
$processing = microtime(true);
if(isset($cache) && (file_exists($cache_dir = dirname($cache)) || mkdir($cache_dir, 0755, true)))
{
file_put_contents($cache, $str);
}
}
// stop here for not existing file path-traversal for both file and cache here
if(empty($str) || strpos($path, '..') !== false)
{
http_response_code(404);
exit;
}
// headers to allow caching, egw_framework specifies etag on url to force reload, even with Expires header
Api\Session::cache_control(86400); // cache for one day
$etag = '"' . md5($str) . '"';
Header('ETag: ' . $etag);
// if servers send a If-None-Match header, response with 304 Not Modified, if etag matches
if(isset($_SERVER['HTTP_IF_NONE_MATCH']) && $_SERVER['HTTP_IF_NONE_MATCH'] === $etag)
{
header("HTTP/1.1 304 Not Modified");
exit;
}
// we run our own gzip compression, to set a correct Content-Length of the encoded content
if(function_exists('gzencode') && in_array('gzip', explode(',', $_SERVER['HTTP_ACCEPT_ENCODING']), true))
{
$gzip_start = microtime(true);
$str = gzencode($str);
header('Content-Encoding: gzip');
$gziping = microtime(true) - $gzip_start;
}
header('X-Timing: header-include=' . number_format($header_include - $GLOBALS['start'], 3) .
(empty($processing) ? ', cache-read=' . number_format($cache_read - $header_include, 3) :
', processing=' . number_format($processing - $header_include, 3)) .
(!empty($gziping) ? ', gziping=' . number_format($gziping, 3) : '') .
', total=' . number_format(microtime(true) - $GLOBALS['start'], 3)
);
// Content-Length header is important, otherwise browsers dont cache!
Header('Content-Length: ' . bytes($str));
echo $str;
exit; // stop further processing eg. redirect to login
}

View File

@ -0,0 +1,130 @@
/**
* EGroupware eTemplate2 - Box widget
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Nathan Gray
*/
import {css, html, LitElement} from "@lion/core";
import {Et2Widget} from "../Et2Widget/Et2Widget";
import {et2_IDetachedDOM} from "../et2_core_interfaces";
export class Et2Box extends Et2Widget(LitElement) implements et2_IDetachedDOM
{
static get styles()
{
return [
...super.styles,
css`
:host {
display: block;
}
:host > div {
display: flex;
flex-wrap: nowrap;
justify-content: flex-start;
align-items: stretch;
}
/* CSS for child elements */
::slotted(*) {
margin: 0px 2px;
flex: 1 0 auto;
}
::slotted([align="left"]) {
margin-right: auto;
order: -1;
}
::slotted([align="right"]) {
margin-left: auto;
order: 1;
}
`,
];
}
render()
{
return html`
<div ${this.id ? html`id="${this.id}"` : ''}>
<slot></slot>
</div> `;
}
set_label(new_label)
{
// Boxes don't have labels
}
_createNamespace() : boolean
{
return true;
}
/**
* Code for implementing et2_IDetachedDOM
*
* Individual widgets are detected and handled by the grid, but the interface is needed for this to happen
*
* @param {array} _attrs array to add further attributes to
*/
getDetachedAttributes(_attrs)
{
_attrs.push('data');
}
getDetachedNodes()
{
return [this.getDOMNode()];
}
setDetachedAttributes(_nodes, _values)
{
if(_values.data)
{
var pairs = _values.data.split(/,/g);
for(var i = 0; i < pairs.length; ++i)
{
var name_value = pairs[i].split(':');
jQuery(_nodes[0]).attr('data-' + name_value[0], name_value[1]);
}
}
}
}
customElements.define("et2-box", Et2Box);
export class Et2HBox extends Et2Box
{
static get styles()
{
return [
...super.styles,
css`
:host > div {
flex-direction: row;
}`
];
}
}
customElements.define("et2-hbox", Et2HBox);
export class Et2VBox extends Et2Box
{
static get styles()
{
return [
...super.styles,
css`
:host > div {
flex-direction: column;
}`
];
}
}
customElements.define("et2-vbox", Et2VBox);

View File

@ -0,0 +1,32 @@
/**
* Test file for Etemplate webComponent base widget Et2Box
*/
import {assert, fixture} from '@open-wc/testing';
import {Et2Box} from "../Et2Box";
import {html} from "lit-element";
describe("Box widget", () =>
{
// Reference to component under test
let element : Et2Box;
// Setup run before each test
beforeEach(async() =>
{
// Create an element to test with, and wait until it's ready
element = await fixture<Et2Box>(html`
<et2-box></et2-box>
`);
});
it('is defined', () =>
{
assert.instanceOf(element, Et2Box);
});
it('has no label', () =>
{
element.set_label("Nope");
assert.isEmpty(element.shadowRoot.querySelectorAll('.et2_label'));
})
});

View File

@ -0,0 +1,292 @@
/**
* EGroupware eTemplate2 - Button widget
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Nathan Gray
*/
import {css, html} from "@lion/core";
import {LionButton} from "@lion/button";
import {SlotMixin} from "@lion/core";
import {Et2InputWidget} from "../Et2InputWidget/Et2InputWidget";
export class Et2Button extends Et2InputWidget(SlotMixin(LionButton))
{
protected _created_icon_node : HTMLImageElement;
protected clicked : boolean = false;
private _image : string;
/**
* images to be used as background-image, if none is explicitly applied and id matches given regular expression
*/
static readonly default_background_images : object = {
save: /save(&|\]|$)/,
apply: /apply(&|\]|$)/,
cancel: /cancel(&|\]|$)/,
delete: /delete(&|\]|$)/,
discard: /discard(&|\]|$)/,
edit: /edit(&|\[\]|$)/,
next: /(next|continue)(&|\]|$)/,
finish: /finish(&|\]|$)/,
back: /(back|previous)(&|\]|$)/,
copy: /copy(&|\]|$)/,
more: /more(&|\]|$)/,
check: /(yes|check)(&|\]|$)/,
cancelled: /no(&|\]|$)/,
ok: /ok(&|\]|$)/,
close: /close(&|\]|$)/,
add: /(add(&|\]|$)|create)/ // customfields use create*
};
/**
* Classnames added automatically to buttons to set certain hover background colors
*/
static readonly default_classes : object = {
et2_button_cancel: /cancel(&|\]|$)/, // yellow
et2_button_question: /(yes|no)(&|\]|$)/, // yellow
et2_button_delete: /delete(&|\]|$)/ // red
};
static get styles()
{
return [
...super.styles,
css`
:host {
padding: 1px 8px;
/* These should probably come from somewhere else */
border-radius: 3px;
background-color: #e6e6e6;
max-width: 125px;
}
:host([readonly]) {
display: none;
}
/* Set size for icon */
::slotted([slot="icon"][src]) {
width: 20px;
padding-right: 3px;
}
::slotted([slot="icon"][src='']) {
display: none;
}
`,
];
}
static get properties()
{
return {
...super.properties,
image: {type: String}
}
}
get slots()
{
return {
...super.slots,
icon: () =>
{
return document.createElement("img");
}
}
}
constructor()
{
super();
// Property default values
this._image = '';
// Do not add icon here, no children can be added in constructor
}
connectedCallback()
{
super.connectedCallback();
//this.classList.add("et2_button")
}
set image(new_image : string)
{
let oldValue = this._image;
if(new_image.indexOf("http") >= 0)
{
this._image = new_image
}
else
{
this._image = this.egw().image(new_image, 'etemplate');
}
this.requestUpdate("image", oldValue);
}
_handleClick(event : MouseEvent) : boolean
{
// ignore click on readonly button
if(this.disabled || this.readonly)
{
return false;
}
this.clicked = true;
// Cancel buttons don't trigger the close confirmation prompt
if(this.classList.contains("et2_button_cancel"))
{
this.getInstanceManager()?.skip_close_prompt();
}
if(!super._handleClick(event))
{
this.clicked = false;
return false;
}
// Submit the form
if(this.getType() !== "buttononly")
{
return this.getInstanceManager().submit();
}
this.clicked = false;
this.getInstanceManager()?.skip_close_prompt(false);
return true;
}
/**
* Handle changes that have to happen based on changes to properties
*
*/
requestUpdate(name : PropertyKey, oldValue)
{
super.requestUpdate(name, oldValue);
// Default image & class are determined based on ID
if(name == "id" && this._widget_id)
{
// Check against current value to avoid triggering another update
if(!this.image)
{
let image = this._get_default_image(this._widget_id);
if(image != this._image)
{
this.image = image;
}
}
let default_class = this._get_default_class(this._widget_id);
if(default_class && !this.classList.contains(default_class))
{
this.classList.add(default_class);
}
}
}
render()
{
if(this.readonly)
{
return '';
}
this._iconNode.src = this._image;
return html`
<div class="button-content et2_button" id="${this._buttonId}">
<slot name="icon"></slot>
<slot>${this._label}</slot>
</div> `;
}
/**
* Get a default image for the button based on ID
*
* @param {string} check_id
*/
_get_default_image(check_id : string) : string
{
if(!check_id)
{
return "";
}
if(typeof this.image == 'undefined')
{
for(const image in Et2Button.default_background_images)
{
if(check_id.match(Et2Button.default_background_images[image]))
{
return image;
}
}
}
return "";
}
/**
* Get a default class for the button based on ID
*
* @param check_id
* @returns {string}
*/
_get_default_class(check_id)
{
if(!check_id)
{
return "";
}
for(var name in Et2Button.default_classes)
{
if(check_id.match(Et2Button.default_classes[name]))
{
return name;
}
}
return "";
}
get _iconNode() : HTMLImageElement
{
return <HTMLImageElement>(Array.from(this.children)).find(
el => (<HTMLElement>el).slot === "icon",
);
}
/**
* Implementation of the et2_IInput interface
*/
/**
* Always return false as a button is never dirty
*/
isDirty()
{
return false;
}
resetDirty()
{
}
getValue()
{
if(this.clicked)
{
return true;
}
// If "null" is returned, the result is not added to the submitted
// array.
return null;
}
}
// @ts-ignore TypeScript is not recognizing that Et2Button is a LitElement
customElements.define("et2-button", Et2Button);

View File

@ -0,0 +1,69 @@
/**
* Test file for Etemplate webComponent base widget Et2Box
*/
import {assert, fixture} from '@open-wc/testing';
import {Et2Button} from "../Et2Button";
import type {Et2Widget} from "../../Et2Widget/Et2Widget";
import {html} from "lit-element";
import * as sinon from 'sinon';
describe("Button widget", () =>
{
// Reference to component under test
let element : Et2Button;
// Setup run before each test
beforeEach(async() =>
{
// Create an element to test with, and wait until it's ready
element = await fixture<Et2Button>(html`
<et2-button label="I'm a button"></et2-button>
`);
// Stub egw()
sinon.stub(element, "egw").returns({
tooltipUnbind: () => {},
// Image always give check mark. Use data URL to avoid having to serve an actual image
image: i => ""
});
});
// Make sure it works
it('is defined', () =>
{
assert.instanceOf(element, Et2Button);
});
it('has a label', () =>
{
element.set_label("Label set");
assert.equal(element.textContent, "Label set");
})
it("click happens", () =>
{
// Setup
let clickSpy = sinon.spy();
element.onclick = clickSpy;
// Click
element.dispatchEvent(new MouseEvent("click"));
// Check for once & only once
assert(clickSpy.calledOnce, "Click only once");
})
it("gets an icon", async() =>
{
element.image = "check";
// Wait for the render to finish
await element.updateComplete;
let image = element.querySelectorAll("img");
assert.equal(image.length, 1);
assert.equal(image[0].src, element.egw().image("check"));
})
});

View File

@ -0,0 +1,151 @@
/**
* EGroupware eTemplate2 - Colorpicker widget (WebComponent)
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Hadi Nategh
*/
import {css, html, SlotMixin, render, RenderOptions} from "@lion/core";
import {LionInput} from "@lion/input";
import {Et2Widget} from "../Et2Widget/Et2Widget";
import {Et2InputWidget} from "../Et2InputWidget/Et2InputWidget";
export class Et2Colorpicker extends Et2InputWidget(Et2Widget(SlotMixin(LionInput)))
{
private cleared : boolean = true;
static get styles()
{
return [
...super.styles,
css`
:host {
display: flex;
}
.input-group__suffix{
height: 12px;
}
.input-group__container {
align-items: center
}
:host(:hover) ::slotted([slot="suffix"]) {
width: 12px;
height: 12px;
display: inline-flex;
background-image: url(pixelegg/images/close.svg);
background-size: 10px;
background-position: center;
background-repeat: no-repeat;
cursor: pointer;
}
`,
];
}
get slots()
{
return {
...super.slots,
input: () => this.__getInputNode(),
suffix: () => this.__getClearButtonNode()
}
}
constructor()
{
super();
// Override the default type of "text"
this.type = 'color';
// Bind the handlers, since slots bind without this as context
this._handleChange = this._handleChange.bind(this);
this._handleClickClear = this._handleClickClear.bind(this);
}
connectedCallback()
{
super.connectedCallback();
}
__getInputNode()
{
const renderParent = document.createElement('div');
render(
this._inputTemplate(),
renderParent,
<RenderOptions>({
scopeName: this.localName,
eventContext: this,
}),
);
return renderParent.firstElementChild;
}
_inputTemplate()
{
return html`
<input type="color" onchange="${this._handleChange}"/>
`;
}
/**
* Get the clear button node
* @returns {Element|null}
*/
__getClearButtonNode()
{
const renderParent = document.createElement('div');
render(
this._clearButtonTemplate(),
renderParent,
<RenderOptions>({
scopeName: this.localName,
eventContext: this,
}),
);
return renderParent.firstElementChild;
}
_clearButtonTemplate()
{
return html`
<span class="clear-icon" @click="${this._handleClickClear}"></span>
`;
}
_handleChange(e)
{
this.set_value(e.target.value);
}
_handleClickClear()
{
this.set_value('');
}
getValue()
{
let value = this._inputNode.value;
if (this.cleared || value === '#FFFFFF' || value === '#ffffff') {
return '';
}
return value;
}
set_value(color)
{
if(!color)
{
color = '';
}
this.cleared = !color;
this._inputNode.value = color;
}
}
customElements.define('et2-colorpicker', Et2Colorpicker);

View File

@ -0,0 +1,38 @@
/**
* Test file for Etemplate webComponent base widget Et2Colorpicker
*/
import {assert, fixture} from '@open-wc/testing';
import {Et2Colorpicker} from "../Et2Colorpicker";
import {html} from "lit-element";
describe("Colorpicker widget", () =>
{
// Reference to component under test
let element : Et2Colorpicker;
// Setup run before each test
beforeEach(async() =>
{
// Create an element to test with, and wait until it's ready
element = await fixture<Et2Colorpicker>(html`
<et2-colorpicker></et2-colorpicker>
`);
});
it('is defined', () =>
{
assert.instanceOf(element, Et2Colorpicker);
});
it('clearing value', () =>
{
// set a value
element.set_value("11111");
// trigger the clear button
element.__getClearButtonNode().dispatchEvent(new MouseEvent('click'));
assert.equal(element.getValue(), "");
});
});

View File

@ -0,0 +1,441 @@
/**
* EGroupware eTemplate2 - Date widget (WebComponent)
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Nathan Gray
*/
import {css, html} from "@lion/core";
import {LionInputDatepicker} from "@lion/input-datepicker";
import {Unparseable} from "@lion/form-core";
import {Et2InputWidget} from "../Et2InputWidget/Et2InputWidget";
/**
* Parse a date string into a Date object
* Time will be 00:00:00 UTC
*
* @param {string} dateString
* @returns {Date | undefined}
*/
export function parseDate(dateString)
{
// First try the server format
if(dateString.substr(-1) === "Z")
{
try
{
let date = new Date(dateString);
if(date instanceof Date)
{
return date;
}
}
catch(e)
{
// Nope, that didn't parse directly
}
}
let formatString = <string>(window.egw.preference("dateformat") || 'Y-m-d');
formatString = formatString.replaceAll(new RegExp('[-/\.]', 'ig'), '-');
let parsedString = "";
switch(formatString)
{
case 'd-m-Y':
parsedString = `${dateString.slice(6, 10)}/${dateString.slice(3, 5,)}/${dateString.slice(0, 2)}`;
break;
case 'm-d-Y':
parsedString = `${dateString.slice(6, 10)}/${dateString.slice(0, 2,)}/${dateString.slice(3, 5)}`;
break;
case 'Y-m-d':
parsedString = `${dateString.slice(0, 4)}/${dateString.slice(5, 7,)}/${dateString.slice(8, 10)}`;
break;
case 'Y-d-m':
parsedString = `${dateString.slice(0, 4)}/${dateString.slice(8, 10)}/${dateString.slice(5, 7)}`;
break;
case 'd-M-Y':
parsedString = `${dateString.slice(6, 10)}/${dateString.slice(3, 5,)}/${dateString.slice(0, 2)}`;
break;
default:
parsedString = '0000/00/00';
}
const [year, month, day] = parsedString.split('/').map(Number);
const parsedDate = new Date(`${year}-${month < 10 ? "0" + month : month}-${day < 10 ? "0" + day : day}T00:00:00Z`);
// Check if parsedDate is not `Invalid Date` or that the date has changed (e.g. the not existing 31.02.2020)
if(
year > 0 &&
month > 0 &&
day > 0 &&
parsedDate.getUTCDate() === day &&
parsedDate.getUTCMonth() === month - 1
)
{
return parsedDate;
}
return undefined;
}
/**
* To parse a time into a Date object
* Date will be 1970-01-01, time is in UTC to avoid browser issues
*
* @param {string} timeString
* @returns {Date | undefined}
*/
export function parseTime(timeString)
{
// First try the server format
if(timeString.substr(-1) === "Z")
{
try
{
let date = new Date(timeString);
if(date instanceof Date)
{
return date;
}
}
catch(e)
{
// Nope, that didn't parse directly
}
}
let am_pm = timeString.endsWith("pm") || timeString.endsWith("PM") ? 12 : 0;
let strippedString = timeString.replaceAll(/[^0-9:]/gi, '');
if(timeString.startsWith("12") && strippedString != timeString)
{
// 12:xx am -> 0:xx, 12:xx pm -> 12:xx
am_pm -= 12;
}
const [hour, minute] = strippedString.split(':').map(Number);
const parsedDate = new Date("1970-01-01T00:00:00Z");
parsedDate.setUTCHours(hour + am_pm);
parsedDate.setUTCMinutes(minute);
// Check if parsedDate is not `Invalid Date` or that the time has changed
if(
parsedDate.getUTCHours() === hour + am_pm &&
parsedDate.getUTCMinutes() === minute
)
{
return parsedDate;
}
return undefined;
}
/**
* To parse a date+time into an object
* Time is in UTC to avoid browser issues
*
* @param {string} dateTimeString
* @returns {Date | undefined}
*/
export function parseDateTime(dateTimeString)
{
// First try some common invalid values
if(dateTimeString === "" || dateTimeString === "0" || dateTimeString === 0)
{
return undefined;
}
// Next try server format
if(typeof dateTimeString === "string" && dateTimeString.substr(-1) === "Z" || !isNaN(dateTimeString))
{
if(!isNaN(dateTimeString) && parseInt(dateTimeString) == dateTimeString)
{
this.egw().debug("warn", "Invalid date/time string: " + dateTimeString);
dateTimeString *= 1000;
}
try
{
let date = new Date(dateTimeString);
if(date instanceof Date)
{
return date;
}
}
catch(e)
{
// Nope, that didn't parse directly
}
}
const date = parseDate(dateTimeString);
let explody = dateTimeString.split(" ");
explody.shift();
const time = parseTime(explody.join(" "));
if(typeof date === "undefined" || typeof time === "undefined")
{
return undefined;
}
date.setUTCHours(time.getUTCHours());
date.setUTCMinutes(time.getUTCMinutes());
date.setUTCSeconds(time.getUTCSeconds());
return date;
}
/**
* Format dates according to user preference
*
* @param {Date} date
* @param {import('@lion/localize/types/LocalizeMixinTypes').FormatDateOptions} [options] Intl options are available
* set 'dateFormat': "Y-m-d" to specify a particular format
* @returns {string}
*/
export function formatDate(date : Date, options = {dateFormat: ""}) : string
{
if(!date || !(date instanceof Date))
{
return "";
}
let _value = '';
// Add timezone offset back in, or formatDate will lose those hours
let formatDate = new Date(date.valueOf() - date.getTimezoneOffset() * 60 * 1000);
let dateformat = options.dateFormat || <string>window.egw.preference("dateformat") || 'Y-m-d';
var replace_map = {
d: (date.getUTCDate() < 10 ? "0" : "") + date.getUTCDate(),
m: (date.getUTCMonth() < 9 ? "0" : "") + (date.getUTCMonth() + 1),
Y: "" + date.getUTCFullYear()
}
var re = new RegExp(Object.keys(replace_map).join("|"), "gi");
_value = dateformat.replace(re, function(matched)
{
return replace_map[matched];
});
return _value;
}
/**
* Format dates according to user preference
*
* @param {Date} date
* @param {import('@lion/localize/types/LocalizeMixinTypes').FormatDateOptions} [options] Intl options are available
* set 'timeFormat': "12" to specify a particular format
* @returns {string}
*/
export function formatTime(date : Date, options = {timeFormat: ""}) : string
{
if(!date || !(date instanceof Date))
{
return "";
}
let _value = '';
let timeformat = options.timeFormat || <string>window.egw.preference("timeformat") || "24";
let hours = (timeformat == "12" && date.getUTCHours() > 12) ? (date.getUTCHours() - 12) : date.getUTCHours();
if(timeformat == "12" && hours == 0)
{
// 00:00 is 12:00 am
hours = 12;
}
_value = (timeformat == "24" && hours < 10 ? "0" : "") + hours + ":" +
(date.getUTCMinutes() < 10 ? "0" : "") + (date.getUTCMinutes()) +
(timeformat == "24" ? "" : (date.getUTCHours() < 12 ? " am" : " pm"));
return _value;
}
/**
* Format date+time according to user preference
*
* @param {Date} date
* @param {import('@lion/localize/types/LocalizeMixinTypes').FormatDateOptions} [options] Intl options are available
* set 'dateFormat': "Y-m-d", 'timeFormat': "12" to specify a particular format
* @returns {string}
*/
export function formatDateTime(date : Date, options = {dateFormat: "", timeFormat: ""}) : string
{
if(!date || !(date instanceof Date))
{
return "";
}
return formatDate(date, options) + " " + formatTime(date, options);
}
export class Et2Date extends Et2InputWidget(LionInputDatepicker)
{
static get styles()
{
return [
...super.styles,
css`
:host([focused]) ::slotted(button), :host(:hover) ::slotted(button) {
display: inline-block;
}
::slotted(.calendar_button) {
border: none;
background: transparent;
margin-left: -20px;
display: none;
}
`,
];
}
static get properties()
{
return {
...super.properties
}
}
constructor()
{
super();
this.parser = parseDate;
this.formatter = formatDate;
}
connectedCallback()
{
super.connectedCallback();
}
/**
* @param {Date} modelValue
*/
// eslint-disable-next-line class-methods-use-this
serializer(modelValue : Date)
{
// isValidDate() is hidden inside LionInputDate, and not exported
// @ts-ignore Can't call isNan(Date), but we're just checking
if(!(modelValue instanceof Date) || isNaN(modelValue))
{
return '';
}
// modelValue is localized, so we take the timezone offset in milliseconds and subtract it
// before converting it to ISO string.
const offset = modelValue.getTimezoneOffset() * 60000;
return new Date(modelValue.getTime() - offset).toJSON().replace(/\.\d{3}Z$/, 'Z');
}
set_value(value)
{
this.modelValue = this.parser(value);
}
getValue()
{
if(this.readOnly)
{
return null;
}
// The supplied value was not understandable, return null
if(this.modelValue instanceof Unparseable || !this.modelValue)
{
return null;
}
this.modelValue.setUTCHours(0);
this.modelValue.setUTCMinutes(0);
this.modelValue.setSeconds(0, 0);
return this.modelValue.toJSON();
}
get _overlayReferenceNode()
{
return this.getInputNode();
}
/**
* @override Configures OverlayMixin
* @desc overrides default configuration options for this component
* @returns {Object}
*/
_defineOverlayConfig()
{
this.hasArrow = false;
if(window.innerWidth >= 600)
{
return {
hidesOnOutsideClick: true,
placementMode: 'local',
popperConfig: {
placement: 'bottom-end',
},
};
}
return super.withBottomSheetConfig();
}
/**
* The LionCalendar shouldn't know anything about the modelValue;
* it can't handle Unparseable dates, but does handle 'undefined'
* @param {?} modelValue
* @returns {Date|undefined} a 'guarded' modelValue
*/
static __getSyncDownValue(modelValue)
{
if(!(modelValue instanceof Date))
{
return undefined;
}
const offset = modelValue.getTimezoneOffset() * 60000;
return new Date(modelValue.getTime() + offset);
}
/**
* Overriding from parent for read-only
*
* @return {TemplateResult}
* @protected
*/
// eslint-disable-next-line class-methods-use-this
_inputGroupInputTemplate()
{
if(this.readOnly)
{
return this.formattedValue;
}
else
{
return super._inputGroupInputTemplate();
}
}
/**
* Overriding parent to add class to button, and use an image instead of unicode emoji
*/
// eslint-disable-next-line class-methods-use-this
_invokerTemplate()
{
if(this.readOnly)
{
return '';
}
let img = this.egw() ? this.egw().image("calendar") || '' : '';
return html`
<button
type="button"
class="calendar_button"
@click="${this.__openCalendarOverlay}"
id="${this.__invokerId}"
aria-label="${this.msgLit('lion-input-datepicker:openDatepickerLabel')}"
title="${this.msgLit('lion-input-datepicker:openDatepickerLabel')}"
>
<img src="${img}" style="width:16px"/>
</button>
`;
}
}
// @ts-ignore TypeScript is not recognizing that Et2Date is a LitElement
customElements.define("et2-date", Et2Date);

View File

@ -0,0 +1,69 @@
/**
* EGroupware eTemplate2 - Date+Time widget (WebComponent)
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Nathan Gray
*/
import {css, html} from "@lion/core";
import {Et2Date, formatDateTime, parseDateTime} from "./Et2Date";
import {Unparseable} from "@lion/form-core";
export class Et2DateTime extends Et2Date
{
static get styles()
{
return [
...super.styles,
css`
:host([focused]) ::slotted(button), :host(:hover) ::slotted(button) {
display: inline-block;
}
::slotted(.calendar_button) {
border: none;
background: transparent;
margin-left: -20px;
display: none;
}
`,
];
}
static get properties()
{
return {
...super.properties
}
}
constructor()
{
super();
this.parser = parseDateTime;
this.formatter = formatDateTime;
}
getValue()
{
if(this.readOnly)
{
return null;
}
// The supplied value was not understandable, return null
if(this.modelValue instanceof Unparseable || !this.modelValue)
{
return null;
}
return this.modelValue.toJSON();
}
}
// @ts-ignore TypeScript is not recognizing that Et2DateTime is a LitElement
customElements.define("et2-datetime", Et2DateTime);

View File

@ -0,0 +1,125 @@
/**
* Test file for Etemplate webComponent Date
*/
import {assert, elementUpdated, fixture} from '@open-wc/testing';
import {Et2Date} from "../Et2Date";
import {html} from "lit-element";
import * as sinon from 'sinon';
describe("Date widget", () =>
{
// Reference to component under test
let element : Et2Date;
// Setup run before each test
beforeEach(async() =>
{
// Create an element to test with, and wait until it's ready
// @ts-ignore
element = await fixture<Et2Date>(html`
<et2-date label="I'm a date"></et2-date>
`);
// Stub egw()
sinon.stub(element, "egw").returns({
tooltipUnbind: () => {},
// Image always give check mark. Use data URL to avoid having to serve an actual image
image: i => ""
});
// Stub global egw for preference
// @ts-ignore
window.egw = {
preference: () => 'Y-m-d'
};
});
// Make sure it works
it('is defined', () =>
{
assert.instanceOf(element, Et2Date);
});
it('has a label', () =>
{
element.set_label("Label set");
assert.equal(element.querySelector("[slot='label']").textContent, "Label set");
})
it('Readonly does not return a value', async() =>
{
element.readOnly = true;
let test_time_string = '2008-09-22T12:00:00.000Z';
element.set_value(test_time_string);
// wait for asychronous changes to the DOM
await elementUpdated(element);
// Read-only widget returns null
assert.equal(element.getValue(), null);
});
it('No value shows no value', () =>
{
assert.equal(element.querySelector("input").textContent, "");
assert.equal(element.get_value(), null);
});
it("'0' shows no value", async() =>
{
element.set_value("0");
// wait for asychronous changes to the DOM
await elementUpdated(element);
assert.equal(element.querySelector("input").value, "");
assert.equal(element.get_value(), null);
});
const tz_list = [
{name: "America/Edmonton", offset: -600},
{name: "UTC", offset: 0},
{name: "Australia/Adelaide", offset: 630}
];
for(let tz of tz_list)
{
describe("Timezone: " + tz.name, () =>
{
// TODO: Figure out how to mock timezone...
// Stub timezone offset to return a different value
let tz_offset_stub = sinon.stub(Date.prototype, "getTimezoneOffset").returns(
tz.offset
);
let test_time_string = '2008-09-22T12:00:00.000Z';
let test_time = new Date(test_time_string);
it('Can accept a value', async() =>
{
element.set_value(test_time_string);
// wait for asychronous changes to the DOM
await elementUpdated(element);
// Widget gives time as a string so we can send to server, but zeros the time
assert.equal(element.getValue().substr(0, 11), test_time_string.substr(0, 11));
});
/* Doesn't work yet
it("Can be modified", () =>
{
element.getInputNode().value = "2008-09-22";
let event = new Event("change");
element.getInputNode().dispatchEvent(event);
// Use a Promise to wait for asychronous changes to the DOM
return Promise.resolve().then(() =>
{
assert.equal(element.getValue(), "2008-09-22T00:00:00.000Z");
});
});
*/
// Put timezone offset back
tz_offset_stub.restore();
});
}
});

View File

@ -0,0 +1,125 @@
/**
* Test file for Etemplate webComponent Date
*/
import {assert, elementUpdated, fixture, oneEvent} from '@open-wc/testing';
import {html} from "lit-element";
import * as sinon from 'sinon';
import {Et2DateTime} from "../Et2DateTime";
describe("DateTime widget", () =>
{
// Reference to component under test
let element : Et2DateTime;
// Setup run before each test
beforeEach(async() =>
{
// Create an element to test with, and wait until it's ready
// @ts-ignore
element = await fixture<Et2DateTime>(html`
<et2-datetime label="I'm a date-time"></et2-datetime>
`);
// Stub egw()
sinon.stub(element, "egw").returns({
tooltipUnbind: () => {},
// Image always give check mark. Use data URL to avoid having to serve an actual image
image: i => ""
});
// Stub global egw for preference
// @ts-ignore
window.egw = {
preference: () => 'Y-m-d'
};
});
// Make sure it works
it('is defined', () =>
{
assert.instanceOf(element, Et2DateTime);
});
it('has a label', () =>
{
element.set_label("Label set");
assert.equal(element.querySelector("[slot='label']").textContent, "Label set");
})
it('Readonly does not return a value', async() =>
{
element.readOnly = true;
let test_time_string = '2008-09-22T12:00:00.000Z';
element.set_value(test_time_string);
// wait for asychronous changes to the DOM
await elementUpdated(<Element><unknown>element);
// Read-only widget returns null
assert.equal(element.getValue(), null);
});
it('No value shows no value', () =>
{
assert.equal(element.querySelector("input").textContent, "");
assert.equal(element.get_value(), null);
});
it("'0' shows no value", async() =>
{
element.set_value("0");
// wait for asychronous changes to the DOM
await elementUpdated(element);
assert.equal(element.querySelector("input").value, "");
assert.equal(element.get_value(), null);
});
const tz_list = [
{name: "America/Edmonton", offset: -600},
{name: "UTC", offset: 0},
{name: "Australia/Adelaide", offset: 630}
];
for(let tz of tz_list)
{
describe("Timezone: " + tz.name, () =>
{
// TODO: Figure out how to mock timezone...
// Stub timezone offset to return a different value
let tz_offset_stub = sinon.stub(Date.prototype, "getTimezoneOffset").returns(
tz.offset
);
let test_time_string = '2008-09-22T12:00:00.000Z';
let test_time = new Date(test_time_string);
it('Can accept a value', async() =>
{
element.set_value(test_time_string);
// wait for asychronous changes to the DOM
await elementUpdated(element);
// Widget gives time as a string so we can send to server, but zeros the time
assert.equal(element.getValue().substr(0, 11), test_time_string.substr(0, 11));
});
/* Doesn't work yet
it("Can be modified", () =>
{
element.getInputNode().value = "2008-09-22";
let event = new Event("change");
element.getInputNode().dispatchEvent(event);
// Use a Promise to wait for asychronous changes to the DOM
return Promise.resolve().then(() =>
{
assert.equal(element.getValue(), "2008-09-22T00:00:00.000Z");
});
});
*/
// Put timezone offset back
tz_offset_stub.restore();
});
}
});

View File

@ -0,0 +1,100 @@
/**
* Test file for Etemplate date formatting
*
* For now, the best way to check if different timezones work is to change the TZ of your
* computer, then run the tests,
*/
import {assert} from '@open-wc/testing';
import {formatDate, formatTime} from "../Et2Date";
describe("Date formatting", () =>
{
// Function under test
let formatter = formatDate;
// Setup run before each test
beforeEach(async() =>
{
// Stub global egw for preference
// @ts-ignore
window.egw = {
preference: () => 'Y-m-d'
};
});
it("Handles Y-m-d", () =>
{
let test_string = '2021-09-22';
let test_date = new Date("2021-09-22T12:34:56Z");
let formatted = formatter(test_date);
assert.equal(formatted, test_string);
});
it("Handles Y.d.m", () =>
{
let test_string = '2021.22.09';
let test_date = new Date("2021-09-22T12:34:56Z");
//@ts-ignore
window.egw = {
preference: () => 'Y.d.m'
};
let formatted = formatter(test_date);
assert.equal(formatted, test_string);
});
});
describe("Time formatting", () =>
{
// Function under test
let formatter = formatTime;
// Setup run before each test
beforeEach(async() =>
{
// Stub global egw for preference
// @ts-ignore
window.egw = {
preference: () => 'Y-m-d'
};
});
it("Handles 12h", () =>
{
const test_data = {
"9:15 am": new Date('2021-09-22T09:15:00Z'),
"12:00 am": new Date('2021-09-22T00:00:00Z'),
"12:00 pm": new Date('2021-09-22T12:00:00Z'),
"5:00 pm": new Date('2021-09-22T17:00:00Z'),
};
for(let test_string of Object.keys(test_data))
{
let test_date = test_data[test_string];
let formatted = formatter(test_date, {timeFormat: "12"});
assert.equal(formatted, test_string);
}
});
it("Handles 24h", () =>
{
const test_data = {
"09:15": new Date('2021-09-22T09:15:00Z'),
"00:00": new Date('2021-09-22T00:00:00Z'),
"12:00": new Date('2021-09-22T12:00:00Z'),
"17:00": new Date('2021-09-22T17:00:00Z'),
};
for(let test_string of Object.keys(test_data))
{
let test_date = test_data[test_string];
let formatted = formatter(test_date, {timeFormat: "24"});
assert.equal(formatted, test_string);
}
});
});

View File

@ -0,0 +1,150 @@
/**
* Test file for Etemplate date parsing
*/
import {assert} from '@open-wc/testing';
import {parseDate, parseTime} from "../Et2Date";
describe("Date parsing", () =>
{
// Function under test
let parser = parseDate;
// Setup run before each test
beforeEach(async() =>
{
// Stub global egw for preference
// @ts-ignore
window.egw = {
preference: () => 'Y-m-d'
};
});
it("Handles server format", () =>
{
let test_string = '2021-09-22T19:22:00Z';
let test_date = new Date(test_string);
let parsed = parser(test_string);
// Can't compare results - different objects
//assert.equal(parsed, test_date);
assert.equal(parsed.toJSON(), test_date.toJSON());
});
it("Handles Y-m-d", () =>
{
let test_string = '2021-09-22';
let test_date = new Date("2021-09-22T00:00:00Z");
let parsed = parser(test_string);
assert.equal(parsed.toJSON(), test_date.toJSON());
});
it("Handles Y.d.m", () =>
{
let test_string = '2021.22.09';
let test_date = new Date("2021-09-22T00:00:00Z");
//@ts-ignore
window.egw = {
preference: () => 'Y.d.m'
};
let parsed = parser(test_string);
assert.equal(parsed.toJSON(), test_date.toJSON());
});
it("Handles '0'", () =>
{
let test_string = '0';
let test_date = undefined;
//@ts-ignore
window.egw = {
preference: () => 'Y.d.m'
};
let parsed = parser(test_string);
assert.equal(parsed, test_date);
});
});
describe("Time parsing", () =>
{
// Setup run before each test
beforeEach(async() =>
{
// Stub global egw for preference
// @ts-ignore
window.egw = {
preference: () => 'Y-m-d'
};
});
it("Handles 12h", () =>
{
const test_data = {
// As expected
"9:15 am": new Date('1970-01-01T09:15:00Z'),
"12:00 am": new Date('1970-01-01T00:00:00Z'),
"12:00 pm": new Date('1970-01-01T12:00:00Z'),
"5:00 pm": new Date('1970-01-01T17:00:00Z'),
"11:59 pm": new Date('1970-01-01T23:59:00Z'),
// Not valid, should be undefined
"invalid": undefined,
"23:45 pm": undefined,
"0": undefined,
"": undefined
};
for(let test_string of Object.keys(test_data))
{
let test_date = test_data[test_string];
let parsed = parseTime(test_string);
if(typeof test_date == "undefined")
{
assert.isUndefined(parsed);
}
else
{
assert.equal(parsed.toJSON(), test_date.toJSON());
}
}
});
it("Handles 24h", () =>
{
const test_data = {
"09:15": new Date('1970-01-01T09:15:00Z'),
"00:00": new Date('1970-01-01T00:00:00Z'),
"12:00": new Date('1970-01-01T12:00:00Z'),
"17:00": new Date('1970-01-01T17:00:00Z'),
"23:59": new Date('1970-01-01T23:59:00Z'),
// Not valid, should be undefined
"invalid": undefined,
"23:45 pm": undefined,
"0": undefined,
"": undefined
};
for(let test_string of Object.keys(test_data))
{
let test_date = test_data[test_string];
let parsed = parseTime(test_string);
if(typeof test_date == "undefined")
{
assert.isUndefined(parsed);
}
else
{
assert.equal(parsed.toJSON(), test_date.toJSON());
}
}
});
});

View File

@ -0,0 +1,164 @@
/**
* EGroupware eTemplate2 - Iframe widget (WebComponent)
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Hadi Nategh
*/
import {css, html, LitElement, SlotMixin} from "@lion/core";
import {Et2Widget} from "../Et2Widget/Et2Widget";
export class Et2Iframe extends Et2Widget(SlotMixin(LitElement))
{
static get styles()
{
return [
...super.styles,
css`
:host {
display: flex;
}
:host > iframe {
width: 100%;
height: 100%;
}
/* Custom CSS */
`,
];
}
static get properties()
{
return {
...super.properties,
label: {type: String},
seamless: {type: Boolean},
name: {type: String},
fullscreen: {type: Boolean},
needed: {type: Boolean},
src: {type:String},
allow: {type: String}
}
}
constructor(...args : any[])
{
super(...args);
}
get slots()
{
return {
...super.slots
};
}
connectedCallback()
{
super.connectedCallback();
}
render() {
return html`
<iframe ${this.id ? html`id="${this.id}"` : ''} allowfullscreen="${this.fullscreen}" seamless="${this.seamless}" name="${this.name}" allow="${this.allow}"></iframe>
<slot>${this._label}</slot>
`;
}
__getIframeNode()
{
return this.shadowRoot.querySelector('iframe');
}
/**
* Set the URL for the iframe
*
* Sets the src attribute to the given value
*
* @param _value String URL
*/
set_src(_value)
{
if(_value.trim() != "")
{
if(_value.trim() == 'about:blank')
{
this.__getIframeNode().src = _value;
}
else
{
// Load the new page, but display a loader
let loader = jQuery('<div class="et2_iframe loading"/>');
this.__getIframeNode().before(loader);
window.setTimeout(function() {
this.__getIframeNode().src = _value;
this.__getIframeNode().addEventListener('load',function() {
loader.remove();
});
}.bind(this),0);
}
}
}
/**
* Set name of iframe (to be used as target for links)
*
* @param _name
*/
set_name(_name)
{
this.options.name = _name;
this.__getIframeNode().attribute('name', _name);
}
set_allow (_allow)
{
this.options.allow = _allow;
this.__getIframeNode().attribute('allow', _allow);
}
/**
* Make it look like part of the containing document
*
* @param _seamless boolean
*/
set_seamless(_seamless)
{
this.options.seamless = _seamless;
this.__getIframeNode().attribute("seamless", _seamless);
}
set_value(_value)
{
if(typeof _value == "undefined") _value = "";
if(_value.trim().indexOf("http") == 0 || _value.indexOf('about:') == 0 || _value[0] == '/')
{
// Value is a URL
this.set_src(_value);
}
else
{
// Value is content
this.set_srcdoc(_value);
}
}
/**
* Sets the content of the iframe
*
* Sets the srcdoc attribute to the given value
*
* @param _value String Content of a document
*/
set_srcdoc(_value)
{
this.__getIframeNode().attribute("srcdoc", _value);
}
}
// @ts-ignore TypeScript is not recognizing that Et2Iframe is a LitElement
customElements.define("et2-iframe", Et2Iframe);

View File

@ -0,0 +1,156 @@
import {et2_IInput, et2_IInputNode} from "../et2_core_interfaces";
import {Et2Widget} from "../Et2Widget/Et2Widget";
import {dedupeMixin} from "@lion/core";
/**
* This mixin will allow any LitElement to become an Et2InputWidget
*
* Usage:
* export class Et2Button extends Et2InputWidget(LitWidget)) {...}
*/
/**
* Need to define the interface first, to get around TypeScript issues with protected/public
* This must match the public API for Et2InputWidgetClass
* @see https://lit.dev/docs/composition/mixins/#typing-the-subclass
*/
export declare class Et2InputWidgetInterface
{
readOnly : boolean;
protected value : string | number | Object;
public set_value(any) : void;
public get_value() : any;
public getValue() : any;
public isDirty() : boolean;
public resetDirty() : void;
public isValid(messages : string[]) : boolean;
}
const Et2InputWidgetMixin = (superclass) =>
{
class Et2InputWidgetClass extends Et2Widget(superclass) implements et2_IInput, et2_IInputNode
{
protected value : string | number | Object;
protected _oldValue : string | number | Object;
protected node : HTMLElement;
/** WebComponent **/
static get styles()
{
return [
...super.styles
];
}
static get properties()
{
return {
...super.properties,
// readOnly is what the property is in Lion, readonly is the attribute
readOnly: {
type: Boolean,
attribute: 'readonly',
reflect: true,
},
// readonly is what is in the templates
// I put this in here so loadWebComponent finds it when it tries to set it from the template
readonly: {
type: Boolean
}
};
}
constructor(...args : any[])
{
super(...args);
}
connectedCallback()
{
super.connectedCallback();
this.node = this.getInputNode();
}
set_value(new_value)
{
this.value = new_value;
}
get_value()
{
return this.getValue();
}
getValue()
{
return typeof this.serializedValue !== "undefined" ? this.serializedValue : this.modalValue;
}
isDirty()
{
let value = this.getValue();
if(typeof value !== typeof this._oldValue)
{
return true;
}
if(this._oldValue === value)
{
return false;
}
switch(typeof this._oldValue)
{
case "object":
if(Array.isArray(this._oldValue) &&
this._oldValue.length !== value.length
)
{
return true;
}
for(let key in this._oldValue)
{
if(this._oldValue[key] !== value[key])
{
return true;
}
}
return false;
default:
return this._oldValue != value;
}
}
resetDirty()
{
this._oldValue = this.getValue();
}
isValid(messages)
{
var ok = true;
// Check for required
if(this.options && this.options.needed && !this.options.readonly && !this.disabled &&
(this.getValue() == null || this.getValue().valueOf() == ''))
{
messages.push(this.egw().lang('Field must not be empty !!!'));
ok = false;
}
return ok;
}
getInputNode()
{
// From LionInput
return this._inputNode;
}
}
return Et2InputWidgetClass;
}
export const Et2InputWidget = dedupeMixin(Et2InputWidgetMixin);

View File

@ -0,0 +1,39 @@
/**
* Common base for easily running some standard tests on all input widgets
*
* This file should not get run on its own, extend it
*
* TODO: Not sure exactly how to make this happen yet. Maybe:
* https://github.com/mochajs/mocha/wiki/Shared-Behaviours
* <code>
* shared:
* exports.shouldBehaveLikeAUser = function() {
* it('should have .name.first', function() {
* this.user.name.first.should.equal('tobi');
* })
*
* it('should have .name.last', function() {
* this.user.name.last.should.equal('holowaychuk');
* })
*
* describe('.fullname()', function() {
* it('should return the full name', function() {
* this.user.fullname().should.equal('tobi holowaychuk');
* })
* })
* };
* test.js:
*
* var User = require('./user').User
* , Admin = require('./user').Admin
* , shared = require('./shared');
*
* describe('User', function() {
* beforeEach(function() {
* this.user = new User('tobi', 'holowaychuk');
* })
*
* shared.shouldBehaveLikeAUser();
* })
* </code>
*/

View File

@ -0,0 +1,128 @@
/**
* EGroupware eTemplate2 - Textbox widget (WebComponent)
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Nathan Gray
*/
import {css, html} from "@lion/core";
import {LionTextarea} from "@lion/textarea";
import {Et2InputWidget} from "../Et2InputWidget/Et2InputWidget";
import {Et2Widget} from "../Et2Widget/Et2Widget";
export class Et2Textarea extends Et2InputWidget(LionTextarea)
{
static get styles()
{
return [
...super.styles,
css`
:host {
display: flex;
flex-direction: column;
width: 100%;
height: 100%;
}
/* Get text area to fill its space */
.form-field__group-two {
height: 100%;
display: flex;
flex-direction: column;
}
.input-group {
height: 100%;
display: flex;
}
.input-group__container {
width: 100%;
}
`,
];
}
static get properties()
{
return {
...super.properties,
/**
* Specify the width of the text area.
* If not set, it will expand to fill the space available.
*/
width: {type: String},
/**
* Specify the height of the text area.
* If not set, it will expand to fill the space available.
*/
height: {type: String},
onkeypress: Function,
}
}
constructor()
{
super();
this.rows = "";
}
connectedCallback()
{
super.connectedCallback();
if(this._width && this._inputNode)
{
this._inputNode.style.width = this._width;
}
if(this._height && this._inputNode)
{
this._inputNode.style.height = this._height;
}
}
/**
* Use width and height attributes to affect style
* It would be better to deprecate these and just use CSS
*
* @param value
*/
set width(value)
{
let oldValue = this._width;
this._width = value;
this.requestUpdate("width", oldValue);
}
set height(value)
{
let oldValue = this._height;
this._height = value;
this.requestUpdate("height", oldValue);
}
/** Override some parent stuff to get sizing how we like it **/
setTextareaMaxHeight()
{
this._inputNode.style.maxHeight = 'inherit';
}
__initializeAutoresize()
{
return;
}
__startAutoresize()
{
return;
}
}
// @ts-ignore TypeScript is not recognizing that Et2Textarea is a LitElement
customElements.define("et2-textarea", Et2Textarea);

View File

@ -0,0 +1,49 @@
/**
* EGroupware eTemplate2 - Textbox widget (WebComponent)
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Nathan Gray
*/
import {css, html} from "@lion/core";
import {LionInput} from "@lion/input";
import {Et2InputWidget} from "../Et2InputWidget/Et2InputWidget";
export class Et2Textbox extends Et2InputWidget(LionInput)
{
static get styles()
{
return [
...super.styles,
css`
/* Custom CSS */
`,
];
}
static get properties()
{
return {
...super.properties,
onkeypress: Function,
}
}
constructor(...args : any[])
{
super(...args);
}
connectedCallback()
{
super.connectedCallback();
}
}
// @ts-ignore TypeScript is not recognizing that Et2Textbox is a LitElement
customElements.define("et2-textbox", Et2Textbox);

View File

@ -0,0 +1,32 @@
/**
* Test file for Etemplate webComponent Textbox
*/
import {assert, fixture} from '@open-wc/testing';
import {Et2Textbox} from "../Et2Textbox";
import {html} from "lit-element";
describe("Textbox widget", () =>
{
// Reference to component under test
let element : Et2Textbox;
// Setup run before each test
beforeEach(async() =>
{
// Create an element to test with, and wait until it's ready
element = await fixture<Et2Textbox>(html`
<et2-textbox></et2-textbox>
`);
});
it('is defined', () =>
{
assert.instanceOf(element, Et2Textbox);
});
it('has a label', () =>
{
element.set_label("Yay label");
assert.isEmpty(element.shadowRoot.querySelectorAll('.et2_label'));
})
});

View File

@ -0,0 +1,34 @@
/**
* Testing Et2Textbox input / values
*/
import {assert, fixture} from "@open-wc/testing";
import {html} from "lit-html";
import {Et2Textbox} from "../Et2Textbox";
describe("Textbox input / values", () =>
{
// Reference to component under test
let element : Et2Textbox;
// Setup run before each test
beforeEach(async() =>
{
// Create an element to test with, and wait until it's ready
element = await fixture<Et2Textbox>(html`
<et2-textbox></et2-textbox>
`);
});
it("Takes a value", () =>
{
/*
Complains about set_value() being missing?
let test_value = "test value";
debugger;
element.set_value(test_value);
assert(document.querySelector('input').should.have.text(test_value))
*/
})
});

File diff suppressed because it is too large Load Diff

View File

@ -1,21 +0,0 @@
/**
* EGroupware eTemplate2 - Button widget
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Nathan Gray
*/
/* Commented out while we work on rollup
import {LitElement,html} from "https://cdn.skypack.dev/lit-element";
import {SlButton} from "https://cdn.jsdelivr.net/npm/@shoelace-style/shoelace@2.0.0-beta.44/dist/shoelace.js";
export class Et2Button extends SlButton
{
size='small';
}
customElements.define("et2-button",Et2Button);
*/

View File

@ -238,7 +238,7 @@ export abstract class et2_DOMWidget extends et2_widget implements et2_IDOMNode
// Append this node at its index // Append this node at its index
var idx = this.getDOMIndex(); var idx = this.getDOMIndex();
if (idx < 0 || idx >= this.parentNode.childNodes.length - 1) if(idx < 0 || idx > this.parentNode.childNodes.length - 1)
{ {
this.parentNode.appendChild(node); this.parentNode.appendChild(node);
} }
@ -285,7 +285,17 @@ export abstract class et2_DOMWidget extends et2_widget implements et2_IDOMNode
// _node is actually a Web Component // _node is actually a Web Component
else if(_node instanceof Element) else if(_node instanceof Element)
{ {
this.getDOMNode().append(_node); if(this.getDOMNode(_node))
{
this.getDOMNode(_node).append(_node);
}
else
{
// Warn about it. This slows down loading, as it requires a second pass (loadingFinished) to get the child
// properly added.
console.warn("Legacy widget " + this.getType() + "[#" + this.options.id + "] could not handle adding a child (" +
_node.getType() + (_node.id ? "#" + _node.id : "") + ")");
}
} }
} }

View File

@ -12,73 +12,10 @@
et2_core_common; et2_core_common;
*/ */
import {egw} from "../jsapi/egw_global"; import {egw, IegwAppLocal} from "../jsapi/egw_global";
import {et2_checkType, et2_no_init, et2_validateAttrib} from "./et2_core_common"; import {et2_checkType, et2_cloneObject, et2_no_init, et2_validateAttrib} from "./et2_core_common";
import {et2_implements_registry} from "./et2_core_interfaces"; import {et2_IDOMNode, et2_IInput, et2_IInputNode, et2_implements_registry} from "./et2_core_interfaces";
// Needed for mixin
export function mix (superclass)
{
return new MixinBuilder(superclass);
}
export class MixinBuilder {
constructor(superclass) {
this.superclass = superclass;
}
with(...mixins) {
return mixins.reduce(this.applyMixins, this.superclass);
}
applyMixins(derivedConstructor: any, baseConstructor: any) {
Object.getOwnPropertyNames(baseConstructor.prototype)
.forEach(name => {
Object.defineProperty(derivedConstructor.prototype,
name,
Object.
getOwnPropertyDescriptor(
baseConstructor.prototype,
name
)
);
});
}
copyProperties(target, source) {
for (let key of Reflect.ownKeys(source)) {
if (key !== "constructor" && key !== "prototype" && key !== "name") {
let desc = Object.getOwnPropertyDescriptor(source, key);
Object.defineProperty(target, key, desc);
}
}
}
}
// This one from Typescript docs
export function applyMixins(derivedCtor: any, constructors: any[]) {
constructors.forEach((baseCtor) => {
Object.getOwnPropertyNames(baseCtor.prototype).forEach((name) => {
Object.defineProperty(
derivedCtor.prototype,
name,
Object.getOwnPropertyDescriptor(baseCtor.prototype, name) ||
Object.create(null)
);
});
});
}
/*
Experiments in using mixins to combine et2_widget & LitElement
Note that this "works", in that it mixes the code properly.
It does not work in that the resulting class does not work with et2's inheritance & class checking stuff
// This one to make TypeScript happy?
interface et2_textbox extends et2_textbox, LitElement {}
// This one to make the inheritance magic happen
applyMixins(et2_textbox, [et2_textbox,LitElement]);
// Make it a real WebComponent
customElements.define("et2-textbox",et2_textbox);
*/
export class ClassWithInterfaces export class ClassWithInterfaces
{ {
/** /**
@ -133,13 +70,19 @@ export class ClassWithAttributes extends ClassWithInterfaces
getAttribute(_name) getAttribute(_name)
{ {
if (typeof this.attributes[_name] != "undefined" && if (typeof this.attributes[_name] != "undefined" &&
!this.attributes[_name].ignore) { !this.attributes[_name].ignore)
if (typeof this["get_" + _name] == "function") { {
if (typeof this["get_" + _name] == "function")
{
return this["get_" + _name](); return this["get_" + _name]();
} else { }
else
{
return this[_name]; return this[_name];
} }
} else { }
else
{
egw.debug("error", this, "Attribute '" + _name + "' does not exist!"); egw.debug("error", this, "Attribute '" + _name + "' does not exist!");
} }
} }
@ -156,22 +99,30 @@ export class ClassWithAttributes extends ClassWithInterfaces
*/ */
setAttribute(_name, _value, _override) setAttribute(_name, _value, _override)
{ {
if (typeof this.attributes[_name] != "undefined") { if (typeof this.attributes[_name] != "undefined")
if (!this.attributes[_name].ignore) { {
if (typeof _override == "undefined") { if (!this.attributes[_name].ignore)
{
if (typeof _override == "undefined")
{
_override = true; _override = true;
} }
var val = et2_checkType(_value, this.attributes[_name].type, var val = et2_checkType(_value, this.attributes[_name].type,
_name, this); _name, this);
if (typeof this["set_" + _name] == "function") { if (typeof this["set_" + _name] == "function")
{
this["set_" + _name](val); this["set_" + _name](val);
} else if (_override || typeof this[_name] == "undefined") { }
else if (_override || typeof this[_name] == "undefined")
{
this[_name] = val; this[_name] = val;
} }
} }
} else { }
else
{
egw.debug("warn", this, "Attribute '" + _name + "' does not exist!"); egw.debug("warn", this, "Attribute '" + _name + "' does not exist!");
} }
} }
@ -186,13 +137,18 @@ export class ClassWithAttributes extends ClassWithInterfaces
static generateAttributeSet(widget, _attrs) static generateAttributeSet(widget, _attrs)
{ {
// Sanity check and validation // Sanity check and validation
for (var key in _attrs) { for (var key in _attrs)
if (typeof widget[key] != "undefined") { {
if (!widget[key].ignore) { if (typeof widget[key] != "undefined")
{
if (!widget[key].ignore)
{
_attrs[key] = et2_checkType(_attrs[key], widget[key].type, _attrs[key] = et2_checkType(_attrs[key], widget[key].type,
key, this); key, this);
} }
} else { }
else
{
// Key does not exist - delete it and issue a warning // Key does not exist - delete it and issue a warning
delete (_attrs[key]); delete (_attrs[key]);
egw.debug("warn", this, "Attribute '" + key + egw.debug("warn", this, "Attribute '" + key +
@ -201,10 +157,13 @@ export class ClassWithAttributes extends ClassWithInterfaces
} }
// Include default values or already set values for this attribute // Include default values or already set values for this attribute
for (var key in widget) { for (var key in widget)
if (typeof _attrs[key] == "undefined") { {
if (typeof _attrs[key] == "undefined")
{
var _default = widget[key]["default"]; var _default = widget[key]["default"];
if (_default == et2_no_init) { if (_default == et2_no_init)
{
_default = undefined; _default = undefined;
} }
@ -225,8 +184,10 @@ export class ClassWithAttributes extends ClassWithInterfaces
*/ */
initAttributes(_attrs) initAttributes(_attrs)
{ {
for (var key in _attrs) { for (var key in _attrs)
if (typeof this.attributes[key] != "undefined" && !this.attributes[key].ignore && !(_attrs[key] == undefined)) { {
if (typeof this.attributes[key] != "undefined" && !this.attributes[key].ignore && !(_attrs[key] == undefined))
{
this.setAttribute(key, _attrs[key], false); this.setAttribute(key, _attrs[key], false);
} }
} }
@ -237,13 +198,16 @@ export class ClassWithAttributes extends ClassWithInterfaces
let class_tree = []; let class_tree = [];
let attributes = {}; let attributes = {};
let n = 0; let n = 0;
do { do
{
n++; n++;
class_tree.push(class_prototype); class_tree.push(class_prototype);
class_prototype = Object.getPrototypeOf(class_prototype); class_prototype = Object.getPrototypeOf(class_prototype);
} while (class_prototype !== ClassWithAttributes && n < 50); }
while (class_prototype !== ClassWithAttributes && n < 50);
for (let i = class_tree.length - 1; i >= 0; i--) { for (let i = class_tree.length - 1; i >= 0; i--)
{
attributes = ClassWithAttributes.extendAttributes(attributes, class_tree[i]._attributes); attributes = ClassWithAttributes.extendAttributes(attributes, class_tree[i]._attributes);
} }
return attributes; return attributes;
@ -264,15 +228,19 @@ export class ClassWithAttributes extends ClassWithInterfaces
var result = {}; var result = {};
// Copy the new object // Copy the new object
if (typeof _new != "undefined") { if (typeof _new != "undefined")
for (var key in _new) { {
for (var key in _new)
{
result[key] = _new[key]; result[key] = _new[key];
} }
} }
// Merge the old object // Merge the old object
for (var key in _old) { for (var key in _old)
if (typeof result[key] == "undefined") { {
if (typeof result[key] == "undefined")
{
result[key] = _old[key]; result[key] = _old[key];
} }
} }
@ -283,23 +251,27 @@ export class ClassWithAttributes extends ClassWithInterfaces
var attributes = {}; var attributes = {};
// Copy the old attributes // Copy the old attributes
for (var key in _attributes) { for (var key in _attributes)
{
attributes[key] = _copyMerge({}, _attributes[key]); attributes[key] = _copyMerge({}, _attributes[key]);
} }
// Add the old attributes to the new ones. If the attributes already // Add the old attributes to the new ones. If the attributes already
// exist, they are merged. // exist, they are merged.
for (var key in _parent) { for (var key in _parent)
{
var _old = _parent[key]; var _old = _parent[key];
attributes[key] = _copyMerge(attributes[key], _old); attributes[key] = _copyMerge(attributes[key], _old);
} }
// Validate the attributes // Validate the attributes
for (var key in attributes) { for (var key in attributes)
{
et2_validateAttrib(key, attributes[key]); et2_validateAttrib(key, attributes[key]);
} }
return attributes; return attributes;
} }
} }

View File

@ -18,14 +18,16 @@ import {et2_no_init} from "./et2_core_common";
import {ClassWithAttributes} from "./et2_core_inheritance"; import {ClassWithAttributes} from "./et2_core_inheritance";
import {et2_widget, WidgetConfig} from "./et2_core_widget"; import {et2_widget, WidgetConfig} from "./et2_core_widget";
import {et2_valueWidget} from './et2_core_valueWidget' import {et2_valueWidget} from './et2_core_valueWidget'
import {et2_IInput, et2_ISubmitListener} from "./et2_core_interfaces"; import {et2_IInput, et2_IInputNode, et2_ISubmitListener} from "./et2_core_interfaces";
import {et2_compileLegacyJS} from "./et2_core_legacyJSFunctions"; import {et2_compileLegacyJS} from "./et2_core_legacyJSFunctions";
// fixing circular dependencies by only importing the type (not in compiled .js) // fixing circular dependencies by only importing the type (not in compiled .js)
import type {et2_tabbox} from "./et2_widget_tabs"; import type {et2_tabbox} from "./et2_widget_tabs";
export interface et2_input { export interface et2_input
{
getInputNode(): HTMLInputElement | HTMLElement; getInputNode(): HTMLInputElement | HTMLElement;
} }
/** /**
* et2_inputWidget derrives from et2_simpleWidget and implements the IInput * et2_inputWidget derrives from et2_simpleWidget and implements the IInput
* interface. When derriving from this class, call setDOMNode with an input * interface. When derriving from this class, call setDOMNode with an input
@ -141,10 +143,12 @@ export class et2_inputWidget extends et2_valueWidget implements et2_IInput, et2_
{ {
jQuery(node) jQuery(node)
.off('.et2_inputWidget') .off('.et2_inputWidget')
.bind("change.et2_inputWidget", this, function(e) { .bind("change.et2_inputWidget", this, function (e)
{
e.data.change.call(e.data, this); e.data.change.call(e.data, this);
}) })
.bind("focus.et2_inputWidget", this, function(e) { .bind("focus.et2_inputWidget", this, function (e)
{
e.data.focus.call(e.data, this); e.data.focus.call(e.data, this);
}); });
} }
@ -180,7 +184,9 @@ export class et2_inputWidget extends et2_valueWidget implements et2_IInput, et2_
if (args.indexOf(this) == -1) args.push(this); if (args.indexOf(this) == -1) args.push(this);
return this.onchange.apply(this, args); return this.onchange.apply(this, args);
} else { }
else
{
return (et2_compileLegacyJS(this.options.onchange, this, _node))(); return (et2_compileLegacyJS(this.options.onchange, this, _node))();
} }
} }
@ -249,9 +255,12 @@ export class et2_inputWidget extends et2_valueWidget implements et2_IInput, et2_
var node = this.getInputNode(); var node = this.getInputNode();
if (node) if (node)
{ {
if(_value && !this.options.readonly) { if (_value && !this.options.readonly)
{
jQuery(node).attr("required", "required"); jQuery(node).attr("required", "required");
} else { }
else
{
node.removeAttribute("required"); node.removeAttribute("required");
} }
} }
@ -382,4 +391,3 @@ export class et2_inputWidget extends et2_valueWidget implements et2_IInput, et2_
return valid; return valid;
} }
} }

View File

@ -71,7 +71,8 @@ export function et2_compileLegacyJS(_code, _widget, _context)
} }
} }
if (typeof parent[existing_func] === "function") { if (typeof parent[existing_func] === "function") {
return parent[existing_func]; // Bind the object so no matter what happens, context is correct
return parent[existing_func].bind(parent);
} }
} }

View File

@ -22,10 +22,10 @@ import {egw, IegwAppLocal} from "../jsapi/egw_global";
import {et2_cloneObject, et2_csvSplit} from "./et2_core_common"; import {et2_cloneObject, et2_csvSplit} from "./et2_core_common";
import {et2_compileLegacyJS} from "./et2_core_legacyJSFunctions"; import {et2_compileLegacyJS} from "./et2_core_legacyJSFunctions";
import {et2_IDOMNode, et2_IInputNode} from "./et2_core_interfaces"; import {et2_IDOMNode, et2_IInputNode} from "./et2_core_interfaces";
import {loadWebComponent} from "./Et2Widget/Et2Widget";
// fixing circular dependencies by only importing type // fixing circular dependencies by only importing type
import type {et2_container} from "./et2_core_baseWidget"; import type {et2_container} from "./et2_core_baseWidget";
import type {et2_inputWidget, et2_input} from "./et2_core_inputWidget"; import type {et2_inputWidget} from "./et2_core_inputWidget";
import {decorateLanguageService} from "ts-lit-plugin/lib/decorate-language-service";
/** /**
* The registry contains all XML tag names and the corresponding widget * The registry contains all XML tag names and the corresponding widget
@ -119,10 +119,12 @@ export function et2_createWidget(_name : string, _attrs : object, _parent? : any
// make et2_createWidget publicly available as we need to call it from stylite/js/gantt.js (maybe others) // make et2_createWidget publicly available as we need to call it from stylite/js/gantt.js (maybe others)
if (typeof window.et2_createWidget === 'undefined') window['et2_createWidget'] = et2_createWidget; if (typeof window.et2_createWidget === 'undefined') window['et2_createWidget'] = et2_createWidget;
export interface WidgetConfig { export interface WidgetConfig
{
type?: string; type?: string;
readonly?: boolean; readonly?: boolean;
width?: number; width?: number;
[propName: string]: any; [propName: string]: any;
} }
@ -214,15 +216,18 @@ export class et2_widget extends ClassWithAttributes
this.attributes = ClassWithAttributes.extendAttributes(et2_widget._attributes, _child || {}); this.attributes = ClassWithAttributes.extendAttributes(et2_widget._attributes, _child || {});
// Check whether all attributes are available // Check whether all attributes are available
if (typeof _parent == "undefined") { if (typeof _parent == "undefined")
{
_parent = null; _parent = null;
} }
if (typeof _attrs == "undefined") { if (typeof _attrs == "undefined")
{
_attrs = {}; _attrs = {};
} }
if (_attrs.attributes) { if (_attrs.attributes)
{
jQuery.extend(_attrs, _attrs.attributes); jQuery.extend(_attrs, _attrs.attributes);
} }
// Initialize all important parameters // Initialize all important parameters
@ -233,22 +238,26 @@ export class et2_widget extends ClassWithAttributes
this.id = _attrs["id"]; this.id = _attrs["id"];
// Add this widget to the given parent widget // Add this widget to the given parent widget
if (_parent != null) { if (_parent != null)
{
_parent.addChild(this); _parent.addChild(this);
} }
// The supported widget classes array defines a whitelist for all widget // The supported widget classes array defines a whitelist for all widget
// classes or interfaces child widgets have to support. // classes or interfaces child widgets have to support.
this.supportedWidgetClasses = [et2_widget]; this.supportedWidgetClasses = [et2_widget, HTMLElement];
if (_attrs["id"]) { if (_attrs["id"])
{
// Create a namespace for this object // Create a namespace for this object
if (this._createNamespace()) { if (this._createNamespace())
{
this.checkCreateNamespace(_attrs); this.checkCreateNamespace(_attrs);
} }
} }
if (this.id) { if (this.id)
{
//this.id = this.id.replace(/\[/g,'&#x5B;').replace(/]/g,'&#x5D;'); //this.id = this.id.replace(/\[/g,'&#x5B;').replace(/]/g,'&#x5D;');
} }
@ -271,18 +280,22 @@ export class et2_widget extends ClassWithAttributes
destroy() destroy()
{ {
// Call the destructor of all children // Call the destructor of all children
for (var i = this._children.length - 1; i >= 0; i--) { for (var i = this._children.length - 1; i >= 0; i--)
{
this._children[i].destroy(); this._children[i].destroy();
} }
// Remove this element from the parent, if it exists // Remove this element from the parent, if it exists
if (typeof this._parent != "undefined" && this._parent !== null) { if (typeof this._parent != "undefined" && this._parent !== null)
{
this._parent.removeChild(this); this._parent.removeChild(this);
} }
// Free the array managers if they belong to this widget // Free the array managers if they belong to this widget
for (var key in this._mgrs) { for (var key in this._mgrs)
if (this._mgrs[key] && this._mgrs[key].owner == this) { {
if (this._mgrs[key] && this._mgrs[key].owner == this)
{
this._mgrs[key].destroy(); this._mgrs[key].destroy();
} }
} }
@ -308,7 +321,8 @@ export class et2_widget extends ClassWithAttributes
clone(_parent) clone(_parent)
{ {
// Default _parent to null // Default _parent to null
if (typeof _parent == "undefined") { if (typeof _parent == "undefined")
{
_parent = null; _parent = null;
} }
@ -323,12 +337,14 @@ export class et2_widget extends ClassWithAttributes
assign(_obj) assign(_obj)
{ {
if (typeof _obj._children == "undefined") { if (typeof _obj._children == "undefined")
{
this.egw().debug("log", "Foo!"); this.egw().debug("log", "Foo!");
} }
// Create a clone of all child elements of the given object // Create a clone of all child elements of the given object
for (var i = 0; i < _obj._children.length; i++) { for (var i = 0; i < _obj._children.length; i++)
{
_obj._children[i].clone(this); _obj._children[i].clone(this);
} }
@ -361,9 +377,12 @@ export class et2_widget extends ClassWithAttributes
*/ */
getRoot(): et2_container getRoot(): et2_container
{ {
if (this._parent != null) { if (this._parent != null)
{
return this._parent.getRoot(); return this._parent.getRoot();
} else { }
else
{
return <et2_container><unknown>this; return <et2_container><unknown>this;
} }
} }
@ -389,15 +408,19 @@ export class et2_widget extends ClassWithAttributes
insertChild(_node: et2_widget, _idx: number) insertChild(_node: et2_widget, _idx: number)
{ {
// Check whether the node is one of the supported widget classes. // Check whether the node is one of the supported widget classes.
if (this.isOfSupportedWidgetClass(_node)) { if (this.isOfSupportedWidgetClass(_node))
{
// Remove the node from its original parent // Remove the node from its original parent
if (_node._parent) { if (_node._parent)
{
_node._parent.removeChild(_node); _node._parent.removeChild(_node);
} }
_node._parent = this; _node._parent = this;
this._children.splice(_idx, 0, _node); this._children.splice(_idx, 0, _node);
} else { }
else
{
this.egw().debug("error", "Widget " + _node._type + " is not supported by this widget class", this); this.egw().debug("error", "Widget " + _node._type + " is not supported by this widget class", this);
// throw("Widget is not supported by this widget class!"); // throw("Widget is not supported by this widget class!");
} }
@ -413,7 +436,8 @@ export class et2_widget extends ClassWithAttributes
// Retrieve the child from the child list // Retrieve the child from the child list
var idx = this._children.indexOf(_node); var idx = this._children.indexOf(_node);
if (idx >= 0) { if (idx >= 0)
{
// This element is no longer parent of the child // This element is no longer parent of the child
_node._parent = null; _node._parent = null;
@ -428,22 +452,27 @@ export class et2_widget extends ClassWithAttributes
*/ */
getWidgetById(_id): et2_widget | null getWidgetById(_id): et2_widget | null
{ {
if (this.id == _id) { if (this.id == _id)
{
return this; return this;
} }
if (!this._children) return null; if (!this._children) return null;
for (var i = 0; i < this._children.length; i++) { for (var i = 0; i < this._children.length; i++)
{
var elem = this._children[i].getWidgetById(_id); var elem = this._children[i].getWidgetById(_id);
if (elem != null) { if (elem != null)
{
return elem; return elem;
} }
} }
if (this.id && _id.indexOf('[') > -1 && this._children.length) { if (this.id && _id.indexOf('[') > -1 && this._children.length)
{
var ids = (new et2_arrayMgr()).explodeKey(_id); var ids = (new et2_arrayMgr()).explodeKey(_id);
var widget: et2_widget = this; var widget: et2_widget = this;
for (var i = 0; i < ids.length && widget !== null; i++) { for (var i = 0; i < ids.length && widget !== null; i++)
{
widget = widget.getWidgetById(ids[i]); widget = widget.getWidgetById(ids[i]);
} }
return widget; return widget;
@ -462,15 +491,18 @@ export class et2_widget extends ClassWithAttributes
*/ */
iterateOver(_callback, _context, _type?) iterateOver(_callback, _context, _type?)
{ {
if (typeof _type == "undefined") { if (typeof _type == "undefined")
{
_type = et2_widget; _type = et2_widget;
} }
if (this.isInTree() && this.instanceOf(_type)) { if (this.isInTree() && this.instanceOf(_type))
{
_callback.call(_context, this); _callback.call(_context, this);
} }
for (var i = 0; i < this._children.length; i++) { for (var i = 0; i < this._children.length; i++)
{
this._children[i].iterateOver(_callback, _context, _type); this._children[i].iterateOver(_callback, _context, _type);
} }
} }
@ -488,11 +520,13 @@ export class et2_widget extends ClassWithAttributes
*/ */
isInTree(_sender?, _vis?: boolean) isInTree(_sender?, _vis?: boolean)
{ {
if (typeof _vis == "undefined") { if (typeof _vis == "undefined")
{
_vis = true; _vis = true;
} }
if (this._parent) { if (this._parent)
{
return _vis && this._parent.isInTree(this); return _vis && this._parent.isInTree(this);
} }
@ -501,8 +535,10 @@ export class et2_widget extends ClassWithAttributes
isOfSupportedWidgetClass(_obj) isOfSupportedWidgetClass(_obj)
{ {
for (var i = 0; i < this.supportedWidgetClasses.length; i++) { for (var i = 0; i < this.supportedWidgetClasses.length; i++)
if (_obj instanceof this.supportedWidgetClasses[i]) { {
if (_obj instanceof this.supportedWidgetClasses[i])
{
return true; return true;
} }
} }
@ -521,47 +557,55 @@ export class et2_widget extends ClassWithAttributes
parseXMLAttrs(_attrsObj, _target, _proto) parseXMLAttrs(_attrsObj, _target, _proto)
{ {
// Check whether the attributes object is really existing, if not abort // Check whether the attributes object is really existing, if not abort
if (typeof _attrsObj == "undefined") { if (typeof _attrsObj == "undefined")
{
return; return;
} }
// Iterate over the given attributes and parse them // Iterate over the given attributes and parse them
var mgr = this.getArrayMgr("content"); var mgr = this.getArrayMgr("content");
for (var i = 0; i < _attrsObj.length; i++) { for (var i = 0; i < _attrsObj.length; i++)
{
var attrName = _attrsObj[i].name; var attrName = _attrsObj[i].name;
var attrValue = _attrsObj[i].value; var attrValue = _attrsObj[i].value;
// Special handling for the legacy options // Special handling for the legacy options
if (attrName == "options" && _proto.constructor.legacyOptions && _proto.constructor.legacyOptions.length > 0) { if (attrName == "options" && _proto.constructor.legacyOptions && _proto.constructor.legacyOptions.length > 0)
{
let legacy = _proto.constructor.legacyOptions || []; let legacy = _proto.constructor.legacyOptions || [];
let attrs = et2_attribute_registry[Object.getPrototypeOf(_proto).constructor.name] || {}; let attrs = et2_attribute_registry[Object.getPrototypeOf(_proto).constructor.name] || {};
// Check for modifications on legacy options here. Normal modifications // Check for modifications on legacy options here. Normal modifications
// are handled in widget constructor, but it's too late for legacy options then // are handled in widget constructor, but it's too late for legacy options then
if (_target.id && this.getArrayMgr("modifications").getEntry(_target.id)) { if (_target.id && this.getArrayMgr("modifications").getEntry(_target.id))
{
var mod: any = this.getArrayMgr("modifications").getEntry(_target.id); var mod: any = this.getArrayMgr("modifications").getEntry(_target.id);
if (typeof mod.options != "undefined") attrValue = _attrsObj[i].value = mod.options; if (typeof mod.options != "undefined") attrValue = _attrsObj[i].value = mod.options;
} }
// expand legacyOptions with content // expand legacyOptions with content
if (attrValue.charAt(0) == '@' || attrValue.indexOf('$') != -1) { if (attrValue.charAt(0) == '@' || attrValue.indexOf('$') != -1)
{
attrValue = mgr.expandName(attrValue); attrValue = mgr.expandName(attrValue);
} }
// Parse the legacy options (as a string, other types not allowed) // Parse the legacy options (as a string, other types not allowed)
var splitted = et2_csvSplit(attrValue + ""); var splitted = et2_csvSplit(attrValue + "");
for (var j = 0; j < splitted.length && j < legacy.length; j++) { for (var j = 0; j < splitted.length && j < legacy.length; j++)
{
// Blank = not set, unless there's more legacy options provided after // Blank = not set, unless there's more legacy options provided after
if (splitted[j].trim().length === 0 && legacy.length >= splitted.length) continue; if (splitted[j].trim().length === 0 && legacy.length >= splitted.length) continue;
// Check to make sure we don't overwrite a current option with a legacy option // Check to make sure we don't overwrite a current option with a legacy option
if (typeof _target[legacy[j]] === "undefined") { if (typeof _target[legacy[j]] === "undefined")
{
attrValue = splitted[j]; attrValue = splitted[j];
/** /**
If more legacy options than expected, stuff them all in the last legacy option If more legacy options than expected, stuff them all in the last legacy option
Some legacy options take a comma separated list. Some legacy options take a comma separated list.
*/ */
if (j == legacy.length - 1 && splitted.length > legacy.length) { if (j == legacy.length - 1 && splitted.length > legacy.length)
{
attrValue = splitted.slice(j); attrValue = splitted.slice(j);
} }
@ -569,26 +613,37 @@ export class et2_widget extends ClassWithAttributes
// If the attribute is marked as boolean, parse the // If the attribute is marked as boolean, parse the
// expression as bool expression. // expression as bool expression.
if (attr.type == "boolean") { if (attr.type == "boolean")
{
attrValue = mgr.parseBoolExpression(attrValue); attrValue = mgr.parseBoolExpression(attrValue);
} else if (typeof attrValue != "object") { }
else if (typeof attrValue != "object")
{
attrValue = mgr.expandName(attrValue); attrValue = mgr.expandName(attrValue);
} }
_target[legacy[j]] = attrValue; _target[legacy[j]] = attrValue;
} }
} }
} else if (attrName == "readonly" && typeof _target[attrName] != "undefined") { }
else if (attrName == "readonly" && typeof _target[attrName] != "undefined")
{
// do NOT overwrite already evaluated readonly attribute // do NOT overwrite already evaluated readonly attribute
} else { }
else
{
let attrs = et2_attribute_registry[_proto.constructor.name] || {}; let attrs = et2_attribute_registry[_proto.constructor.name] || {};
if (mgr != null && typeof attrs[attrName] != "undefined") { if (mgr != null && typeof attrs[attrName] != "undefined")
{
var attr = attrs[attrName]; var attr = attrs[attrName];
// If the attribute is marked as boolean, parse the // If the attribute is marked as boolean, parse the
// expression as bool expression. // expression as bool expression.
if (attr.type == "boolean") { if (attr.type == "boolean")
{
attrValue = mgr.parseBoolExpression(attrValue); attrValue = mgr.parseBoolExpression(attrValue);
} else { }
else
{
attrValue = mgr.expandName(attrValue); attrValue = mgr.expandName(attrValue);
} }
} }
@ -609,20 +664,26 @@ export class et2_widget extends ClassWithAttributes
{ {
// Apply the content of the modifications array // Apply the content of the modifications array
if (this.id) { if (this.id)
if (typeof this.id != "string") { {
if (typeof this.id != "string")
{
console.log(this.id); console.log(this.id);
} }
if (this.getArrayMgr("modifications")) { if (this.getArrayMgr("modifications"))
{
var data = this.getArrayMgr("modifications").getEntry(this.id); var data = this.getArrayMgr("modifications").getEntry(this.id);
// Check for already inside namespace // Check for already inside namespace
if (this._createNamespace() && this.getArrayMgr("modifications").perspectiveData.owner == this) { if (this._createNamespace() && this.getArrayMgr("modifications").perspectiveData.owner == this)
{
data = this.getArrayMgr("modifications").data; data = this.getArrayMgr("modifications").data;
} }
if (typeof data === 'object') { if (typeof data === 'object')
for (var key in data) { {
for (var key in data)
{
_attrs[key] = data[key]; _attrs[key] = data[key];
} }
} }
@ -630,10 +691,13 @@ export class et2_widget extends ClassWithAttributes
} }
// Translate the attributes // Translate the attributes
for (var key in _attrs) { for (var key in _attrs)
if (_attrs[key] && typeof this.attributes[key] != "undefined") { {
if (_attrs[key] && typeof this.attributes[key] != "undefined")
{
if (this.attributes[key].translate === true || if (this.attributes[key].translate === true ||
(this.attributes[key].translate === "!no_lang" && !_attrs["no_lang"])) { (this.attributes[key].translate === "!no_lang" && !_attrs["no_lang"]))
{
let value = _attrs[key]; let value = _attrs[key];
// allow statustext to contain multiple translated sub-strings eg: {Firstname}.{Lastname} // allow statustext to contain multiple translated sub-strings eg: {Firstname}.{Lastname}
if (value.indexOf('{') !== -1) if (value.indexOf('{') !== -1)
@ -682,20 +746,26 @@ export class et2_widget extends ClassWithAttributes
// Check to see if modifications change type // Check to see if modifications change type
var modifications = this.getArrayMgr("modifications"); var modifications = this.getArrayMgr("modifications");
if (modifications && _node.getAttribute("id")) { if (modifications && _node.getAttribute("id"))
{
var entry: any = modifications.getEntry(_node.getAttribute("id")); var entry: any = modifications.getEntry(_node.getAttribute("id"));
if (entry == null) { if (entry == null)
{
// Try again, but skip the fancy stuff // Try again, but skip the fancy stuff
// TODO: Figure out why the getEntry() call doesn't always work // TODO: Figure out why the getEntry() call doesn't always work
var entry = modifications.data[_node.getAttribute("id")]; var entry = modifications.data[_node.getAttribute("id")];
if (entry) { if (entry)
{
this.egw().debug("warn", "getEntry(" + _node.getAttribute("id") + ") failed, but the data is there.", modifications, entry); this.egw().debug("warn", "getEntry(" + _node.getAttribute("id") + ") failed, but the data is there.", modifications, entry);
} else { }
else
{
// Try the root, in case a namespace got missed // Try the root, in case a namespace got missed
entry = modifications.getRoot().getEntry(_node.getAttribute("id")); entry = modifications.getRoot().getEntry(_node.getAttribute("id"));
} }
} }
if (entry && entry.type && typeof entry.type === 'string') { if (entry && entry.type && typeof entry.type === 'string')
{
_nodeName = attributes["type"] = entry.type; _nodeName = attributes["type"] = entry.type;
} }
entry = null; entry = null;
@ -703,25 +773,27 @@ export class et2_widget extends ClassWithAttributes
// if _nodeName / type-attribute contains something to expand (eg. type="@${row}[type]"), // if _nodeName / type-attribute contains something to expand (eg. type="@${row}[type]"),
// we need to expand it now as it defines the constructor and by that attributes parsed via parseXMLAttrs! // we need to expand it now as it defines the constructor and by that attributes parsed via parseXMLAttrs!
if (_nodeName.charAt(0) == '@' || _nodeName.indexOf('$') >= 0) { if (_nodeName.charAt(0) == '@' || _nodeName.indexOf('$') >= 0)
{
_nodeName = attributes["type"] = this.getArrayMgr('content').expandName(_nodeName); _nodeName = attributes["type"] = this.getArrayMgr('content').expandName(_nodeName);
} }
// Get the constructor - if the widget is readonly, use the special "_ro" // Get the constructor - if the widget is readonly, use the special "_ro"
// constructor if it is available // constructor if it is available
var constructor = et2_registry[typeof et2_registry[_nodeName] == "undefined" ? 'placeholder' : _nodeName]; var constructor = et2_registry[typeof et2_registry[_nodeName] == "undefined" ? 'placeholder' : _nodeName];
if (readonly === true && typeof et2_registry[_nodeName + "_ro"] != "undefined") { if (readonly === true && typeof et2_registry[_nodeName + "_ro"] != "undefined")
{
constructor = et2_registry[_nodeName + "_ro"]; constructor = et2_registry[_nodeName + "_ro"];
} }
if (undefined == window.customElements.get(_nodeName))
{
// Parse the attributes from the given XML attributes object // Parse the attributes from the given XML attributes object
this.parseXMLAttrs(_node.attributes, attributes, constructor.prototype); this.parseXMLAttrs(_node.attributes, attributes, constructor.prototype);
// Do an sanity check for the attributes // Do an sanity check for the attributes
ClassWithAttributes.generateAttributeSet(et2_attribute_registry[constructor.name], attributes); ClassWithAttributes.generateAttributeSet(et2_attribute_registry[constructor.name], attributes);
if(undefined == window.customElements.get(_nodeName))
{
// Creates the new widget, passes this widget as an instance and // Creates the new widget, passes this widget as an instance and
// passes the widgetType. Then it goes on loading the XML for it. // passes the widgetType. Then it goes on loading the XML for it.
var widget = new constructor(this, attributes); var widget = new constructor(this, attributes);
@ -731,30 +803,8 @@ export class et2_widget extends ClassWithAttributes
} }
else else
{ {
widget = this.loadWebComponent(_nodeName, _node, attributes); widget = loadWebComponent(_nodeName, _node, this);
if(this.addChild)
{
// webcomponent going into old et2_widget
this.addChild(widget);
} }
}
return widget;
}
/**
* Load a Web Component
* @param _nodeName
* @param _node
*/
loadWebComponent(_nodeName : string, _node, attributes : Object) : HTMLElement
{
let widget = document.createElement(_nodeName);
widget.textContent = _node.textContent;
// Apply any set attributes
_node.getAttributeNames().forEach(attribute => widget.setAttribute(attribute, attributes[attribute]));
return widget; return widget;
} }
@ -766,16 +816,20 @@ export class et2_widget extends ClassWithAttributes
loadFromXML(_node) loadFromXML(_node)
{ {
// Load the child nodes. // Load the child nodes.
for (var i = 0; i < _node.childNodes.length; i++) { for (var i = 0; i < _node.childNodes.length; i++)
{
var node = _node.childNodes[i]; var node = _node.childNodes[i];
var widgetType = node.nodeName.toLowerCase(); var widgetType = node.nodeName.toLowerCase();
if (widgetType == "#comment") { if (widgetType == "#comment")
{
continue; continue;
} }
if (widgetType == "#text") { if (widgetType == "#text")
if (node.data.replace(/^\s+|\s+$/g, '')) { {
if (node.data.replace(/^\s+|\s+$/g, ''))
{
this.loadContent(node.data); this.loadContent(node.data);
} }
continue; continue;
@ -823,29 +877,39 @@ export class et2_widget extends ClassWithAttributes
// Make sure promises is defined to avoid errors. // Make sure promises is defined to avoid errors.
// We'll warn (below) if programmer should have passed it. // We'll warn (below) if programmer should have passed it.
if (typeof promises == "undefined") { if (typeof promises == "undefined")
{
promises = []; promises = [];
var warn_if_deferred = true; var warn_if_deferred = true;
} }
var loadChildren = function () { var loadChildren = function ()
{
// Descend recursively into the tree // Descend recursively into the tree
for (var i = 0; i < this._children.length; i++) { for (var i = 0; i < this._children.length; i++)
try { {
try
{
this._children[i].loadingFinished(promises); this._children[i].loadingFinished(promises);
} catch (e) { }
catch (e)
{
egw.debug("error", "There was an error with a widget:\nError:%o\nProblem widget:%o", e.valueOf(), this._children[i], e.stack); egw.debug("error", "There was an error with a widget:\nError:%o\nProblem widget:%o", e.valueOf(), this._children[i], e.stack);
} }
} }
}; };
var result = this.doLoadingFinished(); var result = this.doLoadingFinished();
if (typeof result == "boolean" && result) { if (typeof result == "boolean" && result)
{
// Simple widget finishes nicely // Simple widget finishes nicely
loadChildren.apply(this, arguments); loadChildren.apply(this, arguments);
} else if (typeof result == "object" && result.done) { }
else if (typeof result == "object" && result.done)
{
// Warn if list was not provided // Warn if list was not provided
if (warn_if_deferred) { if (warn_if_deferred)
{
// Might not be a problem, but if you need the widget to be really loaded, it could be // Might not be a problem, but if you need the widget to be really loaded, it could be
egw.debug("warn", "Loading was deferred for widget %o, but creator is not checking. Pass a list to loadingFinished().", this); egw.debug("warn", "Loading was deferred for widget %o, but creator is not checking. Pass a list to loadingFinished().", this);
} }
@ -868,11 +932,14 @@ export class et2_widget extends ClassWithAttributes
*/ */
initAttributes(_attrs) initAttributes(_attrs)
{ {
for (var key in _attrs) { for (var key in _attrs)
if (typeof this.attributes[key] != "undefined" && !this.attributes[key].ignore && !(_attrs[key] == undefined)) { {
if (typeof this.attributes[key] != "undefined" && !this.attributes[key].ignore && !(_attrs[key] == undefined))
{
var val = _attrs[key]; var val = _attrs[key];
// compile string values of attribute type "js" to functions // compile string values of attribute type "js" to functions
if (this.attributes[key].type == 'js' && typeof _attrs[key] == 'string') { if (this.attributes[key].type == 'js' && typeof _attrs[key] == 'string')
{
val = et2_compileLegacyJS(val, this, val = et2_compileLegacyJS(val, this,
this.implements(et2_IInputNode) ? (<et2_inputWidget><unknown>this).getInputNode() : this.implements(et2_IInputNode) ? (<et2_inputWidget><unknown>this).getInputNode() :
(this.implements(et2_IDOMNode) ? (<et2_IDOMNode><unknown>this).getDOMNode() : null)); (this.implements(et2_IDOMNode) ? (<et2_IDOMNode><unknown>this).getDOMNode() : null));
@ -906,16 +973,20 @@ export class et2_widget extends ClassWithAttributes
egw(): IegwAppLocal egw(): IegwAppLocal
{ {
// The _egw property is not set // The _egw property is not set
if (typeof this._egw === 'undefined') { if (typeof this._egw === 'undefined')
if (this._parent != null) { {
if (this._parent != null)
{
return this._parent.egw(); return this._parent.egw();
} }
// Get the window this object belongs to // Get the window this object belongs to
var wnd = null; var wnd = null;
if (this.implements(et2_IDOMNode)) { if (this.implements(et2_IDOMNode))
{
var node = (<et2_IDOMNode><unknown>this).getDOMNode(); var node = (<et2_IDOMNode><unknown>this).getDOMNode();
if (node && node.ownerDocument) { if (node && node.ownerDocument)
{
wnd = node.ownerDocument.parentNode || node.ownerDocument.defaultView; wnd = node.ownerDocument.parentNode || node.ownerDocument.defaultView;
} }
} }
@ -958,20 +1029,24 @@ export class et2_widget extends ClassWithAttributes
*/ */
getArrayMgrs(_mgrs?: object) getArrayMgrs(_mgrs?: object)
{ {
if (typeof _mgrs == "undefined") { if (typeof _mgrs == "undefined")
{
_mgrs = {}; _mgrs = {};
} }
// Add all managers of this object to the result, if they have not already // Add all managers of this object to the result, if they have not already
// been set in the result // been set in the result
for (var key in this._mgrs) { for (var key in this._mgrs)
if (typeof _mgrs[key] == "undefined") { {
if (typeof _mgrs[key] == "undefined")
{
_mgrs[key] = this._mgrs[key]; _mgrs[key] = this._mgrs[key];
} }
} }
// Recursively applies this function to the parent widget // Recursively applies this function to the parent widget
if (this._parent) { if (this._parent)
{
this._parent.getArrayMgrs(_mgrs); this._parent.getArrayMgrs(_mgrs);
} }
@ -996,9 +1071,12 @@ export class et2_widget extends ClassWithAttributes
*/ */
getArrayMgr(managed_array_type: string): et2_arrayMgr | null getArrayMgr(managed_array_type: string): et2_arrayMgr | null
{ {
if (this._mgrs && typeof this._mgrs[managed_array_type] != "undefined") { if (this._mgrs && typeof this._mgrs[managed_array_type] != "undefined")
{
return this._mgrs[managed_array_type]; return this._mgrs[managed_array_type];
} else if (this._parent) { }
else if (this._parent)
{
return this._parent.getArrayMgr(managed_array_type); return this._parent.getArrayMgr(managed_array_type);
} }
@ -1017,22 +1095,27 @@ export class et2_widget extends ClassWithAttributes
// Get the content manager // Get the content manager
var mgrs = this.getArrayMgrs(); var mgrs = this.getArrayMgrs();
for (var key in mgrs) { for (var key in mgrs)
{
var mgr = mgrs[key]; var mgr = mgrs[key];
// Get the original content manager if we have already created a // Get the original content manager if we have already created a
// perspective for this node // perspective for this node
if (typeof this._mgrs[key] != "undefined" && mgr.perspectiveData.owner == this) { if (typeof this._mgrs[key] != "undefined" && mgr.perspectiveData.owner == this)
{
mgr = mgr.parentMgr; mgr = mgr.parentMgr;
} }
// Check whether the manager has a namespace for the id of this object // Check whether the manager has a namespace for the id of this object
var entry = mgr.getEntry(this.id); var entry = mgr.getEntry(this.id);
if (typeof entry === 'object' && entry !== null || this.id) { if (typeof entry === 'object' && entry !== null || this.id)
{
// The content manager has an own node for this object, so // The content manager has an own node for this object, so
// create an own perspective. // create an own perspective.
this._mgrs[key] = mgr.openPerspective(this, this.id); this._mgrs[key] = mgr.openPerspective(this, this.id);
} else { }
else
{
// The current content manager does not have an own namespace for // The current content manager does not have an own namespace for
// this element, so use the content manager of the parent. // this element, so use the content manager of the parent.
delete (this._mgrs[key]); delete (this._mgrs[key]);
@ -1077,9 +1160,12 @@ export class et2_widget extends ClassWithAttributes
*/ */
getInstanceManager() getInstanceManager()
{ {
if (this._inst != null) { if (this._inst != null)
{
return this._inst; return this._inst;
} else if (this._parent) { }
else if (this._parent)
{
return this._parent.getInstanceManager(); return this._parent.getInstanceManager();
} }

View File

@ -1759,17 +1759,9 @@ export class et2_nextmatch extends et2_DOMWidget implements et2_IResizeable, et2
{ {
continue; continue;
} }
if(typeof _row[x] != "undefined" && _row[x].widget)
{
columnWidgets[x] = _row[x].widget; columnWidgets[x] = _row[x].widget;
// Append the widget to this container
this.addChild(_row[x].widget);
}
else
{
columnWidgets[x] = _row[x].widget;
}
// Pass along column alignment // Pass along column alignment
if(_row[x].align && columnWidgets[x]) if(_row[x].align && columnWidgets[x])
{ {
@ -3848,14 +3840,27 @@ export class et2_nextmatch_header_bar extends et2_DOMWidget implements et2_INext
return this.filter_div[0]; return this.filter_div[0];
} }
} }
if(_sender == this.et2_searchbox) return this.search_box[0]; if(_sender == this.et2_searchbox)
if(_sender.id == 'export') return this.right_div[0]; {
return this.search_box[0];
}
if(_sender == this.favorites)
{
return egwIsMobile() ? this.search_box.find('.nm_favorites_div').show()[0] : this.right_div[0];
}
if(_sender.id == 'export')
{
return this.right_div[0];
}
if(_sender && _sender._type == "template") if(_sender && _sender._type == "template")
{ {
for(let i = 0; i < this.headers.length; i++) for(let i = 0; i < this.headers.length; i++)
{ {
if(_sender.id == this.headers[i].id && _sender._parent == this) return i == 2 ? this.header_row[0] : this.header_div[0]; if(_sender.id == this.headers[i].id && _sender._parent == this)
{
return i == 2 ? this.header_row[0] : this.header_div[0];
}
} }
} }
return null; return null;

View File

@ -193,7 +193,7 @@ export class et2_nextmatch_rowProvider
} }
// Adjust data for that row // Adjust data for that row
entry.widget.transformAttributes.call(entry.widget,data); entry.widget.transformAttributes?.call(entry.widget, data);
// Call the setDetachedAttributes function // Call the setDetachedAttributes function
entry.widget.setDetachedAttributes(nodes, data, _data); entry.widget.setDetachedAttributes(nodes, data, _data);
@ -282,22 +282,35 @@ export class et2_nextmatch_rowProvider
// Get all attribute values // Get all attribute values
for (const key in _widget.attributes) for (const key in _widget.attributes)
{ {
if(typeof _widget.attributes[key] !== "object")
{
continue;
}
let attr_name = key;
let val;
if(!_widget.attributes[key].ignore && if(!_widget.attributes[key].ignore &&
typeof _widget.options != "undefined" &&
typeof _widget.options[key] != "undefined") typeof _widget.options[key] != "undefined")
{ {
const val = _widget.options[key]; val = _widget.options[key];
}
// Handle web components
else if(_widget.attributes[key].value)
{
val = _widget.attributes[key].value;
attr_name = _widget.attributes[key].name;
}
// TODO: Improve detection // TODO: Improve detection
if(typeof val == "string" && val.indexOf("$") >= 0) if(typeof val == "string" && val.indexOf("$") >= 0)
{ {
hasAttr = true; hasAttr = true;
widgetData.data.push({ widgetData.data.push({
"attribute": key, "attribute": attr_name,
"expression": val "expression": val
}); });
} }
} }
}
// Add the entry if there is any data in it // Add the entry if there is any data in it
if(hasAttr) if(hasAttr)

View File

@ -66,7 +66,7 @@ export class et2_box extends et2_baseWidget implements et2_IDetachedDOM
* @param {object} _node * @param {object} _node
*/ */
loadFromXML(_node) { loadFromXML(_node) {
if(this.getType() != "box") if(this.getType() != "old-box")
{ {
return super.loadFromXML(_node); return super.loadFromXML(_node);
} }
@ -162,7 +162,7 @@ export class et2_box extends et2_baseWidget implements et2_IDetachedDOM
} }
} }
et2_register_widget(et2_box, ["vbox", "box"]); et2_register_widget(et2_box, ["old-vbox", "old-box"]);
/** /**
* Details widget implementation * Details widget implementation

View File

@ -20,7 +20,7 @@ import {et2_register_widget, WidgetConfig} from "./et2_core_widget";
import {et2_baseWidget} from './et2_core_baseWidget' import {et2_baseWidget} from './et2_core_baseWidget'
import {et2_inputWidget} from "./et2_core_inputWidget"; import {et2_inputWidget} from "./et2_core_inputWidget";
import {expose} from "./expose"; import {expose} from "./expose";
import {et2_IDetachedDOM, et2_IExposable} from "./et2_core_interfaces"; import {et2_IDetachedDOM, et2_IExposable, et2_IInputNode} from "./et2_core_interfaces";
import {egw} from "../jsapi/egw_global"; import {egw} from "../jsapi/egw_global";
/** /**
@ -168,10 +168,10 @@ export class et2_description extends expose(class et2_description extends et2_ba
(for_widget = this.getRoot().getWidgetById(this.options.for)) (for_widget = this.getRoot().getWidgetById(this.options.for))
) && for_widget && for_widget.id) ) && for_widget && for_widget.id)
{ {
if(for_widget.dom_id) if(for_widget.dom_id || for_widget.getDOMNode().id)
{ {
for_id = for_widget.dom_id; for_id = for_widget.dom_id || for_widget.getDOMNode().id;
if(for_widget.instanceOf(et2_inputWidget) && for_widget.getInputNode() && for_widget.dom_id !== for_widget.getInputNode().id) if(for_widget.instanceOf(et2_IInputNode) && for_widget.getInputNode() && for_id !== for_widget.getInputNode().id)
{ {
for_id = for_widget.getInputNode().id; for_id = for_widget.getInputNode().id;
} }

View File

@ -119,6 +119,7 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
private lastRowNode : null; private lastRowNode : null;
private sortablejs : Sortable = null; private sortablejs : Sortable = null;
/** /**
* Constructor * Constructor
* *
@ -139,7 +140,8 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
.appendTo(this.table); .appendTo(this.table);
} }
_initCells(_colData : ColumnEntry[], _rowData : RowEntry[]) { _initCells(_colData : ColumnEntry[], _rowData : RowEntry[])
{
// Copy the width and height // Copy the width and height
const w = _colData.length; const w = _colData.length;
const h = _rowData.length; const h = _rowData.length;
@ -234,7 +236,8 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
} }
// Parse the columns tag // Parse the columns tag
et2_filteredNodeIterator(columns, function(node, nodeName) { et2_filteredNodeIterator(columns, function(node, nodeName)
{
const colDataEntry = this._getColDataEntry(); const colDataEntry = this._getColDataEntry();
// This cannot be done inside a nm, it will expand it later // This cannot be done inside a nm, it will expand it later
colDataEntry["disabled"] = nm ? colDataEntry["disabled"] = nm ?
@ -266,7 +269,8 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
}, this); }, this);
// Parse the rows tag // Parse the rows tag
et2_filteredNodeIterator(rows, function(node, nodeName) { et2_filteredNodeIterator(rows, function(node, nodeName)
{
const rowDataEntry = this._getRowDataEntry(); const rowDataEntry = this._getRowDataEntry();
rowDataEntry["disabled"] = this.getArrayMgr("content") rowDataEntry["disabled"] = this.getArrayMgr("content")
.parseBoolExpression(et2_readAttrWithDefault(node, "disabled", "")); .parseBoolExpression(et2_readAttrWithDefault(node, "disabled", ""));
@ -328,34 +332,42 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
// Not in a nextmatch, so we can expand with abandon // Not in a nextmatch, so we can expand with abandon
const currentPerspective = jQuery.extend({}, content.perspectiveData); const currentPerspective = jQuery.extend({}, content.perspectiveData);
const check = function (node, nodeName) { const check = function(node, nodeName)
if (nodeName == 'box' || nodeName == 'hbox' || nodeName == 'vbox') { {
if(nodeName == 'box' || nodeName == 'hbox' || nodeName == 'vbox')
{
return et2_filteredNodeIterator(node, check, this); return et2_filteredNodeIterator(node, check, this);
} }
content.perspectiveData.row = rowIndex; content.perspectiveData.row = rowIndex;
for (let attr in node.attributes) { for(let attr in node.attributes)
{
const value = et2_readAttrWithDefault(node, node.attributes[attr].name, ""); const value = et2_readAttrWithDefault(node, node.attributes[attr].name, "");
// Don't include first char, those should be handled by normal means // Don't include first char, those should be handled by normal means
// and it would break nextmatch // and it would break nextmatch
if (value.indexOf('@') > 0 || value.indexOf('$') > 0) { if(value.indexOf('@') > 0 || value.indexOf('$') > 0)
{
// Ok, we found something. How many? Check for values. // Ok, we found something. How many? Check for values.
let ident = content.expandName(value); let ident = content.expandName(value);
// expandName() handles index into content (@), but we have to look up // expandName() handles index into content (@), but we have to look up
// regular values // regular values
if (value[0] != '@') { if(value[0] != '@')
{
// Returns null if there isn't an actual value // Returns null if there isn't an actual value
ident = content.getEntry(ident, false, true); ident = content.getEntry(ident, false, true);
} }
while (ident != null && rowIndex < 1000) { while(ident != null && rowIndex < 1000)
{
rowData[rowIndex] = jQuery.extend({}, rowDataEntry); rowData[rowIndex] = jQuery.extend({}, rowDataEntry);
content.perspectiveData.row = ++rowIndex; content.perspectiveData.row = ++rowIndex;
ident = content.expandName(value); ident = content.expandName(value);
if (value[0] != '@') { if(value[0] != '@')
{
// Returns null if there isn't an actual value // Returns null if there isn't an actual value
ident = content.getEntry(ident, false, true); ident = content.getEntry(ident, false, true);
} }
} }
if (rowIndex >= 1000) { if(rowIndex >= 1000)
{
egw.debug("error", "Problem in autorepeat fallback: too many rows for '%s'. Use a nextmatch, or start debugging.", value); egw.debug("error", "Problem in autorepeat fallback: too many rows for '%s'. Use a nextmatch, or start debugging.", value);
} }
return; return;
@ -381,7 +393,8 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
} }
} }
_fillCells(cells, columns : object[], rows : object[]) { _fillCells(cells, columns : object[], rows : object[])
{
const h = cells.length; const h = cells.length;
const w = (h > 0) ? cells[0].length : 0; const w = (h > 0) ? cells[0].length : 0;
const currentPerspective = jQuery.extend({}, this.getArrayMgr("content").perspectiveData); const currentPerspective = jQuery.extend({}, this.getArrayMgr("content").perspectiveData);
@ -389,9 +402,11 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
// Read the elements inside the columns // Read the elements inside the columns
let x = 0; let x = 0;
et2_filteredNodeIterator(columns, function(node, nodeName) { et2_filteredNodeIterator(columns, function(node, nodeName)
{
function _readColNode(node, nodeName) { function _readColNode(node, nodeName)
{
if(y >= h) if(y >= h)
{ {
this.egw().debug("warn", "Skipped grid cell in column, '" + this.egw().debug("warn", "Skipped grid cell in column, '" +
@ -455,9 +470,11 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
nm = (widget.getType() == 'nextmatch'); nm = (widget.getType() == 'nextmatch');
widget = widget.getParent(); widget = widget.getParent();
} }
et2_filteredNodeIterator(rows, function(node, nodeName) { et2_filteredNodeIterator(rows, function(node, nodeName)
{
readRowNode = function _readRowNode(node, nodeName) { readRowNode = function _readRowNode(node, nodeName)
{
if(x >= w) if(x >= w)
{ {
if(nodeName != "description") if(nodeName != "description")
@ -595,7 +612,8 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
}, this); }, this);
// Extra content rows // Extra content rows
for(y; y < h; y++) { for(y; y < h; y++)
{
x = 0; x = 0;
et2_filteredNodeIterator(this.lastRowNode, readRowNode, this); et2_filteredNodeIterator(this.lastRowNode, readRowNode, this);
@ -655,6 +673,7 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
{ {
return true; return true;
} }
/** /**
* As the does not fit very well into the default widget structure, we're * As the does not fit very well into the default widget structure, we're
* overwriting the loadFromXML function and doing a two-pass reading - * overwriting the loadFromXML function and doing a two-pass reading -
@ -662,7 +681,8 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
* *
* @param {object} _node xml node to process * @param {object} _node xml node to process
*/ */
loadFromXML(_node ) { loadFromXML(_node)
{
// Keep the node for later changing / reloading // Keep the node for later changing / reloading
this.template_node = _node; this.template_node = _node;
@ -766,8 +786,16 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
if(cell.disabled) if(cell.disabled)
{ {
td.hide(); td.hide();
// Need to do different things with webComponents
if(typeof cell.widget.options !== "undefined")
{
cell.widget.options.disabled = cell.disabled; cell.widget.options.disabled = cell.disabled;
} }
else if(cell.widget.nodeName)
{
cell.widget.disabled = cell.disabled;
}
}
if(cell.width != "auto") if(cell.width != "auto")
{ {
@ -791,11 +819,13 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
const rs = (y == h - 1) ? h - y : Math.min(h - y, cell.rowSpan); const rs = (y == h - 1) ? h - y : Math.min(h - y, cell.rowSpan);
// Set the col and row span values // Set the col and row span values
if (cs > 1) { if(cs > 1)
{
td.attr("colspan", cs); td.attr("colspan", cs);
} }
if (rs > 1) { if(rs > 1)
{
td.attr("rowspan", rs); td.attr("rowspan", rs);
} }
@ -889,7 +919,8 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
} }
wrapper.css('overflow', _value); wrapper.css('overflow', _value);
if(wrapper.length && (!_value || _value === 'visible')) { if(wrapper.length && (!_value || _value === 'visible'))
{
this.table.unwrap(); this.table.unwrap();
} }
} }
@ -914,6 +945,7 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
const cell = this.managementArray[i]; const cell = this.managementArray[i];
if(cell.widget) if(cell.widget)
{ {
this.removeChild(cell.widget);
cell.widget.destroy(); cell.widget.destroy();
} }
} }
@ -968,17 +1000,22 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
filter: this.options.sortable_cancel, filter: this.options.sortable_cancel,
ghostClass: this.options.sortable_placeholder, ghostClass: this.options.sortable_placeholder,
dataIdAttr: 'id', dataIdAttr: 'id',
onAdd:function (event) { onAdd: function(event)
if (typeof self.options.sortable_recieveCallback == 'function') { {
if(typeof self.options.sortable_recieveCallback == 'function')
{
self.options.sortable_recieveCallback.call(self, event, this, self.id); self.options.sortable_recieveCallback.call(self, event, this, self.id);
} }
}, },
onStart: function (event, ui) { onStart: function(event, ui)
if (typeof self.options.sortable_startCallback == 'function') { {
if(typeof self.options.sortable_startCallback == 'function')
{
self.options.sortable_startCallback.call(self, event, this, self.id); self.options.sortable_startCallback.call(self, event, this, self.id);
} }
}, },
onSort: function (event) { onSort: function(event)
{
self.egw().json(sortable, [ self.egw().json(sortable, [
self.getInstanceManager().etemplate_exec_id, self.getInstanceManager().etemplate_exec_id,
self.sortablejs.toArray(), self.sortablejs.toArray(),
@ -1004,7 +1041,8 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
let objectManager = egw_getAppObjectManager(true, this.getInstanceManager().app); let objectManager = egw_getAppObjectManager(true, this.getInstanceManager().app);
objectManager = objectManager.getObjectById(this.getInstanceManager().uniqueId, 2) || objectManager; objectManager = objectManager.getObjectById(this.getInstanceManager().uniqueId, 2) || objectManager;
let widget_object = objectManager.getObjectById(this.id); let widget_object = objectManager.getObjectById(this.id);
if (widget_object == null) { if(widget_object == null)
{
// Add a new container to the object manager which will hold the widget // Add a new container to the object manager which will hold the widget
// objects // objects
widget_object = objectManager.insertObject(false, new egwActionObject( widget_object = objectManager.insertObject(false, new egwActionObject(
@ -1129,7 +1167,10 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
if(!row[c].widget) continue; if(!row[c].widget) continue;
let found = row[c].widget === _sender; let found = row[c].widget === _sender;
if (!found) row[c].widget.iterateOver(function(_widget) {if (_widget === _sender) found = true;}); if(!found) row[c].widget.iterateOver(function(_widget)
{
if(_widget === _sender) found = true;
});
if(found) if(found)
{ {
// return a fake row object allowing to iterate over it's children // return a fake row object allowing to iterate over it's children
@ -1140,7 +1181,8 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
} }
row_obj.isInTree = jQuery.proxy(this.isInTree, this); row_obj.isInTree = jQuery.proxy(this.isInTree, this);
// we must not free the children! // we must not free the children!
row_obj.destroy = function(){ row_obj.destroy = function()
{
// @ts-ignore // @ts-ignore
delete row_obj._children; delete row_obj._children;
}; };
@ -1153,10 +1195,12 @@ export class et2_grid extends et2_DOMWidget implements et2_IDetachedDOM, et2_IAl
/** /**
* Needed for the align interface, but we're not doing anything with it... * Needed for the align interface, but we're not doing anything with it...
*/ */
get_align(): string { get_align() : string
{
return ""; return "";
} }
} }
et2_register_widget(et2_grid, ["grid"]); et2_register_widget(et2_grid, ["grid"]);
interface ColumnEntry interface ColumnEntry

View File

@ -199,6 +199,6 @@ export class et2_hbox extends et2_baseWidget
} }
} }
} }
et2_register_widget(et2_hbox, ["hbox"]); et2_register_widget(et2_hbox, ["old-hbox"]);

View File

@ -23,7 +23,12 @@ import {et2_nextmatch, et2_nextmatch_header_bar} from "./et2_extension_nextmatch
import {et2_tabbox} from "./et2_widget_tabs"; import {et2_tabbox} from "./et2_widget_tabs";
import '../jsapi/egw_json.js'; import '../jsapi/egw_json.js';
import {egwIsMobile} from "../egw_action/egw_action_common.js"; import {egwIsMobile} from "../egw_action/egw_action_common.js";
//import './et2-button'; import './Et2Box/Et2Box';
import './Et2Button/Et2Button';
import './Et2Date/Et2DateTime';
import './Et2Textarea/Et2Textarea';
import './Et2Textbox/Et2Textbox';
import './Et2Colorpicker/Et2Colorpicker';
/* Include all widget classes here, we only care about them registering, not importing anything*/ /* Include all widget classes here, we only care about them registering, not importing anything*/
import './et2_widget_vfs'; // Vfs must be first (before et2_widget_file) due to import cycle import './et2_widget_vfs'; // Vfs must be first (before et2_widget_file) due to import cycle
import './et2_widget_template'; import './et2_widget_template';
@ -165,13 +170,15 @@ export class etemplate2
* *
* @param {jQuery.event} e * @param {jQuery.event} e
*/ */
public resize(e) { public resize(e)
{
const event = e; const event = e;
const self = this; const self = this;
let excess_height: number | boolean = false; let excess_height: number | boolean = false;
// Check if the framework has an specific excess height calculation // Check if the framework has an specific excess height calculation
if (typeof window.framework != 'undefined' && typeof window.framework.get_wExcessHeight != 'undefined') { if (typeof window.framework != 'undefined' && typeof window.framework.get_wExcessHeight != 'undefined')
{
excess_height = window.framework.get_wExcessHeight(window); excess_height = window.framework.get_wExcessHeight(window);
} }
@ -507,12 +514,15 @@ export class etemplate2
// require necessary translations from server AND the app.js file, if not already loaded // require necessary translations from server AND the app.js file, if not already loaded
let promisses = [window.egw_ready]; // to wait for legacy-loaded JS let promisses = [window.egw_ready]; // to wait for legacy-loaded JS
if (Array.isArray(_data.langRequire)) { if (Array.isArray(_data.langRequire))
{
promisses.push(egw(currentapp, window).langRequire(window, _data.langRequire)); promisses.push(egw(currentapp, window).langRequire(window, _data.langRequire));
} }
return Promise.all(promisses).catch((err) => { return Promise.all(promisses).catch((err) =>
{
console.log("et2.load(): error loading lang-files and app.js: " + err.message); console.log("et2.load(): error loading lang-files and app.js: " + err.message);
}).then(() => { }).then(() =>
{
this.clear(); this.clear();
// Initialize application js // Initialize application js
@ -564,7 +574,8 @@ export class etemplate2
const _load = function () const _load = function ()
{ {
egw.debug("log", "Loading template..."); egw.debug("log", "Loading template...");
if (egw.debug_level() >= 4 && console.timeStamp) { if (egw.debug_level() >= 4 && console.timeStamp)
{
console.timeStamp("Begin rendering template"); console.timeStamp("Begin rendering template");
} }
@ -670,10 +681,12 @@ export class etemplate2
// Profiling // Profiling
if (egw.debug_level() >= 4) if (egw.debug_level() >= 4)
{ {
if (console.timeEnd) { if (console.timeEnd)
{
console.timeEnd(_name); console.timeEnd(_name);
} }
if (console.profileEnd) { if (console.profileEnd)
{
console.profileEnd(_name); console.profileEnd(_name);
} }
const end_time = (new Date).getTime(); const end_time = (new Date).getTime();
@ -819,7 +832,8 @@ export class etemplate2
indexes = [indexes.shift(), indexes.join('[')]; indexes = [indexes.shift(), indexes.join('[')];
indexes[1] = indexes[1].substring(0, indexes[1].length - 1); indexes[1] = indexes[1].substring(0, indexes[1].length - 1);
const children = indexes[1].split(']['); const children = indexes[1].split('][');
if (children.length) { if (children.length)
{
indexes = jQuery.merge([indexes[0]], children); indexes = jQuery.merge([indexes[0]], children);
} }
} }
@ -1009,7 +1023,8 @@ export class etemplate2
_root.iterateOver(function (_widget) _root.iterateOver(function (_widget)
{ {
// The widget must have an id to be included in the values array // The widget must have an id to be included in the values array
if (_widget.id === undefined || _widget.id === "") { if (_widget.id === undefined || _widget.id === "")
{
return; return;
} }
@ -1024,7 +1039,8 @@ export class etemplate2
indexes = [indexes.shift(), indexes.join('[')]; indexes = [indexes.shift(), indexes.join('[')];
indexes[1] = indexes[1].substring(0, indexes[1].length - 1); indexes[1] = indexes[1].substring(0, indexes[1].length - 1);
const children = indexes[1].split(']['); const children = indexes[1].split('][');
if (children.length) { if (children.length)
{
indexes = jQuery.merge([indexes[0]], children); indexes = jQuery.merge([indexes[0]], children);
} }
path = path.concat(indexes); path = path.concat(indexes);
@ -1425,38 +1441,48 @@ export class etemplate2
* @returns {Boolean} Handled by this plugin * @returns {Boolean} Handled by this plugin
* @throws Invalid parameters if the required res.data parameters are missing * @throws Invalid parameters if the required res.data parameters are missing
*/ */
public handle_assign(type, res, req) { public handle_assign(type, res, req)
{
//type, req; // unused, but required by plugin signature //type, req; // unused, but required by plugin signature
//Check whether all needed parameters have been passed and call the alertHandler function //Check whether all needed parameters have been passed and call the alertHandler function
if ((typeof res.data.id != 'undefined') && if ((typeof res.data.id != 'undefined') &&
(typeof res.data.key != 'undefined') && (typeof res.data.key != 'undefined') &&
(typeof res.data.value != 'undefined') (typeof res.data.value != 'undefined')
) { )
{
if (typeof res.data.etemplate_exec_id == 'undefined' || if (typeof res.data.etemplate_exec_id == 'undefined' ||
res.data.etemplate_exec_id != this._etemplate_exec_id) { res.data.etemplate_exec_id != this._etemplate_exec_id)
{
// Not for this etemplate, but not an error // Not for this etemplate, but not an error
return false; return false;
} }
if (res.data.key == 'etemplate_exec_id') { if (res.data.key == 'etemplate_exec_id')
{
this._etemplate_exec_id = res.data.value; this._etemplate_exec_id = res.data.value;
return true; return true;
} }
if (this._widgetContainer == null) { if (this._widgetContainer == null)
{
// Right etemplate, but it's already been cleared. // Right etemplate, but it's already been cleared.
egw.debug('warn', "Tried to call assign on an un-loaded etemplate", res.data); egw.debug('warn', "Tried to call assign on an un-loaded etemplate", res.data);
return false; return false;
} }
const widget = this._widgetContainer.getWidgetById(res.data.id); const widget = this._widgetContainer.getWidgetById(res.data.id);
if (widget) { if (widget)
if (typeof widget['set_' + res.data.key] != 'function') { {
if (typeof widget['set_' + res.data.key] != 'function')
{
egw.debug('warn', "Cannot set %s attribute %s via JSON assign, no set_%s()", res.data.id, res.data.key, res.data.key); egw.debug('warn', "Cannot set %s attribute %s via JSON assign, no set_%s()", res.data.id, res.data.key, res.data.key);
return false; return false;
} }
try { try
{
widget['set_' + res.data.key].call(widget, res.data.value); widget['set_' + res.data.key].call(widget, res.data.value);
return true; return true;
} catch (e) { }
catch (e)
{
egw.debug("error", "When assigning %s on %s via AJAX, \n" + (e.message || e + ""), res.data.key, res.data.id, widget); egw.debug("error", "When assigning %s on %s via AJAX, \n" + (e.message || e + ""), res.data.key, res.data.id, widget);
} }
} }

View File

@ -161,11 +161,11 @@ egw.extend('tooltip', egw.MODULE_WND_LOCAL, function(_app, _wnd)
* It is important to remove all tooltips from all elements which are * It is important to remove all tooltips from all elements which are
* no longer needed, in order to prevent memory leaks. * no longer needed, in order to prevent memory leaks.
* *
* @param _elem is the element to which the tooltip should get bound. It * @param _elem is the element to which the tooltip should get bound.
* has to be a jQuery node.
* @param _html is the html code which should be shown as tooltip. * @param _html is the html code which should be shown as tooltip.
*/ */
tooltipBind: function(_elem, _html, _isHtml) { tooltipBind: function(_elem, _html, _isHtml) {
_elem = jQuery(_elem);
if (_html != '') if (_html != '')
{ {
_elem.bind('mouseenter.tooltip', function(e) { _elem.bind('mouseenter.tooltip', function(e) {
@ -222,6 +222,7 @@ egw.extend('tooltip', egw.MODULE_WND_LOCAL, function(_app, _wnd)
* removed. _elem has to be a jQuery node. * removed. _elem has to be a jQuery node.
*/ */
tooltipUnbind: function(_elem) { tooltipUnbind: function(_elem) {
_elem = jQuery(_elem);
if (current_elem == _elem) if (current_elem == _elem)
{ {
hide(); hide();

View File

@ -76,4 +76,4 @@ class Button extends Etemplate\Widget
} }
} }
} }
Etemplate\Widget::registerWidget(__NAMESPACE__.'\\Button', array('button','buttononly')); Etemplate\Widget::registerWidget(__NAMESPACE__.'\\Button', array('et2-button','button','buttononly'));

View File

@ -87,14 +87,20 @@ class Date extends Transformer
* @param array $expand * @param array $expand
* @param array $data Row data * @param array $data Row data
*/ */
public function set_row_value($cname, Array $expand, Array &$data) public function set_row_value($cname, array $expand, array &$data)
{ {
if($this->type == 'date-duration') return; if($this->type == 'date-duration')
{
return;
}
$form_name = self::form_name($cname, $this->id, $expand); $form_name = self::form_name($cname, $this->id, $expand);
$value =& $this->get_array($data, $form_name, true); $value =& $this->get_array($data, $form_name, true);
if (true) $value = $this->format_date($value); if(true)
{
$value = $this->format_date($value);
}
} }
/** /**
@ -104,7 +110,10 @@ class Date extends Transformer
*/ */
public function format_date($value) public function format_date($value)
{ {
if (!$value) return $value; // otherwise we will get current date or 1970-01-01 instead of an empty value if(!$value)
{
return $value;
} // otherwise we will get current date or 1970-01-01 instead of an empty value
// for DateTime objects (regular PHP and Api\DateTime ones), set user timezone // for DateTime objects (regular PHP and Api\DateTime ones), set user timezone
if($value instanceof \DateTime) if($value instanceof \DateTime)
@ -158,7 +167,10 @@ class Date extends Transformer
{ {
try try
{ {
if (substr($value, -1) === 'Z') $value = substr($value, 0, -1); if(substr($value, -1) === 'Z')
{
$value = substr($value, 0, -1);
}
$date = new Api\DateTime($value); $date = new Api\DateTime($value);
} }
catch (\Exception $e) catch (\Exception $e)
@ -193,7 +205,11 @@ class Date extends Transformer
elseif(preg_match('/[+-][[:digit:]]+[ymwd]/', $this->attrs['min'])) elseif(preg_match('/[+-][[:digit:]]+[ymwd]/', $this->attrs['min']))
{ {
// Relative date with periods // Relative date with periods
$min = new Api\DateTime(strtotime(str_replace(array('y','m','w','d'), array('years','months','weeks','days'), $this->attrs['min']))); $min = new Api\DateTime(strtotime(str_replace(array('y', 'm', 'w', 'd'), array('years', 'months',
'weeks',
'days'), $this->attrs['min'])
)
);
} }
else else
{ {
@ -204,7 +220,8 @@ class Date extends Transformer
self::set_validation_error($form_name, lang( self::set_validation_error($form_name, lang(
"Value has to be at least '%1' !!!", "Value has to be at least '%1' !!!",
$min->format($this->type != 'date') $min->format($this->type != 'date')
),''); ), ''
);
$value = $min; $value = $min;
} }
} }
@ -217,7 +234,11 @@ class Date extends Transformer
elseif(preg_match('/[+-][[:digit:]]+[ymwd]/', $this->attrs['max'])) elseif(preg_match('/[+-][[:digit:]]+[ymwd]/', $this->attrs['max']))
{ {
// Relative date with periods // Relative date with periods
$max = new Api\DateTime(strtotime(str_replace(array('y','m','w','d'), array('years','months','weeks','days'), $this->attrs['max']))); $max = new Api\DateTime(strtotime(str_replace(array('y', 'm', 'w', 'd'), array('years', 'months',
'weeks',
'days'), $this->attrs['max'])
)
);
} }
else else
{ {
@ -228,7 +249,8 @@ class Date extends Transformer
self::set_validation_error($form_name, lang( self::set_validation_error($form_name, lang(
"Value has to be at maximum '%1' !!!", "Value has to be at maximum '%1' !!!",
$max->format($this->type != 'date') $max->format($this->type != 'date')
),''); ), ''
);
$value = $max; $value = $max;
} }
} }
@ -259,4 +281,7 @@ class Date extends Transformer
} }
} }
} }
\EGroupware\Api\Etemplate\Widget::registerWidget(__NAMESPACE__.'\\Date', array('time_or_date'));
\EGroupware\Api\Etemplate\Widget::registerWidget(__NAMESPACE__ . '\\Date',
array('et2-date', 'et2-datetime', 'time_or_date')
);

View File

@ -205,29 +205,16 @@ class Template extends Etemplate\Widget
/** /**
* Convert relative template path from relPath to an url incl. cache-buster modification time postfix * Convert relative template path from relPath to an url incl. cache-buster modification time postfix
* *
* This adds the server-side modification of eTemplates for web-components /api/etemplate.php.
*
* @param string $path * @param string $path
* @return string url * @return string url
*/ */
public static function rel2url($path) public static function rel2url($path)
{ {
if ($path) return $GLOBALS['egw_info']['server']['webserver_url'].'/api/etemplate.php'.
{ ($path[0] === '/' ? $path : preg_replace('#^'.self::VFS_TEMPLATE_PATH.'#', '',
if ($path[0] === '/') Api\Vfs::parse_url($path, PHP_URL_PATH))).'?'.filemtime(self::rel2path($path));
{
$url = $GLOBALS['egw_info']['server']['webserver_url'].$path.'?'.filemtime(self::rel2path($path));
}
else
{
$url = Api\Vfs::download_url($path);
if ($url[0] == '/') $url = Api\Framework::link($url);
// mtime postfix has to use '?download=', as our WebDAV treats everything else literal and not ignore them like Apache for static files!
$url .= '?download='.filemtime($path);
}
}
//error_log(__METHOD__."('$path') returning $url");
return $url;
} }
/** /**

View File

@ -179,4 +179,6 @@ class Textbox extends Etemplate\Widget
} }
} }
} }
Etemplate\Widget::registerWidget(__NAMESPACE__.'\\Textbox', array('textbox','text','int','integer','float','hidden','colorpicker','hidden')); Etemplate\Widget::registerWidget(__NAMESPACE__ . '\\Textbox', array('et2-textarea', 'et2-textbox', 'textbox', 'text',
'int', 'integer', 'float', 'hidden', 'colorpicker',
'hidden'));

View File

@ -26,6 +26,7 @@
padding: 0px; padding: 0px;
background-color: white; background-color: white;
height: 99%; height: 99%;
max-width: 100%
} }
.et2_container > div:not([class]) { .et2_container > div:not([class]) {
height: 100%; height: 100%;
@ -315,6 +316,9 @@ button.et2_button_with_image {
button.et2_button_with_image.et2_button_text { button.et2_button_with_image.et2_button_text {
background-position: 4px center; background-position: 4px center;
} }
button.et2_buttonFitContent, et2-button.et2_buttonFitContent {
max-width: fit-content;
}
/* et2_box_widget ###*/ /* et2_box_widget ###*/
button[id="cancel"], button[id="cancel"],
button#cancel { button#cancel {

View File

@ -85,10 +85,8 @@
</rows> </rows>
</grid> </grid>
<vbox class="filemanager_config"> <vbox class="filemanager_config">
<button label="Mount /etemplates to allow customizing of eTemplates" id="etemplates"/> <button label="Mount /etemplates to allow customizing of eTemplates" id="etemplates" class="et2_buttonFitContent"/>
<menulist> <select id="allow_delete_versions" onchange="1" label="Who should be allowed to finally delete deleted files or old versions of a file:" empty_label="Noone" disabled="!@versioning"/>
<menupopup id="allow_delete_versions" onchange="1" label="Who should be allowed to finally delete deleted files or old versions of a file:" empty_label="Noone" disabled="!@versioning"/>
</menulist>
<hbox disabled="!@versioning"> <hbox disabled="!@versioning">
<integer id="mtime" label="Delete all older versions and deleted files older then %s days" statustext="0 means all, -N newer then N days"/> <integer id="mtime" label="Delete all older versions and deleted files older then %s days" statustext="0 means all, -N newer then N days"/>
<textbox size="30" label="under directory" id="versionedpath" statustext="/ = everywhere"/> <textbox size="30" label="under directory" id="versionedpath" statustext="/ = everywhere"/>

View File

@ -66,9 +66,11 @@ button.infologExtraButton {
button.infologExtraButton:hover { button.infologExtraButton:hover {
background-size: 16px; background-size: 16px;
} }
div.et2_hbox.tab_toolbar {
.tab_toolbar {
position: relative; position: relative;
} }
button#infolog-edit_encrypt { button#infolog-edit_encrypt {
position: absolute; position: absolute;
right: 0px; right: 0px;

View File

@ -3,7 +3,7 @@
<!-- $Id$ --> <!-- $Id$ -->
<overlay> <overlay>
<template id="infolog.edit.description" template="" lang="" group="0" version="1.6.001"> <template id="infolog.edit.description" template="" lang="" group="0" version="1.6.001">
<textbox multiline="true" id="info_des" no_lang="1" width="99.7%" height="245px"/> <et2-textarea id="info_des" no_lang="1"/>
<checkbox id="clean_history"/> <checkbox id="clean_history"/>
</template> </template>
<template id="infolog.edit.links" template="" lang="" group="0" version="1.3.001"> <template id="infolog.edit.links" template="" lang="" group="0" version="1.3.001">
@ -154,7 +154,7 @@
<rows> <rows>
<row class="dialogHeader"> <row class="dialogHeader">
<description value="Title" for="info_subject"/> <description value="Title" for="info_subject"/>
<textbox statustext="a short subject for the entry" id="info_subject" class="et2_fullWidth et2_required" maxlength="255" span="4" tabindex="1"/> <et2-textbox statustext="a short subject for the entry" id="info_subject" class="et2_fullWidth et2_required" maxlength="255" span="4" tabindex="1"></et2-textbox>
<textbox type="integer" id="info_number" readonly="true"/> <textbox type="integer" id="info_number" readonly="true"/>
<appicon src="infolog" for="info_number"/> <appicon src="infolog" for="info_number"/>
</row> </row>
@ -165,7 +165,9 @@
</menulist> </menulist>
<description/> <description/>
<description value="Startdate" for="info_startdate"/> <description value="Startdate" for="info_startdate"/>
<date-time statustext="when should the ToDo or Phonecall be started, it shows up from that date in the filter open or own open (startpage)" id="info_startdate" class="et2_fullWidth"/> <et2-datetime
statustext="when should the ToDo or Phonecall be started, it shows up from that date in the filter open or own open (startpage)"
id="info_startdate" class="et2_fullWidth"></et2-datetime>
</row> </row>
<row class="dialogHeader3"> <row class="dialogHeader3">
<description value="Contact"/> <description value="Contact"/>
@ -217,32 +219,33 @@
</row> </row>
<row disabled="!@info_owner" class="dialogOperators"> <row disabled="!@info_owner" class="dialogOperators">
<description value="Owner"/> <description value="Owner"/>
<hbox width="100%"> <et2-hbox>
<menulist> <select-account id="info_owner" readonly="true"/>
<menupopup type="select-account" id="info_owner" readonly="true"/>
</menulist>
<date-time id="info_created" readonly="true" align="right"/> <date-time id="info_created" readonly="true" align="right"/>
</hbox> </et2-hbox>
<description/> <description/>
<description value="Last modified"/> <description value="Last modified"/>
<hbox width="100%"> <et2-hbox>
<menulist> <select-account id="info_modifier" readonly="true"/>
<menupopup type="select-account" id="info_modifier" readonly="true"/>
</menulist>
<date-time id="info_datemodified" readonly="true" align="right"/> <date-time id="info_datemodified" readonly="true" align="right"/>
</hbox> </et2-hbox>
</row> </row>
<row class="dialogFooterToolbar"> <row class="dialogFooterToolbar">
<hbox span="6"> <hbox span="all">
<button statustext="Saves this entry" label="Save" id="button[save]" image="save" background_image="1"/> <button statustext="Saves this entry" label="Save" id="button[save]" image="save"
<button statustext="Apply the changes" label="Apply" id="button[apply]" image="apply" background_image="1"/> background_image="1"/>
<button statustext="leave without saveing the entry" label="Cancel" id="button[cancel]" onclick="window.close();" image="cancel" background_image="1"/> <button statustext="Apply the changes" label="Apply" id="button[apply]" image="apply"
<menulist> background_image="1"></button>
<menupopup statustext="Execute a further action for this entry" id="action" onchange="app.infolog.edit_actions()" options="Actions..."/> <button statustext="leave without saveing the entry" label="Cancel" id="button[cancel]"
</menulist> onclick="window.close();" image="cancel" background_image="1"/>
<checkbox label="Do not notify" id="no_notifications" statustext="Do not notify of these changes"/> <select statustext="Execute a further action for this entry" id="action"
onchange="app.infolog.edit_actions()" empty_label="Actions..."/>
<checkbox label="Do not notify" id="no_notifications"
statustext="Do not notify of these changes"/>
<button align="right" statustext="delete this entry" label="Delete" id="button[delete]"
onclick="if($cont[info_anz_subs]) return $cont[info_anz_subs]; et2_dialog.confirm(widget,'Delete this entry','Delete');"
image="delete" background_image="1" span="all"/>
</hbox> </hbox>
<button align="right" statustext="delete this entry" label="Delete" id="button[delete]" onclick="if($cont[info_anz_subs]) return $cont[info_anz_subs]; et2_dialog.confirm(widget,'Delete this entry','Delete');" image="delete" background_image="1" span="all"/>
</row> </row>
</rows> </rows>
</grid> </grid>

View File

@ -226,7 +226,7 @@
* Add / remove link or category popup used for actions on multiple entries * Add / remove link or category popup used for actions on multiple entries
*/ */
div.action_popup[id] { .action_popup[id] {
position: fixed; position: fixed;
top: 200px; top: 200px;
left: 450px; left: 450px;

View File

@ -132,7 +132,7 @@ button.infologExtraButton {
button.infologExtraButton:hover { button.infologExtraButton:hover {
background-size: 16px; background-size: 16px;
} }
div.et2_hbox.tab_toolbar { .tab_toolbar {
position: relative; position: relative;
} }
button#infolog-edit_encrypt { button#infolog-edit_encrypt {

View File

@ -1,47 +0,0 @@
<!--
@license
Copyright IBM Corp. 2019
This source code is licensed under the Apache-2.0 license found in the
LICENSE file in the root directory of this source tree.
-->
<html>
<head>
<title>carbon-web-components example</title>
<meta charset="UTF-8" />
<style type="text/css">
body {
font-family: 'IBM Plex Sans', 'Helvetica Neue', Arial, sans-serif;
}
#app {
width: 300px;
}
bx-dropdown,
bx-dropdown-item {
visibility: hidden;
}
bx-dropdown:defined,
bx-dropdown-item:defined {
visibility: inherit;
}
</style>
</head>
<body>
<h1>Hello World! 👋</h1>
<div id="app">
<bx-dropdown trigger-content="Select an item">
<bx-dropdown-item value="all">Option 1</bx-dropdown-item>
<bx-dropdown-item value="cloudFoundry">Option 2</bx-dropdown-item>
<bx-dropdown-item value="staging">Option 3</bx-dropdown-item>
<bx-dropdown-item value="dea">Option 4</bx-dropdown-item>
<bx-dropdown-item value="router">Option 5</bx-dropdown-item>
</bx-dropdown>
</div>
<script type="module" src="js/app.min.js"></script>
</body>
</html>

View File

@ -1,11 +0,0 @@
/**
* @license
*
* Copyright IBM Corp. 2020
*
* This source code is licensed under the Apache-2.0 license found in the
* LICENSE file in the root directory of this source tree.
*/
import 'carbon-web-components/es/components/dropdown/dropdown.js';
import 'carbon-web-components/es/components/dropdown/dropdown-item.js';

View File

@ -319,7 +319,7 @@ app.classes.mail = AppJS.extend(
{ {
textAreaWidget.tinymce.then(()=>{ textAreaWidget.tinymce.then(()=>{
that.compose_resizeHandler(); that.compose_resizeHandler();
jQuery(textAreaWidget.editor.iframeElement.contentWindow.document).on('dragenter', function(){ if (textAreaWidget.editor) jQuery(textAreaWidget.editor.iframeElement.contentWindow.document).on('dragenter', function(){
// anything to bind on tinymce iframe // anything to bind on tinymce iframe
}); });
}); });

View File

@ -637,7 +637,7 @@ div.mailDisplayHeaders div.mail_extraEmails.visible
box-shadow: 5px 5px 5px #aaa; box-shadow: 5px 5px 5px #aaa;
border: 1px solid gray; border: 1px solid gray;
} }
div.mailComposeBody { .mailComposeBody {
white-space: normal !important; white-space: normal !important;
} }
#mail-compose_mail_plaintext { #mail-compose_mail_plaintext {
@ -884,11 +884,11 @@ span#mail-compose_replyto_expander {
padding: 0; padding: 0;
} }
/*Make file uploads in compose dialog invisible*/ /*Make file uploads in compose dialog invisible*/
div.mail-compose_toolbar_assist div.mail-compose_fileselector, #mail-compose_selectFromVFSForCompose, div.mail-compose_toolbar_assist { .mail-compose_toolbar_assist div.mail-compose_fileselector, #mail-compose_selectFromVFSForCompose, .mail-compose_toolbar_assist {
display:none; display:none;
} }
/*Make file uploads in compose dialog invisible*/ /*Make file uploads in compose dialog invisible*/
div.mail-compose_toolbar_assist div.mail-compose_fileselector, #mail-compose_selectFromVFSForCompose, div.mail-compose_toolbar_assist { .mail-compose_toolbar_assist div.mail-compose_fileselector, #mail-compose_selectFromVFSForCompose, .mail-compose_toolbar_assist {
display:none; display:none;
} }

View File

@ -86,12 +86,12 @@
</grid> </grid>
</vbox> </vbox>
<vbox class="mailComposeBodySection" width="100%"> <vbox class="mailComposeBodySection" width="100%">
<hbox disabled="@is_plain" class="mailComposeBody mailComposeHtmlContainer"> <old-hbox disabled="@is_plain" class="mailComposeBody mailComposeHtmlContainer">
<htmlarea name="mail_htmltext" id="mail_htmltext" statusbar="false" menubar="false" toolbar="@html_toolbar" imageUpload="link_to" expand_toolbar="true" height="478px" width="100%" resize_ratio="0"/> <htmlarea name="mail_htmltext" id="mail_htmltext" statusbar="false" menubar="false" toolbar="@html_toolbar" imageUpload="link_to" expand_toolbar="true" height="478px" width="100%" resize_ratio="0"/>
</hbox> </old-hbox>
<hbox disabled="@is_html" class="mailComposeBody mailComposeTextContainer"> <old-hbox disabled="@is_html" class="mailComposeBody mailComposeTextContainer">
<textbox multiline="true" rows="40" cols="120" width="100%" span="all" no_lang="1" name="mail_plaintext" id="mail_plaintext" resize_ratio="0"/> <textbox multiline="true" rows="40" cols="120" width="100%" span="all" no_lang="1" name="mail_plaintext" id="mail_plaintext" resize_ratio="0"/>
</hbox> </old-hbox>
<vbox class="et2_file mailUploadSection" disabled="@no_griddata"> <vbox class="et2_file mailUploadSection" disabled="@no_griddata">
<hbox> <hbox>
<select id="filemode" label="Send files as" onchange="app.mail.check_sharing_filemode"/> <select id="filemode" label="Send files as" onchange="app.mail.check_sharing_filemode"/>

View File

@ -5,43 +5,43 @@
<template id="mail.index.splitter" height="100%" template="" lang="" group="0" version="1.9.001"> <template id="mail.index.splitter" height="100%" template="" lang="" group="0" version="1.9.001">
<split dock_side="bottomDock" id="mailSplitter"> <split dock_side="bottomDock" id="mailSplitter">
<nextmatch id="nm" onselect="app.mail.mail_preview" class="" template="mail.index.rows" header_left="mail.index.add" header_right="mail.index.header_right" disable_selection_advance="true"/> <nextmatch id="nm" onselect="app.mail.mail_preview" class="" template="mail.index.rows" header_left="mail.index.add" header_right="mail.index.header_right" disable_selection_advance="true"/>
<vbox id="mailPreview" width="100%"> <old-vbox id="mailPreview" width="100%">
<toolbar id="toolbar"/> <toolbar id="toolbar"/>
<box id="blank" disabled="false"> <old-box id="blank" disabled="false">
<image src="mail"/> <image src="mail"/>
<description value="Select an item to read"/> <description value="Select an item to read"/>
</box> </old-box>
<hbox width="100%" id="mailPreviewHeadersFrom" class="mailPreviewHeaders"> <old-hbox width="100%" id="mailPreviewHeadersFrom" class="mailPreviewHeaders">
<description value="From"/> <description value="From"/>
<hbox id="additionalFromAddress" class="mail_extraEmails"> <old-hbox id="additionalFromAddress" class="mail_extraEmails">
</hbox> </old-hbox>
<buttononly class="et2_button ui-button" label="Show all Addresses" image="foldertree_nolines_plus" onclick="app.mail.showAllHeader"/> <buttononly class="et2_button ui-button" label="Show all Addresses" image="foldertree_nolines_plus" onclick="app.mail.showAllHeader"/>
</hbox> </old-hbox>
<hbox id="mailPreviewHeadersSubject" class="mailPreviewHeaders"> <old-hbox id="mailPreviewHeadersSubject" class="mailPreviewHeaders">
<description value="Subject"/> <description value="Subject"/>
<description align="left" id="previewSubject" readonly="true" hover_action="app.mail.modifyMessageSubjectDialog" hover_action_title="Modify subject of this message"/> <description align="left" id="previewSubject" readonly="true" hover_action="app.mail.modifyMessageSubjectDialog" hover_action_title="Modify subject of this message"/>
</hbox> </old-hbox>
<hbox id="mailPreviewHeadersDate" class="mailPreviewHeaders"> <old-hbox id="mailPreviewHeadersDate" class="mailPreviewHeaders">
<description value="Date"/> <description value="Date"/>
<date-time align="left" id="previewDate" readonly="true"/> <date-time align="left" id="previewDate" readonly="true"/>
</hbox> </old-hbox>
<hbox width="100%" id="mailPreviewHeadersTo" class="mailPreviewHeaders"> <old-hbox width="100%" id="mailPreviewHeadersTo" class="mailPreviewHeaders">
<description value="To"/> <description value="To"/>
<hbox id="additionalToAddress" class="mail_extraEmails"> <old-hbox id="additionalToAddress" class="mail_extraEmails">
</hbox> </old-hbox>
<buttononly class="et2_button ui-button" label="Show all Addresses" image="foldertree_nolines_plus" onclick="app.mail.showAllHeader"/> <buttononly class="et2_button ui-button" label="Show all Addresses" image="foldertree_nolines_plus" onclick="app.mail.showAllHeader"/>
</hbox> </old-hbox>
<hbox id="mailPreviewHeadersCC" class="mailPreviewHeaders"> <old-hbox id="mailPreviewHeadersCC" class="mailPreviewHeaders">
<description value="CC"/> <description value="CC"/>
<hbox id="additionalCCAddress" class="mail_extraEmails"> <old-hbox id="additionalCCAddress" class="mail_extraEmails">
</hbox> </old-hbox>
<buttononly class="et2_button ui-button" label="Show all Addresses" image="foldertree_nolines_plus" onclick="app.mail.showAllHeader"/> <buttononly class="et2_button ui-button" label="Show all Addresses" image="foldertree_nolines_plus" onclick="app.mail.showAllHeader"/>
</hbox> </old-hbox>
<hbox class="mailPreviewHeaders smimeIcons"> <old-hbox class="mailPreviewHeaders smimeIcons">
<image id="smime_signature" src="smime_sign" statustext="Smime signed message" disabled="true" align="right" width="24"/> <image id="smime_signature" src="smime_sign" statustext="Smime signed message" disabled="true" align="right" width="24"/>
<image id="smime_encryption" src="smime_encrypt" statustext="Smime encrypted message" disabled="true" align="right" width="24"/> <image id="smime_encryption" src="smime_encrypt" statustext="Smime encrypted message" disabled="true" align="right" width="24"/>
</hbox> </old-hbox>
<hbox id="mailPreviewHeadersAttachments" class="mailPreviewHeaders"> <old-hbox id="mailPreviewHeadersAttachments" class="mailPreviewHeaders">
<description value="Attachments"/> <description value="Attachments"/>
<grid disabled="@no_griddata" id="previewAttachmentArea" class="previewAttachmentArea egwGridView_grid"> <grid disabled="@no_griddata" id="previewAttachmentArea" class="previewAttachmentArea egwGridView_grid">
<columns> <columns>
@ -72,11 +72,11 @@
</rows> </rows>
</grid> </grid>
<buttononly class="et2_button ui-button" label="Show all attachments" image="foldertree_nolines_plus" width="16px" height="16px" onclick="app.mail.showAllHeader"/> <buttononly class="et2_button ui-button" label="Show all attachments" image="foldertree_nolines_plus" width="16px" height="16px" onclick="app.mail.showAllHeader"/>
</hbox> </old-hbox>
<box id="mailPreviewContainer"> <old-box id="mailPreviewContainer">
<iframe frameborder="1" id="messageIFRAME" scrolling="auto"/> <iframe frameborder="1" id="messageIFRAME" scrolling="auto"/>
</box> </old-box>
</vbox> </old-vbox>
</split> </split>
</template> </template>
<template id="mail.index.nosplitter" template="" lang="" group="0" version="1.9.001"> <template id="mail.index.nosplitter" template="" lang="" group="0" version="1.9.001">

View File

@ -850,15 +850,15 @@ span#mail-compose_replyto_expander {
padding: 0; padding: 0;
} }
/*Make file uploads in compose dialog invisible*/ /*Make file uploads in compose dialog invisible*/
div.mail-compose_toolbar_assist div.mail-compose_fileselector, .mail-compose_toolbar_assist div.mail-compose_fileselector,
#mail-compose_selectFromVFSForCompose, #mail-compose_selectFromVFSForCompose,
div.mail-compose_toolbar_assist { .mail-compose_toolbar_assist {
display: none; display: none;
} }
/*Make file uploads in compose dialog invisible*/ /*Make file uploads in compose dialog invisible*/
div.mail-compose_toolbar_assist div.mail-compose_fileselector, .mail-compose_toolbar_assist div.mail-compose_fileselector,
#mail-compose_selectFromVFSForCompose, #mail-compose_selectFromVFSForCompose,
div.mail-compose_toolbar_assist { .mail-compose_toolbar_assist {
display: none; display: none;
} }
div.mailComposeHeaderSection > table { div.mailComposeHeaderSection > table {
@ -2072,6 +2072,7 @@ table#mail-index_previewAttachmentArea::-webkit-scrollbar-thumb:hover {
#mail-index_nm.et2_nextmatch .egwGridView_outer thead tr { #mail-index_nm.et2_nextmatch .egwGridView_outer thead tr {
border-left: 12px solid #B4B4B4; border-left: 12px solid #B4B4B4;
} }
#mail-index_nm.et2_nextmatch.et2_nextmatch .egwGridView_outer .egwGridView_scrollarea tbody tr.row_category td:first-child > div { #mail-index_nm.et2_nextmatch .egwGridView_outer thead tr.row_category td:first-child > div {
margin-left: 0px; margin-left: 0px;
padding-left: 0px;
} }

8603
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@ -6,14 +6,23 @@
"repository": {}, "repository": {},
"scripts": { "scripts": {
"build": "rollup -c", "build": "rollup -c",
"build:watch": "rollup -cw" "build:watch": "rollup -cw",
"jstest": "tsc &> /dev/null; web-test-runner",
"jstest:watch": "web-test-runner --watch"
}, },
"devDependencies": { "devDependencies": {
"@babel/core": "^7.14.6", "@babel/core": "^7.14.6",
"@babel/preset-typescript": "^7.14.5", "@babel/preset-typescript": "^7.14.5",
"@open-wc/testing": "^2.5.33",
"@rollup/plugin-babel": "^5.3.0", "@rollup/plugin-babel": "^5.3.0",
"@rollup/plugin-node-resolve": "^13.0.0", "@rollup/plugin-node-resolve": "^13.0.0",
"@rollup/plugin-typescript": "^8.2.1", "@rollup/plugin-typescript": "^8.2.1",
"@types/chai": "^4.2.21",
"@types/mocha": "^8.2.3",
"@web/dev-server-esbuild": "^0.2.14",
"@web/dev-server-rollup": "^0.3.9",
"@web/test-runner": "^0.13.16",
"@web/test-runner-playwright": "^0.8.8",
"grunt": "^1.3.0", "grunt": "^1.3.0",
"grunt-contrib-cssmin": "^2.2.1", "grunt-contrib-cssmin": "^2.2.1",
"grunt-newer": "^1.3.0", "grunt-newer": "^1.3.0",
@ -21,6 +30,7 @@
"rimraf": "^3.0.2", "rimraf": "^3.0.2",
"rollup": "^2.52.2", "rollup": "^2.52.2",
"rollup-plugin-terser": "^7.0.2", "rollup-plugin-terser": "^7.0.2",
"sinon": "^11.1.2",
"terser": "^4.8.0", "terser": "^4.8.0",
"typescript": "^3.9.7" "typescript": "^3.9.7"
}, },
@ -44,9 +54,13 @@
}, },
"dependencies": { "dependencies": {
"@andxor/jquery-ui-touch-punch-fix": "^1.0.2", "@andxor/jquery-ui-touch-punch-fix": "^1.0.2",
"@lion/button": "^0.14.2",
"@lion/core": "^0.18.2",
"@lion/input": "^0.15.4",
"@lion/input-date": "^0.12.6",
"@lion/input-datepicker": "^0.23.6",
"@lion/textarea": "^0.13.4",
"jquery-ui-timepicker-addon": "^1.6.3", "jquery-ui-timepicker-addon": "^1.6.3",
"lit-element": "^2.5.1",
"lit-html": "^1.4.1",
"sortablejs": "^1.14.0" "sortablejs": "^1.14.0"
}, },
"engines": { "engines": {

View File

@ -28,7 +28,19 @@ readdirSync('./chunks').forEach(name => {
}); });
// Turn on minification // Turn on minification
const do_minify = true; const do_minify = false;
function isBareSpecifier (id) {
if (id.startsWith("./") || id.startsWith("../") || id.startsWith("/"))
return false;
try {
new URL(id);
return false;
}
catch {
return true;
}
}
const config = { const config = {
treeshake: false, treeshake: false,
@ -65,6 +77,11 @@ const config = {
}, },
plugins: [{ plugins: [{
resolveId (id, parentId) { resolveId (id, parentId) {
// Delegate bare specifiers to node_modules resolver
if (isBareSpecifier(id))
{
return;
}
if (!parentId || parentId.indexOf(path.sep + 'node_modules' + path.sep) !== -1) if (!parentId || parentId.indexOf(path.sep + 'node_modules' + path.sep) !== -1)
{ {
return; return;
@ -93,7 +110,9 @@ const config = {
} }
}, },
// resolve (external) node modules from node_modules directory // resolve (external) node modules from node_modules directory
resolve(), resolve({
browser: true
}),
{ {
transform (code, id) { transform (code, id) {
if (id.endsWith('.ts')) if (id.endsWith('.ts'))

View File

@ -0,0 +1,75 @@
/**
* This is the configuration file for automatic TypeScript testing
*
* It uses "web-test-runner" to run the tests, which are written using
* Mocha (https://mochajs.org/) & Chai Assertion Library (https://www.chaijs.com/api/assert/)
* Playwright (https://playwright.dev/docs/intro) runs the tests in actual browsers.
*
* Trouble getting tests to run? Try manually compiling TypeScript (source & tests), that seems to help.
*/
import fs from 'fs';
import {playwrightLauncher} from '@web/test-runner-playwright';
import {esbuildPlugin} from '@web/dev-server-esbuild';
// Get tests for web components (in their own directory)
const webComponents = fs.readdirSync('api/js/etemplate')
.filter(
dir => fs.statSync(`api/js/etemplate/${dir}`).isDirectory() && fs.existsSync(`api/js/etemplate/${dir}/test`),
)
.map(dir => `api/js/etemplate/${dir}/test`);
// Add any test files in app/js/test/
const appJS = fs.readdirSync('.')
.filter(
dir => fs.existsSync(`${dir}/js`) && fs.existsSync(`${dir}/js/test`) && fs.statSync(`${dir}/js/test`).isDirectory(),
)
export default {
nodeResolve: true,
coverageConfig: {
report: true,
reportDir: 'coverage',
threshold: {
statements: 90,
branches: 65,
functions: 80,
lines: 90,
},
},
testFramework: {
config: {
timeout: '3000',
},
},
browsers: [
playwrightLauncher({product: 'firefox', concurrency: 1}),
playwrightLauncher({product: 'chromium'}),
// Dependant on specific versions of shared libraries (libicuuc.so.66, latest is .67)
//playwrightLauncher({ product: 'webkit' }),
],
groups:
webComponents.map(pkg =>
{
return {
name: `${pkg}`,
files: `${pkg}/*.test.ts`
};
}).concat(
appJS.map(app =>
{
return {
name: app,
files: `${app}/js/**/*.test.ts`
}
}
)
)
,
plugins: [
// Handles typescript
esbuildPlugin({ts: true})
],
};