Files
AFFiNE-Mirror/blocksuite/affine/components/src/toolbar/menu-button.ts
2024-12-20 15:38:06 +08:00

227 lines
4.9 KiB
TypeScript

import { PANEL_BASE } from '@blocksuite/affine-shared/styles';
import { createButtonPopper } from '@blocksuite/affine-shared/utils';
import { WithDisposable } from '@blocksuite/global/utils';
import {
css,
html,
LitElement,
type PropertyValues,
type TemplateResult,
} from 'lit';
import { property, query } from 'lit/decorators.js';
import type { EditorIconButton } from './icon-button.js';
export class EditorMenuButton extends WithDisposable(LitElement) {
static override styles = css`
:host {
display: flex;
align-items: center;
justify-content: center;
gap: 8px;
}
`;
private _popper!: ReturnType<typeof createButtonPopper>;
override firstUpdated() {
this._popper = createButtonPopper(
this._trigger,
this._content,
({ display }) => {
const opened = display === 'show';
this._trigger.showTooltip = !opened;
this.dispatchEvent(
new CustomEvent('toggle', {
detail: opened,
bubbles: false,
cancelable: false,
composed: true,
})
);
},
{
mainAxis: 12,
ignoreShift: true,
}
);
this._disposables.addFromEvent(this, 'keydown', (e: KeyboardEvent) => {
e.stopPropagation();
if (e.key === 'Escape') {
this._popper.hide();
}
});
this._disposables.addFromEvent(this._trigger, 'click', (_: MouseEvent) => {
this._popper.toggle();
if (this._popper.state === 'show') {
this._content.focus({ preventScroll: true });
}
});
this._disposables.add(this._popper);
}
hide() {
this._popper?.hide();
}
override render() {
return html`
${this.button}
<editor-menu-content role="menu" tabindex="-1">
<slot></slot>
</editor-menu-content>
`;
}
show(force = false) {
this._popper?.show(force);
}
override willUpdate(changedProperties: PropertyValues) {
if (changedProperties.has('contentPadding')) {
this.style.setProperty('--content-padding', this.contentPadding ?? '');
}
}
@query('editor-menu-content')
private accessor _content!: EditorMenuContent;
@query('editor-icon-button')
private accessor _trigger!: EditorIconButton;
@property({ attribute: false })
accessor button!: string | TemplateResult<1>;
@property({ attribute: false })
accessor contentPadding: string | undefined = undefined;
}
export class EditorMenuContent extends LitElement {
static override styles = css`
:host {
--packed-height: 6px;
--offset-height: calc(-1 * var(--packed-height));
display: none;
outline: none;
}
:host::before,
:host::after {
content: '';
display: block;
position: absolute;
height: var(--packed-height);
width: 100%;
}
:host::before {
top: var(--offset-height);
}
:host::after {
bottom: var(--offset-height);
}
:host([data-show]) {
${PANEL_BASE};
justify-content: center;
padding: var(--content-padding, 0 6px);
}
::slotted(:not(.custom)) {
display: flex;
align-items: center;
align-self: stretch;
gap: 8px;
min-height: 36px;
}
::slotted([data-size]) {
min-width: 146px;
}
::slotted([data-size='small']) {
min-width: 164px;
}
::slotted([data-size='large']) {
min-width: 176px;
}
::slotted([data-orientation='vertical']) {
flex-direction: column;
align-items: stretch;
gap: unset;
min-height: unset;
}
`;
override render() {
return html`<slot></slot>`;
}
}
export class EditorMenuAction extends LitElement {
static override styles = css`
:host {
display: flex;
width: 100%;
align-items: center;
white-space: nowrap;
box-sizing: border-box;
padding: 4px 8px;
border-radius: 4px;
overflow: hidden;
text-overflow: ellipsis;
cursor: pointer;
gap: 8px;
color: var(--affine-text-primary-color);
font-weight: 400;
min-height: 30px; // 22 + 8
}
:host(:hover),
:host([data-selected]) {
background-color: var(--affine-hover-color);
}
:host([data-selected]) {
pointer-events: none;
}
:host(:hover.delete),
:host(:hover.delete) ::slotted(svg) {
background-color: var(--affine-background-error-color);
color: var(--affine-error-color);
}
:host([disabled]) {
pointer-events: none;
cursor: not-allowed;
color: var(--affine-text-disable-color);
}
::slotted(svg) {
color: var(--affine-icon-color);
}
`;
override connectedCallback() {
super.connectedCallback();
this.role = 'button';
}
override render() {
return html`<slot></slot>`;
}
}
declare global {
interface HTMLElementTagNameMap {
'editor-menu-button': EditorMenuButton;
'editor-menu-content': EditorMenuContent;
'editor-menu-action': EditorMenuAction;
}
}