@c8y/ngx-components
Version:
Angular modules for Cumulocity IoT applications
115 lines • 15.3 kB
JavaScript
import { Component, NgZone, ViewChild, ElementRef, forwardRef, Input, Output, EventEmitter } from '@angular/core';
import { NG_VALUE_ACCESSOR } from '@angular/forms';
import { LoadingComponent, ThemeSwitcherService } from '@c8y/ngx-components';
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
import { NgIf } from '@angular/common';
import * as i0 from "@angular/core";
import * as i1 from "@c8y/ngx-components";
export async function loadMonacoEditor() {
const monaco = await import('monaco-editor');
return monaco.default || monaco;
}
export class EditorComponent {
constructor(zone, theme) {
this.zone = zone;
this.theme = theme;
this.editorOptions = {};
this.editorInit = new EventEmitter();
this._readOnly = false;
this._currentValue = '';
this.themeChanges$ = this.theme.currentlyAppliedTheme$.pipe(takeUntilDestroyed());
}
writeValue(obj) {
if (typeof obj === 'string') {
this._currentValue = obj;
}
else {
this._currentValue = JSON.stringify(obj, null, 2);
}
if (this.editor) {
this.editor.setValue(this._currentValue);
}
}
registerOnChange(fn) {
this._onChanges = fn;
this.register();
}
registerOnTouched(fn) {
this._onTouched = fn;
this.register();
}
setDisabledState(isDisabled) {
this._readOnly = isDisabled;
}
async ngAfterViewInit() {
this.monaco = await loadMonacoEditor();
this.editor = this.monaco.editor.create(this._editorContainer.nativeElement, Object.assign({
value: this._currentValue || '',
language: 'json',
automaticLayout: true,
readOnly: this._readOnly
}, this.editorOptions));
window.monacoEditor = this.editor;
this.editorInit.emit(this.editor);
this.register();
this.themeChanges$.subscribe(theme => {
const themeToSet = theme === 'dark' ? `vs-dark` : 'vs';
this.monaco.editor.setTheme(themeToSet);
});
}
register() {
if (!this.editor) {
return;
}
if (this._onChanges) {
const onChanges = this._onChanges;
this.editor.onDidChangeModelContent(() => {
this.zone.run(() => {
const value = this.editor.getValue();
this._currentValue = value;
onChanges(value);
});
});
this._onChanges = undefined;
}
if (this._onTouched) {
const onTouched = this._onTouched;
this.editor.onDidChangeModelContent(() => {
this.zone.run(() => {
onTouched();
});
});
this._onTouched = undefined;
}
}
ngOnDestroy() {
this.editor.getModel().dispose();
this.editor.dispose();
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: EditorComponent, deps: [{ token: i0.NgZone }, { token: i1.ThemeSwitcherService }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: EditorComponent, isStandalone: true, selector: "c8y-editor", inputs: { editorOptions: "editorOptions" }, outputs: { editorInit: "editorInit" }, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => EditorComponent),
multi: true
}
], viewQueries: [{ propertyName: "_editorContainer", first: true, predicate: ["editorContainer"], descendants: true, static: true }], ngImport: i0, template: "<c8y-loading *ngIf=\"!monaco\"></c8y-loading>\n<div class=\"editor-container\" style=\"height: 100%\" #editorContainer></div>\n", dependencies: [{ kind: "component", type: LoadingComponent, selector: "c8y-loading", inputs: ["layout", "progress", "message"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: EditorComponent, decorators: [{
type: Component,
args: [{ selector: 'c8y-editor', standalone: true, imports: [LoadingComponent, NgIf], providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => EditorComponent),
multi: true
}
], template: "<c8y-loading *ngIf=\"!monaco\"></c8y-loading>\n<div class=\"editor-container\" style=\"height: 100%\" #editorContainer></div>\n" }]
}], ctorParameters: () => [{ type: i0.NgZone }, { type: i1.ThemeSwitcherService }], propDecorators: { editorOptions: [{
type: Input
}], editorInit: [{
type: Output
}], _editorContainer: [{
type: ViewChild,
args: ['editorContainer', { static: true }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"editor.component.js","sourceRoot":"","sources":["../../../editor/editor.component.ts","../../../editor/editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  AfterViewInit,\n  OnDestroy,\n  NgZone,\n  ViewChild,\n  ElementRef,\n  forwardRef,\n  Input,\n  Output,\n  EventEmitter\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport type * as Monaco from 'monaco-editor';\nimport { LoadingComponent, ThemeOptions, ThemeSwitcherService } from '@c8y/ngx-components';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { Observable } from 'rxjs';\nimport { NgIf } from '@angular/common';\n\nexport async function loadMonacoEditor() {\n  const monaco = await import('monaco-editor');\n  return monaco.default || monaco;\n}\n\n@Component({\n  selector: 'c8y-editor',\n  templateUrl: './editor.component.html',\n  standalone: true,\n  imports: [LoadingComponent, NgIf],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => EditorComponent),\n      multi: true\n    }\n  ]\n})\nexport class EditorComponent implements AfterViewInit, OnDestroy, ControlValueAccessor {\n  @Input() editorOptions: Monaco.editor.IStandaloneEditorConstructionOptions = {};\n  @Output() editorInit = new EventEmitter<Monaco.editor.IStandaloneCodeEditor>();\n  monaco: typeof Monaco;\n  editor: Monaco.editor.IStandaloneCodeEditor;\n\n  private _readOnly = false;\n  private _currentValue = '';\n  private _onChanges: (value: string) => void;\n  private _onTouched: () => void;\n  @ViewChild('editorContainer', { static: true }) _editorContainer: ElementRef;\n  private themeChanges$: Observable<ThemeOptions>;\n\n  constructor(\n    private zone: NgZone,\n    private theme: ThemeSwitcherService\n  ) {\n    this.themeChanges$ = this.theme.currentlyAppliedTheme$.pipe(takeUntilDestroyed());\n  }\n\n  writeValue(obj: string | object): void {\n    if (typeof obj === 'string') {\n      this._currentValue = obj;\n    } else {\n      this._currentValue = JSON.stringify(obj, null, 2);\n    }\n\n    if (this.editor) {\n      this.editor.setValue(this._currentValue);\n    }\n  }\n\n  registerOnChange(fn: (value: string) => void): void {\n    this._onChanges = fn;\n    this.register();\n  }\n\n  registerOnTouched(fn: () => void): void {\n    this._onTouched = fn;\n    this.register();\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this._readOnly = isDisabled;\n  }\n\n  async ngAfterViewInit(): Promise<void> {\n    this.monaco = await loadMonacoEditor();\n    this.editor = this.monaco.editor.create(\n      this._editorContainer.nativeElement,\n      Object.assign(\n        {\n          value: this._currentValue || '',\n          language: 'json',\n          automaticLayout: true,\n          readOnly: this._readOnly\n        },\n        this.editorOptions\n      )\n    );\n    (<any>window).monacoEditor = this.editor;\n    this.editorInit.emit(this.editor);\n    this.register();\n    this.themeChanges$.subscribe(theme => {\n      const themeToSet = theme === 'dark' ? `vs-dark` : 'vs';\n      this.monaco.editor.setTheme(themeToSet);\n    });\n  }\n\n  register() {\n    if (!this.editor) {\n      return;\n    }\n    if (this._onChanges) {\n      const onChanges = this._onChanges;\n      this.editor.onDidChangeModelContent(() => {\n        this.zone.run(() => {\n          const value = this.editor.getValue();\n          this._currentValue = value;\n          onChanges(value);\n        });\n      });\n      this._onChanges = undefined;\n    }\n\n    if (this._onTouched) {\n      const onTouched = this._onTouched;\n      this.editor.onDidChangeModelContent(() => {\n        this.zone.run(() => {\n          onTouched();\n        });\n      });\n      this._onTouched = undefined;\n    }\n  }\n\n  ngOnDestroy() {\n    this.editor.getModel().dispose();\n    this.editor.dispose();\n  }\n}\n","<c8y-loading *ngIf=\"!monaco\"></c8y-loading>\n<div class=\"editor-container\" style=\"height: 100%\" #editorContainer></div>\n"]}