egroupware/api/js/etemplate/Et2Vfs/Et2VfsPath.ts

309 lines
9.2 KiB
TypeScript
Raw Normal View History

2024-01-25 23:24:11 +01:00
/**
* EGroupware eTemplate2 - Vfs path WebComponent
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package api
* @link https://www.egroupware.org
* @author Nathan Gray
*/
import {Et2InputWidget} from "../Et2InputWidget/Et2InputWidget";
import {html, LitElement, nothing} from "lit";
import shoelace from "../Styles/shoelace";
import styles from "./Et2VfsPath.styles";
import {property} from "lit/decorators/property.js";
import {state} from "lit/decorators/state.js";
import {classMap} from "lit/directives/class-map.js";
import {repeat} from "lit/directives/repeat.js";
import {FileInfo} from "./Et2VfsSelectDialog";
2024-01-25 23:24:11 +01:00
import {SlBreadcrumbItem} from "@shoelace-style/shoelace";
import {HasSlotController} from "../Et2Widget/slot";
/**
* @summary Display an editable path from the VFS
* @since
*
2024-02-21 17:11:45 +01:00
* @slot label - The inputs label. Alternatively, you can use the label attribute.
2024-01-25 23:24:11 +01:00
* @slot prefix - Before the path
* @slot suffix - Like prefix, but after
2024-02-21 17:11:45 +01:00
* @slot edit-icon - The icon that switches to editing the path as text.
2024-01-25 23:24:11 +01:00
* @slot help-text - Text that describes how to use the input. Alternatively, you can use the `help-text` attribute.
*
* @event change - Emitted when the control's value changes.
2024-02-21 17:11:45 +01:00
* @event {CustomEvent} click - Emitted when the user clicks on part of the path. `event.detail` contains the path.
2024-01-25 23:24:11 +01:00
*
* @csspart form-control-input - The textbox's wrapper.
* @csspart form-control-help-text - The help text's wrapper.
* @csspart prefix - The container that wraps the prefix slot.
* @csspart suffix - The container that wraps the suffix slot.
*
*/
export class Et2VfsPath extends Et2InputWidget(LitElement)
{
static get styles()
{
return [
shoelace,
...super.styles,
styles
];
}
/** The component's help text. If you need to display HTML, use the `help-text` slot instead. */
@property({attribute: 'help-text'}) helpText = '';
/* User is directly editing the path as a string */
@state() editing = false;
protected readonly hasSlotController = new HasSlotController(this, 'help-text', 'label');
private _value = ""
get _edit() : HTMLInputElement { return this.shadowRoot.querySelector("input");}
constructor()
{
super();
this.handleEditMouseDown = this.handleEditMouseDown.bind(this);
this.handleKeyDown = this.handleKeyDown.bind(this);
this.handlePathClick = this.handlePathClick.bind(this);
}
@property()
set value(_value : string)
{
try
{
_value = this.egw().decodePath(<string>_value);
}
catch(e)
{
this.set_validation_error('Error! ' + _value);
return;
}
const oldValue = this._value;
this._value = <string>_value;
this.requestUpdate("value", oldValue);
}
get value() { return this._value;}
setValue(_value : string | FileInfo)
{
if(typeof _value != "string" && _value.path)
{
_value = _value.path;
}
this.value = <string>_value;
}
getValue()
{
return (this.readonly || this.disabled) ? null : (this.egw().encodePath(this._value || ''));
}
public focus()
{
this.edit();
}
public blur()
{
this.editing = false;
this.requestUpdate("editing");
let oldValue = this.value;
this.value = this._edit.value;
if(oldValue != this.value)
{
this.updateComplete.then(() =>
{
this.dispatchEvent(new Event("change"));
})
}
}
public edit()
{
const oldValue = this.editing;
this.editing = true;
this.requestUpdate("editing", oldValue);
this.updateComplete.then(() =>
{
this._edit?.focus();
})
}
protected handleLabelClick()
{
this.edit();
}
protected handleEditMouseDown(event : MouseEvent)
{
this.edit();
}
protected handleKeyDown(event : KeyboardEvent)
{
switch(event.key)
{
case "Enter":
event.stopPropagation();
event.preventDefault();
this.editing = !this.editing;
this.requestUpdate("editing");
break;
case "Escape":
event.stopPropagation();
event.preventDefault();
this.blur();
break;
}
}
protected handlePathClick(event : MouseEvent)
{
if(event.target instanceof SlBreadcrumbItem && event.composedPath().includes(this))
{
event.preventDefault();
event.stopPropagation();
const dirs = Array.from(event.target.parentElement.querySelectorAll('sl-breadcrumb-item')) ?? [];
let stopIndex = dirs.indexOf(event.target) + 1;
2024-02-06 22:37:30 +01:00
let newPath = dirs.slice(0, stopIndex)
// Strip out any extra space
.map(d => d.textContent.trim().replace(/\/*$/, '').trim() + "/")
.filter(p => p);
if(newPath[0] !== '/')
{
// Make sure we start at /, breadcrumb parsing above might lose it
newPath.unshift('/');
}
2024-01-25 23:24:11 +01:00
if(!(this.disabled || this.readonly))
{
const oldValue = this.value;
2024-02-06 22:37:30 +01:00
// No trailing slash in the value
this.value = newPath.join("").replace(/\/*$/, '');
2024-01-25 23:24:11 +01:00
if(oldValue != this.value)
{
this.updateComplete.then(() =>
{
this.dispatchEvent(new Event("change"));
})
}
}
// Can still click on it when disabled I guess
if(!this.disabled)
{
this.dispatchEvent(new CustomEvent("click", {
bubbles: true,
cancelable: true,
detail: newPath.join("")
}));
}
}
}
render()
{
const hasLabelSlot = this.hasSlotController.test('label');
const hasHelpTextSlot = this.hasSlotController.test('help-text');
const hasLabel = this.label ? true : !!hasLabelSlot;
const hasHelpText = this.helpText ? true : !!hasHelpTextSlot;
2024-02-06 22:37:30 +01:00
// No trailing slash in the path
const pathParts = this.value === "/" ? [""] : this.value
// Remove trailing /
.replace(/\/*$/, '')
.split('/');
2024-01-25 23:24:11 +01:00
const isEditable = !(this.disabled || this.readonly);
const editing = this.editing && isEditable;
return html`
<div
part="form-control"
class=${classMap({
'vfs-path': true,
'vfs-path__readonly': !isEditable,
'vfs-path__disabled': this.disabled,
'form-control': true,
'form-control--medium': true,
'form-control--has-label': hasLabel,
'form-control--has-help-text': hasHelpText
})}
>
<label
id="label"
part="form-control-label"
class="form-control__label"
aria-hidden=${hasLabel ? 'false' : 'true'}
@click=${this.handleLabelClick}
>
<slot name="label">${this.label}</slot>
</label>
<div part="form-control-input" class="form-control-input"
@click=${() => this.focus()}
>
<slot part="prefix" name="prefix"></slot>
${editing ? html`
<input
class="vfs-path__value-input"
type="text"
?disabled=${this.disabled}
?required=${this.required}
.value=${this.value}
tabindex="-1"
aria-hidden="true"
@blur=${() => this.blur()}
@keydown=${this.handleKeyDown}
/>` : html`
<sl-breadcrumb
class="vfs-path__breadcrumb"
@click=${this.handlePathClick}
>
<span slot="separator">/</span>
${repeat(pathParts, (path) =>
{
return html`
2024-02-06 22:37:30 +01:00
<sl-breadcrumb-item class="vfs-path__directory">${path.trim()}
<span slot="separator">/</span>
</sl-breadcrumb-item>`;
2024-01-25 23:24:11 +01:00
})}
</sl-breadcrumb>
${!isEditable ? nothing : html`
<button
part="edit-button"
class="vfs-path__edit"
type="button"
aria-label=${this.egw().lang('edit')}
@mousedown=${this.handleEditMouseDown}
tabindex="-1"
>
<slot name="edit-icon">
<sl-icon name="pencil"></sl-icon>
</slot>
</button>`
}
`}
<slot part="suffix" name="suffix"></slot>
</div>
<div
part="form-control-help-text"
id="help-text"
class="form-control__help-text"
aria-hidden=${hasHelpText ? 'false' : 'true'}
>
<slot name="help-text">${this.helpText}</slot>
</div>
</div>
`;
}
}
customElements.define("et2-vfs-path", Et2VfsPath);