META-INF.dirigible.dev-tools.ui.ListWidget.js Maven / Gradle / Ivy
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import * as Common from '../common/common.js';
import * as ARIAUtils from './ARIAUtils.js';
import {Toolbar, ToolbarButton} from './Toolbar.js';
import {createInput, createTextButton, ElementFocusRestorer} from './UIUtils.js';
import {VBox} from './Widget.js';
/**
* @template T
*/
export class ListWidget extends VBox {
/**
* @param {!Delegate} delegate
*/
constructor(delegate) {
super(true, true /* delegatesFocus */);
this.registerRequiredCSS('ui/listWidget.css');
this._delegate = delegate;
this._list = this.contentElement.createChild('div', 'list');
this._list.addEventListener('keydown', event => this._onKeyDown(event));
this._lastSeparator = false;
/** @type {?ElementFocusRestorer} */
this._focusRestorer = null;
/** @type {!Array} */
this._items = [];
/** @type {!Array} */
this._editable = [];
/** @type {!Array} */
this._elements = [];
/** @type {?Editor} */
this._editor = null;
/** @type {?T} */
this._editItem = null;
/** @type {?Element} */
this._editElement = null;
this._selectedIndex = -1;
/** @type {?Element} */
this._emptyPlaceholder = null;
this._updatePlaceholder();
}
clear() {
this._items = [];
this._editable = [];
this._elements = [];
this._lastSeparator = false;
this._list.removeChildren();
this._updatePlaceholder();
this._stopEditing();
}
/**
* @param {!T} item
* @param {boolean} editable
*/
appendItem(item, editable) {
if (this._lastSeparator && this._items.length) {
this._list.appendChild(createElementWithClass('div', 'list-separator'));
}
this._lastSeparator = false;
this._items.push(item);
this._editable.push(editable);
const element = this._list.createChild('div', 'list-item');
element.appendChild(this._delegate.renderItem(item, editable));
if (editable) {
element.classList.add('editable');
element.appendChild(this._createControls(item, element));
}
const index = this._items.length - 1;
element.addEventListener('click', () => {
this._select(index, /* takeFocus */ true);
});
this._elements.push(element);
if (this._selectedIndex === -1 || this._selectedIndex === index) {
this._select(index, /* takeFocus */ false);
}
this._updatePlaceholder();
}
appendSeparator() {
this._lastSeparator = true;
}
/**
* @param {number} index
*/
removeItem(index) {
if (this._editItem === this._items[index]) {
this._stopEditing();
}
const element = this._elements[index];
const previous = element.previousElementSibling;
const previousIsSeparator = previous && previous.classList.contains('list-separator');
const next = element.nextElementSibling;
const nextIsSeparator = next && next.classList.contains('list-separator');
if (previousIsSeparator && (nextIsSeparator || !next)) {
previous.remove();
}
if (nextIsSeparator && !previous) {
next.remove();
}
element.remove();
if (this._selectedIndex === index) {
this._selectNext();
}
this._elements.splice(index, 1);
this._items.splice(index, 1);
this._editable.splice(index, 1);
this._updatePlaceholder();
}
/**
* @param {number} index
* @param {!T} item
*/
addNewItem(index, item) {
this._startEditing(item, null, this._elements[index] || null);
}
/**
* @param {?Element} element
*/
setEmptyPlaceholder(element) {
this._emptyPlaceholder = element;
this._updatePlaceholder();
}
/**
* @param {!Event} event
*/
_onKeyDown(event) {
if (this._editor || this._elements.length < 1) {
return;
}
if (event.key === 'ArrowUp' || event.key === 'ArrowDown') {
if (this._selectedIndex < 0) {
return;
}
const offset = event.key === 'ArrowUp' ? -1 : 1;
const newIndex = this._selectedIndex + offset;
if (newIndex < 0 || newIndex >= this._elements.length) {
return;
}
this._select(newIndex, /* takeFocus */ true);
event.consume(true);
}
}
/**
* @param {number} index
* @param {boolean} takeFocus
*/
_select(index, takeFocus) {
if (index < 0 || index >= this._elements.length) {
return;
}
if (this._selectedIndex >= 0) {
const oldSelectedElement = this._elements[this._selectedIndex].firstElementChild;
oldSelectedElement.tabIndex = -1;
}
const newSelectedElement = this._elements[index].firstElementChild;
newSelectedElement.tabIndex = 0;
this._selectedIndex = index;
if (takeFocus) {
newSelectedElement.focus();
}
}
_selectNext() {
if (this._selectedIndex < 0 || this._list.length === 0) {
return;
}
const offset = this._selectedIndex < this._list.length ? 1 : -1;
const nextIndex = this._selectedIndex + offset;
this._select(nextIndex, /* takeFocus */ false);
}
/**
* @param {!T} item
* @param {!Element} element
* @return {!Element}
*/
_createControls(item, element) {
const controls = createElementWithClass('div', 'controls-container fill');
controls.createChild('div', 'controls-gradient');
const buttons = controls.createChild('div', 'controls-buttons');
const toolbar = new Toolbar('', buttons);
const editButton = new ToolbarButton(Common.UIString.UIString('Edit'), 'largeicon-edit');
editButton.addEventListener(ToolbarButton.Events.Click, onEditClicked.bind(this));
toolbar.appendToolbarItem(editButton);
const removeButton = new ToolbarButton(Common.UIString.UIString('Remove'), 'largeicon-trash-bin');
removeButton.addEventListener(ToolbarButton.Events.Click, onRemoveClicked.bind(this));
toolbar.appendToolbarItem(removeButton);
return controls;
/**
* @this {ListWidget}
*/
function onEditClicked() {
const index = this._elements.indexOf(element);
const insertionPoint = this._elements[index + 1] || null;
this._startEditing(item, element, insertionPoint);
}
/**
* @this {ListWidget}
*/
function onRemoveClicked() {
const index = this._elements.indexOf(element);
this.element.focus();
this._delegate.removeItemRequested(this._items[index], index);
}
}
/**
* @override
*/
wasShown() {
super.wasShown();
this._stopEditing();
}
_updatePlaceholder() {
if (!this._emptyPlaceholder) {
return;
}
if (!this._elements.length && !this._editor) {
this._list.appendChild(this._emptyPlaceholder);
} else {
this._emptyPlaceholder.remove();
}
}
/**
* @param {!T} item
* @param {?Element} element
* @param {?Element} insertionPoint
*/
_startEditing(item, element, insertionPoint) {
if (element && this._editElement === element) {
return;
}
this._stopEditing();
this._focusRestorer = new ElementFocusRestorer(this.element);
this._list.classList.add('list-editing');
this._editItem = item;
this._editElement = element;
if (element) {
element.classList.add('hidden');
}
const index = element ? this._elements.indexOf(element) : -1;
this._editor = this._delegate.beginEdit(item);
this._updatePlaceholder();
this._list.insertBefore(this._editor.element, insertionPoint);
this._editor.beginEdit(
item, index, element ? Common.UIString.UIString('Save') : Common.UIString.UIString('Add'),
this._commitEditing.bind(this), this._stopEditing.bind(this));
}
_commitEditing() {
const editItem = this._editItem;
const isNew = !this._editElement;
const editor = /** @type {!Editor} */ (this._editor);
this._stopEditing();
this._delegate.commitEdit(editItem, editor, isNew);
}
_stopEditing() {
this._list.classList.remove('list-editing');
if (this._focusRestorer) {
this._focusRestorer.restore();
}
if (this._editElement) {
this._editElement.classList.remove('hidden');
}
if (this._editor && this._editor.element.parentElement) {
this._editor.element.remove();
}
this._editor = null;
this._editItem = null;
this._editElement = null;
this._updatePlaceholder();
}
}
/**
* @template T
* @interface
*/
export class Delegate {
/**
* @param {!T} item
* @param {boolean} editable
* @return {!Element}
*/
renderItem(item, editable) {
}
/**
* @param {!T} item
* @param {number} index
*/
removeItemRequested(item, index) {
}
/**
* @param {!T} item
* @return {!Editor}
*/
beginEdit(item) {
}
/**
* @param {!T} item
* @param {!Editor} editor
* @param {boolean} isNew
*/
commitEdit(item, editor, isNew) {}
}
/**
* @template T
*/
export class Editor {
constructor() {
this.element = createElementWithClass('div', 'editor-container');
this.element.addEventListener('keydown', onKeyDown.bind(null, isEscKey, this._cancelClicked.bind(this)), false);
this.element.addEventListener('keydown', onKeyDown.bind(null, isEnterKey, this._commitClicked.bind(this)), false);
this._contentElement = this.element.createChild('div', 'editor-content');
const buttonsRow = this.element.createChild('div', 'editor-buttons');
this._commitButton = createTextButton('', this._commitClicked.bind(this), '', true /* primary */);
buttonsRow.appendChild(this._commitButton);
this._cancelButton = createTextButton(Common.UIString.UIString('Cancel'), this._cancelClicked.bind(this));
this._cancelButton.addEventListener(
'keydown', onKeyDown.bind(null, isEnterKey, this._cancelClicked.bind(this)), false);
buttonsRow.appendChild(this._cancelButton);
this._errorMessageContainer = this.element.createChild('div', 'list-widget-input-validation-error');
ARIAUtils.markAsAlert(this._errorMessageContainer);
/**
* @param {function(!Event):boolean} predicate
* @param {function()} callback
* @param {!Event} event
*/
function onKeyDown(predicate, callback, event) {
if (predicate(event)) {
event.consume(true);
callback();
}
}
/** @type {!Array} */
this._controls = [];
/** @type {!Map} */
this._controlByName = new Map();
/** @type {!Array} */
this._validators = [];
/** @type {?function()} */
this._commit = null;
/** @type {?function()} */
this._cancel = null;
/** @type {?T} */
this._item = null;
/** @type {number} */
this._index = -1;
}
/**
* @return {!Element}
*/
contentElement() {
return this._contentElement;
}
/**
* @param {string} name
* @param {string} type
* @param {string} title
* @param {function(!T, number, (!HTMLInputElement|!HTMLSelectElement)): !ValidatorResult} validator
* @return {!HTMLInputElement}
*/
createInput(name, type, title, validator) {
const input = /** @type {!HTMLInputElement} */ (createInput('', type));
input.placeholder = title;
input.addEventListener('input', this._validateControls.bind(this, false), false);
input.addEventListener('blur', this._validateControls.bind(this, false), false);
ARIAUtils.setAccessibleName(input, title);
this._controlByName.set(name, input);
this._controls.push(input);
this._validators.push(validator);
return input;
}
/**
* @param {string} name
* @param {!Array} options
* @param {function(!T, number, (!HTMLInputElement|!HTMLSelectElement)): !ValidatorResult} validator
* @param {string=} title
* @return {!HTMLSelectElement}
*/
createSelect(name, options, validator, title) {
const select = /** @type {!HTMLSelectElement} */ (createElementWithClass('select', 'chrome-select'));
for (let index = 0; index < options.length; ++index) {
const option = select.createChild('option');
option.value = options[index];
option.textContent = options[index];
}
if (title) {
select.title = title;
ARIAUtils.setAccessibleName(select, title);
}
select.addEventListener('input', this._validateControls.bind(this, false), false);
select.addEventListener('blur', this._validateControls.bind(this, false), false);
this._controlByName.set(name, select);
this._controls.push(select);
this._validators.push(validator);
return select;
}
/**
* @param {string} name
* @return {!HTMLInputElement|!HTMLSelectElement}
*/
control(name) {
return /** @type {!HTMLInputElement|!HTMLSelectElement} */ (this._controlByName.get(name));
}
/**
* @param {boolean} forceValid
*/
_validateControls(forceValid) {
let allValid = true;
this._errorMessageContainer.textContent = '';
for (let index = 0; index < this._controls.length; ++index) {
const input = this._controls[index];
const {valid, errorMessage} = this._validators[index].call(null, this._item, this._index, input);
input.classList.toggle('error-input', !valid && !forceValid);
if (valid || forceValid) {
ARIAUtils.setInvalid(input, false);
} else {
ARIAUtils.setInvalid(input, true);
}
if (!forceValid && errorMessage && !this._errorMessageContainer.textContent) {
this._errorMessageContainer.textContent = errorMessage;
}
allValid &= valid;
}
this._commitButton.disabled = !allValid;
}
/**
* @param {!T} item
* @param {number} index
* @param {string} commitButtonTitle
* @param {function()} commit
* @param {function()} cancel
*/
beginEdit(item, index, commitButtonTitle, commit, cancel) {
this._commit = commit;
this._cancel = cancel;
this._item = item;
this._index = index;
this._commitButton.textContent = commitButtonTitle;
this.element.scrollIntoViewIfNeeded(false);
if (this._controls.length) {
this._controls[0].focus();
}
this._validateControls(true);
}
_commitClicked() {
if (this._commitButton.disabled) {
return;
}
const commit = this._commit;
this._commit = null;
this._cancel = null;
this._item = null;
this._index = -1;
commit();
}
_cancelClicked() {
const cancel = this._cancel;
this._commit = null;
this._cancel = null;
this._item = null;
this._index = -1;
cancel();
}
}
/** @typedef {{valid: boolean, errorMessage: (string|undefined)}} */
export let ValidatorResult;
© 2015 - 2025 Weber Informatics LLC | Privacy Policy