@eclipse-scout/core
Version:
Eclipse Scout runtime
127 lines (109 loc) • 4.49 kB
text/typescript
/*
* Copyright (c) 2010, 2024 BSI Business Systems Integration AG
*
* This program and the accompanying materials are made
* available under the terms of the Eclipse Public License 2.0
* which is available at https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*/
import {Action, aria, KeyStrokeContext, Menu, NavigateButtonModel, Outline, Page, SomeRequired} from '../../../index';
import $ from 'jquery';
/**
* The outline navigation works mostly browser-side. The navigation logic is implemented in JavaScript.
* When a navigation button is clicked, we process that click browser-side first and send an event to
* the server which nodes have been selected. We do that for better user experience. In a first attempt
* the whole navigation logic was on the server, which caused a lag and flickering in the UI.
*/
export abstract class NavigateButton extends Menu implements NavigateButtonModel {
declare model: NavigateButtonModel;
declare initModel: SomeRequired<this['model'], 'parent' | 'node' | 'outline'>;
node: Page;
outline: Outline;
altKeyStrokeContext: KeyStrokeContext;
overflow: boolean;
protected _defaultText: string;
protected _defaultIconId: string;
constructor() {
super();
this.node = null;
this.outline = null;
this.actionStyle = Action.ActionStyle.BUTTON;
this._addCloneProperties(['node', 'outline', 'altKeyStrokeContext']);
this.inheritAccessibility = false;
}
protected override _render() {
if (this.overflow) {
this.text = this.session.text(this._defaultText);
this.iconId = null;
} else {
this.text = null;
this.iconId = this._defaultIconId;
}
this.updateEnabled();
super._render();
this.$container.addClass('navigate-button small');
this.altKeyStrokeContext.registerKeyStroke(this);
}
protected override _renderProperties() {
super._renderProperties();
aria.label(this.$container, this.session.text(this._defaultText));
}
protected override _remove() {
super._remove();
this.altKeyStrokeContext.unregisterKeyStroke(this);
}
/**
* Toggles the visibility of the detail form according to {@link _toggleDetail}. Called when the button is clicked and {@link _isDetail} is `true`.
*/
protected _setDetailVisible() {
let detailVisible = this._toggleDetail();
$.log.isDebugEnabled() && $.log.debug('show detail-' + (detailVisible ? 'form' : 'table'));
this.outline.setDetailFormVisibleByUi(this.node, detailVisible);
}
/**
* @returns `true` if the page has a detail form and the detail form is marked as visible.
*/
protected _hasDetailForm(): boolean {
return this.node.detailFormVisible && !!this.node.detailForm;
}
/**
* @returns `true` if the page has a detail table and the detail table is marked as visible and the detail table contains
* at least one row. The last requirement can be lifted by setting the optional parameter `requireRows` to false.
*/
protected _hasDetailTable(requireRows = true): boolean {
return this.node.detailTableVisible && !!this.node.detailTable && (this.node.detailTable.rows.length > 0 || !requireRows);
}
protected override _doAction() {
super._doAction();
if (this._isDetail()) {
this._setDetailVisible();
} else {
this._drill();
}
}
/**
* Returns `true` if the button is in "detail toggle mode", i .e. the page has both a detail form and a detail table and the button
* should toggle between these two detail views instead of changing the selected page. In other words, when this is `true`, clicking
* the button does not {@link _drill drill} up or down like normal, but only changes the displayed "detail content".
*/
protected abstract _isDetail(): boolean;
/**
* Sets the outline selection to the child or parent page. Called when the button is clicked and {@link _isDetail} is `false`.
*/
protected abstract _drill();
/**
* Specifies the new value of {@link Page#detailFormVisibleByUi} when toggling the detail content instead of changing the selected page.
*/
protected abstract _toggleDetail(): boolean;
/**
* Computes whether this button should be enabled in the current state of the application.
*/
protected abstract _buttonEnabled(): boolean;
/**
* Called when enabled state must be re-calculated and probably rendered.
*/
updateEnabled() {
this.setEnabled(this._buttonEnabled());
}
}