@kolkov/angular-editor
Advanced tools
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
+412
| import { HttpEvent, HttpClient, HttpResponse } from '@angular/common/http'; | ||
| import { Observable } from 'rxjs'; | ||
| import * as i0 from '@angular/core'; | ||
| import { OnInit, EventEmitter, ElementRef, Renderer2, AfterViewInit, OnDestroy, TemplateRef, ChangeDetectorRef } from '@angular/core'; | ||
| import * as i7 from '@angular/forms'; | ||
| import { ControlValueAccessor } from '@angular/forms'; | ||
| import { DomSanitizer } from '@angular/platform-browser'; | ||
| import * as i6 from '@angular/common'; | ||
| interface CustomClass { | ||
| name: string; | ||
| class: string; | ||
| tag?: string; | ||
| } | ||
| interface Font { | ||
| name: string; | ||
| class: string; | ||
| } | ||
| interface AngularEditorConfig { | ||
| editable?: boolean; | ||
| spellcheck?: boolean; | ||
| height?: 'auto' | string; | ||
| minHeight?: '0' | string; | ||
| maxHeight?: 'auto' | string; | ||
| width?: 'auto' | string; | ||
| minWidth?: '0' | string; | ||
| translate?: 'yes' | 'now' | string; | ||
| enableToolbar?: boolean; | ||
| showToolbar?: boolean; | ||
| placeholder?: string; | ||
| defaultParagraphSeparator?: string; | ||
| defaultFontName?: string; | ||
| defaultFontSize?: '1' | '2' | '3' | '4' | '5' | '6' | '7' | string; | ||
| uploadUrl?: string; | ||
| upload?: (file: File) => Observable<HttpEvent<UploadResponse>>; | ||
| uploadWithCredentials?: boolean; | ||
| fonts?: Font[]; | ||
| customClasses?: CustomClass[]; | ||
| sanitize?: boolean; | ||
| toolbarPosition?: 'top' | 'bottom'; | ||
| outline?: boolean; | ||
| toolbarHiddenButtons?: string[][]; | ||
| rawPaste?: boolean; | ||
| } | ||
| interface UploadResponse { | ||
| imageUrl: string; | ||
| } | ||
| declare class AngularEditorService { | ||
| private http; | ||
| private doc; | ||
| savedSelection: Range | null; | ||
| selectedText: string; | ||
| uploadUrl: string; | ||
| uploadWithCredentials: boolean; | ||
| constructor(http: HttpClient, doc: any); | ||
| /** | ||
| * Executed command from editor header buttons exclude toggleEditorMode | ||
| * @param command string from triggerCommand | ||
| * @param value | ||
| */ | ||
| executeCommand(command: string, value?: string): void; | ||
| /** | ||
| * Create URL link | ||
| * @param url string from UI prompt | ||
| */ | ||
| createLink(url: string): void; | ||
| /** | ||
| * insert color either font or background | ||
| * | ||
| * @param color color to be inserted | ||
| * @param where where the color has to be inserted either text/background | ||
| */ | ||
| insertColor(color: string, where: string): void; | ||
| /** | ||
| * Set font name | ||
| * @param fontName string | ||
| */ | ||
| setFontName(fontName: string): void; | ||
| /** | ||
| * Set font size | ||
| * @param fontSize string | ||
| */ | ||
| setFontSize(fontSize: string): void; | ||
| /** | ||
| * Create raw HTML | ||
| * @param html HTML string | ||
| */ | ||
| insertHtml(html: string): void; | ||
| /** | ||
| * save selection when the editor is focussed out | ||
| */ | ||
| saveSelection: () => void; | ||
| /** | ||
| * restore selection when the editor is focused in | ||
| * | ||
| * saved selection when the editor is focused out | ||
| */ | ||
| restoreSelection(): boolean; | ||
| /** | ||
| * setTimeout used for execute 'saveSelection' method in next event loop iteration | ||
| */ | ||
| executeInNextQueueIteration(callbackFn: (...args: any[]) => any, timeout?: number): void; | ||
| /** check any selection is made or not */ | ||
| private checkSelection; | ||
| /** | ||
| * Upload file to uploadUrl | ||
| * @param file The file | ||
| */ | ||
| uploadImage(file: File): Observable<HttpEvent<UploadResponse>>; | ||
| /** | ||
| * Insert image with Url | ||
| * @param imageUrl The imageUrl. | ||
| */ | ||
| insertImage(imageUrl: string): void; | ||
| setDefaultParagraphSeparator(separator: string): void; | ||
| createCustomClass(customClass: CustomClass): void; | ||
| insertVideo(videoUrl: string): void; | ||
| private insertYouTubeVideoTag; | ||
| private insertVimeoVideoTag; | ||
| nextNode(node: any): any; | ||
| getRangeSelectedNodes(range: any, includePartiallySelectedContainers: any): any[]; | ||
| getSelectedNodes(): any[]; | ||
| replaceWithOwnChildren(el: any): void; | ||
| removeSelectedElements(tagNames: any): void; | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AngularEditorService, never>; | ||
| static ɵprov: i0.ɵɵInjectableDeclaration<AngularEditorService>; | ||
| } | ||
| interface SelectOption { | ||
| label: string; | ||
| value: string; | ||
| } | ||
| declare class AeSelectComponent implements OnInit, ControlValueAccessor { | ||
| private elRef; | ||
| private r; | ||
| options: SelectOption[]; | ||
| isHidden: boolean; | ||
| selectedOption: SelectOption; | ||
| disabled: boolean; | ||
| optionId: number; | ||
| get label(): string; | ||
| opened: boolean; | ||
| get value(): string; | ||
| hidden: string; | ||
| changeEvent: EventEmitter<any>; | ||
| labelButton: ElementRef; | ||
| constructor(elRef: ElementRef, r: Renderer2); | ||
| ngOnInit(): void; | ||
| hide(): void; | ||
| optionSelect(option: SelectOption, event: MouseEvent): void; | ||
| toggleOpen(event: MouseEvent): void; | ||
| onClick($event: MouseEvent): void; | ||
| close(): void; | ||
| get isOpen(): boolean; | ||
| writeValue(value: any): void; | ||
| setValue(value: any): void; | ||
| onChange: any; | ||
| onTouched: any; | ||
| registerOnChange(fn: any): void; | ||
| registerOnTouched(fn: any): void; | ||
| setDisabledState(isDisabled: boolean): void; | ||
| handleKeyDown($event: KeyboardEvent): void; | ||
| _handleArrowDown($event: any): void; | ||
| _handleArrowUp($event: any): void; | ||
| _handleSpace($event: any): void; | ||
| _handleEnter($event: any): void; | ||
| _handleTab($event: any): void; | ||
| _handleBackspace(): void; | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AeSelectComponent, never>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AeSelectComponent, "ae-select", never, { "options": { "alias": "options"; "required": false; }; "isHidden": { "alias": "hidden"; "required": false; }; }, { "changeEvent": "change"; }, never, never, false, never>; | ||
| } | ||
| declare class AeToolbarComponent { | ||
| private r; | ||
| private editorService; | ||
| private er; | ||
| private doc; | ||
| htmlMode: boolean; | ||
| linkSelected: boolean; | ||
| block: string; | ||
| fontName: string; | ||
| fontSize: string; | ||
| foreColour: any; | ||
| backColor: any; | ||
| headings: SelectOption[]; | ||
| fontSizes: SelectOption[]; | ||
| customClassId: string; | ||
| _customClasses: CustomClass[]; | ||
| customClassList: SelectOption[]; | ||
| tagMap: { | ||
| BLOCKQUOTE: string; | ||
| A: string; | ||
| }; | ||
| select: string[]; | ||
| buttons: string[]; | ||
| id: string; | ||
| uploadUrl: string; | ||
| upload: (file: File) => Observable<HttpEvent<UploadResponse>>; | ||
| showToolbar: boolean; | ||
| fonts: SelectOption[]; | ||
| set customClasses(classes: CustomClass[]); | ||
| set defaultFontName(value: string); | ||
| set defaultFontSize(value: string); | ||
| hiddenButtons: string[][]; | ||
| execute: EventEmitter<string>; | ||
| myInputFile: ElementRef; | ||
| get isLinkButtonDisabled(): boolean; | ||
| constructor(r: Renderer2, editorService: AngularEditorService, er: ElementRef, doc: any); | ||
| /** | ||
| * Trigger command from editor header buttons | ||
| * @param command string from toolbar buttons | ||
| */ | ||
| triggerCommand(command: string): void; | ||
| /** | ||
| * highlight editor buttons when cursor moved or positioning | ||
| */ | ||
| triggerButtons(): void; | ||
| /** | ||
| * trigger highlight editor buttons when cursor moved or positioning in block | ||
| */ | ||
| triggerBlocks(nodes: Node[]): void; | ||
| /** | ||
| * insert URL link | ||
| */ | ||
| insertUrl(): void; | ||
| /** | ||
| * insert Video link | ||
| */ | ||
| insertVideo(): void; | ||
| /** insert color */ | ||
| insertColor(color: string, where: string): void; | ||
| /** | ||
| * set font Name/family | ||
| * @param foreColor string | ||
| */ | ||
| setFontName(foreColor: string): void; | ||
| /** | ||
| * set font Size | ||
| * @param fontSize string | ||
| */ | ||
| setFontSize(fontSize: string): void; | ||
| /** | ||
| * toggle editor mode (WYSIWYG or SOURCE) | ||
| * @param m boolean | ||
| */ | ||
| setEditorMode(m: boolean): void; | ||
| /** | ||
| * Upload image when file is selected. | ||
| */ | ||
| onFileChanged(event: any): void; | ||
| watchUploadImage(response: HttpResponse<{ | ||
| imageUrl: string; | ||
| }>, event: any): void; | ||
| /** | ||
| * Set custom class | ||
| */ | ||
| setCustomClass(classId: string): void; | ||
| isButtonHidden(name: string): boolean; | ||
| focus(): void; | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AeToolbarComponent, never>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AeToolbarComponent, "angular-editor-toolbar, ae-toolbar, div[aeToolbar]", never, { "id": { "alias": "id"; "required": false; }; "uploadUrl": { "alias": "uploadUrl"; "required": false; }; "upload": { "alias": "upload"; "required": false; }; "showToolbar": { "alias": "showToolbar"; "required": false; }; "fonts": { "alias": "fonts"; "required": false; }; "customClasses": { "alias": "customClasses"; "required": false; }; "defaultFontName": { "alias": "defaultFontName"; "required": false; }; "defaultFontSize": { "alias": "defaultFontSize"; "required": false; }; "hiddenButtons": { "alias": "hiddenButtons"; "required": false; }; }, { "execute": "execute"; }, never, ["*"], false, never>; | ||
| } | ||
| declare class AngularEditorComponent implements OnInit, ControlValueAccessor, AfterViewInit, OnDestroy { | ||
| private r; | ||
| private editorService; | ||
| private doc; | ||
| private sanitizer; | ||
| private cdRef; | ||
| private autoFocus; | ||
| private onChange; | ||
| private onTouched; | ||
| modeVisual: boolean; | ||
| showPlaceholder: boolean; | ||
| disabled: boolean; | ||
| focused: boolean; | ||
| touched: boolean; | ||
| changed: boolean; | ||
| focusInstance: any; | ||
| blurInstance: any; | ||
| id: string; | ||
| config: AngularEditorConfig; | ||
| placeholder: string; | ||
| tabIndex: number | null; | ||
| html: any; | ||
| textArea: ElementRef; | ||
| editorWrapper: ElementRef; | ||
| editorToolbar: AeToolbarComponent; | ||
| customButtonsTemplateRef?: TemplateRef<any>; | ||
| executeCommandFn: any; | ||
| viewMode: EventEmitter<boolean>; | ||
| /** emits `blur` event when focused out from the textarea */ | ||
| blurEvent: EventEmitter<FocusEvent>; | ||
| /** emits `focus` event when focused in to the textarea */ | ||
| focusEvent: EventEmitter<FocusEvent>; | ||
| tabindex: number; | ||
| onFocus(): void; | ||
| constructor(r: Renderer2, editorService: AngularEditorService, doc: any, sanitizer: DomSanitizer, cdRef: ChangeDetectorRef, defaultTabIndex: string, autoFocus: any); | ||
| ngOnInit(): void; | ||
| ngAfterViewInit(): void; | ||
| onPaste(event: ClipboardEvent): string; | ||
| /** | ||
| * Executed command from editor header buttons | ||
| * @param command string from triggerCommand | ||
| * @param value | ||
| */ | ||
| executeCommand(command: string, value?: string): void; | ||
| /** | ||
| * focus event | ||
| */ | ||
| onTextAreaFocus(event: FocusEvent): void; | ||
| /** | ||
| * @description fires when cursor leaves textarea | ||
| */ | ||
| onTextAreaMouseOut(event: MouseEvent): void; | ||
| /** | ||
| * blur event | ||
| */ | ||
| onTextAreaBlur(event: FocusEvent): void; | ||
| /** | ||
| * focus the text area when the editor is focused | ||
| */ | ||
| focus(): void; | ||
| /** | ||
| * Executed from the contenteditable section while the input property changes | ||
| * @param element html element from contenteditable | ||
| */ | ||
| onContentChange(element: HTMLElement): void; | ||
| /** | ||
| * Set the function to be called | ||
| * when the control receives a change event. | ||
| * | ||
| * @param fn a function | ||
| */ | ||
| registerOnChange(fn: any): void; | ||
| /** | ||
| * Set the function to be called | ||
| * when the control receives a touch event. | ||
| * | ||
| * @param fn a function | ||
| */ | ||
| registerOnTouched(fn: any): void; | ||
| /** | ||
| * Write a new value to the element. | ||
| * | ||
| * @param value value to be executed when there is a change in contenteditable | ||
| */ | ||
| writeValue(value: any): void; | ||
| /** | ||
| * refresh view/HTML of the editor | ||
| * | ||
| * @param value html string from the editor | ||
| */ | ||
| refreshView(value: string): void; | ||
| /** | ||
| * toggles placeholder based on input string | ||
| * | ||
| * @param value A HTML string from the editor | ||
| */ | ||
| togglePlaceholder(value: boolean): void; | ||
| /** | ||
| * Implements disabled state for this element | ||
| * | ||
| * @param isDisabled Disabled flag | ||
| */ | ||
| setDisabledState(isDisabled: boolean): void; | ||
| /** | ||
| * toggles editor mode based on bToSource bool | ||
| * | ||
| * @param bToSource A boolean value from the editor | ||
| */ | ||
| toggleEditorMode(bToSource: boolean): void; | ||
| /** | ||
| * toggles editor buttons when cursor moved or positioning | ||
| * | ||
| * Send a node array from the contentEditable of the editor | ||
| */ | ||
| exec(): void; | ||
| private configure; | ||
| getFonts(): { | ||
| label: string; | ||
| value: string; | ||
| }[]; | ||
| getCustomTags(): string; | ||
| ngOnDestroy(): void; | ||
| filterStyles(html: string): string; | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AngularEditorComponent, [null, null, null, null, null, { attribute: "tabindex"; }, { attribute: "autofocus"; }]>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AngularEditorComponent, "angular-editor", never, { "id": { "alias": "id"; "required": false; }; "config": { "alias": "config"; "required": false; }; "placeholder": { "alias": "placeholder"; "required": false; }; "tabIndex": { "alias": "tabIndex"; "required": false; }; }, { "html": "html"; "viewMode": "viewMode"; "blurEvent": "blur"; "focusEvent": "focus"; }, ["customButtonsTemplateRef"], never, false, never>; | ||
| } | ||
| declare class AeButtonComponent { | ||
| iconName: string; | ||
| constructor(); | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AeButtonComponent, never>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AeButtonComponent, "ae-button, button[aeButton]", never, { "iconName": { "alias": "iconName"; "required": false; }; }, {}, never, ["*"], false, never>; | ||
| } | ||
| declare class AeToolbarSetComponent { | ||
| constructor(); | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AeToolbarSetComponent, never>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AeToolbarSetComponent, "ae-toolbar-set, [aeToolbarSet]", never, {}, {}, never, ["*"], false, never>; | ||
| } | ||
| declare class AngularEditorModule { | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AngularEditorModule, never>; | ||
| static ɵmod: i0.ɵɵNgModuleDeclaration<AngularEditorModule, [typeof AngularEditorComponent, typeof AeToolbarComponent, typeof AeSelectComponent, typeof AeButtonComponent, typeof AeToolbarSetComponent], [typeof i6.CommonModule, typeof i7.FormsModule, typeof i7.ReactiveFormsModule], [typeof AngularEditorComponent, typeof AeToolbarComponent, typeof AeButtonComponent, typeof AeToolbarSetComponent]>; | ||
| static ɵinj: i0.ɵɵInjectorDeclaration<AngularEditorModule>; | ||
| } | ||
| export { AeButtonComponent, AeSelectComponent, AeToolbarComponent, AeToolbarSetComponent, AngularEditorComponent, AngularEditorModule, AngularEditorService }; | ||
| export type { AngularEditorConfig, CustomClass, SelectOption, UploadResponse }; |
@@ -80,3 +80,3 @@ <?xml version="1.0" encoding="UTF-8"?> | ||
| <symbol id="unorderedList" viewBox="0 0 16 16"> | ||
| <g id="SVGRepo_iconCarrier"> | ||
| <g id="SVGRepo_iconCarrier" transform="translate(1.6 1.6) scale(0.8)"> | ||
| <rect width="16" height="16" id="icon-bound" fill="none"></rect> | ||
@@ -87,3 +87,3 @@ <path d="M4,3v2h12V3H4z M0,5h2V3H0V5z M4,9h12V7H4V9z M0,9h2V7H0V9z M4,13h12v-2H4V13z M0,13h2v-2H0V13z"></path> | ||
| <symbol id="orderedList" viewBox="0 0 16 16"> | ||
| <g id="SVGRepo_iconCarrier"> | ||
| <g id="SVGRepo_iconCarrier" transform="translate(1.6 1.6) scale(0.8)"> | ||
| <rect width="16" height="16" id="icon-bound" fill="none"></rect> | ||
@@ -90,0 +90,0 @@ <path id="list-numbered" |
+57
-1
@@ -0,3 +1,59 @@ | ||
| <a name="3.0.0"></a> | ||
| ## [3.0.0](https://github.com/kolkov/angular-editor/compare/v2.0.0...v3.0.0) (2025-11-22) Major Angular 20 Upgrade | ||
| 🎉 **Stable Release** - Production Ready! | ||
| ### Breaking Changes | ||
| * **Angular Version:** Minimum required version is now Angular 20.0.0 | ||
| * **RxJS:** Requires RxJS 7.8.0 or higher (upgraded from 6.5.5) | ||
| * **TypeScript:** Requires TypeScript 5.4 or higher | ||
| * **zone.js:** Updated to 0.15.1 | ||
| ### Features | ||
| * **Angular 20 Support:** Full compatibility with Angular 20.3.13 (v20-lts) | ||
| * **Angular 21 Ready:** Forward compatible with Angular 21.x | ||
| * **Modern Build System:** Updated to latest ng-packagr 20.3.2 | ||
| * **Enhanced Type Safety:** Improved TypeScript strict mode compliance | ||
| * **Font Awesome Removed:** No external icon dependencies - using pure SVG icons (27 icons) | ||
| * **Zero External Icon Dependencies:** Fully self-contained icon system | ||
| ### Migration Path | ||
| * Migrated through: Angular 13 → 18 → 19 → 20 | ||
| * All Angular CLI migrations applied successfully | ||
| * Updated DOCUMENT import from @angular/core (Angular 20 requirement) | ||
| * Modernized test infrastructure (waitForAsync) | ||
| ### Developer Experience | ||
| * **ESLint:** Updated to @angular-eslint 20.x | ||
| * **Linting:** All files pass linting (0 errors) | ||
| * **Build:** Both development and production builds verified | ||
| * **Tests:** 13/13 tests passing (100% success rate) | ||
| ### Bug Fixes | ||
| * **Icons:** Fixed list icons (unordered/ordered) display consistency in toolbar | ||
| * **Tests:** Fixed AeSelectComponent tests for mousedown event handling | ||
| * **Demo:** Updated demo app for Angular 20 compatibility | ||
| ### Technical Details | ||
| * Removed deprecated `async` test helper (use `waitForAsync`) | ||
| * Fixed TypeScript strict type checking for event handlers | ||
| * Disabled new strict rules for backward compatibility (prefer-standalone, prefer-inject) | ||
| * Updated moduleResolution to 'bundler' (Angular 20 standard) | ||
| ### Peer Dependencies | ||
| ```json | ||
| { | ||
| "@angular/common": "^20.0.0 || ^21.0.0", | ||
| "@angular/core": "^20.0.0 || ^21.0.0", | ||
| "@angular/forms": "^20.0.0 || ^21.0.0", | ||
| "rxjs": "^7.8.0" | ||
| } | ||
| ``` | ||
| <a name="3.0.0-beta.2"></a> | ||
| ## [3.0.0-beta.2](https://github.com/kolkov/angular-editor/compare/v3.0.0-beta.1...v3.0.0-beta.2) (2025-01-10) | ||
| * Refactor ae-select component (button → span) | ||
| <a name="2.0.0"></a> | ||
| ## [1.0.2](https://github.com/kolkov/angular-editor/compare/v1.2.0...v2.0.0) (2022-01-06) Majore release | ||
| ## [2.0.0](https://github.com/kolkov/angular-editor/compare/v1.2.0...v2.0.0) (2022-01-06) Major release | ||
| * Update to Angular v.13 and new Ivy compatible package format | ||
@@ -4,0 +60,0 @@ |
+10
-16
| { | ||
| "name": "@kolkov/angular-editor", | ||
| "version": "3.0.0-beta.2", | ||
| "description": "A simple native WYSIWYG editor for Angular 13+. Rich Text editor component for Angular.", | ||
| "version": "3.0.0", | ||
| "description": "A simple native WYSIWYG editor for Angular 20+. Rich Text editor component for Angular.", | ||
| "author": "Andrey Kolkov <a.kolkov@gmail.com>", | ||
@@ -13,4 +13,6 @@ "repository": "https://github.com/kolkov/angular-editor", | ||
| "peerDependencies": { | ||
| "@angular/common": ">=13", | ||
| "@angular/core": ">=13" | ||
| "@angular/common": "^20.0.0 || ^21.0.0", | ||
| "@angular/core": "^20.0.0 || ^21.0.0", | ||
| "@angular/forms": "^20.0.0 || ^21.0.0", | ||
| "rxjs": "^7.8.0" | ||
| }, | ||
@@ -31,8 +33,4 @@ "dependencies": { | ||
| ], | ||
| "module": "fesm2015/kolkov-angular-editor.mjs", | ||
| "es2020": "fesm2020/kolkov-angular-editor.mjs", | ||
| "esm2020": "esm2020/kolkov-angular-editor.mjs", | ||
| "fesm2020": "fesm2020/kolkov-angular-editor.mjs", | ||
| "fesm2015": "fesm2015/kolkov-angular-editor.mjs", | ||
| "typings": "kolkov-angular-editor.d.ts", | ||
| "module": "fesm2022/kolkov-angular-editor.mjs", | ||
| "typings": "index.d.ts", | ||
| "exports": { | ||
@@ -43,8 +41,4 @@ "./package.json": { | ||
| ".": { | ||
| "types": "./kolkov-angular-editor.d.ts", | ||
| "esm2020": "./esm2020/kolkov-angular-editor.mjs", | ||
| "es2020": "./fesm2020/kolkov-angular-editor.mjs", | ||
| "es2015": "./fesm2015/kolkov-angular-editor.mjs", | ||
| "node": "./fesm2015/kolkov-angular-editor.mjs", | ||
| "default": "./fesm2020/kolkov-angular-editor.mjs" | ||
| "types": "./index.d.ts", | ||
| "default": "./fesm2022/kolkov-angular-editor.mjs" | ||
| } | ||
@@ -51,0 +45,0 @@ }, |
+15
-10
| <p align="center"> | ||
| <img width="150px" src="https://raw.githubusercontent.com/kolkov/angular-editor/master/docs/angular-editor-logo.png?raw=true" alt="AngularEditor logo"/> | ||
| <img width="150px" src="https://raw.githubusercontent.com/kolkov/angular-editor/main/docs/angular-editor-logo.png?raw=true" alt="AngularEditor logo"/> | ||
| </p> | ||
@@ -8,10 +8,10 @@ | ||
| [](https://stackblitz.com/edit/angular-editor-wysiwyg) | ||
| [](https://travis-ci.com/kolkov/angular-editor) | ||
| [](https://travis-ci.com/kolkov/angular-editor) | ||
| [](https://www.npmjs.com/package/@kolkov/angular-editor) | ||
| [](https://www.jsdelivr.com/package/npm/@kolkov/angular-editor) | ||
| [](https://coveralls.io/github/kolkov/angular-editor?branch=master) | ||
| [](https://codecov.io/gh/kolkov/angular-editor) | ||
| [](https://coveralls.io/github/kolkov/angular-editor?branch=main) | ||
| [](https://codecov.io/gh/kolkov/angular-editor) | ||
| [](https://paypal.me/AndreyKolkov) | ||
| A simple native WYSIWYG/Rich Text editor for Angular 6-19+ | ||
| A simple native WYSIWYG/Rich Text editor for Angular 20+ | ||
@@ -34,7 +34,7 @@  | ||
| ### Versions | ||
| 3.0.0 and above - for Angular v13+ (removed Font Awesome icons deps + CSS variables) | ||
| 3.0.0 and above - for Angular v20+ (CSS variables, modern Angular 20) | ||
| 2.0.0 and above - for Angular v13.x.x and above | ||
| 2.0.0 and above - for Angular v13-19 | ||
| 1.0.0 and above - for Angular v8.x.x and above | ||
| 1.0.0 and above - for Angular v8-12 | ||
@@ -45,2 +45,7 @@ 0.18.4 and above - for Angular v7.x.x | ||
| **Note:** Version 3.0.0 requires: | ||
| - Angular 20.0.0 or higher (also compatible with Angular 21) | ||
| - RxJS 7.8.0 or higher | ||
| - TypeScript 5.4 or higher | ||
| Attention! `alpha` and `beta` versions may contain breaking changes. | ||
@@ -329,5 +334,5 @@ | ||
| Please read through our [contributing guidelines](https://github.com/kolkov/angular-editor/blob/master/CONTRIBUTING.md). Included are directions for opening issues, coding standards, and notes on development. | ||
| Please read through our [contributing guidelines](https://github.com/kolkov/angular-editor/blob/main/CONTRIBUTING.md). Included are directions for opening issues, coding standards, and notes on development. | ||
| Editor preferences are available in the [editor config](https://github.com/kolkov/angular-editor/blob/master/.editorconfig) for easy use in common text editors. Read more and download plugins at <http://editorconfig.org>. | ||
| Editor preferences are available in the [editor config](https://github.com/kolkov/angular-editor/blob/main/.editorconfig) for easy use in common text editors. Read more and download plugins at <http://editorconfig.org>. | ||
@@ -334,0 +339,0 @@ ## Versioning |
@@ -0,0 +0,0 @@ :root { |
| /** | ||
| * Generated bundle index. Do not edit. | ||
| */ | ||
| export * from './public-api'; | ||
| //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoia29sa292LWFuZ3VsYXItZWRpdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vcHJvamVjdHMvYW5ndWxhci1lZGl0b3Ivc3JjL2tvbGtvdi1hbmd1bGFyLWVkaXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19 |
| import { Component, Input } from '@angular/core'; | ||
| import * as i0 from "@angular/core"; | ||
| import * as i1 from "@angular/common"; | ||
| export class AeButtonComponent { | ||
| constructor() { | ||
| this.iconName = ''; | ||
| } | ||
| } | ||
| AeButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AeButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); | ||
| AeButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AeButtonComponent, selector: "ae-button, button[aeButton]", inputs: { iconName: "iconName" }, host: { properties: { "tabIndex": "-1", "type": "\"button\"" }, classAttribute: "angular-editor-button" }, ngImport: i0, template: "<ng-container *ngIf=\"iconName; else contentTemplate\">\n <svg>\n <use [attr.href]=\"'assets/ae-icons/icons.svg#' + iconName\" [attr.xlink:href]=\"'assets/ae-icons/icons.svg#' + iconName\"></use>\n </svg>\n</ng-container>\n<ng-template #contentTemplate>\n <ng-content></ng-content>\n</ng-template>\n", styles: ["a{cursor:pointer}:host.angular-editor-button{background-color:var(--ae-button-bg-color, white);vertical-align:middle;border:var(--ae-button-border, 1px solid #ddd);border-radius:var(--ae-button-radius, 4px);padding:.4rem;float:left;width:2rem;height:2rem}:host.angular-editor-button svg{width:100%;height:100%}:host.angular-editor-button:hover{cursor:pointer;background-color:var(--ae-button-hover-bg-color, #f1f1f1);transition:.2s ease}:host.angular-editor-button:focus,:host.angular-editor-button.focus{outline:0}:host.angular-editor-button:disabled{background-color:var(--ae-button-disabled-bg-color, #f5f5f5);pointer-events:none;cursor:not-allowed}:host.angular-editor-button:disabled>.color-label{pointer-events:none;cursor:not-allowed}:host.angular-editor-button:disabled>.color-label.foreground :after{background:#555555}:host.angular-editor-button:disabled>.color-label.background{background:#555555}:host.angular-editor-button.active{background:var(--ae-button-active-bg-color, #fffbd3)}:host.angular-editor-button.active:hover{background-color:var(--ae-button-active-hover-bg-color, #fffaad)}\n"], directives: [{ type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] }); | ||
| i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AeButtonComponent, decorators: [{ | ||
| type: Component, | ||
| args: [{ selector: 'ae-button, button[aeButton]', host: { | ||
| 'class': 'angular-editor-button', | ||
| '[tabIndex]': '-1', | ||
| '[type]': '"button"', | ||
| }, template: "<ng-container *ngIf=\"iconName; else contentTemplate\">\n <svg>\n <use [attr.href]=\"'assets/ae-icons/icons.svg#' + iconName\" [attr.xlink:href]=\"'assets/ae-icons/icons.svg#' + iconName\"></use>\n </svg>\n</ng-container>\n<ng-template #contentTemplate>\n <ng-content></ng-content>\n</ng-template>\n", styles: ["a{cursor:pointer}:host.angular-editor-button{background-color:var(--ae-button-bg-color, white);vertical-align:middle;border:var(--ae-button-border, 1px solid #ddd);border-radius:var(--ae-button-radius, 4px);padding:.4rem;float:left;width:2rem;height:2rem}:host.angular-editor-button svg{width:100%;height:100%}:host.angular-editor-button:hover{cursor:pointer;background-color:var(--ae-button-hover-bg-color, #f1f1f1);transition:.2s ease}:host.angular-editor-button:focus,:host.angular-editor-button.focus{outline:0}:host.angular-editor-button:disabled{background-color:var(--ae-button-disabled-bg-color, #f5f5f5);pointer-events:none;cursor:not-allowed}:host.angular-editor-button:disabled>.color-label{pointer-events:none;cursor:not-allowed}:host.angular-editor-button:disabled>.color-label.foreground :after{background:#555555}:host.angular-editor-button:disabled>.color-label.background{background:#555555}:host.angular-editor-button.active{background:var(--ae-button-active-bg-color, #fffbd3)}:host.angular-editor-button.active:hover{background-color:var(--ae-button-active-hover-bg-color, #fffaad)}\n"] }] | ||
| }], ctorParameters: function () { return []; }, propDecorators: { iconName: [{ | ||
| type: Input | ||
| }] } }); | ||
| //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWUtYnV0dG9uLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2FuZ3VsYXItZWRpdG9yL3NyYy9saWIvYWUtYnV0dG9uL2FlLWJ1dHRvbi5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hbmd1bGFyLWVkaXRvci9zcmMvbGliL2FlLWJ1dHRvbi9hZS1idXR0b24uY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLFNBQVMsRUFBRSxLQUFLLEVBQW9CLE1BQU0sZUFBZSxDQUFDOzs7QUFhbEUsTUFBTSxPQUFPLGlCQUFpQjtJQUk1QjtRQUZTLGFBQVEsR0FBRyxFQUFFLENBQUM7SUFHdkIsQ0FBQzs7OEdBTFUsaUJBQWlCO2tHQUFqQixpQkFBaUIsZ05DYjlCLG1UQVFBOzJGREthLGlCQUFpQjtrQkFYN0IsU0FBUzsrQkFDRSw2QkFBNkIsUUFJakM7d0JBQ0osT0FBTyxFQUFFLHVCQUF1Qjt3QkFDaEMsWUFBWSxFQUFFLElBQUk7d0JBQ2xCLFFBQVEsRUFBRSxVQUFVO3FCQUNyQjswRUFJUSxRQUFRO3NCQUFoQixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtDb21wb25lbnQsIElucHV0LCBWaWV3RW5jYXBzdWxhdGlvbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2FlLWJ1dHRvbiwgYnV0dG9uW2FlQnV0dG9uXScsXG4gIHRlbXBsYXRlVXJsOiAnLi9hZS1idXR0b24uY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9hZS1idXR0b24uY29tcG9uZW50LnNjc3MnXSxcbiAgLy9lbmNhcHN1bGF0aW9uOiBWaWV3RW5jYXBzdWxhdGlvbi5Ob25lLFxuICBob3N0OiB7XG4gICAgJ2NsYXNzJzogJ2FuZ3VsYXItZWRpdG9yLWJ1dHRvbicsXG4gICAgJ1t0YWJJbmRleF0nOiAnLTEnLFxuICAgICdbdHlwZV0nOiAnXCJidXR0b25cIicsXG4gIH1cbn0pXG5leHBvcnQgY2xhc3MgQWVCdXR0b25Db21wb25lbnQge1xuXG4gIEBJbnB1dCgpIGljb25OYW1lID0gJyc7XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gIH1cblxufVxuIiwiPG5nLWNvbnRhaW5lciAqbmdJZj1cImljb25OYW1lOyBlbHNlIGNvbnRlbnRUZW1wbGF0ZVwiPlxuICA8c3ZnPlxuICAgIDx1c2UgW2F0dHIuaHJlZl09XCInYXNzZXRzL2FlLWljb25zL2ljb25zLnN2ZyMnICsgaWNvbk5hbWVcIiBbYXR0ci54bGluazpocmVmXT1cIidhc3NldHMvYWUtaWNvbnMvaWNvbnMuc3ZnIycgKyBpY29uTmFtZVwiPjwvdXNlPlxuICA8L3N2Zz5cbjwvbmctY29udGFpbmVyPlxuPG5nLXRlbXBsYXRlICNjb250ZW50VGVtcGxhdGU+XG4gIDxuZy1jb250ZW50PjwvbmctY29udGVudD5cbjwvbmctdGVtcGxhdGU+XG4iXX0= |
| import { Component, EventEmitter, forwardRef, HostBinding, HostListener, Input, Output, ViewChild } from '@angular/core'; | ||
| import { NG_VALUE_ACCESSOR } from '@angular/forms'; | ||
| import { isDefined } from '../utils'; | ||
| import * as i0 from "@angular/core"; | ||
| import * as i1 from "@angular/common"; | ||
| export class AeSelectComponent { | ||
| constructor(elRef, r) { | ||
| this.elRef = elRef; | ||
| this.r = r; | ||
| this.options = []; | ||
| this.disabled = false; | ||
| this.optionId = 0; | ||
| this.opened = false; | ||
| this.hidden = 'inline-block'; | ||
| // eslint-disable-next-line @angular-eslint/no-output-native, @angular-eslint/no-output-rename | ||
| this.changeEvent = new EventEmitter(); | ||
| this.onChange = () => { | ||
| }; | ||
| this.onTouched = () => { | ||
| }; | ||
| } | ||
| get label() { | ||
| return this.selectedOption && this.selectedOption.hasOwnProperty('label') ? this.selectedOption.label : 'Select'; | ||
| } | ||
| get value() { | ||
| return this.selectedOption.value; | ||
| } | ||
| ngOnInit() { | ||
| this.selectedOption = this.options[0]; | ||
| if (isDefined(this.isHidden) && this.isHidden) { | ||
| this.hide(); | ||
| } | ||
| } | ||
| hide() { | ||
| this.hidden = 'none'; | ||
| } | ||
| optionSelect(option, event) { | ||
| event.stopPropagation(); | ||
| this.setValue(option.value); | ||
| this.onChange(this.selectedOption.value); | ||
| this.changeEvent.emit(this.selectedOption.value); | ||
| this.onTouched(); | ||
| this.opened = false; | ||
| } | ||
| toggleOpen(event) { | ||
| // event.stopPropagation(); | ||
| if (this.disabled) { | ||
| return; | ||
| } | ||
| this.opened = !this.opened; | ||
| } | ||
| onClick($event) { | ||
| if (!this.elRef.nativeElement.contains($event.target)) { | ||
| this.close(); | ||
| } | ||
| } | ||
| close() { | ||
| this.opened = false; | ||
| } | ||
| get isOpen() { | ||
| return this.opened; | ||
| } | ||
| writeValue(value) { | ||
| if (!value || typeof value !== 'string') { | ||
| return; | ||
| } | ||
| this.setValue(value); | ||
| } | ||
| setValue(value) { | ||
| let index = 0; | ||
| const selectedEl = this.options.find((el, i) => { | ||
| index = i; | ||
| return el.value === value; | ||
| }); | ||
| if (selectedEl) { | ||
| this.selectedOption = selectedEl; | ||
| this.optionId = index; | ||
| } | ||
| } | ||
| registerOnChange(fn) { | ||
| this.onChange = fn; | ||
| } | ||
| registerOnTouched(fn) { | ||
| this.onTouched = fn; | ||
| } | ||
| setDisabledState(isDisabled) { | ||
| this.labelButton.nativeElement.disabled = isDisabled; | ||
| const div = this.labelButton.nativeElement; | ||
| const action = isDisabled ? 'addClass' : 'removeClass'; | ||
| this.r[action](div, 'disabled'); | ||
| this.disabled = isDisabled; | ||
| } | ||
| handleKeyDown($event) { | ||
| if (!this.opened) { | ||
| return; | ||
| } | ||
| // console.log($event.key); | ||
| // if (KeyCode[$event.key]) { | ||
| switch ($event.key) { | ||
| case 'ArrowDown': | ||
| this._handleArrowDown($event); | ||
| break; | ||
| case 'ArrowUp': | ||
| this._handleArrowUp($event); | ||
| break; | ||
| case 'Space': | ||
| this._handleSpace($event); | ||
| break; | ||
| case 'Enter': | ||
| this._handleEnter($event); | ||
| break; | ||
| case 'Tab': | ||
| this._handleTab($event); | ||
| break; | ||
| case 'Escape': | ||
| this.close(); | ||
| $event.preventDefault(); | ||
| break; | ||
| case 'Backspace': | ||
| this._handleBackspace(); | ||
| break; | ||
| } | ||
| // } else if ($event.key && $event.key.length === 1) { | ||
| // this._keyPress$.next($event.key.toLocaleLowerCase()); | ||
| // } | ||
| } | ||
| _handleArrowDown($event) { | ||
| if (this.optionId < this.options.length - 1) { | ||
| this.optionId++; | ||
| } | ||
| } | ||
| _handleArrowUp($event) { | ||
| if (this.optionId >= 1) { | ||
| this.optionId--; | ||
| } | ||
| } | ||
| _handleSpace($event) { | ||
| } | ||
| _handleEnter($event) { | ||
| this.optionSelect(this.options[this.optionId], $event); | ||
| } | ||
| _handleTab($event) { | ||
| } | ||
| _handleBackspace() { | ||
| } | ||
| } | ||
| AeSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AeSelectComponent, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component }); | ||
| AeSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AeSelectComponent, selector: "ae-select", inputs: { options: "options", isHidden: ["hidden", "isHidden"] }, outputs: { changeEvent: "change" }, host: { listeners: { "document:click": "onClick($event)", "keydown": "handleKeyDown($event)" }, properties: { "style.display": "this.hidden" } }, providers: [ | ||
| { | ||
| provide: NG_VALUE_ACCESSOR, | ||
| useExisting: forwardRef(() => AeSelectComponent), | ||
| multi: true, | ||
| } | ||
| ], viewQueries: [{ propertyName: "labelButton", first: true, predicate: ["labelButton"], descendants: true, static: true }], ngImport: i0, template: "<span class=\"ae-picker\" [ngClass]=\"{'ae-expanded':isOpen}\">\n <button [tabIndex]=\"-1\" #labelButton tabindex=\"-1\" type=\"button\" role=\"button\" class=\"ae-picker-label\" (click)=\"toggleOpen($event);\">{{label}}\n <svg>\n <use [attr.href]=\"'assets/ae-icons/icons.svg#sort'\" [attr.xlink:href]=\"'assets/ae-icons/icons.svg#sort'\"></use>\n </svg>\n </button>\n <span class=\"ae-picker-options\">\n <button tabindex=\"-1\" type=\"button\" role=\"button\" class=\"ae-picker-item\"\n *ngFor=\"let item of options; let i = index\"\n [ngClass]=\"{'selected': item.value === value, 'focused': i === optionId}\"\n (click)=\"optionSelect(item, $event)\">\n {{item.label}}\n </button>\n <span class=\"dropdown-item\" *ngIf=\"!options.length\">No items for select</span>\n </span>\n</span>\n", styles: ["a{cursor:pointer}svg{width:100%;height:100%}.ae-picker{color:var(--ae-picker-color, #444);display:inline-block;float:left;width:100%;position:relative;vertical-align:middle}.ae-picker-label{cursor:pointer;display:inline-block;padding-left:8px;padding-right:10px;position:relative;width:100%;line-height:1.8rem;vertical-align:middle;font-size:85%;text-align:left;background-color:var(--ae-picker-label-color, white);min-width:2rem;float:left;border:1px solid #ddd;border-radius:var(--ae-button-radius, 4px);text-overflow:clip;overflow:hidden;white-space:nowrap;height:2rem}.ae-picker-label:before{content:\"\";position:absolute;right:0;top:0;width:20px;height:100%;background:linear-gradient(to right,var(--ae-picker-label-color, white),var(--ae-picker-label-color, white) 100%)}.ae-picker-label:focus{outline:none}.ae-picker-label:hover{cursor:pointer;background-color:#f1f1f1;transition:.2s ease}.ae-picker-label:hover:before{background:linear-gradient(to right,#f5f5f5 100%,#ffffff 100%)}.ae-picker-label:disabled{background-color:#f5f5f5;pointer-events:none;cursor:not-allowed}.ae-picker-label:disabled:before{background:linear-gradient(to right,#f5f5f5 100%,#ffffff 100%)}.ae-picker-label svg{position:absolute;right:0;width:1rem}.ae-picker-label svg:not(:root){overflow:hidden}.ae-picker-label svg .ae-stroke{fill:none;stroke:#444;stroke-linecap:round;stroke-linejoin:round;stroke-width:2}.ae-picker-options{background-color:var(--ae-picker-option-bg-color, #fff);display:none;min-width:100%;position:absolute;white-space:nowrap;z-index:3;border:1px solid transparent;box-shadow:#0003 0 2px 8px}.ae-picker-options .ae-picker-item{cursor:pointer;display:block;padding-bottom:5px;padding-top:5px;padding-left:5px;z-index:3;text-align:left;background-color:transparent;min-width:2rem;width:100%;border:0 solid #ddd}.ae-picker-options .ae-picker-item.selected{color:#06c;background-color:var(--ae-picker-option-active-bg-color, #fff4c2)}.ae-picker-options .ae-picker-item.focused{background-color:var(--ae-picker-option-focused-bg-color, #fbf9b0)}.ae-picker-options .ae-picker-item:hover{background-color:var(--ae-picker-option-hover-bg-color, #fffa98)}.ae-expanded{display:block;margin-top:-1px;z-index:1}.ae-expanded .ae-picker-label{color:#ccc;z-index:2}.ae-expanded .ae-picker-label svg{color:#ccc;z-index:2}.ae-expanded .ae-picker-label svg .ae-stroke{stroke:#ccc}.ae-expanded .ae-picker-options{display:flex;flex-direction:column;margin-top:-1px;top:100%;z-index:3;border-color:#ccc}\n"], directives: [{ type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] }); | ||
| i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AeSelectComponent, decorators: [{ | ||
| type: Component, | ||
| args: [{ selector: 'ae-select', providers: [ | ||
| { | ||
| provide: NG_VALUE_ACCESSOR, | ||
| useExisting: forwardRef(() => AeSelectComponent), | ||
| multi: true, | ||
| } | ||
| ], template: "<span class=\"ae-picker\" [ngClass]=\"{'ae-expanded':isOpen}\">\n <button [tabIndex]=\"-1\" #labelButton tabindex=\"-1\" type=\"button\" role=\"button\" class=\"ae-picker-label\" (click)=\"toggleOpen($event);\">{{label}}\n <svg>\n <use [attr.href]=\"'assets/ae-icons/icons.svg#sort'\" [attr.xlink:href]=\"'assets/ae-icons/icons.svg#sort'\"></use>\n </svg>\n </button>\n <span class=\"ae-picker-options\">\n <button tabindex=\"-1\" type=\"button\" role=\"button\" class=\"ae-picker-item\"\n *ngFor=\"let item of options; let i = index\"\n [ngClass]=\"{'selected': item.value === value, 'focused': i === optionId}\"\n (click)=\"optionSelect(item, $event)\">\n {{item.label}}\n </button>\n <span class=\"dropdown-item\" *ngIf=\"!options.length\">No items for select</span>\n </span>\n</span>\n", styles: ["a{cursor:pointer}svg{width:100%;height:100%}.ae-picker{color:var(--ae-picker-color, #444);display:inline-block;float:left;width:100%;position:relative;vertical-align:middle}.ae-picker-label{cursor:pointer;display:inline-block;padding-left:8px;padding-right:10px;position:relative;width:100%;line-height:1.8rem;vertical-align:middle;font-size:85%;text-align:left;background-color:var(--ae-picker-label-color, white);min-width:2rem;float:left;border:1px solid #ddd;border-radius:var(--ae-button-radius, 4px);text-overflow:clip;overflow:hidden;white-space:nowrap;height:2rem}.ae-picker-label:before{content:\"\";position:absolute;right:0;top:0;width:20px;height:100%;background:linear-gradient(to right,var(--ae-picker-label-color, white),var(--ae-picker-label-color, white) 100%)}.ae-picker-label:focus{outline:none}.ae-picker-label:hover{cursor:pointer;background-color:#f1f1f1;transition:.2s ease}.ae-picker-label:hover:before{background:linear-gradient(to right,#f5f5f5 100%,#ffffff 100%)}.ae-picker-label:disabled{background-color:#f5f5f5;pointer-events:none;cursor:not-allowed}.ae-picker-label:disabled:before{background:linear-gradient(to right,#f5f5f5 100%,#ffffff 100%)}.ae-picker-label svg{position:absolute;right:0;width:1rem}.ae-picker-label svg:not(:root){overflow:hidden}.ae-picker-label svg .ae-stroke{fill:none;stroke:#444;stroke-linecap:round;stroke-linejoin:round;stroke-width:2}.ae-picker-options{background-color:var(--ae-picker-option-bg-color, #fff);display:none;min-width:100%;position:absolute;white-space:nowrap;z-index:3;border:1px solid transparent;box-shadow:#0003 0 2px 8px}.ae-picker-options .ae-picker-item{cursor:pointer;display:block;padding-bottom:5px;padding-top:5px;padding-left:5px;z-index:3;text-align:left;background-color:transparent;min-width:2rem;width:100%;border:0 solid #ddd}.ae-picker-options .ae-picker-item.selected{color:#06c;background-color:var(--ae-picker-option-active-bg-color, #fff4c2)}.ae-picker-options .ae-picker-item.focused{background-color:var(--ae-picker-option-focused-bg-color, #fbf9b0)}.ae-picker-options .ae-picker-item:hover{background-color:var(--ae-picker-option-hover-bg-color, #fffa98)}.ae-expanded{display:block;margin-top:-1px;z-index:1}.ae-expanded .ae-picker-label{color:#ccc;z-index:2}.ae-expanded .ae-picker-label svg{color:#ccc;z-index:2}.ae-expanded .ae-picker-label svg .ae-stroke{stroke:#ccc}.ae-expanded .ae-picker-options{display:flex;flex-direction:column;margin-top:-1px;top:100%;z-index:3;border-color:#ccc}\n"] }] | ||
| }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }]; }, propDecorators: { options: [{ | ||
| type: Input | ||
| }], isHidden: [{ | ||
| type: Input, | ||
| args: ['hidden'] | ||
| }], hidden: [{ | ||
| type: HostBinding, | ||
| args: ['style.display'] | ||
| }], changeEvent: [{ | ||
| type: Output, | ||
| args: ['change'] | ||
| }], labelButton: [{ | ||
| type: ViewChild, | ||
| args: ['labelButton', { static: true }] | ||
| }], onClick: [{ | ||
| type: HostListener, | ||
| args: ['document:click', ['$event']] | ||
| }], handleKeyDown: [{ | ||
| type: HostListener, | ||
| args: ['keydown', ['$event']] | ||
| }] } }); | ||
| //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ae-select.component.js","sourceRoot":"","sources":["../../../../../projects/angular-editor/src/lib/ae-select/ae-select.component.ts","../../../../../projects/angular-editor/src/lib/ae-select/ae-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  HostBinding,\n  HostListener,\n  Input,\n  OnInit,\n  Output,\n  Renderer2,\n  ViewChild,\n  ViewEncapsulation\n} from '@angular/core';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\nimport {isDefined} from '../utils';\n\nexport interface SelectOption {\n  label: string;\n  value: string;\n}\n\n@Component({\n  selector: 'ae-select',\n  templateUrl: './ae-select.component.html',\n  styleUrls: ['./ae-select.component.scss'],\n  //encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => AeSelectComponent),\n      multi: true,\n    }\n  ]\n})\nexport class AeSelectComponent implements OnInit, ControlValueAccessor {\n  @Input() options: SelectOption[] = [];\n  // eslint-disable-next-line @angular-eslint/no-input-rename\n  @Input('hidden') isHidden: boolean;\n\n  selectedOption: SelectOption;\n  disabled = false;\n  optionId = 0;\n\n  get label(): string {\n    return this.selectedOption && this.selectedOption.hasOwnProperty('label') ? this.selectedOption.label : 'Select';\n  }\n\n  opened = false;\n\n  get value(): string {\n    return this.selectedOption.value;\n  }\n\n  @HostBinding('style.display') hidden = 'inline-block';\n\n  // eslint-disable-next-line @angular-eslint/no-output-native, @angular-eslint/no-output-rename\n  @Output('change') changeEvent = new EventEmitter();\n\n  @ViewChild('labelButton', {static: true}) labelButton: ElementRef;\n\n  constructor(private elRef: ElementRef,\n              private r: Renderer2,\n  ) {\n  }\n\n  ngOnInit() {\n    this.selectedOption = this.options[0];\n    if (isDefined(this.isHidden) && this.isHidden) {\n      this.hide();\n    }\n  }\n\n  hide() {\n    this.hidden = 'none';\n  }\n\n  optionSelect(option: SelectOption, event: MouseEvent) {\n    event.stopPropagation();\n    this.setValue(option.value);\n    this.onChange(this.selectedOption.value);\n    this.changeEvent.emit(this.selectedOption.value);\n    this.onTouched();\n    this.opened = false;\n  }\n\n  toggleOpen(event: MouseEvent) {\n    // event.stopPropagation();\n    if (this.disabled) {\n      return;\n    }\n    this.opened = !this.opened;\n  }\n\n  @HostListener('document:click', ['$event'])\n  onClick($event: MouseEvent) {\n    if (!this.elRef.nativeElement.contains($event.target)) {\n      this.close();\n    }\n  }\n\n  close() {\n    this.opened = false;\n  }\n\n  get isOpen(): boolean {\n    return this.opened;\n  }\n\n  writeValue(value) {\n    if (!value || typeof value !== 'string') {\n      return;\n    }\n    this.setValue(value);\n  }\n\n  setValue(value) {\n    let index = 0;\n    const selectedEl = this.options.find((el, i) => {\n      index = i;\n      return el.value === value;\n    });\n    if (selectedEl) {\n      this.selectedOption = selectedEl;\n      this.optionId = index;\n    }\n  }\n\n  onChange: any = () => {\n  }\n  onTouched: any = () => {\n  }\n\n  registerOnChange(fn) {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn) {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this.labelButton.nativeElement.disabled = isDisabled;\n    const div = this.labelButton.nativeElement;\n    const action = isDisabled ? 'addClass' : 'removeClass';\n    this.r[action](div, 'disabled');\n    this.disabled = isDisabled;\n  }\n\n  @HostListener('keydown', ['$event'])\n  handleKeyDown($event: KeyboardEvent) {\n    if (!this.opened) {\n      return;\n    }\n    // console.log($event.key);\n    // if (KeyCode[$event.key]) {\n    switch ($event.key) {\n      case 'ArrowDown':\n        this._handleArrowDown($event);\n        break;\n      case 'ArrowUp':\n        this._handleArrowUp($event);\n        break;\n      case 'Space':\n        this._handleSpace($event);\n        break;\n      case 'Enter':\n        this._handleEnter($event);\n        break;\n      case 'Tab':\n        this._handleTab($event);\n        break;\n      case 'Escape':\n        this.close();\n        $event.preventDefault();\n        break;\n      case 'Backspace':\n        this._handleBackspace();\n        break;\n    }\n    // } else if ($event.key && $event.key.length === 1) {\n    // this._keyPress$.next($event.key.toLocaleLowerCase());\n    // }\n  }\n\n  _handleArrowDown($event) {\n    if (this.optionId < this.options.length - 1) {\n      this.optionId++;\n    }\n  }\n\n  _handleArrowUp($event) {\n    if (this.optionId >= 1) {\n      this.optionId--;\n    }\n  }\n\n  _handleSpace($event) {\n\n  }\n\n  _handleEnter($event) {\n    this.optionSelect(this.options[this.optionId], $event);\n  }\n\n  _handleTab($event) {\n\n  }\n\n  _handleBackspace() {\n\n  }\n}\n","<span class=\"ae-picker\" [ngClass]=\"{'ae-expanded':isOpen}\">\n  <button [tabIndex]=\"-1\" #labelButton tabindex=\"-1\" type=\"button\" role=\"button\" class=\"ae-picker-label\" (click)=\"toggleOpen($event);\">{{label}}\n    <svg>\n      <use [attr.href]=\"'assets/ae-icons/icons.svg#sort'\" [attr.xlink:href]=\"'assets/ae-icons/icons.svg#sort'\"></use>\n    </svg>\n  </button>\n  <span class=\"ae-picker-options\">\n    <button tabindex=\"-1\" type=\"button\" role=\"button\" class=\"ae-picker-item\"\n          *ngFor=\"let item of options; let i = index\"\n          [ngClass]=\"{'selected': item.value === value, 'focused': i === optionId}\"\n          (click)=\"optionSelect(item, $event)\">\n          {{item.label}}\n    </button>\n    <span class=\"dropdown-item\" *ngIf=\"!options.length\">No items for select</span>\n  </span>\n</span>\n"]} |
| import { Component } from '@angular/core'; | ||
| import * as i0 from "@angular/core"; | ||
| export class AeToolbarSetComponent { | ||
| constructor() { | ||
| } | ||
| } | ||
| AeToolbarSetComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AeToolbarSetComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); | ||
| AeToolbarSetComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AeToolbarSetComponent, selector: "ae-toolbar-set, [aeToolbarSet]", host: { classAttribute: "angular-editor-toolbar-set" }, ngImport: i0, template: "<ng-content></ng-content>\n", styles: ["a{cursor:pointer}:host.angular-editor-toolbar-set{display:flex;gap:1px;width:fit-content;vertical-align:baseline}\n"] }); | ||
| i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AeToolbarSetComponent, decorators: [{ | ||
| type: Component, | ||
| args: [{ selector: 'ae-toolbar-set, [aeToolbarSet]', host: { | ||
| 'class': 'angular-editor-toolbar-set' | ||
| }, template: "<ng-content></ng-content>\n", styles: ["a{cursor:pointer}:host.angular-editor-toolbar-set{display:flex;gap:1px;width:fit-content;vertical-align:baseline}\n"] }] | ||
| }], ctorParameters: function () { return []; } }); | ||
| //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWUtdG9vbGJhci1zZXQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvYW5ndWxhci1lZGl0b3Ivc3JjL2xpYi9hZS10b29sYmFyLXNldC9hZS10b29sYmFyLXNldC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hbmd1bGFyLWVkaXRvci9zcmMvbGliL2FlLXRvb2xiYXItc2V0L2FlLXRvb2xiYXItc2V0LmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxTQUFTLEVBQW9CLE1BQU0sZUFBZSxDQUFDOztBQVczRCxNQUFNLE9BQU8scUJBQXFCO0lBRWhDO0lBQ0EsQ0FBQzs7a0hBSFUscUJBQXFCO3NHQUFyQixxQkFBcUIsOEhDWGxDLDZCQUNBOzJGRFVhLHFCQUFxQjtrQkFUakMsU0FBUzsrQkFDRSxnQ0FBZ0MsUUFJcEM7d0JBQ0osT0FBTyxFQUFFLDRCQUE0QjtxQkFDdEMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0NvbXBvbmVudCwgVmlld0VuY2Fwc3VsYXRpb259IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdhZS10b29sYmFyLXNldCwgW2FlVG9vbGJhclNldF0nLFxuICB0ZW1wbGF0ZVVybDogJy4vYWUtdG9vbGJhci1zZXQuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9hZS10b29sYmFyLXNldC5jb21wb25lbnQuc2NzcyddLFxuICAvL2VuY2Fwc3VsYXRpb246IFZpZXdFbmNhcHN1bGF0aW9uLk5vbmUsXG4gIGhvc3Q6IHtcbiAgICAnY2xhc3MnOiAnYW5ndWxhci1lZGl0b3ItdG9vbGJhci1zZXQnXG4gIH1cbn0pXG5leHBvcnQgY2xhc3MgQWVUb29sYmFyU2V0Q29tcG9uZW50IHtcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgfVxuXG59XG4iLCI8bmctY29udGVudD48L25nLWNvbnRlbnQ+XG4iXX0= |
Sorry, the diff of this file is too big to display
| import { NgModule } from '@angular/core'; | ||
| import { AngularEditorComponent } from './editor/angular-editor.component'; | ||
| import { AeToolbarComponent } from './ae-toolbar/ae-toolbar.component'; | ||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms'; | ||
| import { CommonModule } from '@angular/common'; | ||
| import { AeSelectComponent } from './ae-select/ae-select.component'; | ||
| import { AeButtonComponent } from "./ae-button/ae-button.component"; | ||
| import { AeToolbarSetComponent } from './ae-toolbar-set/ae-toolbar-set.component'; | ||
| import * as i0 from "@angular/core"; | ||
| export class AngularEditorModule { | ||
| } | ||
| AngularEditorModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AngularEditorModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); | ||
| AngularEditorModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AngularEditorModule, declarations: [AngularEditorComponent, AeToolbarComponent, AeSelectComponent, AeButtonComponent, AeToolbarSetComponent], imports: [CommonModule, FormsModule, ReactiveFormsModule], exports: [AngularEditorComponent, AeToolbarComponent, AeButtonComponent, AeToolbarSetComponent] }); | ||
| AngularEditorModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AngularEditorModule, imports: [[ | ||
| CommonModule, FormsModule, ReactiveFormsModule | ||
| ]] }); | ||
| i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AngularEditorModule, decorators: [{ | ||
| type: NgModule, | ||
| args: [{ | ||
| imports: [ | ||
| CommonModule, FormsModule, ReactiveFormsModule | ||
| ], | ||
| declarations: [AngularEditorComponent, AeToolbarComponent, AeSelectComponent, AeButtonComponent, AeToolbarSetComponent], | ||
| exports: [AngularEditorComponent, AeToolbarComponent, AeButtonComponent, AeToolbarSetComponent] | ||
| }] | ||
| }] }); | ||
| //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW5ndWxhci1lZGl0b3IubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvYW5ndWxhci1lZGl0b3Ivc3JjL2xpYi9hbmd1bGFyLWVkaXRvci5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLFFBQVEsRUFBQyxNQUFNLGVBQWUsQ0FBQztBQUN2QyxPQUFPLEVBQUMsc0JBQXNCLEVBQUMsTUFBTSxtQ0FBbUMsQ0FBQztBQUN6RSxPQUFPLEVBQUMsa0JBQWtCLEVBQUMsTUFBTSxtQ0FBbUMsQ0FBQztBQUNyRSxPQUFPLEVBQUMsV0FBVyxFQUFFLG1CQUFtQixFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFDaEUsT0FBTyxFQUFDLFlBQVksRUFBQyxNQUFNLGlCQUFpQixDQUFDO0FBQzdDLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQ3BFLE9BQU8sRUFBQyxpQkFBaUIsRUFBQyxNQUFNLGlDQUFpQyxDQUFDO0FBQ2xFLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDJDQUEyQyxDQUFDOztBQVNsRixNQUFNLE9BQU8sbUJBQW1COztnSEFBbkIsbUJBQW1CO2lIQUFuQixtQkFBbUIsaUJBSGYsc0JBQXNCLEVBQUUsa0JBQWtCLEVBQUUsaUJBQWlCLEVBQUUsaUJBQWlCLEVBQUUscUJBQXFCLGFBRnBILFlBQVksRUFBRSxXQUFXLEVBQUUsbUJBQW1CLGFBR3RDLHNCQUFzQixFQUFFLGtCQUFrQixFQUFFLGlCQUFpQixFQUFFLHFCQUFxQjtpSEFFbkYsbUJBQW1CLFlBTnJCO1lBQ1AsWUFBWSxFQUFFLFdBQVcsRUFBRSxtQkFBbUI7U0FDL0M7MkZBSVUsbUJBQW1CO2tCQVAvQixRQUFRO21CQUFDO29CQUNSLE9BQU8sRUFBRTt3QkFDUCxZQUFZLEVBQUUsV0FBVyxFQUFFLG1CQUFtQjtxQkFDL0M7b0JBQ0QsWUFBWSxFQUFFLENBQUMsc0JBQXNCLEVBQUUsa0JBQWtCLEVBQUUsaUJBQWlCLEVBQUUsaUJBQWlCLEVBQUUscUJBQXFCLENBQUM7b0JBQ3ZILE9BQU8sRUFBRSxDQUFDLHNCQUFzQixFQUFFLGtCQUFrQixFQUFFLGlCQUFpQixFQUFFLHFCQUFxQixDQUFDO2lCQUNoRyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7TmdNb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtBbmd1bGFyRWRpdG9yQ29tcG9uZW50fSBmcm9tICcuL2VkaXRvci9hbmd1bGFyLWVkaXRvci5jb21wb25lbnQnO1xuaW1wb3J0IHtBZVRvb2xiYXJDb21wb25lbnR9IGZyb20gJy4vYWUtdG9vbGJhci9hZS10b29sYmFyLmNvbXBvbmVudCc7XG5pbXBvcnQge0Zvcm1zTW9kdWxlLCBSZWFjdGl2ZUZvcm1zTW9kdWxlfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQge0NvbW1vbk1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEFlU2VsZWN0Q29tcG9uZW50IH0gZnJvbSAnLi9hZS1zZWxlY3QvYWUtc2VsZWN0LmNvbXBvbmVudCc7XG5pbXBvcnQge0FlQnV0dG9uQ29tcG9uZW50fSBmcm9tIFwiLi9hZS1idXR0b24vYWUtYnV0dG9uLmNvbXBvbmVudFwiO1xuaW1wb3J0IHsgQWVUb29sYmFyU2V0Q29tcG9uZW50IH0gZnJvbSAnLi9hZS10b29sYmFyLXNldC9hZS10b29sYmFyLXNldC5jb21wb25lbnQnO1xuXG5ATmdNb2R1bGUoe1xuICBpbXBvcnRzOiBbXG4gICAgQ29tbW9uTW9kdWxlLCBGb3Jtc01vZHVsZSwgUmVhY3RpdmVGb3Jtc01vZHVsZVxuICBdLFxuICBkZWNsYXJhdGlvbnM6IFtBbmd1bGFyRWRpdG9yQ29tcG9uZW50LCBBZVRvb2xiYXJDb21wb25lbnQsIEFlU2VsZWN0Q29tcG9uZW50LCBBZUJ1dHRvbkNvbXBvbmVudCwgQWVUb29sYmFyU2V0Q29tcG9uZW50XSxcbiAgZXhwb3J0czogW0FuZ3VsYXJFZGl0b3JDb21wb25lbnQsIEFlVG9vbGJhckNvbXBvbmVudCwgQWVCdXR0b25Db21wb25lbnQsIEFlVG9vbGJhclNldENvbXBvbmVudF1cbn0pXG5leHBvcnQgY2xhc3MgQW5ndWxhckVkaXRvck1vZHVsZSB7XG59XG4iXX0= |
| import { Inject, Injectable } from '@angular/core'; | ||
| import { DOCUMENT } from '@angular/common'; | ||
| import * as i0 from "@angular/core"; | ||
| import * as i1 from "@angular/common/http"; | ||
| export class AngularEditorService { | ||
| constructor(http, doc) { | ||
| this.http = http; | ||
| this.doc = doc; | ||
| /** | ||
| * save selection when the editor is focussed out | ||
| */ | ||
| this.saveSelection = () => { | ||
| if (this.doc.getSelection) { | ||
| const sel = this.doc.getSelection(); | ||
| if (sel.getRangeAt && sel.rangeCount) { | ||
| this.savedSelection = sel.getRangeAt(0); | ||
| this.selectedText = sel.toString(); | ||
| } | ||
| } | ||
| else if (this.doc.getSelection && this.doc.createRange) { | ||
| this.savedSelection = document.createRange(); | ||
| } | ||
| else { | ||
| this.savedSelection = null; | ||
| } | ||
| }; | ||
| } | ||
| /** | ||
| * Executed command from editor header buttons exclude toggleEditorMode | ||
| * @param command string from triggerCommand | ||
| * @param value | ||
| */ | ||
| executeCommand(command, value) { | ||
| const commands = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'p', 'pre']; | ||
| if (commands.includes(command)) { | ||
| this.doc.execCommand('formatBlock', false, command); | ||
| return; | ||
| } | ||
| this.doc.execCommand(command, false, value); | ||
| } | ||
| /** | ||
| * Create URL link | ||
| * @param url string from UI prompt | ||
| */ | ||
| createLink(url) { | ||
| if (!url.includes('http')) { | ||
| this.doc.execCommand('createlink', false, url); | ||
| } | ||
| else { | ||
| const newUrl = '<a href="' + url + '" target="_blank">' + this.selectedText + '</a>'; | ||
| this.insertHtml(newUrl); | ||
| } | ||
| } | ||
| /** | ||
| * insert color either font or background | ||
| * | ||
| * @param color color to be inserted | ||
| * @param where where the color has to be inserted either text/background | ||
| */ | ||
| insertColor(color, where) { | ||
| const restored = this.restoreSelection(); | ||
| if (restored) { | ||
| if (where === 'textColor') { | ||
| this.doc.execCommand('foreColor', false, color); | ||
| } | ||
| else { | ||
| this.doc.execCommand('hiliteColor', false, color); | ||
| } | ||
| } | ||
| } | ||
| /** | ||
| * Set font name | ||
| * @param fontName string | ||
| */ | ||
| setFontName(fontName) { | ||
| this.doc.execCommand('fontName', false, fontName); | ||
| } | ||
| /** | ||
| * Set font size | ||
| * @param fontSize string | ||
| */ | ||
| setFontSize(fontSize) { | ||
| this.doc.execCommand('fontSize', false, fontSize); | ||
| } | ||
| /** | ||
| * Create raw HTML | ||
| * @param html HTML string | ||
| */ | ||
| insertHtml(html) { | ||
| const isHTMLInserted = this.doc.execCommand('insertHTML', false, html); | ||
| if (!isHTMLInserted) { | ||
| throw new Error('Unable to perform the operation'); | ||
| } | ||
| } | ||
| /** | ||
| * restore selection when the editor is focused in | ||
| * | ||
| * saved selection when the editor is focused out | ||
| */ | ||
| restoreSelection() { | ||
| if (this.savedSelection) { | ||
| if (this.doc.getSelection) { | ||
| const sel = this.doc.getSelection(); | ||
| sel.removeAllRanges(); | ||
| sel.addRange(this.savedSelection); | ||
| return true; | ||
| } | ||
| else if (this.doc.getSelection /*&& this.savedSelection.select*/) { | ||
| // this.savedSelection.select(); | ||
| return true; | ||
| } | ||
| } | ||
| else { | ||
| return false; | ||
| } | ||
| } | ||
| /** | ||
| * setTimeout used for execute 'saveSelection' method in next event loop iteration | ||
| */ | ||
| executeInNextQueueIteration(callbackFn, timeout = 1e2) { | ||
| setTimeout(callbackFn, timeout); | ||
| } | ||
| /** check any selection is made or not */ | ||
| checkSelection() { | ||
| const selectedText = this.savedSelection.toString(); | ||
| if (selectedText.length === 0) { | ||
| throw new Error('No Selection Made'); | ||
| } | ||
| return true; | ||
| } | ||
| /** | ||
| * Upload file to uploadUrl | ||
| * @param file The file | ||
| */ | ||
| uploadImage(file) { | ||
| const uploadData = new FormData(); | ||
| uploadData.append('file', file, file.name); | ||
| return this.http.post(this.uploadUrl, uploadData, { | ||
| reportProgress: true, | ||
| observe: 'events', | ||
| withCredentials: this.uploadWithCredentials, | ||
| }); | ||
| } | ||
| /** | ||
| * Insert image with Url | ||
| * @param imageUrl The imageUrl. | ||
| */ | ||
| insertImage(imageUrl) { | ||
| this.doc.execCommand('insertImage', false, imageUrl); | ||
| } | ||
| setDefaultParagraphSeparator(separator) { | ||
| this.doc.execCommand('defaultParagraphSeparator', false, separator); | ||
| } | ||
| createCustomClass(customClass) { | ||
| let newTag = this.selectedText; | ||
| if (customClass) { | ||
| const tagName = customClass.tag ? customClass.tag : 'span'; | ||
| newTag = '<' + tagName + ' class="' + customClass.class + '">' + this.selectedText + '</' + tagName + '>'; | ||
| } | ||
| this.insertHtml(newTag); | ||
| } | ||
| insertVideo(videoUrl) { | ||
| if (videoUrl.match('www.youtube.com')) { | ||
| this.insertYouTubeVideoTag(videoUrl); | ||
| } | ||
| if (videoUrl.match('vimeo.com')) { | ||
| this.insertVimeoVideoTag(videoUrl); | ||
| } | ||
| } | ||
| insertYouTubeVideoTag(videoUrl) { | ||
| const id = videoUrl.split('v=')[1]; | ||
| const imageUrl = `https://img.youtube.com/vi/${id}/0.jpg`; | ||
| const thumbnail = ` | ||
| <div style='position: relative'> | ||
| <a href='${videoUrl}' target='_blank'> | ||
| <img src="${imageUrl}" alt="click to watch"/> | ||
| <img style='position: absolute; left:200px; top:140px' | ||
| src="https://img.icons8.com/color/96/000000/youtube-play.png"/> | ||
| </a> | ||
| </div>`; | ||
| this.insertHtml(thumbnail); | ||
| } | ||
| insertVimeoVideoTag(videoUrl) { | ||
| const sub = this.http.get(`https://vimeo.com/api/oembed.json?url=${videoUrl}`).subscribe(data => { | ||
| const imageUrl = data.thumbnail_url_with_play_button; | ||
| const thumbnail = `<div> | ||
| <a href='${videoUrl}' target='_blank'> | ||
| <img src="${imageUrl}" alt="${data.title}"/> | ||
| </a> | ||
| </div>`; | ||
| this.insertHtml(thumbnail); | ||
| sub.unsubscribe(); | ||
| }); | ||
| } | ||
| nextNode(node) { | ||
| if (node.hasChildNodes()) { | ||
| return node.firstChild; | ||
| } | ||
| else { | ||
| while (node && !node.nextSibling) { | ||
| node = node.parentNode; | ||
| } | ||
| if (!node) { | ||
| return null; | ||
| } | ||
| return node.nextSibling; | ||
| } | ||
| } | ||
| getRangeSelectedNodes(range, includePartiallySelectedContainers) { | ||
| let node = range.startContainer; | ||
| const endNode = range.endContainer; | ||
| let rangeNodes = []; | ||
| // Special case for a range that is contained within a single node | ||
| if (node === endNode) { | ||
| rangeNodes = [node]; | ||
| } | ||
| else { | ||
| // Iterate nodes until we hit the end container | ||
| while (node && node !== endNode) { | ||
| rangeNodes.push(node = this.nextNode(node)); | ||
| } | ||
| // Add partially selected nodes at the start of the range | ||
| node = range.startContainer; | ||
| while (node && node !== range.commonAncestorContainer) { | ||
| rangeNodes.unshift(node); | ||
| node = node.parentNode; | ||
| } | ||
| } | ||
| // Add ancestors of the range container, if required | ||
| if (includePartiallySelectedContainers) { | ||
| node = range.commonAncestorContainer; | ||
| while (node) { | ||
| rangeNodes.push(node); | ||
| node = node.parentNode; | ||
| } | ||
| } | ||
| return rangeNodes; | ||
| } | ||
| getSelectedNodes() { | ||
| const nodes = []; | ||
| if (this.doc.getSelection) { | ||
| const sel = this.doc.getSelection(); | ||
| for (let i = 0, len = sel.rangeCount; i < len; ++i) { | ||
| nodes.push.apply(nodes, this.getRangeSelectedNodes(sel.getRangeAt(i), true)); | ||
| } | ||
| } | ||
| return nodes; | ||
| } | ||
| replaceWithOwnChildren(el) { | ||
| const parent = el.parentNode; | ||
| while (el.hasChildNodes()) { | ||
| parent.insertBefore(el.firstChild, el); | ||
| } | ||
| parent.removeChild(el); | ||
| } | ||
| removeSelectedElements(tagNames) { | ||
| const tagNamesArray = tagNames.toLowerCase().split(','); | ||
| this.getSelectedNodes().forEach((node) => { | ||
| if (node.nodeType === 1 && | ||
| tagNamesArray.indexOf(node.tagName.toLowerCase()) > -1) { | ||
| // Remove the node and replace it with its children | ||
| this.replaceWithOwnChildren(node); | ||
| } | ||
| }); | ||
| } | ||
| } | ||
| AngularEditorService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AngularEditorService, deps: [{ token: i1.HttpClient }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Injectable }); | ||
| AngularEditorService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AngularEditorService }); | ||
| i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AngularEditorService, decorators: [{ | ||
| type: Injectable | ||
| }], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: undefined, decorators: [{ | ||
| type: Inject, | ||
| args: [DOCUMENT] | ||
| }] }]; } }); | ||
| //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"angular-editor.service.js","sourceRoot":"","sources":["../../../../projects/angular-editor/src/lib/angular-editor.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {Inject, Injectable} from '@angular/core';\nimport {HttpClient, HttpEvent} from '@angular/common/http';\nimport {Observable} from 'rxjs';\nimport {DOCUMENT} from '@angular/common';\nimport {CustomClass} from './config';\n\nexport interface UploadResponse {\n  imageUrl: string;\n}\n\n@Injectable()\nexport class AngularEditorService {\n\n  savedSelection: Range | null;\n  selectedText: string;\n  uploadUrl: string;\n  uploadWithCredentials: boolean;\n\n  constructor(\n    private http: HttpClient,\n    @Inject(DOCUMENT) private doc: any\n  ) { }\n\n  /**\n   * Executed command from editor header buttons exclude toggleEditorMode\n   * @param command string from triggerCommand\n   * @param value\n   */\n  executeCommand(command: string, value?: string) {\n    const commands = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'p', 'pre'];\n    if (commands.includes(command)) {\n      this.doc.execCommand('formatBlock', false, command);\n      return;\n    }\n    this.doc.execCommand(command, false, value);\n  }\n\n  /**\n   * Create URL link\n   * @param url string from UI prompt\n   */\n  createLink(url: string) {\n    if (!url.includes('http')) {\n      this.doc.execCommand('createlink', false, url);\n    } else {\n      const newUrl = '<a href=\"' + url + '\" target=\"_blank\">' + this.selectedText + '</a>';\n      this.insertHtml(newUrl);\n    }\n  }\n\n  /**\n   * insert color either font or background\n   *\n   * @param color color to be inserted\n   * @param where where the color has to be inserted either text/background\n   */\n  insertColor(color: string, where: string): void {\n    const restored = this.restoreSelection();\n    if (restored) {\n      if (where === 'textColor') {\n        this.doc.execCommand('foreColor', false, color);\n      } else {\n        this.doc.execCommand('hiliteColor', false, color);\n      }\n    }\n  }\n\n  /**\n   * Set font name\n   * @param fontName string\n   */\n  setFontName(fontName: string) {\n    this.doc.execCommand('fontName', false, fontName);\n  }\n\n  /**\n   * Set font size\n   * @param fontSize string\n   */\n  setFontSize(fontSize: string) {\n    this.doc.execCommand('fontSize', false, fontSize);\n  }\n\n  /**\n   * Create raw HTML\n   * @param html HTML string\n   */\n  insertHtml(html: string): void {\n\n    const isHTMLInserted = this.doc.execCommand('insertHTML', false, html);\n\n    if (!isHTMLInserted) {\n      throw new Error('Unable to perform the operation');\n    }\n  }\n\n  /**\n   * save selection when the editor is focussed out\n   */\n  public saveSelection = (): void => {\n    if (this.doc.getSelection) {\n      const sel = this.doc.getSelection();\n      if (sel.getRangeAt && sel.rangeCount) {\n        this.savedSelection = sel.getRangeAt(0);\n        this.selectedText = sel.toString();\n      }\n    } else if (this.doc.getSelection && this.doc.createRange) {\n      this.savedSelection = document.createRange();\n    } else {\n      this.savedSelection = null;\n    }\n  }\n\n  /**\n   * restore selection when the editor is focused in\n   *\n   * saved selection when the editor is focused out\n   */\n  restoreSelection(): boolean {\n    if (this.savedSelection) {\n      if (this.doc.getSelection) {\n        const sel = this.doc.getSelection();\n        sel.removeAllRanges();\n        sel.addRange(this.savedSelection);\n        return true;\n      } else if (this.doc.getSelection /*&& this.savedSelection.select*/) {\n        // this.savedSelection.select();\n        return true;\n      }\n    } else {\n      return false;\n    }\n  }\n\n  /**\n   * setTimeout used for execute 'saveSelection' method in next event loop iteration\n   */\n  public executeInNextQueueIteration(callbackFn: (...args: any[]) => any, timeout = 1e2): void {\n    setTimeout(callbackFn, timeout);\n  }\n\n  /** check any selection is made or not */\n  private checkSelection(): any {\n\n    const selectedText = this.savedSelection.toString();\n\n    if (selectedText.length === 0) {\n      throw new Error('No Selection Made');\n    }\n    return true;\n  }\n\n  /**\n   * Upload file to uploadUrl\n   * @param file The file\n   */\n  uploadImage(file: File): Observable<HttpEvent<UploadResponse>> {\n\n    const uploadData: FormData = new FormData();\n\n    uploadData.append('file', file, file.name);\n\n    return this.http.post<UploadResponse>(this.uploadUrl, uploadData, {\n      reportProgress: true,\n      observe: 'events',\n      withCredentials: this.uploadWithCredentials,\n    });\n  }\n\n  /**\n   * Insert image with Url\n   * @param imageUrl The imageUrl.\n   */\n  insertImage(imageUrl: string) {\n    this.doc.execCommand('insertImage', false, imageUrl);\n  }\n\n  setDefaultParagraphSeparator(separator: string) {\n    this.doc.execCommand('defaultParagraphSeparator', false, separator);\n  }\n\n  createCustomClass(customClass: CustomClass) {\n    let newTag = this.selectedText;\n    if (customClass) {\n      const tagName = customClass.tag ? customClass.tag : 'span';\n      newTag = '<' + tagName + ' class=\"' + customClass.class + '\">' + this.selectedText + '</' + tagName + '>';\n    }\n    this.insertHtml(newTag);\n  }\n\n  insertVideo(videoUrl: string) {\n    if (videoUrl.match('www.youtube.com')) {\n      this.insertYouTubeVideoTag(videoUrl);\n    }\n    if (videoUrl.match('vimeo.com')) {\n      this.insertVimeoVideoTag(videoUrl);\n    }\n  }\n\n  private insertYouTubeVideoTag(videoUrl: string): void {\n    const id = videoUrl.split('v=')[1];\n    const imageUrl = `https://img.youtube.com/vi/${id}/0.jpg`;\n    const thumbnail = `\n      <div style='position: relative'>\n        <a href='${videoUrl}' target='_blank'>\n          <img src=\"${imageUrl}\" alt=\"click to watch\"/>\n          <img style='position: absolute; left:200px; top:140px'\n          src=\"https://img.icons8.com/color/96/000000/youtube-play.png\"/>\n        </a>\n      </div>`;\n    this.insertHtml(thumbnail);\n  }\n\n  private insertVimeoVideoTag(videoUrl: string): void {\n    const sub = this.http.get<any>(`https://vimeo.com/api/oembed.json?url=${videoUrl}`).subscribe(data => {\n      const imageUrl = data.thumbnail_url_with_play_button;\n      const thumbnail = `<div>\n        <a href='${videoUrl}' target='_blank'>\n          <img src=\"${imageUrl}\" alt=\"${data.title}\"/>\n        </a>\n      </div>`;\n      this.insertHtml(thumbnail);\n      sub.unsubscribe();\n    });\n  }\n\n  nextNode(node) {\n    if (node.hasChildNodes()) {\n      return node.firstChild;\n    } else {\n      while (node && !node.nextSibling) {\n        node = node.parentNode;\n      }\n      if (!node) {\n        return null;\n      }\n      return node.nextSibling;\n    }\n  }\n\n  getRangeSelectedNodes(range, includePartiallySelectedContainers) {\n    let node = range.startContainer;\n    const endNode = range.endContainer;\n    let rangeNodes = [];\n\n    // Special case for a range that is contained within a single node\n    if (node === endNode) {\n      rangeNodes = [node];\n    } else {\n      // Iterate nodes until we hit the end container\n      while (node && node !== endNode) {\n        rangeNodes.push( node = this.nextNode(node) );\n      }\n\n      // Add partially selected nodes at the start of the range\n      node = range.startContainer;\n      while (node && node !== range.commonAncestorContainer) {\n        rangeNodes.unshift(node);\n        node = node.parentNode;\n      }\n    }\n\n    // Add ancestors of the range container, if required\n    if (includePartiallySelectedContainers) {\n      node = range.commonAncestorContainer;\n      while (node) {\n        rangeNodes.push(node);\n        node = node.parentNode;\n      }\n    }\n\n    return rangeNodes;\n  }\n\n  getSelectedNodes() {\n    const nodes = [];\n    if (this.doc.getSelection) {\n      const sel = this.doc.getSelection();\n      for (let i = 0, len = sel.rangeCount; i < len; ++i) {\n        nodes.push.apply(nodes, this.getRangeSelectedNodes(sel.getRangeAt(i), true));\n      }\n    }\n    return nodes;\n  }\n\n  replaceWithOwnChildren(el) {\n    const parent = el.parentNode;\n    while (el.hasChildNodes()) {\n      parent.insertBefore(el.firstChild, el);\n    }\n    parent.removeChild(el);\n  }\n\n  removeSelectedElements(tagNames) {\n    const tagNamesArray = tagNames.toLowerCase().split(',');\n    this.getSelectedNodes().forEach((node) => {\n      if (node.nodeType === 1 &&\n        tagNamesArray.indexOf(node.tagName.toLowerCase()) > -1) {\n        // Remove the node and replace it with its children\n        this.replaceWithOwnChildren(node);\n      }\n    });\n  }\n}\n"]} |
| export const angularEditorConfig = { | ||
| editable: true, | ||
| spellcheck: true, | ||
| height: 'auto', | ||
| minHeight: '0', | ||
| maxHeight: 'auto', | ||
| width: 'auto', | ||
| minWidth: '0', | ||
| translate: 'yes', | ||
| enableToolbar: true, | ||
| showToolbar: true, | ||
| placeholder: 'Enter text here...', | ||
| defaultParagraphSeparator: '', | ||
| defaultFontName: '', | ||
| defaultFontSize: '', | ||
| fonts: [ | ||
| { class: 'arial', name: 'Arial' }, | ||
| { class: 'times-new-roman', name: 'Times New Roman' }, | ||
| { class: 'calibri', name: 'Calibri' }, | ||
| { class: 'comic-sans-ms', name: 'Comic Sans MS' } | ||
| ], | ||
| uploadUrl: 'v1/image', | ||
| uploadWithCredentials: false, | ||
| sanitize: true, | ||
| toolbarPosition: 'top', | ||
| outline: true, | ||
| /*toolbarHiddenButtons: [ | ||
| ['bold', 'italic', 'underline', 'strikeThrough', 'superscript', 'subscript'], | ||
| ['heading', 'fontName', 'fontSize', 'color'], | ||
| ['justifyLeft', 'justifyCenter', 'justifyRight', 'justifyFull', 'indent', 'outdent'], | ||
| ['cut', 'copy', 'delete', 'removeFormat', 'undo', 'redo'], | ||
| ['paragraph', 'blockquote', 'removeBlockquote', 'horizontalLine', 'orderedList', 'unorderedList'], | ||
| ['link', 'unlink', 'image', 'video'] | ||
| ]*/ | ||
| }; | ||
| //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvYW5ndWxhci1lZGl0b3Ivc3JjL2xpYi9jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMENBLE1BQU0sQ0FBQyxNQUFNLG1CQUFtQixHQUF3QjtJQUN0RCxRQUFRLEVBQUUsSUFBSTtJQUNkLFVBQVUsRUFBRSxJQUFJO0lBQ2hCLE1BQU0sRUFBRSxNQUFNO0lBQ2QsU0FBUyxFQUFFLEdBQUc7SUFDZCxTQUFTLEVBQUUsTUFBTTtJQUNqQixLQUFLLEVBQUUsTUFBTTtJQUNiLFFBQVEsRUFBRSxHQUFHO0lBQ2IsU0FBUyxFQUFFLEtBQUs7SUFDaEIsYUFBYSxFQUFFLElBQUk7SUFDbkIsV0FBVyxFQUFFLElBQUk7SUFDakIsV0FBVyxFQUFFLG9CQUFvQjtJQUNqQyx5QkFBeUIsRUFBRSxFQUFFO0lBQzdCLGVBQWUsRUFBRSxFQUFFO0lBQ25CLGVBQWUsRUFBRSxFQUFFO0lBQ25CLEtBQUssRUFBRTtRQUNMLEVBQUMsS0FBSyxFQUFFLE9BQU8sRUFBRSxJQUFJLEVBQUUsT0FBTyxFQUFDO1FBQy9CLEVBQUMsS0FBSyxFQUFFLGlCQUFpQixFQUFFLElBQUksRUFBRSxpQkFBaUIsRUFBQztRQUNuRCxFQUFDLEtBQUssRUFBRSxTQUFTLEVBQUUsSUFBSSxFQUFFLFNBQVMsRUFBQztRQUNuQyxFQUFDLEtBQUssRUFBRSxlQUFlLEVBQUUsSUFBSSxFQUFFLGVBQWUsRUFBQztLQUNoRDtJQUNELFNBQVMsRUFBRSxVQUFVO0lBQ3JCLHFCQUFxQixFQUFFLEtBQUs7SUFDNUIsUUFBUSxFQUFFLElBQUk7SUFDZCxlQUFlLEVBQUUsS0FBSztJQUN0QixPQUFPLEVBQUUsSUFBSTtJQUNiOzs7Ozs7O09BT0c7Q0FDSixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVXBsb2FkUmVzcG9uc2UgfSBmcm9tICcuL2FuZ3VsYXItZWRpdG9yLnNlcnZpY2UnO1xuaW1wb3J0IHsgSHR0cEV2ZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uL2h0dHAnO1xuaW1wb3J0IHsgT2JzZXJ2YWJsZSB9IGZyb20gJ3J4anMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEN1c3RvbUNsYXNzIHtcbiAgbmFtZTogc3RyaW5nO1xuICBjbGFzczogc3RyaW5nO1xuICB0YWc/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRm9udCB7XG4gIG5hbWU6IHN0cmluZztcbiAgY2xhc3M6IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBbmd1bGFyRWRpdG9yQ29uZmlnIHtcbiAgZWRpdGFibGU/OiBib29sZWFuO1xuICBzcGVsbGNoZWNrPzogYm9vbGVhbjtcbiAgaGVpZ2h0PzogJ2F1dG8nIHwgc3RyaW5nO1xuICBtaW5IZWlnaHQ/OiAnMCcgfCBzdHJpbmc7XG4gIG1heEhlaWdodD86ICdhdXRvJyB8IHN0cmluZztcbiAgd2lkdGg/OiAnYXV0bycgfCBzdHJpbmc7XG4gIG1pbldpZHRoPzogJzAnIHwgc3RyaW5nO1xuICB0cmFuc2xhdGU/OiAneWVzJyB8ICdub3cnIHwgc3RyaW5nO1xuICBlbmFibGVUb29sYmFyPzogYm9vbGVhbjtcbiAgc2hvd1Rvb2xiYXI/OiBib29sZWFuO1xuICBwbGFjZWhvbGRlcj86IHN0cmluZztcbiAgZGVmYXVsdFBhcmFncmFwaFNlcGFyYXRvcj86IHN0cmluZztcbiAgZGVmYXVsdEZvbnROYW1lPzogc3RyaW5nO1xuICBkZWZhdWx0Rm9udFNpemU/OiAnMScgfCAnMicgfCAnMycgfCAnNCcgfCAnNScgfCAnNicgfCAnNycgfCBzdHJpbmc7XG4gIHVwbG9hZFVybD86IHN0cmluZztcbiAgdXBsb2FkPzogKGZpbGU6IEZpbGUpID0+IE9ic2VydmFibGU8SHR0cEV2ZW50PFVwbG9hZFJlc3BvbnNlPj47XG4gIHVwbG9hZFdpdGhDcmVkZW50aWFscz86IGJvb2xlYW47XG4gIGZvbnRzPzogRm9udFtdO1xuICBjdXN0b21DbGFzc2VzPzogQ3VzdG9tQ2xhc3NbXTtcbiAgc2FuaXRpemU/OiBib29sZWFuO1xuICB0b29sYmFyUG9zaXRpb24/OiAndG9wJyB8ICdib3R0b20nO1xuICBvdXRsaW5lPzogYm9vbGVhbjtcbiAgdG9vbGJhckhpZGRlbkJ1dHRvbnM/OiBzdHJpbmdbXVtdO1xuICByYXdQYXN0ZT86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjb25zdCBhbmd1bGFyRWRpdG9yQ29uZmlnOiBBbmd1bGFyRWRpdG9yQ29uZmlnID0ge1xuICBlZGl0YWJsZTogdHJ1ZSxcbiAgc3BlbGxjaGVjazogdHJ1ZSxcbiAgaGVpZ2h0OiAnYXV0bycsXG4gIG1pbkhlaWdodDogJzAnLFxuICBtYXhIZWlnaHQ6ICdhdXRvJyxcbiAgd2lkdGg6ICdhdXRvJyxcbiAgbWluV2lkdGg6ICcwJyxcbiAgdHJhbnNsYXRlOiAneWVzJyxcbiAgZW5hYmxlVG9vbGJhcjogdHJ1ZSxcbiAgc2hvd1Rvb2xiYXI6IHRydWUsXG4gIHBsYWNlaG9sZGVyOiAnRW50ZXIgdGV4dCBoZXJlLi4uJyxcbiAgZGVmYXVsdFBhcmFncmFwaFNlcGFyYXRvcjogJycsXG4gIGRlZmF1bHRGb250TmFtZTogJycsXG4gIGRlZmF1bHRGb250U2l6ZTogJycsXG4gIGZvbnRzOiBbXG4gICAge2NsYXNzOiAnYXJpYWwnLCBuYW1lOiAnQXJpYWwnfSxcbiAgICB7Y2xhc3M6ICd0aW1lcy1uZXctcm9tYW4nLCBuYW1lOiAnVGltZXMgTmV3IFJvbWFuJ30sXG4gICAge2NsYXNzOiAnY2FsaWJyaScsIG5hbWU6ICdDYWxpYnJpJ30sXG4gICAge2NsYXNzOiAnY29taWMtc2Fucy1tcycsIG5hbWU6ICdDb21pYyBTYW5zIE1TJ31cbiAgXSxcbiAgdXBsb2FkVXJsOiAndjEvaW1hZ2UnLFxuICB1cGxvYWRXaXRoQ3JlZGVudGlhbHM6IGZhbHNlLFxuICBzYW5pdGl6ZTogdHJ1ZSxcbiAgdG9vbGJhclBvc2l0aW9uOiAndG9wJyxcbiAgb3V0bGluZTogdHJ1ZSxcbiAgLyp0b29sYmFySGlkZGVuQnV0dG9uczogW1xuICAgIFsnYm9sZCcsICdpdGFsaWMnLCAndW5kZXJsaW5lJywgJ3N0cmlrZVRocm91Z2gnLCAnc3VwZXJzY3JpcHQnLCAnc3Vic2NyaXB0J10sXG4gICAgWydoZWFkaW5nJywgJ2ZvbnROYW1lJywgJ2ZvbnRTaXplJywgJ2NvbG9yJ10sXG4gICAgWydqdXN0aWZ5TGVmdCcsICdqdXN0aWZ5Q2VudGVyJywgJ2p1c3RpZnlSaWdodCcsICdqdXN0aWZ5RnVsbCcsICdpbmRlbnQnLCAnb3V0ZGVudCddLFxuICAgIFsnY3V0JywgJ2NvcHknLCAnZGVsZXRlJywgJ3JlbW92ZUZvcm1hdCcsICd1bmRvJywgJ3JlZG8nXSxcbiAgICBbJ3BhcmFncmFwaCcsICdibG9ja3F1b3RlJywgJ3JlbW92ZUJsb2NrcXVvdGUnLCAnaG9yaXpvbnRhbExpbmUnLCAnb3JkZXJlZExpc3QnLCAndW5vcmRlcmVkTGlzdCddLFxuICAgIFsnbGluaycsICd1bmxpbmsnLCAnaW1hZ2UnLCAndmlkZW8nXVxuICBdKi9cbn07XG4iXX0= |
| import { DOCUMENT } from '@angular/common'; | ||
| import { Attribute, Component, ContentChild, EventEmitter, forwardRef, HostBinding, HostListener, Inject, Input, Output, SecurityContext, ViewChild } from '@angular/core'; | ||
| import { NG_VALUE_ACCESSOR } from '@angular/forms'; | ||
| import { AngularEditorService } from '../angular-editor.service'; | ||
| import { angularEditorConfig } from '../config'; | ||
| import { isDefined } from '../utils'; | ||
| import * as i0 from "@angular/core"; | ||
| import * as i1 from "../angular-editor.service"; | ||
| import * as i2 from "@angular/platform-browser"; | ||
| import * as i3 from "../ae-toolbar/ae-toolbar.component"; | ||
| import * as i4 from "@angular/common"; | ||
| export class AngularEditorComponent { | ||
| constructor(r, editorService, doc, sanitizer, cdRef, defaultTabIndex, autoFocus) { | ||
| this.r = r; | ||
| this.editorService = editorService; | ||
| this.doc = doc; | ||
| this.sanitizer = sanitizer; | ||
| this.cdRef = cdRef; | ||
| this.autoFocus = autoFocus; | ||
| this.modeVisual = true; | ||
| this.showPlaceholder = false; | ||
| this.disabled = false; | ||
| this.focused = false; | ||
| this.touched = false; | ||
| this.changed = false; | ||
| this.id = ''; | ||
| this.config = angularEditorConfig; | ||
| this.placeholder = ''; | ||
| this.executeCommandFn = this.executeCommand.bind(this); | ||
| this.viewMode = new EventEmitter(); | ||
| /** emits `blur` event when focused out from the textarea */ | ||
| // eslint-disable-next-line @angular-eslint/no-output-native, @angular-eslint/no-output-rename | ||
| this.blurEvent = new EventEmitter(); | ||
| /** emits `focus` event when focused in to the textarea */ | ||
| // eslint-disable-next-line @angular-eslint/no-output-rename, @angular-eslint/no-output-native | ||
| this.focusEvent = new EventEmitter(); | ||
| this.tabindex = -1; | ||
| const parsedTabIndex = Number(defaultTabIndex); | ||
| this.tabIndex = (parsedTabIndex || parsedTabIndex === 0) ? parsedTabIndex : null; | ||
| } | ||
| onFocus() { | ||
| this.focus(); | ||
| } | ||
| ngOnInit() { | ||
| this.config.toolbarPosition = this.config.toolbarPosition ? this.config.toolbarPosition : angularEditorConfig.toolbarPosition; | ||
| } | ||
| ngAfterViewInit() { | ||
| if (isDefined(this.autoFocus)) { | ||
| this.focus(); | ||
| } | ||
| } | ||
| onPaste(event) { | ||
| if (this.config.rawPaste) { | ||
| event.preventDefault(); | ||
| const text = event.clipboardData.getData('text/plain'); | ||
| document.execCommand('insertHTML', false, text); | ||
| return text; | ||
| } | ||
| } | ||
| /** | ||
| * Executed command from editor header buttons | ||
| * @param command string from triggerCommand | ||
| * @param value | ||
| */ | ||
| executeCommand(command, value) { | ||
| this.focus(); | ||
| if (command === 'focus') { | ||
| return; | ||
| } | ||
| if (command === 'toggleEditorMode') { | ||
| this.toggleEditorMode(this.modeVisual); | ||
| } | ||
| else if (command !== '') { | ||
| if (command === 'clear') { | ||
| this.editorService.removeSelectedElements(this.getCustomTags()); | ||
| this.onContentChange(this.textArea.nativeElement); | ||
| } | ||
| else if (command === 'default') { | ||
| this.editorService.removeSelectedElements('h1,h2,h3,h4,h5,h6,p,pre'); | ||
| this.onContentChange(this.textArea.nativeElement); | ||
| } | ||
| else { | ||
| this.editorService.executeCommand(command, value); | ||
| } | ||
| this.exec(); | ||
| } | ||
| } | ||
| /** | ||
| * focus event | ||
| */ | ||
| onTextAreaFocus(event) { | ||
| if (this.focused) { | ||
| event.stopPropagation(); | ||
| return; | ||
| } | ||
| this.focused = true; | ||
| this.focusEvent.emit(event); | ||
| if (!this.touched || !this.changed) { | ||
| this.editorService.executeInNextQueueIteration(() => { | ||
| this.configure(); | ||
| this.touched = true; | ||
| }); | ||
| } | ||
| } | ||
| /** | ||
| * @description fires when cursor leaves textarea | ||
| */ | ||
| onTextAreaMouseOut(event) { | ||
| this.editorService.saveSelection(); | ||
| } | ||
| /** | ||
| * blur event | ||
| */ | ||
| onTextAreaBlur(event) { | ||
| /** | ||
| * save selection if focussed out | ||
| */ | ||
| this.editorService.executeInNextQueueIteration(this.editorService.saveSelection); | ||
| if (typeof this.onTouched === 'function') { | ||
| this.onTouched(); | ||
| } | ||
| if (event.relatedTarget !== null) { | ||
| const parent = event.relatedTarget.parentElement; | ||
| if (!parent.classList.contains('angular-editor-toolbar-set') && !parent.classList.contains('ae-picker')) { | ||
| this.blurEvent.emit(event); | ||
| this.focused = false; | ||
| } | ||
| } | ||
| } | ||
| /** | ||
| * focus the text area when the editor is focused | ||
| */ | ||
| focus() { | ||
| if (this.modeVisual) { | ||
| this.textArea.nativeElement.focus(); | ||
| } | ||
| else { | ||
| const sourceText = this.doc.getElementById('sourceText' + this.id); | ||
| sourceText.focus(); | ||
| this.focused = true; | ||
| } | ||
| } | ||
| /** | ||
| * Executed from the contenteditable section while the input property changes | ||
| * @param element html element from contenteditable | ||
| */ | ||
| onContentChange(element) { | ||
| let html = ''; | ||
| if (this.modeVisual) { | ||
| html = element.innerHTML; | ||
| } | ||
| else { | ||
| html = element.innerText; | ||
| } | ||
| if ((!html || html === '<br>')) { | ||
| html = ''; | ||
| } | ||
| if (typeof this.onChange === 'function') { | ||
| this.onChange(this.config.sanitize || this.config.sanitize === undefined ? | ||
| this.sanitizer.sanitize(SecurityContext.HTML, html) : html); | ||
| if ((!html) !== this.showPlaceholder) { | ||
| this.togglePlaceholder(this.showPlaceholder); | ||
| } | ||
| } | ||
| this.changed = true; | ||
| } | ||
| /** | ||
| * Set the function to be called | ||
| * when the control receives a change event. | ||
| * | ||
| * @param fn a function | ||
| */ | ||
| registerOnChange(fn) { | ||
| this.onChange = e => (e === '<br>' ? fn('') : fn(e)); | ||
| } | ||
| /** | ||
| * Set the function to be called | ||
| * when the control receives a touch event. | ||
| * | ||
| * @param fn a function | ||
| */ | ||
| registerOnTouched(fn) { | ||
| this.onTouched = fn; | ||
| } | ||
| /** | ||
| * Write a new value to the element. | ||
| * | ||
| * @param value value to be executed when there is a change in contenteditable | ||
| */ | ||
| writeValue(value) { | ||
| if ((!value || value === '<br>' || value === '') !== this.showPlaceholder) { | ||
| this.togglePlaceholder(this.showPlaceholder); | ||
| } | ||
| if (value === undefined || value === '' || value === '<br>') { | ||
| value = null; | ||
| } | ||
| this.refreshView(value); | ||
| } | ||
| /** | ||
| * refresh view/HTML of the editor | ||
| * | ||
| * @param value html string from the editor | ||
| */ | ||
| refreshView(value) { | ||
| const normalizedValue = value === null ? '' : value; | ||
| this.r.setProperty(this.textArea.nativeElement, 'innerHTML', normalizedValue); | ||
| return; | ||
| } | ||
| /** | ||
| * toggles placeholder based on input string | ||
| * | ||
| * @param value A HTML string from the editor | ||
| */ | ||
| togglePlaceholder(value) { | ||
| if (!value) { | ||
| this.r.addClass(this.editorWrapper.nativeElement, 'show-placeholder'); | ||
| this.showPlaceholder = true; | ||
| } | ||
| else { | ||
| this.r.removeClass(this.editorWrapper.nativeElement, 'show-placeholder'); | ||
| this.showPlaceholder = false; | ||
| } | ||
| } | ||
| /** | ||
| * Implements disabled state for this element | ||
| * | ||
| * @param isDisabled Disabled flag | ||
| */ | ||
| setDisabledState(isDisabled) { | ||
| const div = this.textArea.nativeElement; | ||
| const action = isDisabled ? 'addClass' : 'removeClass'; | ||
| this.r[action](div, 'disabled'); | ||
| this.disabled = isDisabled; | ||
| } | ||
| /** | ||
| * toggles editor mode based on bToSource bool | ||
| * | ||
| * @param bToSource A boolean value from the editor | ||
| */ | ||
| toggleEditorMode(bToSource) { | ||
| let oContent; | ||
| const editableElement = this.textArea.nativeElement; | ||
| if (bToSource) { | ||
| oContent = this.r.createText(editableElement.innerHTML); | ||
| this.r.setProperty(editableElement, 'innerHTML', ''); | ||
| this.r.setProperty(editableElement, 'contentEditable', false); | ||
| const oPre = this.r.createElement('pre'); | ||
| this.r.setStyle(oPre, 'margin', '0'); | ||
| this.r.setStyle(oPre, 'outline', 'none'); | ||
| const oCode = this.r.createElement('code'); | ||
| this.r.setProperty(oCode, 'id', 'sourceText' + this.id); | ||
| this.r.setStyle(oCode, 'display', 'block'); | ||
| this.r.setStyle(oCode, 'white-space', 'pre-wrap'); | ||
| this.r.setStyle(oCode, 'word-break', 'keep-all'); | ||
| this.r.setStyle(oCode, 'outline', 'none'); | ||
| this.r.setStyle(oCode, 'margin', '0'); | ||
| this.r.setStyle(oCode, 'background-color', '#fff5b9'); | ||
| this.r.setProperty(oCode, 'contentEditable', true); | ||
| this.r.appendChild(oCode, oContent); | ||
| this.focusInstance = this.r.listen(oCode, 'focus', (event) => this.onTextAreaFocus(event)); | ||
| this.blurInstance = this.r.listen(oCode, 'blur', (event) => this.onTextAreaBlur(event)); | ||
| this.r.appendChild(oPre, oCode); | ||
| this.r.appendChild(editableElement, oPre); | ||
| // ToDo move to service | ||
| this.doc.execCommand('defaultParagraphSeparator', false, 'div'); | ||
| this.modeVisual = false; | ||
| this.viewMode.emit(false); | ||
| oCode.focus(); | ||
| } | ||
| else { | ||
| if (this.doc.querySelectorAll) { | ||
| this.r.setProperty(editableElement, 'innerHTML', editableElement.innerText); | ||
| } | ||
| else { | ||
| oContent = this.doc.createRange(); | ||
| oContent.selectNodeContents(editableElement.firstChild); | ||
| this.r.setProperty(editableElement, 'innerHTML', oContent.toString()); | ||
| } | ||
| this.r.setProperty(editableElement, 'contentEditable', true); | ||
| this.modeVisual = true; | ||
| this.viewMode.emit(true); | ||
| this.onContentChange(editableElement); | ||
| editableElement.focus(); | ||
| } | ||
| this.editorToolbar.setEditorMode(!this.modeVisual); | ||
| } | ||
| /** | ||
| * toggles editor buttons when cursor moved or positioning | ||
| * | ||
| * Send a node array from the contentEditable of the editor | ||
| */ | ||
| exec() { | ||
| this.editorToolbar.triggerButtons(); | ||
| let userSelection; | ||
| if (this.doc.getSelection) { | ||
| userSelection = this.doc.getSelection(); | ||
| this.editorService.executeInNextQueueIteration(this.editorService.saveSelection); | ||
| } | ||
| let a = userSelection.focusNode; | ||
| const els = []; | ||
| while (a && a.id !== 'editor') { | ||
| els.unshift(a); | ||
| a = a.parentNode; | ||
| } | ||
| this.editorToolbar.triggerBlocks(els); | ||
| } | ||
| configure() { | ||
| this.editorService.uploadUrl = this.config.uploadUrl; | ||
| this.editorService.uploadWithCredentials = this.config.uploadWithCredentials; | ||
| if (this.config.defaultParagraphSeparator) { | ||
| this.editorService.setDefaultParagraphSeparator(this.config.defaultParagraphSeparator); | ||
| } | ||
| if (this.config.defaultFontName) { | ||
| this.editorService.setFontName(this.config.defaultFontName); | ||
| } | ||
| if (this.config.defaultFontSize) { | ||
| this.editorService.setFontSize(this.config.defaultFontSize); | ||
| } | ||
| } | ||
| getFonts() { | ||
| const fonts = this.config.fonts ? this.config.fonts : angularEditorConfig.fonts; | ||
| return fonts.map(x => { | ||
| return { label: x.name, value: x.name }; | ||
| }); | ||
| } | ||
| getCustomTags() { | ||
| const tags = ['span']; | ||
| this.config.customClasses.forEach(x => { | ||
| if (x.tag !== undefined) { | ||
| if (!tags.includes(x.tag)) { | ||
| tags.push(x.tag); | ||
| } | ||
| } | ||
| }); | ||
| return tags.join(','); | ||
| } | ||
| ngOnDestroy() { | ||
| if (this.blurInstance) { | ||
| this.blurInstance(); | ||
| } | ||
| if (this.focusInstance) { | ||
| this.focusInstance(); | ||
| } | ||
| } | ||
| filterStyles(html) { | ||
| html = html.replace('position: fixed;', ''); | ||
| return html; | ||
| } | ||
| } | ||
| AngularEditorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AngularEditorComponent, deps: [{ token: i0.Renderer2 }, { token: i1.AngularEditorService }, { token: DOCUMENT }, { token: i2.DomSanitizer }, { token: i0.ChangeDetectorRef }, { token: 'tabindex', attribute: true }, { token: 'autofocus', attribute: true }], target: i0.ɵɵFactoryTarget.Component }); | ||
| AngularEditorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AngularEditorComponent, selector: "angular-editor", inputs: { id: "id", config: "config", placeholder: "placeholder", tabIndex: "tabIndex" }, outputs: { html: "html", viewMode: "viewMode", blurEvent: "blur", focusEvent: "focus" }, host: { listeners: { "focus": "onFocus()" }, properties: { "attr.tabindex": "this.tabindex" } }, providers: [ | ||
| { | ||
| provide: NG_VALUE_ACCESSOR, | ||
| useExisting: forwardRef(() => AngularEditorComponent), | ||
| multi: true | ||
| }, | ||
| AngularEditorService | ||
| ], queries: [{ propertyName: "customButtonsTemplateRef", first: true, predicate: ["customButtons"], descendants: true }], viewQueries: [{ propertyName: "textArea", first: true, predicate: ["editor"], descendants: true, static: true }, { propertyName: "editorWrapper", first: true, predicate: ["editorWrapper"], descendants: true, static: true }, { propertyName: "editorToolbar", first: true, predicate: ["editorToolbar"], descendants: true }], ngImport: i0, template: "<div\n class=\"angular-editor\"\n #angularEditor\n [style.width]=\"config.width\"\n [style.minWidth]=\"config.minWidth\"\n [ngClass]=\"{\n 'bottom': config.toolbarPosition === 'bottom'\n }\"\n>\n <angular-editor-toolbar\n #editorToolbar\n [id]=\"id\"\n [uploadUrl]=\"config.uploadUrl\"\n [upload]=\"config.upload\"\n [showToolbar]=\"config.showToolbar !== undefined ? config.showToolbar : true\"\n [fonts]=\"getFonts()\"\n [customClasses]=\"config.customClasses\"\n [defaultFontName]=\"config.defaultFontName\"\n [defaultFontSize]=\"config.defaultFontSize\"\n [hiddenButtons]=\"config.toolbarHiddenButtons\"\n (execute)=\"executeCommand($event)\"\n >\n <ng-container\n [ngTemplateOutlet]=\"customButtonsTemplateRef\"\n [ngTemplateOutletContext]=\"{ executeCommandFn: this.executeCommandFn}\"\n >\n </ng-container>\n </angular-editor-toolbar>\n\n <div\n class=\"angular-editor-wrapper\"\n #editorWrapper\n >\n <div\n #editor\n class=\"angular-editor-textarea\"\n [attr.contenteditable]=\"config.editable\"\n [attr.tabindex]=\"disabled ? -1 : tabIndex\"\n [attr.translate]=\"config.translate\"\n [attr.spellcheck]=\"config.spellcheck\"\n [style.height]=\"config.height\"\n [style.minHeight]=\"config.minHeight\"\n [style.maxHeight]=\"config.maxHeight\"\n [style.outline]=\"config.outline === false ? 'none': undefined\"\n (input)=\"onContentChange($event.target)\"\n (focus)=\"onTextAreaFocus($event)\"\n (blur)=\"onTextAreaBlur($event)\"\n (click)=\"exec()\"\n (keyup)=\"exec()\"\n (mouseout)=\"onTextAreaMouseOut($event)\"\n (paste)=\"onPaste($event)\"\n >\n </div>\n <span class=\"angular-editor-placeholder\">{{ placeholder || config['placeholder'] }}</span>\n </div>\n</div>\n", styles: ["a{cursor:pointer}.angular-editor{display:flex;flex-direction:column;gap:var(--ae-gap, 5px)}.angular-editor.bottom{flex-direction:column-reverse}.angular-editor ::ng-deep [contenteditable=true]:empty:before{content:attr(placeholder);color:#868e96;opacity:1}.angular-editor .angular-editor-wrapper{position:relative}.angular-editor .angular-editor-wrapper .angular-editor-textarea{min-height:150px;overflow:auto;resize:vertical}.angular-editor .angular-editor-wrapper .angular-editor-textarea:after{content:\"\";position:absolute;bottom:0;right:0;display:block;width:8px;height:8px;cursor:nwse-resize;background-color:#ffffff80}.angular-editor .angular-editor-wrapper .angular-editor-textarea{min-height:5rem;padding:.5rem .8rem 1rem;border-radius:var(--ae-text-area-border-radius, .3rem);border:var(--ae-text-area-border, 1px solid #ddd);background-color:transparent;overflow-x:hidden;overflow-y:auto;position:relative}.angular-editor .angular-editor-wrapper .angular-editor-textarea:focus,.angular-editor .angular-editor-wrapper .angular-editor-textarea.focus{outline:var(--ae-focus-outline-color, -webkit-focus-ring-color auto 1px)}.angular-editor .angular-editor-wrapper .angular-editor-textarea ::ng-deep blockquote{margin-left:1rem;border-left:.2em solid #dfe2e5;padding-left:.5rem}.angular-editor .angular-editor-wrapper ::ng-deep p{margin-bottom:0}.angular-editor .angular-editor-wrapper .angular-editor-placeholder{display:none;position:absolute;top:0;padding:.6rem .8rem 1rem .9rem;color:#6c757d;opacity:.75}.angular-editor .angular-editor-wrapper.show-placeholder .angular-editor-placeholder{display:block}.angular-editor .angular-editor-wrapper.disabled{cursor:not-allowed;opacity:.5;pointer-events:none}\n"], components: [{ type: i3.AeToolbarComponent, selector: "angular-editor-toolbar, ae-toolbar, div[aeToolbar]", inputs: ["id", "uploadUrl", "upload", "showToolbar", "fonts", "customClasses", "defaultFontName", "defaultFontSize", "hiddenButtons"], outputs: ["execute"] }], directives: [{ type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }] }); | ||
| i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AngularEditorComponent, decorators: [{ | ||
| type: Component, | ||
| args: [{ selector: 'angular-editor', providers: [ | ||
| { | ||
| provide: NG_VALUE_ACCESSOR, | ||
| useExisting: forwardRef(() => AngularEditorComponent), | ||
| multi: true | ||
| }, | ||
| AngularEditorService | ||
| ], template: "<div\n class=\"angular-editor\"\n #angularEditor\n [style.width]=\"config.width\"\n [style.minWidth]=\"config.minWidth\"\n [ngClass]=\"{\n 'bottom': config.toolbarPosition === 'bottom'\n }\"\n>\n <angular-editor-toolbar\n #editorToolbar\n [id]=\"id\"\n [uploadUrl]=\"config.uploadUrl\"\n [upload]=\"config.upload\"\n [showToolbar]=\"config.showToolbar !== undefined ? config.showToolbar : true\"\n [fonts]=\"getFonts()\"\n [customClasses]=\"config.customClasses\"\n [defaultFontName]=\"config.defaultFontName\"\n [defaultFontSize]=\"config.defaultFontSize\"\n [hiddenButtons]=\"config.toolbarHiddenButtons\"\n (execute)=\"executeCommand($event)\"\n >\n <ng-container\n [ngTemplateOutlet]=\"customButtonsTemplateRef\"\n [ngTemplateOutletContext]=\"{ executeCommandFn: this.executeCommandFn}\"\n >\n </ng-container>\n </angular-editor-toolbar>\n\n <div\n class=\"angular-editor-wrapper\"\n #editorWrapper\n >\n <div\n #editor\n class=\"angular-editor-textarea\"\n [attr.contenteditable]=\"config.editable\"\n [attr.tabindex]=\"disabled ? -1 : tabIndex\"\n [attr.translate]=\"config.translate\"\n [attr.spellcheck]=\"config.spellcheck\"\n [style.height]=\"config.height\"\n [style.minHeight]=\"config.minHeight\"\n [style.maxHeight]=\"config.maxHeight\"\n [style.outline]=\"config.outline === false ? 'none': undefined\"\n (input)=\"onContentChange($event.target)\"\n (focus)=\"onTextAreaFocus($event)\"\n (blur)=\"onTextAreaBlur($event)\"\n (click)=\"exec()\"\n (keyup)=\"exec()\"\n (mouseout)=\"onTextAreaMouseOut($event)\"\n (paste)=\"onPaste($event)\"\n >\n </div>\n <span class=\"angular-editor-placeholder\">{{ placeholder || config['placeholder'] }}</span>\n </div>\n</div>\n", styles: ["a{cursor:pointer}.angular-editor{display:flex;flex-direction:column;gap:var(--ae-gap, 5px)}.angular-editor.bottom{flex-direction:column-reverse}.angular-editor ::ng-deep [contenteditable=true]:empty:before{content:attr(placeholder);color:#868e96;opacity:1}.angular-editor .angular-editor-wrapper{position:relative}.angular-editor .angular-editor-wrapper .angular-editor-textarea{min-height:150px;overflow:auto;resize:vertical}.angular-editor .angular-editor-wrapper .angular-editor-textarea:after{content:\"\";position:absolute;bottom:0;right:0;display:block;width:8px;height:8px;cursor:nwse-resize;background-color:#ffffff80}.angular-editor .angular-editor-wrapper .angular-editor-textarea{min-height:5rem;padding:.5rem .8rem 1rem;border-radius:var(--ae-text-area-border-radius, .3rem);border:var(--ae-text-area-border, 1px solid #ddd);background-color:transparent;overflow-x:hidden;overflow-y:auto;position:relative}.angular-editor .angular-editor-wrapper .angular-editor-textarea:focus,.angular-editor .angular-editor-wrapper .angular-editor-textarea.focus{outline:var(--ae-focus-outline-color, -webkit-focus-ring-color auto 1px)}.angular-editor .angular-editor-wrapper .angular-editor-textarea ::ng-deep blockquote{margin-left:1rem;border-left:.2em solid #dfe2e5;padding-left:.5rem}.angular-editor .angular-editor-wrapper ::ng-deep p{margin-bottom:0}.angular-editor .angular-editor-wrapper .angular-editor-placeholder{display:none;position:absolute;top:0;padding:.6rem .8rem 1rem .9rem;color:#6c757d;opacity:.75}.angular-editor .angular-editor-wrapper.show-placeholder .angular-editor-placeholder{display:block}.angular-editor .angular-editor-wrapper.disabled{cursor:not-allowed;opacity:.5;pointer-events:none}\n"] }] | ||
| }], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: i1.AngularEditorService }, { type: undefined, decorators: [{ | ||
| type: Inject, | ||
| args: [DOCUMENT] | ||
| }] }, { type: i2.DomSanitizer }, { type: i0.ChangeDetectorRef }, { type: undefined, decorators: [{ | ||
| type: Attribute, | ||
| args: ['tabindex'] | ||
| }] }, { type: undefined, decorators: [{ | ||
| type: Attribute, | ||
| args: ['autofocus'] | ||
| }] }]; }, propDecorators: { id: [{ | ||
| type: Input | ||
| }], config: [{ | ||
| type: Input | ||
| }], placeholder: [{ | ||
| type: Input | ||
| }], tabIndex: [{ | ||
| type: Input | ||
| }], html: [{ | ||
| type: Output | ||
| }], textArea: [{ | ||
| type: ViewChild, | ||
| args: ['editor', { static: true }] | ||
| }], editorWrapper: [{ | ||
| type: ViewChild, | ||
| args: ['editorWrapper', { static: true }] | ||
| }], editorToolbar: [{ | ||
| type: ViewChild, | ||
| args: ['editorToolbar'] | ||
| }], customButtonsTemplateRef: [{ | ||
| type: ContentChild, | ||
| args: ["customButtons"] | ||
| }], viewMode: [{ | ||
| type: Output | ||
| }], blurEvent: [{ | ||
| type: Output, | ||
| args: ['blur'] | ||
| }], focusEvent: [{ | ||
| type: Output, | ||
| args: ['focus'] | ||
| }], tabindex: [{ | ||
| type: HostBinding, | ||
| args: ['attr.tabindex'] | ||
| }], onFocus: [{ | ||
| type: HostListener, | ||
| args: ['focus'] | ||
| }] } }); | ||
| //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"angular-editor.component.js","sourceRoot":"","sources":["../../../../../projects/angular-editor/src/lib/editor/angular-editor.component.ts","../../../../../projects/angular-editor/src/lib/editor/angular-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {DOCUMENT} from '@angular/common';\nimport {\n  AfterViewInit,\n  Attribute,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  HostBinding,\n  HostListener,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  Renderer2,\n  SecurityContext,\n  TemplateRef,\n  ViewChild, ViewEncapsulation\n} from '@angular/core';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\nimport {DomSanitizer} from '@angular/platform-browser';\nimport {AeToolbarComponent} from '../ae-toolbar/ae-toolbar.component';\nimport {AngularEditorService} from '../angular-editor.service';\nimport {AngularEditorConfig, angularEditorConfig} from '../config';\nimport {isDefined} from '../utils';\n\n@Component({\n  selector: 'angular-editor',\n  templateUrl: './angular-editor.component.html',\n  styleUrls: ['./angular-editor.component.scss'],\n  //encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => AngularEditorComponent),\n      multi: true\n    },\n    AngularEditorService\n  ]\n})\nexport class AngularEditorComponent implements OnInit, ControlValueAccessor, AfterViewInit, OnDestroy {\n\n  private onChange: (value: string) => void;\n  private onTouched: () => void;\n\n  modeVisual = true;\n  showPlaceholder = false;\n  disabled = false;\n  focused = false;\n  touched = false;\n  changed = false;\n\n  focusInstance: any;\n  blurInstance: any;\n\n  @Input() id = '';\n  @Input() config: AngularEditorConfig = angularEditorConfig;\n  @Input() placeholder = '';\n  @Input() tabIndex: number | null;\n\n  @Output() html;\n\n  @ViewChild('editor', {static: true}) textArea: ElementRef;\n  @ViewChild('editorWrapper', {static: true}) editorWrapper: ElementRef;\n  @ViewChild('editorToolbar') editorToolbar: AeToolbarComponent;\n  @ContentChild(\"customButtons\") customButtonsTemplateRef?: TemplateRef<any>;\n  executeCommandFn = this.executeCommand.bind(this);\n\n  @Output() viewMode = new EventEmitter<boolean>();\n\n  /** emits `blur` event when focused out from the textarea */\n    // eslint-disable-next-line @angular-eslint/no-output-native, @angular-eslint/no-output-rename\n  @Output('blur') blurEvent: EventEmitter<FocusEvent> = new EventEmitter<FocusEvent>();\n\n  /** emits `focus` event when focused in to the textarea */\n    // eslint-disable-next-line @angular-eslint/no-output-rename, @angular-eslint/no-output-native\n  @Output('focus') focusEvent: EventEmitter<FocusEvent> = new EventEmitter<FocusEvent>();\n\n  @HostBinding('attr.tabindex') tabindex = -1;\n\n  @HostListener('focus')\n  onFocus() {\n    this.focus();\n  }\n\n  constructor(\n    private r: Renderer2,\n    private editorService: AngularEditorService,\n    @Inject(DOCUMENT) private doc: any,\n    private sanitizer: DomSanitizer,\n    private cdRef: ChangeDetectorRef,\n    @Attribute('tabindex') defaultTabIndex: string,\n    @Attribute('autofocus') private autoFocus: any\n  ) {\n    const parsedTabIndex = Number(defaultTabIndex);\n    this.tabIndex = (parsedTabIndex || parsedTabIndex === 0) ? parsedTabIndex : null;\n  }\n\n  ngOnInit() {\n    this.config.toolbarPosition = this.config.toolbarPosition ? this.config.toolbarPosition : angularEditorConfig.toolbarPosition;\n  }\n\n  ngAfterViewInit() {\n    if (isDefined(this.autoFocus)) {\n      this.focus();\n    }\n  }\n\n  onPaste(event: ClipboardEvent) {\n    if (this.config.rawPaste) {\n      event.preventDefault();\n      const text = event.clipboardData.getData('text/plain');\n      document.execCommand('insertHTML', false, text);\n      return text;\n    }\n  }\n\n  /**\n   * Executed command from editor header buttons\n   * @param command string from triggerCommand\n   * @param value\n   */\n  executeCommand(command: string, value?: string) {\n    this.focus();\n    if (command === 'focus') {\n      return;\n    }\n    if (command === 'toggleEditorMode') {\n      this.toggleEditorMode(this.modeVisual);\n    } else if (command !== '') {\n      if (command === 'clear') {\n        this.editorService.removeSelectedElements(this.getCustomTags());\n        this.onContentChange(this.textArea.nativeElement);\n      } else if (command === 'default') {\n        this.editorService.removeSelectedElements('h1,h2,h3,h4,h5,h6,p,pre');\n        this.onContentChange(this.textArea.nativeElement);\n      } else {\n        this.editorService.executeCommand(command, value);\n      }\n      this.exec();\n    }\n  }\n\n  /**\n   * focus event\n   */\n  onTextAreaFocus(event: FocusEvent): void {\n    if (this.focused) {\n      event.stopPropagation();\n      return;\n    }\n    this.focused = true;\n    this.focusEvent.emit(event);\n    if (!this.touched || !this.changed) {\n      this.editorService.executeInNextQueueIteration(() => {\n        this.configure();\n        this.touched = true;\n      });\n    }\n  }\n\n  /**\n   * @description fires when cursor leaves textarea\n   */\n  public onTextAreaMouseOut(event: MouseEvent): void {\n    this.editorService.saveSelection();\n  }\n\n  /**\n   * blur event\n   */\n  onTextAreaBlur(event: FocusEvent) {\n    /**\n     * save selection if focussed out\n     */\n    this.editorService.executeInNextQueueIteration(this.editorService.saveSelection);\n\n    if (typeof this.onTouched === 'function') {\n      this.onTouched();\n    }\n\n    if (event.relatedTarget !== null) {\n      const parent = (event.relatedTarget as HTMLElement).parentElement;\n      if (!parent.classList.contains('angular-editor-toolbar-set') && !parent.classList.contains('ae-picker')) {\n        this.blurEvent.emit(event);\n        this.focused = false;\n      }\n    }\n  }\n\n  /**\n   *  focus the text area when the editor is focused\n   */\n  focus() {\n    if (this.modeVisual) {\n      this.textArea.nativeElement.focus();\n    } else {\n      const sourceText = this.doc.getElementById('sourceText' + this.id);\n      sourceText.focus();\n      this.focused = true;\n    }\n  }\n\n  /**\n   * Executed from the contenteditable section while the input property changes\n   * @param element html element from contenteditable\n   */\n  onContentChange(element: HTMLElement): void {\n    let html = '';\n    if (this.modeVisual) {\n      html = element.innerHTML;\n    } else {\n      html = element.innerText;\n    }\n    if ((!html || html === '<br>')) {\n      html = '';\n    }\n    if (typeof this.onChange === 'function') {\n      this.onChange(this.config.sanitize || this.config.sanitize === undefined ?\n        this.sanitizer.sanitize(SecurityContext.HTML, html) : html);\n      if ((!html) !== this.showPlaceholder) {\n        this.togglePlaceholder(this.showPlaceholder);\n      }\n    }\n    this.changed = true;\n  }\n\n  /**\n   * Set the function to be called\n   * when the control receives a change event.\n   *\n   * @param fn a function\n   */\n  registerOnChange(fn: any): void {\n    this.onChange = e => (e === '<br>' ? fn('') : fn(e));\n  }\n\n  /**\n   * Set the function to be called\n   * when the control receives a touch event.\n   *\n   * @param fn a function\n   */\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  /**\n   * Write a new value to the element.\n   *\n   * @param value value to be executed when there is a change in contenteditable\n   */\n  writeValue(value: any): void {\n\n    if ((!value || value === '<br>' || value === '') !== this.showPlaceholder) {\n      this.togglePlaceholder(this.showPlaceholder);\n    }\n\n    if (value === undefined || value === '' || value === '<br>') {\n      value = null;\n    }\n\n    this.refreshView(value);\n  }\n\n  /**\n   * refresh view/HTML of the editor\n   *\n   * @param value html string from the editor\n   */\n  refreshView(value: string): void {\n    const normalizedValue = value === null ? '' : value;\n    this.r.setProperty(this.textArea.nativeElement, 'innerHTML', normalizedValue);\n\n    return;\n  }\n\n  /**\n   * toggles placeholder based on input string\n   *\n   * @param value A HTML string from the editor\n   */\n  togglePlaceholder(value: boolean): void {\n    if (!value) {\n      this.r.addClass(this.editorWrapper.nativeElement, 'show-placeholder');\n      this.showPlaceholder = true;\n\n    } else {\n      this.r.removeClass(this.editorWrapper.nativeElement, 'show-placeholder');\n      this.showPlaceholder = false;\n    }\n  }\n\n  /**\n   * Implements disabled state for this element\n   *\n   * @param isDisabled Disabled flag\n   */\n  setDisabledState(isDisabled: boolean): void {\n    const div = this.textArea.nativeElement;\n    const action = isDisabled ? 'addClass' : 'removeClass';\n    this.r[action](div, 'disabled');\n    this.disabled = isDisabled;\n  }\n\n  /**\n   * toggles editor mode based on bToSource bool\n   *\n   * @param bToSource A boolean value from the editor\n   */\n  toggleEditorMode(bToSource: boolean) {\n    let oContent: any;\n    const editableElement = this.textArea.nativeElement;\n\n    if (bToSource) {\n      oContent = this.r.createText(editableElement.innerHTML);\n      this.r.setProperty(editableElement, 'innerHTML', '');\n      this.r.setProperty(editableElement, 'contentEditable', false);\n\n      const oPre = this.r.createElement('pre');\n      this.r.setStyle(oPre, 'margin', '0');\n      this.r.setStyle(oPre, 'outline', 'none');\n\n      const oCode = this.r.createElement('code');\n      this.r.setProperty(oCode, 'id', 'sourceText' + this.id);\n      this.r.setStyle(oCode, 'display', 'block');\n      this.r.setStyle(oCode, 'white-space', 'pre-wrap');\n      this.r.setStyle(oCode, 'word-break', 'keep-all');\n      this.r.setStyle(oCode, 'outline', 'none');\n      this.r.setStyle(oCode, 'margin', '0');\n      this.r.setStyle(oCode, 'background-color', '#fff5b9');\n      this.r.setProperty(oCode, 'contentEditable', true);\n      this.r.appendChild(oCode, oContent);\n      this.focusInstance = this.r.listen(oCode, 'focus', (event) => this.onTextAreaFocus(event));\n      this.blurInstance = this.r.listen(oCode, 'blur', (event) => this.onTextAreaBlur(event));\n      this.r.appendChild(oPre, oCode);\n      this.r.appendChild(editableElement, oPre);\n\n      // ToDo move to service\n      this.doc.execCommand('defaultParagraphSeparator', false, 'div');\n\n      this.modeVisual = false;\n      this.viewMode.emit(false);\n      oCode.focus();\n    } else {\n      if (this.doc.querySelectorAll) {\n        this.r.setProperty(editableElement, 'innerHTML', editableElement.innerText);\n      } else {\n        oContent = this.doc.createRange();\n        oContent.selectNodeContents(editableElement.firstChild);\n        this.r.setProperty(editableElement, 'innerHTML', oContent.toString());\n      }\n      this.r.setProperty(editableElement, 'contentEditable', true);\n      this.modeVisual = true;\n      this.viewMode.emit(true);\n      this.onContentChange(editableElement);\n      editableElement.focus();\n    }\n    this.editorToolbar.setEditorMode(!this.modeVisual);\n  }\n\n  /**\n   * toggles editor buttons when cursor moved or positioning\n   *\n   * Send a node array from the contentEditable of the editor\n   */\n  exec() {\n    this.editorToolbar.triggerButtons();\n\n    let userSelection;\n    if (this.doc.getSelection) {\n      userSelection = this.doc.getSelection();\n      this.editorService.executeInNextQueueIteration(this.editorService.saveSelection);\n    }\n\n    let a = userSelection.focusNode;\n    const els = [];\n    while (a && a.id !== 'editor') {\n      els.unshift(a);\n      a = a.parentNode;\n    }\n    this.editorToolbar.triggerBlocks(els);\n  }\n\n  private configure() {\n    this.editorService.uploadUrl = this.config.uploadUrl;\n    this.editorService.uploadWithCredentials = this.config.uploadWithCredentials;\n    if (this.config.defaultParagraphSeparator) {\n      this.editorService.setDefaultParagraphSeparator(this.config.defaultParagraphSeparator);\n    }\n    if (this.config.defaultFontName) {\n      this.editorService.setFontName(this.config.defaultFontName);\n    }\n    if (this.config.defaultFontSize) {\n      this.editorService.setFontSize(this.config.defaultFontSize);\n    }\n  }\n\n  getFonts() {\n    const fonts = this.config.fonts ? this.config.fonts : angularEditorConfig.fonts;\n    return fonts.map(x => {\n      return {label: x.name, value: x.name};\n    });\n  }\n\n  getCustomTags() {\n    const tags = ['span'];\n    this.config.customClasses.forEach(x => {\n      if (x.tag !== undefined) {\n        if (!tags.includes(x.tag)) {\n          tags.push(x.tag);\n        }\n      }\n    });\n    return tags.join(',');\n  }\n\n  ngOnDestroy() {\n    if (this.blurInstance) {\n      this.blurInstance();\n    }\n    if (this.focusInstance) {\n      this.focusInstance();\n    }\n  }\n\n  filterStyles(html: string): string {\n    html = html.replace('position: fixed;', '');\n    return html;\n  }\n}\n","<div\n  class=\"angular-editor\"\n  #angularEditor\n  [style.width]=\"config.width\"\n  [style.minWidth]=\"config.minWidth\"\n  [ngClass]=\"{\n     'bottom': config.toolbarPosition === 'bottom'\n     }\"\n>\n  <angular-editor-toolbar\n    #editorToolbar\n    [id]=\"id\"\n    [uploadUrl]=\"config.uploadUrl\"\n    [upload]=\"config.upload\"\n    [showToolbar]=\"config.showToolbar !== undefined ? config.showToolbar : true\"\n    [fonts]=\"getFonts()\"\n    [customClasses]=\"config.customClasses\"\n    [defaultFontName]=\"config.defaultFontName\"\n    [defaultFontSize]=\"config.defaultFontSize\"\n    [hiddenButtons]=\"config.toolbarHiddenButtons\"\n    (execute)=\"executeCommand($event)\"\n  >\n    <ng-container\n      [ngTemplateOutlet]=\"customButtonsTemplateRef\"\n      [ngTemplateOutletContext]=\"{ executeCommandFn: this.executeCommandFn}\"\n    >\n    </ng-container>\n  </angular-editor-toolbar>\n\n  <div\n    class=\"angular-editor-wrapper\"\n    #editorWrapper\n  >\n    <div\n      #editor\n      class=\"angular-editor-textarea\"\n      [attr.contenteditable]=\"config.editable\"\n      [attr.tabindex]=\"disabled ? -1 : tabIndex\"\n      [attr.translate]=\"config.translate\"\n      [attr.spellcheck]=\"config.spellcheck\"\n      [style.height]=\"config.height\"\n      [style.minHeight]=\"config.minHeight\"\n      [style.maxHeight]=\"config.maxHeight\"\n      [style.outline]=\"config.outline === false ? 'none': undefined\"\n      (input)=\"onContentChange($event.target)\"\n      (focus)=\"onTextAreaFocus($event)\"\n      (blur)=\"onTextAreaBlur($event)\"\n      (click)=\"exec()\"\n      (keyup)=\"exec()\"\n      (mouseout)=\"onTextAreaMouseOut($event)\"\n      (paste)=\"onPaste($event)\"\n    >\n    </div>\n    <span class=\"angular-editor-placeholder\">{{ placeholder || config['placeholder'] }}</span>\n  </div>\n</div>\n"]} |
| export function isDefined(value) { | ||
| return value !== undefined && value !== null; | ||
| } | ||
| //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9hbmd1bGFyLWVkaXRvci9zcmMvbGliL3V0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sVUFBVSxTQUFTLENBQUMsS0FBVTtJQUNsQyxPQUFPLEtBQUssS0FBSyxTQUFTLElBQUksS0FBSyxLQUFLLElBQUksQ0FBQztBQUMvQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIGlzRGVmaW5lZCh2YWx1ZTogYW55KSB7XG4gIHJldHVybiB2YWx1ZSAhPT0gdW5kZWZpbmVkICYmIHZhbHVlICE9PSBudWxsO1xufVxuIl19 |
| /* | ||
| * Public API Surface of angular-editor | ||
| */ | ||
| export * from './lib/angular-editor.service'; | ||
| export * from './lib/editor/angular-editor.component'; | ||
| export * from './lib/ae-button/ae-button.component'; | ||
| export * from './lib/ae-toolbar-set/ae-toolbar-set.component'; | ||
| export * from './lib/ae-select/ae-select.component'; | ||
| export * from './lib/ae-toolbar/ae-toolbar.component'; | ||
| export * from './lib/angular-editor.module'; | ||
| //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2FuZ3VsYXItZWRpdG9yL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLHVDQUF1QyxDQUFDO0FBQ3RELGNBQWMscUNBQXFDLENBQUM7QUFDcEQsY0FBYywrQ0FBK0MsQ0FBQztBQUM5RCxjQUFjLHFDQUFxQyxDQUFDO0FBQ3BELGNBQWMsdUNBQXVDLENBQUM7QUFDdEQsY0FBYyw2QkFBNkIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2YgYW5ndWxhci1lZGl0b3JcbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2xpYi9hbmd1bGFyLWVkaXRvci5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2VkaXRvci9hbmd1bGFyLWVkaXRvci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvYWUtYnV0dG9uL2FlLWJ1dHRvbi5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvYWUtdG9vbGJhci1zZXQvYWUtdG9vbGJhci1zZXQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2FlLXNlbGVjdC9hZS1zZWxlY3QuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2FlLXRvb2xiYXIvYWUtdG9vbGJhci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvYW5ndWxhci1lZGl0b3IubW9kdWxlJztcbmV4cG9ydCB7IEFuZ3VsYXJFZGl0b3JDb25maWcsIEN1c3RvbUNsYXNzIH0gZnJvbSAnLi9saWIvY29uZmlnJztcbiJdfQ== |
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
| /** | ||
| * Generated bundle index. Do not edit. | ||
| */ | ||
| /// <amd-module name="@kolkov/angular-editor" /> | ||
| export * from './public-api'; |
| import * as i0 from "@angular/core"; | ||
| export declare class AeButtonComponent { | ||
| iconName: string; | ||
| constructor(); | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AeButtonComponent, never>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AeButtonComponent, "ae-button, button[aeButton]", never, { "iconName": "iconName"; }, {}, never, ["*"]>; | ||
| } |
| import { ElementRef, EventEmitter, OnInit, Renderer2 } from '@angular/core'; | ||
| import { ControlValueAccessor } from '@angular/forms'; | ||
| import * as i0 from "@angular/core"; | ||
| export interface SelectOption { | ||
| label: string; | ||
| value: string; | ||
| } | ||
| export declare class AeSelectComponent implements OnInit, ControlValueAccessor { | ||
| private elRef; | ||
| private r; | ||
| options: SelectOption[]; | ||
| isHidden: boolean; | ||
| selectedOption: SelectOption; | ||
| disabled: boolean; | ||
| optionId: number; | ||
| get label(): string; | ||
| opened: boolean; | ||
| get value(): string; | ||
| hidden: string; | ||
| changeEvent: EventEmitter<any>; | ||
| labelButton: ElementRef; | ||
| constructor(elRef: ElementRef, r: Renderer2); | ||
| ngOnInit(): void; | ||
| hide(): void; | ||
| optionSelect(option: SelectOption, event: MouseEvent): void; | ||
| toggleOpen(event: MouseEvent): void; | ||
| onClick($event: MouseEvent): void; | ||
| close(): void; | ||
| get isOpen(): boolean; | ||
| writeValue(value: any): void; | ||
| setValue(value: any): void; | ||
| onChange: any; | ||
| onTouched: any; | ||
| registerOnChange(fn: any): void; | ||
| registerOnTouched(fn: any): void; | ||
| setDisabledState(isDisabled: boolean): void; | ||
| handleKeyDown($event: KeyboardEvent): void; | ||
| _handleArrowDown($event: any): void; | ||
| _handleArrowUp($event: any): void; | ||
| _handleSpace($event: any): void; | ||
| _handleEnter($event: any): void; | ||
| _handleTab($event: any): void; | ||
| _handleBackspace(): void; | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AeSelectComponent, never>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AeSelectComponent, "ae-select", never, { "options": "options"; "isHidden": "hidden"; }, { "changeEvent": "change"; }, never, never>; | ||
| } |
| import * as i0 from "@angular/core"; | ||
| export declare class AeToolbarSetComponent { | ||
| constructor(); | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AeToolbarSetComponent, never>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AeToolbarSetComponent, "ae-toolbar-set, [aeToolbarSet]", never, {}, {}, never, ["*"]>; | ||
| } |
| import { ElementRef, EventEmitter, Renderer2 } from '@angular/core'; | ||
| import { AngularEditorService, UploadResponse } from '../angular-editor.service'; | ||
| import { HttpEvent, HttpResponse } from '@angular/common/http'; | ||
| import { CustomClass } from '../config'; | ||
| import { SelectOption } from '../ae-select/ae-select.component'; | ||
| import { Observable } from 'rxjs'; | ||
| import * as i0 from "@angular/core"; | ||
| export declare class AeToolbarComponent { | ||
| private r; | ||
| private editorService; | ||
| private er; | ||
| private doc; | ||
| htmlMode: boolean; | ||
| linkSelected: boolean; | ||
| block: string; | ||
| fontName: string; | ||
| fontSize: string; | ||
| foreColour: any; | ||
| backColor: any; | ||
| headings: SelectOption[]; | ||
| fontSizes: SelectOption[]; | ||
| customClassId: string; | ||
| _customClasses: CustomClass[]; | ||
| customClassList: SelectOption[]; | ||
| tagMap: { | ||
| BLOCKQUOTE: string; | ||
| A: string; | ||
| }; | ||
| select: string[]; | ||
| buttons: string[]; | ||
| id: string; | ||
| uploadUrl: string; | ||
| upload: (file: File) => Observable<HttpEvent<UploadResponse>>; | ||
| showToolbar: boolean; | ||
| fonts: SelectOption[]; | ||
| set customClasses(classes: CustomClass[]); | ||
| set defaultFontName(value: string); | ||
| set defaultFontSize(value: string); | ||
| hiddenButtons: string[][]; | ||
| execute: EventEmitter<string>; | ||
| myInputFile: ElementRef; | ||
| get isLinkButtonDisabled(): boolean; | ||
| constructor(r: Renderer2, editorService: AngularEditorService, er: ElementRef, doc: any); | ||
| /** | ||
| * Trigger command from editor header buttons | ||
| * @param command string from toolbar buttons | ||
| */ | ||
| triggerCommand(command: string): void; | ||
| /** | ||
| * highlight editor buttons when cursor moved or positioning | ||
| */ | ||
| triggerButtons(): void; | ||
| /** | ||
| * trigger highlight editor buttons when cursor moved or positioning in block | ||
| */ | ||
| triggerBlocks(nodes: Node[]): void; | ||
| /** | ||
| * insert URL link | ||
| */ | ||
| insertUrl(): void; | ||
| /** | ||
| * insert Video link | ||
| */ | ||
| insertVideo(): void; | ||
| /** insert color */ | ||
| insertColor(color: string, where: string): void; | ||
| /** | ||
| * set font Name/family | ||
| * @param foreColor string | ||
| */ | ||
| setFontName(foreColor: string): void; | ||
| /** | ||
| * set font Size | ||
| * @param fontSize string | ||
| */ | ||
| setFontSize(fontSize: string): void; | ||
| /** | ||
| * toggle editor mode (WYSIWYG or SOURCE) | ||
| * @param m boolean | ||
| */ | ||
| setEditorMode(m: boolean): void; | ||
| /** | ||
| * Upload image when file is selected. | ||
| */ | ||
| onFileChanged(event: any): void; | ||
| watchUploadImage(response: HttpResponse<{ | ||
| imageUrl: string; | ||
| }>, event: any): void; | ||
| /** | ||
| * Set custom class | ||
| */ | ||
| setCustomClass(classId: string): void; | ||
| isButtonHidden(name: string): boolean; | ||
| focus(): void; | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AeToolbarComponent, never>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AeToolbarComponent, "angular-editor-toolbar, ae-toolbar, div[aeToolbar]", never, { "id": "id"; "uploadUrl": "uploadUrl"; "upload": "upload"; "showToolbar": "showToolbar"; "fonts": "fonts"; "customClasses": "customClasses"; "defaultFontName": "defaultFontName"; "defaultFontSize": "defaultFontSize"; "hiddenButtons": "hiddenButtons"; }, { "execute": "execute"; }, never, ["*"]>; | ||
| } |
| import * as i0 from "@angular/core"; | ||
| import * as i1 from "./editor/angular-editor.component"; | ||
| import * as i2 from "./ae-toolbar/ae-toolbar.component"; | ||
| import * as i3 from "./ae-select/ae-select.component"; | ||
| import * as i4 from "./ae-button/ae-button.component"; | ||
| import * as i5 from "./ae-toolbar-set/ae-toolbar-set.component"; | ||
| import * as i6 from "@angular/common"; | ||
| import * as i7 from "@angular/forms"; | ||
| export declare class AngularEditorModule { | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AngularEditorModule, never>; | ||
| static ɵmod: i0.ɵɵNgModuleDeclaration<AngularEditorModule, [typeof i1.AngularEditorComponent, typeof i2.AeToolbarComponent, typeof i3.AeSelectComponent, typeof i4.AeButtonComponent, typeof i5.AeToolbarSetComponent], [typeof i6.CommonModule, typeof i7.FormsModule, typeof i7.ReactiveFormsModule], [typeof i1.AngularEditorComponent, typeof i2.AeToolbarComponent, typeof i4.AeButtonComponent, typeof i5.AeToolbarSetComponent]>; | ||
| static ɵinj: i0.ɵɵInjectorDeclaration<AngularEditorModule>; | ||
| } |
| import { HttpClient, HttpEvent } from '@angular/common/http'; | ||
| import { Observable } from 'rxjs'; | ||
| import { CustomClass } from './config'; | ||
| import * as i0 from "@angular/core"; | ||
| export interface UploadResponse { | ||
| imageUrl: string; | ||
| } | ||
| export declare class AngularEditorService { | ||
| private http; | ||
| private doc; | ||
| savedSelection: Range | null; | ||
| selectedText: string; | ||
| uploadUrl: string; | ||
| uploadWithCredentials: boolean; | ||
| constructor(http: HttpClient, doc: any); | ||
| /** | ||
| * Executed command from editor header buttons exclude toggleEditorMode | ||
| * @param command string from triggerCommand | ||
| * @param value | ||
| */ | ||
| executeCommand(command: string, value?: string): void; | ||
| /** | ||
| * Create URL link | ||
| * @param url string from UI prompt | ||
| */ | ||
| createLink(url: string): void; | ||
| /** | ||
| * insert color either font or background | ||
| * | ||
| * @param color color to be inserted | ||
| * @param where where the color has to be inserted either text/background | ||
| */ | ||
| insertColor(color: string, where: string): void; | ||
| /** | ||
| * Set font name | ||
| * @param fontName string | ||
| */ | ||
| setFontName(fontName: string): void; | ||
| /** | ||
| * Set font size | ||
| * @param fontSize string | ||
| */ | ||
| setFontSize(fontSize: string): void; | ||
| /** | ||
| * Create raw HTML | ||
| * @param html HTML string | ||
| */ | ||
| insertHtml(html: string): void; | ||
| /** | ||
| * save selection when the editor is focussed out | ||
| */ | ||
| saveSelection: () => void; | ||
| /** | ||
| * restore selection when the editor is focused in | ||
| * | ||
| * saved selection when the editor is focused out | ||
| */ | ||
| restoreSelection(): boolean; | ||
| /** | ||
| * setTimeout used for execute 'saveSelection' method in next event loop iteration | ||
| */ | ||
| executeInNextQueueIteration(callbackFn: (...args: any[]) => any, timeout?: number): void; | ||
| /** check any selection is made or not */ | ||
| private checkSelection; | ||
| /** | ||
| * Upload file to uploadUrl | ||
| * @param file The file | ||
| */ | ||
| uploadImage(file: File): Observable<HttpEvent<UploadResponse>>; | ||
| /** | ||
| * Insert image with Url | ||
| * @param imageUrl The imageUrl. | ||
| */ | ||
| insertImage(imageUrl: string): void; | ||
| setDefaultParagraphSeparator(separator: string): void; | ||
| createCustomClass(customClass: CustomClass): void; | ||
| insertVideo(videoUrl: string): void; | ||
| private insertYouTubeVideoTag; | ||
| private insertVimeoVideoTag; | ||
| nextNode(node: any): any; | ||
| getRangeSelectedNodes(range: any, includePartiallySelectedContainers: any): any[]; | ||
| getSelectedNodes(): any[]; | ||
| replaceWithOwnChildren(el: any): void; | ||
| removeSelectedElements(tagNames: any): void; | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AngularEditorService, never>; | ||
| static ɵprov: i0.ɵɵInjectableDeclaration<AngularEditorService>; | ||
| } |
| import { UploadResponse } from './angular-editor.service'; | ||
| import { HttpEvent } from '@angular/common/http'; | ||
| import { Observable } from 'rxjs'; | ||
| export interface CustomClass { | ||
| name: string; | ||
| class: string; | ||
| tag?: string; | ||
| } | ||
| export interface Font { | ||
| name: string; | ||
| class: string; | ||
| } | ||
| export interface AngularEditorConfig { | ||
| editable?: boolean; | ||
| spellcheck?: boolean; | ||
| height?: 'auto' | string; | ||
| minHeight?: '0' | string; | ||
| maxHeight?: 'auto' | string; | ||
| width?: 'auto' | string; | ||
| minWidth?: '0' | string; | ||
| translate?: 'yes' | 'now' | string; | ||
| enableToolbar?: boolean; | ||
| showToolbar?: boolean; | ||
| placeholder?: string; | ||
| defaultParagraphSeparator?: string; | ||
| defaultFontName?: string; | ||
| defaultFontSize?: '1' | '2' | '3' | '4' | '5' | '6' | '7' | string; | ||
| uploadUrl?: string; | ||
| upload?: (file: File) => Observable<HttpEvent<UploadResponse>>; | ||
| uploadWithCredentials?: boolean; | ||
| fonts?: Font[]; | ||
| customClasses?: CustomClass[]; | ||
| sanitize?: boolean; | ||
| toolbarPosition?: 'top' | 'bottom'; | ||
| outline?: boolean; | ||
| toolbarHiddenButtons?: string[][]; | ||
| rawPaste?: boolean; | ||
| } | ||
| export declare const angularEditorConfig: AngularEditorConfig; |
| import { AfterViewInit, ChangeDetectorRef, ElementRef, EventEmitter, OnDestroy, OnInit, Renderer2, TemplateRef } from '@angular/core'; | ||
| import { ControlValueAccessor } from '@angular/forms'; | ||
| import { DomSanitizer } from '@angular/platform-browser'; | ||
| import { AeToolbarComponent } from '../ae-toolbar/ae-toolbar.component'; | ||
| import { AngularEditorService } from '../angular-editor.service'; | ||
| import { AngularEditorConfig } from '../config'; | ||
| import * as i0 from "@angular/core"; | ||
| export declare class AngularEditorComponent implements OnInit, ControlValueAccessor, AfterViewInit, OnDestroy { | ||
| private r; | ||
| private editorService; | ||
| private doc; | ||
| private sanitizer; | ||
| private cdRef; | ||
| private autoFocus; | ||
| private onChange; | ||
| private onTouched; | ||
| modeVisual: boolean; | ||
| showPlaceholder: boolean; | ||
| disabled: boolean; | ||
| focused: boolean; | ||
| touched: boolean; | ||
| changed: boolean; | ||
| focusInstance: any; | ||
| blurInstance: any; | ||
| id: string; | ||
| config: AngularEditorConfig; | ||
| placeholder: string; | ||
| tabIndex: number | null; | ||
| html: any; | ||
| textArea: ElementRef; | ||
| editorWrapper: ElementRef; | ||
| editorToolbar: AeToolbarComponent; | ||
| customButtonsTemplateRef?: TemplateRef<any>; | ||
| executeCommandFn: any; | ||
| viewMode: EventEmitter<boolean>; | ||
| /** emits `blur` event when focused out from the textarea */ | ||
| blurEvent: EventEmitter<FocusEvent>; | ||
| /** emits `focus` event when focused in to the textarea */ | ||
| focusEvent: EventEmitter<FocusEvent>; | ||
| tabindex: number; | ||
| onFocus(): void; | ||
| constructor(r: Renderer2, editorService: AngularEditorService, doc: any, sanitizer: DomSanitizer, cdRef: ChangeDetectorRef, defaultTabIndex: string, autoFocus: any); | ||
| ngOnInit(): void; | ||
| ngAfterViewInit(): void; | ||
| onPaste(event: ClipboardEvent): string; | ||
| /** | ||
| * Executed command from editor header buttons | ||
| * @param command string from triggerCommand | ||
| * @param value | ||
| */ | ||
| executeCommand(command: string, value?: string): void; | ||
| /** | ||
| * focus event | ||
| */ | ||
| onTextAreaFocus(event: FocusEvent): void; | ||
| /** | ||
| * @description fires when cursor leaves textarea | ||
| */ | ||
| onTextAreaMouseOut(event: MouseEvent): void; | ||
| /** | ||
| * blur event | ||
| */ | ||
| onTextAreaBlur(event: FocusEvent): void; | ||
| /** | ||
| * focus the text area when the editor is focused | ||
| */ | ||
| focus(): void; | ||
| /** | ||
| * Executed from the contenteditable section while the input property changes | ||
| * @param element html element from contenteditable | ||
| */ | ||
| onContentChange(element: HTMLElement): void; | ||
| /** | ||
| * Set the function to be called | ||
| * when the control receives a change event. | ||
| * | ||
| * @param fn a function | ||
| */ | ||
| registerOnChange(fn: any): void; | ||
| /** | ||
| * Set the function to be called | ||
| * when the control receives a touch event. | ||
| * | ||
| * @param fn a function | ||
| */ | ||
| registerOnTouched(fn: any): void; | ||
| /** | ||
| * Write a new value to the element. | ||
| * | ||
| * @param value value to be executed when there is a change in contenteditable | ||
| */ | ||
| writeValue(value: any): void; | ||
| /** | ||
| * refresh view/HTML of the editor | ||
| * | ||
| * @param value html string from the editor | ||
| */ | ||
| refreshView(value: string): void; | ||
| /** | ||
| * toggles placeholder based on input string | ||
| * | ||
| * @param value A HTML string from the editor | ||
| */ | ||
| togglePlaceholder(value: boolean): void; | ||
| /** | ||
| * Implements disabled state for this element | ||
| * | ||
| * @param isDisabled Disabled flag | ||
| */ | ||
| setDisabledState(isDisabled: boolean): void; | ||
| /** | ||
| * toggles editor mode based on bToSource bool | ||
| * | ||
| * @param bToSource A boolean value from the editor | ||
| */ | ||
| toggleEditorMode(bToSource: boolean): void; | ||
| /** | ||
| * toggles editor buttons when cursor moved or positioning | ||
| * | ||
| * Send a node array from the contentEditable of the editor | ||
| */ | ||
| exec(): void; | ||
| private configure; | ||
| getFonts(): { | ||
| label: string; | ||
| value: string; | ||
| }[]; | ||
| getCustomTags(): string; | ||
| ngOnDestroy(): void; | ||
| filterStyles(html: string): string; | ||
| static ɵfac: i0.ɵɵFactoryDeclaration<AngularEditorComponent, [null, null, null, null, null, { attribute: "tabindex"; }, { attribute: "autofocus"; }]>; | ||
| static ɵcmp: i0.ɵɵComponentDeclaration<AngularEditorComponent, "angular-editor", never, { "id": "id"; "config": "config"; "placeholder": "placeholder"; "tabIndex": "tabIndex"; }, { "html": "html"; "viewMode": "viewMode"; "blurEvent": "blur"; "focusEvent": "focus"; }, ["customButtonsTemplateRef"], never>; | ||
| } |
| export declare function isDefined(value: any): boolean; |
| export * from './lib/angular-editor.service'; | ||
| export * from './lib/editor/angular-editor.component'; | ||
| export * from './lib/ae-button/ae-button.component'; | ||
| export * from './lib/ae-toolbar-set/ae-toolbar-set.component'; | ||
| export * from './lib/ae-select/ae-select.component'; | ||
| export * from './lib/ae-toolbar/ae-toolbar.component'; | ||
| export * from './lib/angular-editor.module'; | ||
| export { AngularEditorConfig, CustomClass } from './lib/config'; |
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
Long strings
Supply chain riskContains long string literals, which may be a sign of obfuscated or packed code.
Found 1 instance in 1 package
Long strings
Supply chain riskContains long string literals, which may be a sign of obfuscated or packed code.
Found 1 instance in 1 package
No v1
QualityPackage is not semver >=1. This means it is not stable and does not support ^ ranges.
Found 1 instance in 1 package
1
-50%355
1.43%15
-63.41%231667
-63.29%5
66.67%9
-71.87%1902
-60.86%1
Infinity%