From 5bd76355d4c1e28d53908608764bbe9049e6d848 Mon Sep 17 00:00:00 2001 From: Freddy Montes <751424+fmontes@users.noreply.github.com> Date: Fri, 20 Oct 2023 12:59:04 -0600 Subject: [PATCH] feat(block-editor): AI Image prompt Block (#26405) (#26485) * Created AI content node * Created ai-content-service * Created AI prompt content extension * Update ai-content.service.ts * Optimised and aligned extension code * Resolved comments on PR * Removed ai content node file * Update ai-content.service.ts * Remove import path for ai-content-node * Updated the ai-prompt form * align ai text prompt extension * Resovled comments on pull * Fix the aiTextPrompt form * Updated extension related code * Update ai-content-prompt.plugin.ts * Remove some extra code * Fixed outside click for aiContentPrompt extension * Created ai content node * Updated ai-content node * Remove unused code, resolve comments on PR * ai-content-prompt extension code optimisation * Update ai-content-prompt.plugin.ts * Integrated with AI api * insert ai node on response * textPrompt extension code optimsation * Update ai-content.service.ts * Resolve comments on pull req * handle close extension on outside click and content flip fix * Added focus field method and type for form * Change name of ai node creation command * Update the ai-content icon in the actions menu list * Update ai-content-prompt.component.scss * Resolved comments on pull * Added pening status and update the name of destroy var * Implement ai-content-actions extension * ai-prompt extension code optimisation * ai-text-prompt block optimisation * Update main.ts * feat: Created ai image prompt extension * Resolve comments on pull * Created loader node and handle loading image * Resolve comments on PR * Code optimisation for ai-content-actions extension * Fix css for p-listbox * Handle ai content actions to work in two diff context, code optimisation * Code optimisation * clean code from extra code --------- Co-authored-by: Nikola Trajkovic <82508651+nikolatrajkovic24@users.noreply.github.com> Co-authored-by: Nikola Co-authored-by: Will Ezell --- .../src/lib/block-editor.module.ts | 7 +- .../dot-block-editor.component.scss | 32 ++++ .../dot-block-editor.component.ts | 8 +- .../action-button/actions-menu.extension.ts | 3 +- .../ai-content-actions.component.ts | 17 +- .../plugins/ai-content-actions.plugin.ts | 31 ++-- .../ai-image-prompt.component.html | 118 ++++++++++++ .../ai-image-prompt.component.scss | 88 +++++++++ .../ai-image-prompt.component.spec.ts | 28 +++ .../ai-image-prompt.component.ts | 73 ++++++++ .../ai-image-prompt.extension.ts | 84 +++++++++ .../plugins/ai-image-prompt.plugin.ts | 175 ++++++++++++++++++ .../extensions/ai-image-prompt/utils/index.ts | 20 ++ .../extensions/bubble-menu/models/index.ts | 1 + .../lib/extensions/bubble-menu/utils/index.ts | 3 +- .../block-editor/src/lib/extensions/index.ts | 3 + .../src/lib/nodes/image-node/helpers/index.ts | 8 +- .../libs/block-editor/src/lib/nodes/index.ts | 1 + .../src/lib/nodes/loader/loader.node.ts | 77 ++++++++ .../suggestions/suggestion-icons.ts | 3 + .../services/ai-content/ai-content.service.ts | 65 ++++++- .../src/lib/shared/utils/suggestion.utils.ts | 8 +- 22 files changed, 819 insertions(+), 34 deletions(-) create mode 100644 core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.html create mode 100644 core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.scss create mode 100644 core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.spec.ts create mode 100644 core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.ts create mode 100644 core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.extension.ts create mode 100644 core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/plugins/ai-image-prompt.plugin.ts create mode 100644 core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/utils/index.ts create mode 100644 core-web/libs/block-editor/src/lib/nodes/loader/loader.node.ts diff --git a/core-web/libs/block-editor/src/lib/block-editor.module.ts b/core-web/libs/block-editor/src/lib/block-editor.module.ts index 49f38a572009..6d74032dda67 100644 --- a/core-web/libs/block-editor/src/lib/block-editor.module.ts +++ b/core-web/libs/block-editor/src/lib/block-editor.module.ts @@ -12,6 +12,7 @@ import { DotEditorCountBarComponent } from './components/dot-editor-count-bar/do import { AIContentPromptComponent, AIContentActionsComponent, + AIImagePromptComponent, BubbleLinkFormComponent, BubbleMenuButtonComponent, BubbleMenuComponent, @@ -54,7 +55,8 @@ import { SharedModule } from './shared/shared.module'; DotEditorCountBarComponent, FloatingButtonComponent, AIContentPromptComponent, - AIContentActionsComponent + AIContentActionsComponent, + AIImagePromptComponent ], providers: [DotUploadFileService, LoggerService, StringUtils, AiContentService], exports: [ @@ -66,7 +68,8 @@ import { SharedModule } from './shared/shared.module'; BubbleFormComponent, DotBlockEditorComponent, AIContentPromptComponent, - AIContentActionsComponent + AIContentActionsComponent, + AIImagePromptComponent ] }) export class BlockEditorModule {} diff --git a/core-web/libs/block-editor/src/lib/components/dot-block-editor/dot-block-editor.component.scss b/core-web/libs/block-editor/src/lib/components/dot-block-editor/dot-block-editor.component.scss index 21d7425722d4..26403a09ba57 100644 --- a/core-web/libs/block-editor/src/lib/components/dot-block-editor/dot-block-editor.component.scss +++ b/core-web/libs/block-editor/src/lib/components/dot-block-editor/dot-block-editor.component.scss @@ -313,6 +313,38 @@ border: 1px solid $color-palette-primary-300; } } + + .loader-style { + display: flex; + justify-content: center; + flex-direction: column; + align-items: center; + min-height: 12.5rem; + min-width: 25rem; + width: 25.5625rem; + height: 17.375rem; + padding: 0.5rem; + border-radius: 0.5rem; + border: 1.5px solid $color-palette-gray-400; + } + + .p-progress-spinner { + border: 5px solid $color-palette-gray-300; + border-radius: 50%; + border-top: 5px solid $color-palette-primary; + width: 2.4rem; + height: 2.4rem; + animation: spin 1s linear infinite; + } + + @keyframes spin { + 0% { + transform: rotate(0deg); + } + 100% { + transform: rotate(360deg); + } + } } tiptap-editor { diff --git a/core-web/libs/block-editor/src/lib/components/dot-block-editor/dot-block-editor.component.ts b/core-web/libs/block-editor/src/lib/components/dot-block-editor/dot-block-editor.component.ts index 73294816a0af..013b319581d5 100644 --- a/core-web/libs/block-editor/src/lib/components/dot-block-editor/dot-block-editor.component.ts +++ b/core-web/libs/block-editor/src/lib/components/dot-block-editor/dot-block-editor.component.ts @@ -51,10 +51,11 @@ import { AssetUploader, DotComands, AIContentPromptExtension, + AIImagePromptExtension, AIContentActionsExtension } from '../../extensions'; import { DotPlaceholder } from '../../extensions/dot-placeholder/dot-placeholder-plugin'; -import { ContentletBlock, ImageNode, VideoNode, AIContentNode } from '../../nodes'; +import { ContentletBlock, ImageNode, VideoNode, AIContentNode, LoaderNode } from '../../nodes'; import { formatHTML, removeInvalidNodes, @@ -62,6 +63,7 @@ import { DotMarketingConfigService, RestoreDefaultDOMAttrs } from '../../shared'; + @Component({ selector: 'dot-block-editor', templateUrl: './dot-block-editor.component.html', @@ -111,7 +113,8 @@ export class DotBlockEditorComponent implements OnInit, OnDestroy { ['image', ImageNode], ['video', VideoNode], ['table', DotTableExtension()], - ['aiContent', AIContentNode] + ['aiContent', AIContentNode], + ['loader', LoaderNode] ]); private destroy$: Subject = new Subject(); @@ -383,6 +386,7 @@ export class DotBlockEditorComponent implements OnInit, OnDestroy { DotBubbleMenuExtension(this.viewContainerRef), BubbleFormExtension(this.viewContainerRef), AIContentPromptExtension(this.viewContainerRef), + AIImagePromptExtension(this.viewContainerRef), AIContentActionsExtension(this.viewContainerRef), DotFloatingButton(this.injector, this.viewContainerRef), DotTableCellExtension(this.viewContainerRef), diff --git a/core-web/libs/block-editor/src/lib/extensions/action-button/actions-menu.extension.ts b/core-web/libs/block-editor/src/lib/extensions/action-button/actions-menu.extension.ts index e3f05f6463bc..6c6d30617e4c 100644 --- a/core-web/libs/block-editor/src/lib/extensions/action-button/actions-menu.extension.ts +++ b/core-web/libs/block-editor/src/lib/extensions/action-button/actions-menu.extension.ts @@ -174,7 +174,8 @@ function execCommand({ superscript: () => editor.chain().setSuperscript().focus().run(), video: () => editor.commands.openAssetForm({ type: 'video' }), aiContentPrompt: () => editor.commands.openAIPrompt(), - aiContent: () => editor.commands.insertAINode() + aiContent: () => editor.commands.insertAINode(), + aiImagePrompt: () => editor.commands.openImagePrompt() }; getCustomActions(customBlocks).forEach((option) => { diff --git a/core-web/libs/block-editor/src/lib/extensions/ai-content-actions/ai-content-actions.component.ts b/core-web/libs/block-editor/src/lib/extensions/ai-content-actions/ai-content-actions.component.ts index 96e8e2615b57..c95252054764 100644 --- a/core-web/libs/block-editor/src/lib/extensions/ai-content-actions/ai-content-actions.component.ts +++ b/core-web/libs/block-editor/src/lib/extensions/ai-content-actions/ai-content-actions.component.ts @@ -1,4 +1,4 @@ -import { Observable, of } from 'rxjs'; +import { Observable } from 'rxjs'; import { Component, EventEmitter, Output, OnInit } from '@angular/core'; @@ -26,6 +26,7 @@ export class AIContentActionsComponent implements OnInit { @Output() actionEmitter = new EventEmitter(); actionOptions!: ActionOption[]; + tooltipContent = 'Describe the size, color palette, style, mood, etc.'; constructor(private aiContentService: AiContentService) {} @@ -56,17 +57,11 @@ export class AIContentActionsComponent implements OnInit { this.actionEmitter.emit(action); } - getLatestContent() { - return this.aiContentService.getLastContentResponse(); + getLatestContent(): string { + return this.aiContentService.getLatestContent(); } - getNewContent(): Observable { - const promptToUse: string = this.aiContentService.getLastUsedPrompt(); - - if (promptToUse) { - return this.aiContentService.getIAContent(promptToUse); - } - - return of(''); + getNewContent(contentType: string): Observable { + return this.aiContentService.getNewContent(contentType); } } diff --git a/core-web/libs/block-editor/src/lib/extensions/ai-content-actions/plugins/ai-content-actions.plugin.ts b/core-web/libs/block-editor/src/lib/extensions/ai-content-actions/plugins/ai-content-actions.plugin.ts index dac816801b4f..198b7c44daf7 100644 --- a/core-web/libs/block-editor/src/lib/extensions/ai-content-actions/plugins/ai-content-actions.plugin.ts +++ b/core-web/libs/block-editor/src/lib/extensions/ai-content-actions/plugins/ai-content-actions.plugin.ts @@ -6,7 +6,7 @@ import tippy, { Instance, Props } from 'tippy.js'; import { ComponentRef } from '@angular/core'; -import { take, takeUntil } from 'rxjs/operators'; +import { takeUntil } from 'rxjs/operators'; import { Editor } from '@tiptap/core'; @@ -88,18 +88,27 @@ export class AIContentActionsView { } private generateContent() { + const nodeType = this.getNodeType(); + this.editor.commands.closeAIContentActions(); - this.component.instance - .getNewContent() - .pipe(take(1), takeUntil(this.destroy$)) - .subscribe((newContent) => { - if (newContent) { - this.editor.commands.deleteSelection(); - this.editor.commands.insertAINode(newContent); - this.editor.commands.openAIContentActions(); - } - }); + this.component.instance.getNewContent(nodeType).subscribe((newContent) => { + if (newContent) { + this.editor.commands.deleteSelection(); + this.editor.commands.insertAINode(newContent); + this.editor.commands.openAIContentActions(); + } + }); + } + + private getNodeType() { + const { state } = this.editor.view; + const { doc, selection } = state; + const { ranges } = selection; + const from = Math.min(...ranges.map((range) => range.$from.pos)); + const node = doc?.nodeAt(from); + + return node.type.name; } private deleteContent() { diff --git a/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.html b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.html new file mode 100644 index 000000000000..303fc31a4c12 --- /dev/null +++ b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.html @@ -0,0 +1,118 @@ +
+

Generate AI Image

+
+
+
+ + + + + + +
+ + Generate an AI image based on your input and requests. + + + + + + +
+
+ +
+ +
+ +
+
+
+ +
+
+ + + + + +
+ Auto-Generate an Image based on the content created within the + block editor. + + + + + +
+
+ +
+ +
+ +
+
+
+
+
diff --git a/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.scss b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.scss new file mode 100644 index 000000000000..fc1f3e306781 --- /dev/null +++ b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.scss @@ -0,0 +1,88 @@ +@use "variables" as *; + +.ai-image-dialog { + position: fixed; + top: 0%; + left: 0%; + transform: translate(-55%, -5%); + width: 90vw; + height: 75vh; + padding: $spacing-4; + margin: $spacing-5; + border: 2px solid $color-palette-gray-400; + box-shadow: 0px 8px 16px 0px rgba(20, 21, 26, 0.08); + border-radius: $border-radius-lg; + z-index: 9999; + background: $white; +} + +.title { + font-size: $font-size-xl; + font-weight: $font-weight-semi-bold; + margin-bottom: $spacing-3 0; +} + +form { + width: 100%; +} + +.input-field-wrapper { + width: 100%; + height: 7rem; + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + margin: $spacing-3 0; + + .input-field { + width: 100%; + height: 100%; + padding: $spacing-1; + line-height: normal; + background-color: $white; + border: 1px solid #ccc; + border-radius: $border-radius-sm; + color: $color-palette-gray-700; + } +} + +.choice-div-container { + display: flex; + justify-content: space-between; + height: 80%; + margin-bottom: $spacing-1; +} + +.choice-div { + height: 100%; + width: 48%; + padding: $spacing-2; + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + border-radius: $border-radius-lg; + border: 2px solid $color-palette-gray-400; + cursor: pointer; + + &:hover { + border: 2px solid #c6b3f9; + } +} + +.choice-div .icon-wrapper, +.submit-btn-container { + margin-bottom: $spacing-3; + display: flex; + justify-content: center; + align-items: center; +} + +span { + text-align: center; +} + +.icon-wrapper { + background-color: $white; +} diff --git a/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.spec.ts b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.spec.ts new file mode 100644 index 000000000000..2c731c999a6f --- /dev/null +++ b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.spec.ts @@ -0,0 +1,28 @@ +import { HttpClientTestingModule } from '@angular/common/http/testing'; +import { ComponentFixture, TestBed } from '@angular/core/testing'; +import { ReactiveFormsModule } from '@angular/forms'; + +import { AIImagePromptComponent } from './ai-image-prompt.component'; + +import { AiContentService } from '../../shared'; + +describe('AIImagePromptComponent', () => { + let component: AIImagePromptComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + imports: [ReactiveFormsModule, HttpClientTestingModule], + declarations: [AIImagePromptComponent], + providers: [AiContentService] + }).compileComponents(); + + fixture = TestBed.createComponent(AIImagePromptComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.ts b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.ts new file mode 100644 index 000000000000..5a2edfe2bacd --- /dev/null +++ b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.component.ts @@ -0,0 +1,73 @@ +import { of } from 'rxjs'; + +import { Component, EventEmitter, Output } from '@angular/core'; +import { FormBuilder, Validators } from '@angular/forms'; + +import { catchError, switchMap } from 'rxjs/operators'; + +import { AiContentService } from '../../shared/services/ai-content/ai-content.service'; + +@Component({ + selector: 'dot-ai-image-prompt', + templateUrl: './ai-image-prompt.component.html', + styleUrls: ['./ai-image-prompt.component.scss'] +}) +export class AIImagePromptComponent { + form = this.fb.group({ + promptGenerate: ['', Validators.required], + promptAutoGenerate: ['', Validators.required] + }); + + isFormSubmitting = false; + showFormOne = true; + showFormTwo = false; + + @Output() formSubmission = new EventEmitter(); + @Output() aiResponse = new EventEmitter(); + + constructor(private fb: FormBuilder, private aiContentService: AiContentService) {} + + onSubmit() { + this.isFormSubmitting = true; + const promptGenerate = this.form.value.promptGenerate; + const promptAutoGenerate = this.form.value.promptAutoGenerate; + const combinedPrompt = `${promptGenerate} ${promptAutoGenerate}`; + + this.isFormSubmitting = false; + this.formSubmission.emit(true); + + if (prompt) { + this.aiContentService + .getAIImage(combinedPrompt) + .pipe( + catchError(() => of(null)), + switchMap((imageId) => { + if (!imageId) { + return of(null); + } + + return this.aiContentService.createAndPublishContentlet(imageId); + }) + ) + .subscribe((contentlet) => { + this.aiResponse.emit(contentlet); + }); + } + } + + openFormOne() { + this.showFormOne = true; + this.showFormTwo = false; + } + + openFormTwo() { + this.showFormTwo = true; + this.showFormOne = false; + } + + cleanForm() { + this.form.reset(); + this.showFormOne = true; + this.showFormTwo = false; + } +} diff --git a/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.extension.ts b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.extension.ts new file mode 100644 index 000000000000..e50c5e537752 --- /dev/null +++ b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/ai-image-prompt.extension.ts @@ -0,0 +1,84 @@ +import { PluginKey } from 'prosemirror-state'; +import { Props } from 'tippy.js'; + +import { ViewContainerRef } from '@angular/core'; + +import { Extension } from '@tiptap/core'; + +import { AIImagePromptComponent } from './ai-image-prompt.component'; +import { aiImagePromptPlugin } from './plugins/ai-image-prompt.plugin'; + +export interface AIImagePromptOptions { + pluginKey: PluginKey; + tippyOptions?: Partial; + element: HTMLElement | null; +} + +declare module '@tiptap/core' { + interface Commands { + AIImagePrompt: { + openImagePrompt: () => ReturnType; + closeImagePrompt: () => ReturnType; + }; + } +} + +export const AI_IMAGE_PROMPT_PLUGIN_KEY = new PluginKey('aiImagePrompt-form'); + +export const AIImagePromptExtension = (viewContainerRef: ViewContainerRef) => { + return Extension.create({ + name: 'aiImagePrompt', + + addOptions() { + return { + element: null, + tippyOptions: {}, + pluginKey: AI_IMAGE_PROMPT_PLUGIN_KEY + }; + }, + + addCommands() { + return { + openImagePrompt: + () => + ({ chain }) => { + return chain() + .command(({ tr }) => { + tr.setMeta(AI_IMAGE_PROMPT_PLUGIN_KEY, { open: true }); + + return true; + }) + .freezeScroll(true) + .run(); + }, + closeImagePrompt: + () => + ({ chain }) => { + return chain() + .command(({ tr }) => { + tr.setMeta(AI_IMAGE_PROMPT_PLUGIN_KEY, { open: false }); + + return true; + }) + .freezeScroll(false) + .run(); + } + }; + }, + + addProseMirrorPlugins() { + const component = viewContainerRef.createComponent(AIImagePromptComponent); + component.changeDetectorRef.detectChanges(); + + return [ + aiImagePromptPlugin({ + pluginKey: this.options.pluginKey, + editor: this.editor, + element: component.location.nativeElement, + tippyOptions: this.options.tippyOptions, + component: component + }) + ]; + } + }); +}; diff --git a/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/plugins/ai-image-prompt.plugin.ts b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/plugins/ai-image-prompt.plugin.ts new file mode 100644 index 000000000000..94fcb3eb81d7 --- /dev/null +++ b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/plugins/ai-image-prompt.plugin.ts @@ -0,0 +1,175 @@ +import { Node } from 'prosemirror-model'; +import { EditorState, Plugin, PluginKey, Transaction } from 'prosemirror-state'; +import { EditorView } from 'prosemirror-view'; +import { Subject } from 'rxjs'; +import tippy, { Instance, Props } from 'tippy.js'; + +import { ComponentRef } from '@angular/core'; + +import { takeUntil } from 'rxjs/operators'; + +import { Editor } from '@tiptap/core'; + +import { AIImagePromptComponent } from '../ai-image-prompt.component'; +import { AI_IMAGE_PROMPT_PLUGIN_KEY } from '../ai-image-prompt.extension'; +import { TIPPY_OPTIONS } from '../utils'; + +interface AIImagePromptProps { + pluginKey: PluginKey; + editor: Editor; + element: HTMLElement; + tippyOptions: Partial; + component: ComponentRef; +} + +interface PluginState { + open: boolean; + form: []; +} + +export type AIImagePromptViewProps = AIImagePromptProps & { + view: EditorView; +}; + +export class AIImagePromptView { + public editor: Editor; + + public node: Node; + + public element: HTMLElement; + + public view: EditorView; + + public tippy: Instance | undefined; + + public tippyOptions: Partial; + + public pluginKey: PluginKey; + + public component: ComponentRef; + + private destroy$ = new Subject(); + + constructor(props: AIImagePromptViewProps) { + const { editor, element, view, tippyOptions = {}, pluginKey, component } = props; + + this.editor = editor; + this.element = element; + this.view = view; + + this.tippyOptions = tippyOptions; + + this.element.remove(); + this.pluginKey = pluginKey; + this.component = component; + + this.view.dom.addEventListener('keydown', this.handleKeyDown.bind(this)); + + this.component.instance.formSubmission.pipe(takeUntil(this.destroy$)).subscribe(() => { + this.editor.commands.closeImagePrompt(); + this.editor.commands.insertLoaderNode(); + }); + + this.component.instance.aiResponse + .pipe(takeUntil(this.destroy$)) + // eslint-disable-next-line @typescript-eslint/no-explicit-any + .subscribe((contentlet: any) => { + this.editor.commands.deleteSelection(); + const data = Object.values(contentlet[0])[0]; + + if (data) { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + this.editor.commands.insertImage(data as any); + this.editor.commands.openAIContentActions(); + } + }); + } + + private handleKeyDown(event: KeyboardEvent) { + if (event.key === 'Backspace') { + this.editor.commands.closeAIContentActions(); + } + } + + update(view: EditorView, prevState?: EditorState) { + const next = this.pluginKey?.getState(view.state); + const prev = prevState ? this.pluginKey?.getState(prevState) : { open: false }; + + if (next?.open === prev?.open) { + this.tippy?.popperInstance?.forceUpdate(); + + return; + } + + if (!next.open) { + this.component.instance.cleanForm(); + } + + this.createTooltip(); + + next.open ? this.show() : this.hide(); + } + + createTooltip() { + const { element: editorElement } = this.editor.options; + const editorIsAttached = !!editorElement.parentElement; + + if (this.tippy || !editorIsAttached) { + return; + } + + this.tippy = tippy(document.body, { + ...TIPPY_OPTIONS, + ...this.tippyOptions, + content: this.element, + onHide: () => { + this.editor.commands.closeImagePrompt(); + } + }); + } + + show() { + this.tippy?.show(); + } + + hide() { + this.tippy?.hide(); + this.editor.view.focus(); + } + + destroy() { + this.tippy?.destroy(); + this.destroy$.next(true); + this.destroy$.complete(); + } +} + +export const aiImagePromptPlugin = (options: AIImagePromptProps) => { + return new Plugin({ + key: options.pluginKey as PluginKey, + view: (view) => new AIImagePromptView({ view, ...options }), + state: { + init(): PluginState { + return { + open: false, + form: [] + }; + }, + + apply( + transaction: Transaction, + value: PluginState, + oldState: EditorState + ): PluginState { + const { open, form } = transaction.getMeta(AI_IMAGE_PROMPT_PLUGIN_KEY) || {}; + const state = AI_IMAGE_PROMPT_PLUGIN_KEY.getState(oldState); + + if (typeof open === 'boolean') { + return { open, form }; + } + + return state || value; + } + } + }); +}; diff --git a/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/utils/index.ts b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/utils/index.ts new file mode 100644 index 000000000000..8caf34a9396f --- /dev/null +++ b/core-web/libs/block-editor/src/lib/extensions/ai-image-prompt/utils/index.ts @@ -0,0 +1,20 @@ +import { Props } from 'tippy.js'; + +export const TIPPY_OPTIONS: Partial = { + duration: [500, 0], + interactive: true, + maxWidth: '100%', + trigger: 'manual', + hideOnClick: true, + placement: 'top', + popperOptions: { + modifiers: [ + { + name: 'preventOverflow', + options: { + altAxis: true + } + } + ] + } +}; diff --git a/core-web/libs/block-editor/src/lib/extensions/bubble-menu/models/index.ts b/core-web/libs/block-editor/src/lib/extensions/bubble-menu/models/index.ts index c11a311b7ece..65fa5381d3fe 100644 --- a/core-web/libs/block-editor/src/lib/extensions/bubble-menu/models/index.ts +++ b/core-web/libs/block-editor/src/lib/extensions/bubble-menu/models/index.ts @@ -84,4 +84,5 @@ export interface HideBubbleMenuExtensions { dotVideo: boolean; youtube: boolean; aiContent: boolean; + loader: boolean; } diff --git a/core-web/libs/block-editor/src/lib/extensions/bubble-menu/utils/index.ts b/core-web/libs/block-editor/src/lib/extensions/bubble-menu/utils/index.ts index 2b42a7644349..f26d2f547871 100644 --- a/core-web/libs/block-editor/src/lib/extensions/bubble-menu/utils/index.ts +++ b/core-web/libs/block-editor/src/lib/extensions/bubble-menu/utils/index.ts @@ -10,7 +10,8 @@ const hideBubbleMenuOn: HideBubbleMenuExtensions = { table: true, youtube: true, dotVideo: true, - aiContent: true + aiContent: true, + loader: true }; /** diff --git a/core-web/libs/block-editor/src/lib/extensions/index.ts b/core-web/libs/block-editor/src/lib/extensions/index.ts index d22b75aa27e9..b752ccd0b069 100644 --- a/core-web/libs/block-editor/src/lib/extensions/index.ts +++ b/core-web/libs/block-editor/src/lib/extensions/index.ts @@ -33,3 +33,6 @@ export * from './ai-content-prompt/plugins/ai-content-prompt.plugin'; export * from './ai-content-actions/ai-content-actions.component'; export * from './ai-content-actions/ai-content-actions.extension'; export * from './ai-content-actions/plugins/ai-content-actions.plugin'; +export * from './ai-image-prompt/ai-image-prompt.component'; +export * from './ai-image-prompt/ai-image-prompt.extension'; +export * from './ai-image-prompt/plugins/ai-image-prompt.plugin'; diff --git a/core-web/libs/block-editor/src/lib/nodes/image-node/helpers/index.ts b/core-web/libs/block-editor/src/lib/nodes/image-node/helpers/index.ts index e815787eb420..b19dc4848d5a 100644 --- a/core-web/libs/block-editor/src/lib/nodes/image-node/helpers/index.ts +++ b/core-web/libs/block-editor/src/lib/nodes/image-node/helpers/index.ts @@ -19,12 +19,14 @@ export const imageElement = (attrs, newAttrs): DOMOutputSpec => { export const addImageLanguageId = (src: string, languageId: number) => src.includes(LANGUAGE_ID) ? src : `${src}?${LANGUAGE_ID}=${languageId}`; -export const getImageAttr = (attrs: DotCMSContentlet | string) => { +export const getImageAttr = ( + attrs: DotCMSContentlet | string | { url: string; base64: string } +) => { if (typeof attrs === 'string') { - return { src: attrs }; + return { src: attrs, data: 'null' }; } - const { fileAsset, asset, title, languageId } = attrs; + const { fileAsset, asset, title, languageId } = attrs as DotCMSContentlet; return { data: attrs, diff --git a/core-web/libs/block-editor/src/lib/nodes/index.ts b/core-web/libs/block-editor/src/lib/nodes/index.ts index 6e8cdfdc74fd..7396ae140b1b 100644 --- a/core-web/libs/block-editor/src/lib/nodes/index.ts +++ b/core-web/libs/block-editor/src/lib/nodes/index.ts @@ -3,3 +3,4 @@ export * from './contentlet-block/contentlet-block.node'; export * from './image-node/image.node'; export * from './video/video.node'; export * from './ai-content/ai-content.node'; +export * from './loader/loader.node'; diff --git a/core-web/libs/block-editor/src/lib/nodes/loader/loader.node.ts b/core-web/libs/block-editor/src/lib/nodes/loader/loader.node.ts new file mode 100644 index 000000000000..fc355b124837 --- /dev/null +++ b/core-web/libs/block-editor/src/lib/nodes/loader/loader.node.ts @@ -0,0 +1,77 @@ +import { Node } from '@tiptap/core'; + +declare module '@tiptap/core' { + interface Commands { + LoaderNode: { + insertLoaderNode: (isLoading?: boolean) => ReturnType; + }; + } +} + +export const LoaderNode = Node.create({ + name: 'loader', + + addAttributes() { + return { + isLoading: { + default: true + } + }; + }, + + parseHTML() { + return [ + { + tag: 'div.p-d-flex.p-jc-center' + } + ]; + }, + + addOptions() { + return { + inline: false + }; + }, + + inline() { + return this.options.inline; + }, + + group() { + return 'block'; + }, + + addCommands() { + return { + ...this.parent?.(), + insertLoaderNode: + (isLoading?: boolean) => + ({ commands }) => { + return commands.insertContent({ + type: this.name, + attrs: { isLoading: isLoading } + }); + } + }; + }, + + renderHTML() { + return ['div', { class: 'p-d-flex p-jc-center' }]; + }, + + addNodeView() { + return ({ node }) => { + const dom = document.createElement('div'); + + dom.classList.add('loader-style'); + + if (node.attrs.isLoading) { + const spinner = document.createElement('div'); + spinner.classList.add('p-progress-spinner'); + dom.append(spinner); + } + + return { dom }; + }; + } +}); diff --git a/core-web/libs/block-editor/src/lib/shared/components/suggestions/suggestion-icons.ts b/core-web/libs/block-editor/src/lib/shared/components/suggestions/suggestion-icons.ts index 5202da8474ed..ad654ffc65e9 100644 --- a/core-web/libs/block-editor/src/lib/shared/components/suggestions/suggestion-icons.ts +++ b/core-web/libs/block-editor/src/lib/shared/components/suggestions/suggestion-icons.ts @@ -32,3 +32,6 @@ export const squarePlus = export const listStarsIcon = ''; + +export const mountsStarsIcon = + ''; diff --git a/core-web/libs/block-editor/src/lib/shared/services/ai-content/ai-content.service.ts b/core-web/libs/block-editor/src/lib/shared/services/ai-content/ai-content.service.ts index 069f67af40d7..bc8c6c60c48f 100644 --- a/core-web/libs/block-editor/src/lib/shared/services/ai-content/ai-content.service.ts +++ b/core-web/libs/block-editor/src/lib/shared/services/ai-content/ai-content.service.ts @@ -3,17 +3,19 @@ import { Observable, throwError } from 'rxjs'; import { HttpClient, HttpHeaders } from '@angular/common/http'; import { Injectable } from '@angular/core'; -import { catchError, map } from 'rxjs/operators'; +import { catchError, map, pluck } from 'rxjs/operators'; + +import { DotCMSContentlet } from '@dotcms/dotcms-models'; interface OpenAIResponse { model: string; prompt: string; response: string; } - @Injectable() export class AiContentService { private lastUsedPrompt: string | null = null; + private lastImagePrompt: string | null = null; private lastContentResponse: string | null = null; constructor(private http: HttpClient) {} @@ -49,4 +51,63 @@ export class AiContentService { }) ); } + + getAIImage(prompt: string) { + const url = 'api/ai/image/generate'; + const body = JSON.stringify({ + prompt + }); + + const headers = new HttpHeaders({ + 'Content-Type': 'application/json' + }); + + return this.http.post(url, body, { headers }).pipe( + catchError(() => { + return throwError('Error fetching AI content'); + }), + map(({ response }) => { + this.lastContentResponse = response; + + return response; + }) + ); + } + + getLatestContent() { + return this.lastContentResponse; + } + + getNewContent(contentType: string): Observable { + if (contentType === 'aiContent') { + return this.getIAContent(this.lastUsedPrompt); + } + + if (contentType === 'dotImage') { + return this.getAIImage(this.lastImagePrompt); + } + } + + createAndPublishContentlet(fileId: string): Observable { + const contentlets = [ + { + contentType: 'dotAsset', + asset: fileId, + hostFolder: '', + indexPolicy: 'WAIT_FOR' + } + ]; + + return this.http + .post('api/v1/workflow/actions/default/fire/PUBLISH', JSON.stringify({ contentlets }), { + headers: { + Origin: window.location.hostname, + 'Content-Type': 'application/json;charset=UTF-8' + } + }) + .pipe( + pluck('entity', 'results'), + catchError((error) => throwError(error)) + ) as Observable; + } } diff --git a/core-web/libs/block-editor/src/lib/shared/utils/suggestion.utils.ts b/core-web/libs/block-editor/src/lib/shared/utils/suggestion.utils.ts index 0bc23f997146..08d31afe48f7 100644 --- a/core-web/libs/block-editor/src/lib/shared/utils/suggestion.utils.ts +++ b/core-web/libs/block-editor/src/lib/shared/utils/suggestion.utils.ts @@ -11,7 +11,8 @@ import { pIcon, quoteIcon, ulIcon, - listStarsIcon + listStarsIcon, + mountsStarsIcon } from '../components/suggestions/suggestion-icons'; import { DotMenuItem } from '../components/suggestions/suggestions.component'; @@ -80,6 +81,11 @@ const block: DotMenuItem[] = [ icon: sanitizeUrl(listStarsIcon), id: 'aiContentPrompt' }, + { + label: 'AI Image', + icon: sanitizeUrl(mountsStarsIcon), + id: 'aiImagePrompt' + }, { label: 'Blockquote', icon: sanitizeUrl(quoteIcon),