- 
      
- 
        Save nul800sebastiaan/6032cff7b6e5aaeed222697cc21a6b46 to your computer and use it in GitHub Desktop. 
    A small example of how to create a custom element to be placed within the Workspace Action part for Umbraco 15+. This takes the existing element that is placed there and adds an icon in the button.
  
        
  
    
      This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
      Learn more about bidirectional Unicode characters
    
  
  
    
  | /* | |
| * This file is a copy of the core version from https://github.com/umbraco/Umbraco-CMS/blob/6bca91a52fabfe96b7ede1837b908b33d78136ed/src/Umbraco.Web.UI.Client/src/packages/core/workspace/components/workspace-action/default/workspace-action-default-kind.element.ts | |
| * with the addition of a document icon in the button! | |
| */ | |
| import type { | |
| ManifestWorkspaceAction, | |
| ManifestWorkspaceActionMenuItem, | |
| MetaWorkspaceActionDefaultKind, | |
| UmbWorkspaceActionDefaultKind, | |
| } from '@umbraco-cms/backoffice/workspace'; | |
| import { UmbActionExecutedEvent } from '@umbraco-cms/backoffice/event'; | |
| import { html, customElement, property, state, when } from '@umbraco-cms/backoffice/external/lit'; | |
| import type { UUIButtonState } from '@umbraco-cms/backoffice/external/uui'; | |
| import { UmbLitElement } from '@umbraco-cms/backoffice/lit-element'; | |
| import { umbExtensionsRegistry } from '@umbraco-cms/backoffice/extension-registry'; | |
| import { | |
| type UmbExtensionElementAndApiInitializer, | |
| UmbExtensionsElementAndApiInitializer, | |
| } from '@umbraco-cms/backoffice/extension-api'; | |
| import { stringOrStringArrayIntersects } from '@umbraco-cms/backoffice/utils'; | |
| @customElement('custom-element') | |
| export class CustomElement< | |
| MetaType extends MetaWorkspaceActionDefaultKind = MetaWorkspaceActionDefaultKind, | |
| ApiType extends UmbWorkspaceActionDefaultKind<MetaType> = UmbWorkspaceActionDefaultKind<MetaType>, | |
| > extends UmbLitElement { | |
| #manifest?: ManifestWorkspaceAction<MetaType>; | |
| #api?: ApiType; | |
| #extensionsController?: UmbExtensionsElementAndApiInitializer< | |
| ManifestWorkspaceActionMenuItem, | |
| 'workspaceActionMenuItem', | |
| ManifestWorkspaceActionMenuItem | |
| >; | |
| @property({ type: Object, attribute: false }) | |
| public set manifest(value: ManifestWorkspaceAction<MetaType> | undefined) { | |
| if (!value) return; | |
| const oldValue = this.#manifest; | |
| if (oldValue !== value) { | |
| this.#manifest = value; | |
| this._href = value?.meta.href; | |
| this._additionalOptions = value?.meta.additionalOptions; | |
| this.#createAliases(); | |
| } | |
| } | |
| public get manifest() { | |
| return this.#manifest; | |
| } | |
| @property({ attribute: false }) | |
| public set api(api: ApiType | undefined) { | |
| this.#api = api; | |
| this.#api?.getHref?.().then((href) => { | |
| this._href = href ?? this.manifest?.meta.href; | |
| }); | |
| this.#api?.hasAdditionalOptions?.().then((additionalOptions) => { | |
| this._additionalOptions = additionalOptions ?? this.manifest?.meta.additionalOptions; | |
| }); | |
| this.#observeIsDisabled(); | |
| } | |
| public get api(): ApiType | undefined { | |
| return this.#api; | |
| } | |
| @state() | |
| private _buttonState?: UUIButtonState; | |
| @state() | |
| private _additionalOptions?: boolean; | |
| @state() | |
| private _href?: string; | |
| @state() | |
| _isDisabled = false; | |
| @state() | |
| private _items: Array<UmbExtensionElementAndApiInitializer<ManifestWorkspaceActionMenuItem>> = []; | |
| /** | |
| * Create a list of original and overwritten aliases of workspace actions for the action. | |
| */ | |
| async #createAliases() { | |
| if (!this.#manifest) return; | |
| const aliases = new Set<string>(); | |
| if (this.#manifest) { | |
| aliases.add(this.#manifest.alias); | |
| // TODO: This works on one level for now, which will be enough for the current use case. However, you can overwrite the overwrites, so we need to make this recursive. Perhaps we could move this to the extensions initializer. | |
| // Add overwrites so that we can show any previously registered actions on the original workspace action | |
| if (this.#manifest.overwrites) { | |
| const overwrites = Array.isArray(this.#manifest.overwrites) | |
| ? this.#manifest.overwrites | |
| : [this.#manifest.overwrites]; | |
| for (const alias of overwrites) { | |
| aliases.add(alias); | |
| } | |
| } | |
| } | |
| this.#observeExtensions(Array.from(aliases)); | |
| } | |
| async #onClick(event: MouseEvent) { | |
| if (this._href) { | |
| event.stopPropagation(); | |
| } | |
| // If its a link or has additional options, then we do not want to display state on the button. [NL] | |
| if (!this._href) { | |
| if (!this._additionalOptions) { | |
| this._buttonState = 'waiting'; | |
| } | |
| try { | |
| if (!this.#api) throw new Error('No api defined'); | |
| await this.#api.execute(); | |
| if (!this._additionalOptions) { | |
| this._buttonState = 'success'; | |
| } | |
| } catch (reason) { | |
| if (reason) { | |
| console.warn(reason); | |
| } | |
| if (!this._additionalOptions) { | |
| this._buttonState = 'failed'; | |
| } | |
| } | |
| } | |
| this.dispatchEvent(new UmbActionExecutedEvent()); | |
| } | |
| #observeIsDisabled() { | |
| this.observe( | |
| this.#api?.isDisabled, | |
| (isDisabled) => { | |
| this._isDisabled = isDisabled || false; | |
| }, | |
| 'isDisabledObserver', | |
| ); | |
| } | |
| #observeExtensions(aliases: string[]): void { | |
| this.#extensionsController?.destroy(); | |
| this.#extensionsController = new UmbExtensionsElementAndApiInitializer< | |
| ManifestWorkspaceActionMenuItem, | |
| 'workspaceActionMenuItem', | |
| ManifestWorkspaceActionMenuItem | |
| >( | |
| this, | |
| umbExtensionsRegistry, | |
| 'workspaceActionMenuItem', | |
| ExtensionApiArgsMethod, | |
| (action) => stringOrStringArrayIntersects(action.forWorkspaceActions, aliases), | |
| (extensionControllers) => { | |
| this._items = extensionControllers; | |
| }, | |
| undefined, // We can leave the alias to undefined, as we destroy this our selfs. | |
| ); | |
| } | |
| #renderButton() { | |
| const label = this.#manifest?.meta.label | |
| ? this.localize.string(this.#manifest.meta.label) | |
| : (this.#manifest?.name ?? ''); | |
| // Let's suppress typescript... | |
| //@ts-ignore | |
| const iconName = this.#manifest?.meta.icon ?? ''; | |
| let iconHtml; | |
| if (iconName) { | |
| iconHtml = html`<span style="display: flex; gap: 7px;"><umb-icon name="${iconName}"></umb-icon> ${label}</span>`; | |
| } | |
| return html` | |
| <uui-button | |
| data-mark="workspace-action:${this.#manifest?.alias}" | |
| .href=${this._href} | |
| look=${this.#manifest?.meta.look ?? 'default'} | |
| color=${this.#manifest?.meta.color ?? 'default'} | |
| label=${this._additionalOptions ? label + '…' : label} | |
| .disabled=${this._isDisabled} | |
| .state=${this._buttonState} | |
| @click=${this.#onClick}>${iconHtml}</uui-button> | |
| `; | |
| } | |
| #renderActionMenu() { | |
| return html` | |
| <umb-workspace-action-menu | |
| .items=${this._items} | |
| color="${this.#manifest?.meta.color ?? 'default'}" | |
| look="${this.#manifest?.meta.look ?? 'default'}"></umb-workspace-action-menu> | |
| `; | |
| } | |
| override render() { | |
| return when( | |
| this._items.length, | |
| () => html` <uui-button-group> ${this.#renderButton()} ${this.#renderActionMenu()} </uui-button-group> `, | |
| () => this.#renderButton(), | |
| ); | |
| } | |
| } | |
| export default CustomElement; | |
| declare global { | |
| interface HTMLElementTagNameMap { | |
| 'custom-element': CustomElement; | |
| } | |
| } | |
| /** | |
| * | |
| * @param manifest | |
| * @returns An array of arguments to pass to the extension API initializer. | |
| */ | |
| function ExtensionApiArgsMethod(manifest: ManifestWorkspaceActionMenuItem) { | |
| return [{ meta: manifest.meta }]; | |
| } | 
  
    
      This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
      Learn more about bidirectional Unicode characters
    
  
  
    
  | import { UMB_WORKSPACE_CONTEXT, UmbWorkspaceActionBase } from '@umbraco-cms/backoffice/workspace'; | |
| export default class CustomWorkspaceAction extends UmbWorkspaceActionBase { | |
| async execute() { | |
| // This gets called when the button is clicked! | |
| const context = await this.getContext(UMB_WORKSPACE_CONTEXT); | |
| let et = context.getEntityType(); | |
| console.log("ET", et); | |
| } | |
| } | 
  
    
      This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
      Learn more about bidirectional Unicode characters
    
  
  
    
  | export const manifests: any = [ | |
| { | |
| type: 'workspaceAction', | |
| kind: 'default', | |
| name: 'Previous page', | |
| alias: 'local.previousPage.action', | |
| weight: 103, | |
| element: () => import('./custom.element'), | |
| api: () => import('./custom.workspace.action'), | |
| meta: { | |
| label: 'My Custom Button', | |
| look: 'primary', | |
| color: 'default', | |
| icon: 'icon-award' | |
| }, | |
| conditions: [ | |
| { | |
| alias: 'Umb.Condition.WorkspaceAlias', | |
| match: 'Umb.Workspace.Document', | |
| }, | |
| ], | |
| } | |
| ]; | 
  
    Sign up for free
    to join this conversation on GitHub.
    Already have an account?
    Sign in to comment