zul/src/main/resources/web/js/zul/sel/Option.ts

Summary

Maintainability
A
0 mins
Test Coverage
/* Option.ts

    Purpose:

    Description:

    History:
        Mon Jun  1 16:43:59     2009, Created by jumperchen

Copyright (C) 2009 Potix Corporation. All Rights Reserved.

This program is distributed under LGPL Version 2.0 in the hope that
it will be useful, but WITHOUT ANY WARRANTY.
*/
/**
 * A HTML option tag.
 */
@zk.WrapClass('zul.sel.Option')
export class Option extends zul.Widget<HTMLOptionElement> {
    // Parent could be null as asserted by `focus`.
    override parent!: zul.sel.Select | undefined; // FIXME: could parent be optgroup?
    // The type of firstChild is determined by the comment in getLabel
    override firstChild!: zul.sel.Listcell | undefined;
    override lastChild!: zul.sel.Listcell | undefined;
    /** @internal */
    _selected = false;
    /** @internal */
    _disabled?: boolean;
    /** @internal */
    _value?: string;
    /** @internal */
    __updating__?: boolean;

    /**
     * @returns whether it is disabled.
     * @defaultValue `false`.
     */
    isDisabled(): boolean {
        return !!this._disabled;
    }

    /**
     * Sets whether it is disabled.
     */
    setDisabled(disabled: boolean, opts?: Record<string, boolean>): this {
        const o = this._disabled;
        this._disabled = disabled;

        if (o !== disabled || opts?.force) {
            var n = this.$n();
            if (n) n.disabled = (disabled ? 'disabled' : '') as unknown as boolean;
        }

        return this;
    }

    /**
     * @returns the value.
     * @defaultValue `null`.
     * <p>Note: the value is application dependent, you can place
     * whatever value you want.
     * <p>If you are using listitem with HTML Form (and with
     * the name attribute), it is better to specify a String-typed
     * value.
     */
    getValue(): string | undefined {
        return this._value;
    }

    /**
     * Sets the value.
     * @param value - the value.
     * <p>Note: the value is application dependent, you can place
     * whatever value you want.
     * <p>If you are using listitem with HTML Form (and with
     * the name attribute), it is better to specify a String-typed
     * value.
     */
    setValue(value: string): this {
        this._value = value;
        return this;
    }

    /**
     * Sets the label of the {@link Option} it contains.
     *
     * <p>If it is not created, we automatically create it.
     * @since 10.0.0
     */
    // To treat as "value" attribute from "label" at client side
    setLabel(label: string): this {
        this._autoFirstCell().setLabel(label);
        return this;
    }

    /** @internal */
    _autoFirstCell(): zul.sel.Listcell {
        if (!this.firstChild)
            this.appendChild(new zul.sel.Listcell(), true);
        return this.firstChild!; // guaranteed to exist because appended in the previous line
    }

    override focus(timeout?: number): boolean {
        var p = this.parent;
        if (p) p.focus(timeout);
        // NOTE: Returning false agrees with the original logic (returning nothing,
        // which is will evaluate to undefined, which is falsey).
        return false;
    }

    override setVisible(visible: boolean, fromServer?: boolean): this {
        if (this._visible != visible) {
            this._visible = visible;
            if (this.desktop)
                this.parent!.requestRerender_(fromServer);
        }
        return this;
    }

    /**
     * Sets whether it is selected.
     */
    setSelected(selected: boolean): this {
        if (this.__updating__) { // for B50-3012466.zul
            delete this.__updating__;
            return this; //nothing to do for second loop triggered by this.parent.toggleItemSelection
        }
        try {
            selected = selected || false;
            this.__updating__ = true;
            if (this._selected != selected) {
                if (this.parent)
                    this.parent.toggleItemSelection(this);
                this._setSelectedDirectly(selected); // always setting for B50-3012466.zul
            }
        } finally {
            delete this.__updating__;
        }
        return this;
    }

    /** @internal */
    _setSelectedDirectly(selected: boolean): void {
        var n = this.$n();
        // Bug ZK-2285, ignore if the status is the same for IE's issue
        if (n && n.selected != selected) {
            n.selected = (selected ? 'selected' : '') as unknown as boolean;
        }
        this._selected = selected;
    }

    /**
     * @returns whether it is selected.
     * @defaultValue `false`.
     */
    isSelected(): boolean {
        return this._selected;
    }

    /**
     * @returns the label of the {@link Listcell} it contains, or null
     * if no such cell.
     */
    getLabel(): string | undefined {
        return this.firstChild ? this.firstChild.getLabel() : undefined;
    }

    /** @internal */
    updateLabel_(): void {
        var n = this.$n();
        // eslint-disable-next-line @microsoft/sdl/no-inner-html
        if (n) n.innerHTML = DOMPurify.sanitize(this.domLabel_());
    }

    /**
     * @returns the maximal length of each item's label.
     * It is a shortcut of {@link Select#getMaxlength}.
     */
    getMaxlength(): number | undefined {
        return this.parent ? this.parent.getMaxlength() : 0;
    }

    /** @internal */
    override bind_(desktop?: zk.Desktop, skipper?: zk.Skipper, after?: CallableFunction[]): void {
        super.bind_(desktop, skipper, after);
        //B60-ZK-1303: force update parent's selected index.
        if (this.isSelected()) {
            this.parent!.updateSelectionDirectly(this);
        }
    }

    /** @internal */
    override doClick_(evt: zk.Event /*, popupOnly?: boolean */): void {
        evt.stop(); // Eats the non-standard onclick event
    }

    /**
     * The index for option widget only , not including the listhead.etc
     * @since 6.0.1
     * @internal
     */
    getOptionIndex_(): number {
        var parent = this.parent, ret = -1;
        if (parent) {
            for (var w = parent.firstChild; w; w = w.nextSibling) {
                if (w instanceof zul.sel.Option) {
                    ret++;
                    if (w == this) break;
                }
            }
        }
        return ret;
    }

    /** @internal */
    domLabel_(): string {
        return zUtl.encodeXML(this.getLabel()!, {maxlength: this.getMaxlength()!});
    }

    /** @internal */
    override domAttrs_(no?: zk.DomAttrsOptions): string {
        var value = this.getValue(),
            shallRenderValue = value && this.parent && this.parent.getName();
        return super.domAttrs_(no) + (this.isDisabled() ? ' disabled="disabled"' : '')
            + (this.isSelected() ? ' selected="selected"' : '') + (shallRenderValue ? ' value="' + value + '"' : '');
    }

    override replaceWidget(newwgt: zul.sel.Option, skipper?: zk.Skipper): void {
        this._syncItems(newwgt);
        super.replaceWidget(newwgt, skipper);
    }

    /** @internal */
    _syncItems(newwgt: zul.sel.Option): void {
        if (this.parent && this.isSelected()) {
            var items = this.parent._selItems;
            if (items && items.$remove(this))
                items.push(newwgt);
        }
    }
}