16 changed files with 440 additions and 17 deletions
File diff suppressed because one or more lines are too long
@ -0,0 +1,25 @@ |
|||
<!-- |
|||
|
|||
Copyright © 2016-2022 The Thingsboard Authors |
|||
|
|||
Licensed under the Apache License, Version 2.0 (the "License"); |
|||
you may not use this file except in compliance with the License. |
|||
You may obtain a copy of the License at |
|||
|
|||
http://www.apache.org/licenses/LICENSE-2.0 |
|||
|
|||
Unless required by applicable law or agreed to in writing, software |
|||
distributed under the License is distributed on an "AS IS" BASIS, |
|||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|||
See the License for the specific language governing permissions and |
|||
limitations under the License. |
|||
|
|||
--> |
|||
<section [formGroup]="htmlCardWidgetSettingsForm" fxLayout="column"> |
|||
<tb-html formControlName="cardHtml" required |
|||
label="{{ 'widgets.html-card.html' | translate }}"> |
|||
</tb-html> |
|||
<tb-css formControlName="cardCss" |
|||
label="{{ 'widgets.html-card.css' | translate }}"> |
|||
</tb-css> |
|||
</section> |
|||
@ -0,0 +1,54 @@ |
|||
///
|
|||
/// Copyright © 2016-2022 The Thingsboard Authors
|
|||
///
|
|||
/// Licensed under the Apache License, Version 2.0 (the "License");
|
|||
/// you may not use this file except in compliance with the License.
|
|||
/// You may obtain a copy of the License at
|
|||
///
|
|||
/// http://www.apache.org/licenses/LICENSE-2.0
|
|||
///
|
|||
/// Unless required by applicable law or agreed to in writing, software
|
|||
/// distributed under the License is distributed on an "AS IS" BASIS,
|
|||
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|||
/// See the License for the specific language governing permissions and
|
|||
/// limitations under the License.
|
|||
///
|
|||
|
|||
import { Component } from '@angular/core'; |
|||
import { WidgetSettings, WidgetSettingsComponent } from '@shared/models/widget.models'; |
|||
import { FormBuilder, FormGroup, Validators } from '@angular/forms'; |
|||
import { Store } from '@ngrx/store'; |
|||
import { AppState } from '@core/core.state'; |
|||
|
|||
@Component({ |
|||
selector: 'tb-html-card-widget-settings', |
|||
templateUrl: './html-card-widget-settings.component.html', |
|||
styleUrls: [] |
|||
}) |
|||
export class HtmlCardWidgetSettingsComponent extends WidgetSettingsComponent { |
|||
|
|||
htmlCardWidgetSettingsForm: FormGroup; |
|||
|
|||
constructor(protected store: Store<AppState>, |
|||
private fb: FormBuilder) { |
|||
super(store); |
|||
} |
|||
|
|||
protected settingsForm(): FormGroup { |
|||
return this.htmlCardWidgetSettingsForm; |
|||
} |
|||
|
|||
protected defaultSettings(): WidgetSettings { |
|||
return { |
|||
cardHtml: '<div class=\'card\'>HTML code here</div>', |
|||
cardCss: '.card {\n font-weight: bold; \n}' |
|||
}; |
|||
} |
|||
|
|||
protected onSettingsSet(settings: WidgetSettings) { |
|||
this.htmlCardWidgetSettingsForm = this.fb.group({ |
|||
cardHtml: [settings.cardHtml, [Validators.required]], |
|||
cardCss: [settings.cardCss, []] |
|||
}); |
|||
} |
|||
} |
|||
@ -0,0 +1,41 @@ |
|||
<!-- |
|||
|
|||
Copyright © 2016-2022 The Thingsboard Authors |
|||
|
|||
Licensed under the Apache License, Version 2.0 (the "License"); |
|||
you may not use this file except in compliance with the License. |
|||
You may obtain a copy of the License at |
|||
|
|||
http://www.apache.org/licenses/LICENSE-2.0 |
|||
|
|||
Unless required by applicable law or agreed to in writing, software |
|||
distributed under the License is distributed on an "AS IS" BASIS, |
|||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|||
See the License for the specific language governing permissions and |
|||
limitations under the License. |
|||
|
|||
--> |
|||
<div class="tb-html" style="background: #fff;" [ngClass]="{'tb-disabled': disabled, 'fill-height': fillHeight}" |
|||
tb-fullscreen |
|||
[fullscreen]="fullscreen" fxLayout="column"> |
|||
<div fxLayout="row" fxLayoutAlign="start center" style="height: 40px;" class="tb-html-toolbar"> |
|||
<label class="tb-title no-padding" [ngClass]="{'tb-error': hasErrors || required && !modelValue, 'tb-required': required}">{{ label }}</label> |
|||
<span fxFlex></span> |
|||
<button type='button' *ngIf="!disabled" mat-button class="tidy" (click)="beautifyHtml()"> |
|||
{{'js-func.tidy' | translate }} |
|||
</button> |
|||
<fieldset style="width: initial"> |
|||
<div matTooltip="{{(fullscreen ? 'fullscreen.exit' : 'fullscreen.expand') | translate}}" |
|||
matTooltipPosition="above" |
|||
style="border-radius: 50%" |
|||
(click)="fullscreen = !fullscreen"> |
|||
<button type='button' mat-button mat-icon-button class="tb-mat-32"> |
|||
<mat-icon class="material-icons">{{ fullscreen ? 'fullscreen_exit' : 'fullscreen' }}</mat-icon> |
|||
</button> |
|||
</div> |
|||
</fieldset> |
|||
</div> |
|||
<div id="tb-html-panel" class="tb-html-content-panel" fxLayout="column"> |
|||
<div #htmlEditor id="tb-html-input" [ngClass]="{'fill-height': fillHeight}"></div> |
|||
</div> |
|||
</div> |
|||
@ -0,0 +1,65 @@ |
|||
/** |
|||
* Copyright © 2016-2022 The Thingsboard Authors |
|||
* |
|||
* Licensed under the Apache License, Version 2.0 (the "License"); |
|||
* you may not use this file except in compliance with the License. |
|||
* You may obtain a copy of the License at |
|||
* |
|||
* http://www.apache.org/licenses/LICENSE-2.0 |
|||
* |
|||
* Unless required by applicable law or agreed to in writing, software |
|||
* distributed under the License is distributed on an "AS IS" BASIS, |
|||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|||
* See the License for the specific language governing permissions and |
|||
* limitations under the License. |
|||
*/ |
|||
.tb-html { |
|||
position: relative; |
|||
|
|||
&.tb-disabled { |
|||
color: rgba(0, 0, 0, .38); |
|||
} |
|||
|
|||
&.fill-height { |
|||
height: 100%; |
|||
} |
|||
|
|||
.tb-html-content-panel { |
|||
height: calc(100% - 40px); |
|||
border: 1px solid #c0c0c0; |
|||
|
|||
#tb-html-input { |
|||
width: 100%; |
|||
min-width: 200px; |
|||
height: 100%; |
|||
|
|||
&:not(.fill-height) { |
|||
min-height: 200px; |
|||
} |
|||
} |
|||
} |
|||
|
|||
.tb-html-toolbar { |
|||
& > * { |
|||
&:not(:last-child) { |
|||
margin-right: 4px; |
|||
} |
|||
} |
|||
button.mat-button, button.mat-icon-button, button.mat-icon-button.tb-mat-32 { |
|||
background: rgba(220, 220, 220, .35); |
|||
align-items: center; |
|||
vertical-align: middle; |
|||
min-width: 32px; |
|||
min-height: 15px; |
|||
padding: 4px; |
|||
font-size: .8rem; |
|||
line-height: 15px; |
|||
&:not(.tb-help-popup-button) { |
|||
color: #7b7b7b; |
|||
} |
|||
} |
|||
.tb-help-popup-button-loading { |
|||
background: #f3f3f3; |
|||
} |
|||
} |
|||
} |
|||
@ -0,0 +1,211 @@ |
|||
///
|
|||
/// Copyright © 2016-2022 The Thingsboard Authors
|
|||
///
|
|||
/// Licensed under the Apache License, Version 2.0 (the "License");
|
|||
/// you may not use this file except in compliance with the License.
|
|||
/// You may obtain a copy of the License at
|
|||
///
|
|||
/// http://www.apache.org/licenses/LICENSE-2.0
|
|||
///
|
|||
/// Unless required by applicable law or agreed to in writing, software
|
|||
/// distributed under the License is distributed on an "AS IS" BASIS,
|
|||
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|||
/// See the License for the specific language governing permissions and
|
|||
/// limitations under the License.
|
|||
///
|
|||
|
|||
import { |
|||
ChangeDetectorRef, |
|||
Component, |
|||
ElementRef, |
|||
forwardRef, |
|||
Input, |
|||
OnDestroy, |
|||
OnInit, |
|||
ViewChild, |
|||
ViewEncapsulation |
|||
} from '@angular/core'; |
|||
import { ControlValueAccessor, FormControl, NG_VALIDATORS, NG_VALUE_ACCESSOR, Validator } from '@angular/forms'; |
|||
import { Ace } from 'ace-builds'; |
|||
import { getAce } from '@shared/models/ace/ace.models'; |
|||
import { coerceBooleanProperty } from '@angular/cdk/coercion'; |
|||
import { Store } from '@ngrx/store'; |
|||
import { AppState } from '@core/core.state'; |
|||
import { UtilsService } from '@core/services/utils.service'; |
|||
import { TranslateService } from '@ngx-translate/core'; |
|||
import { CancelAnimationFrame, RafService } from '@core/services/raf.service'; |
|||
import { ResizeObserver } from '@juggle/resize-observer'; |
|||
import { beautifyHtml } from '@shared/models/beautify.models'; |
|||
|
|||
@Component({ |
|||
selector: 'tb-html', |
|||
templateUrl: './html.component.html', |
|||
styleUrls: ['./html.component.scss'], |
|||
providers: [ |
|||
{ |
|||
provide: NG_VALUE_ACCESSOR, |
|||
useExisting: forwardRef(() => HtmlComponent), |
|||
multi: true |
|||
}, |
|||
{ |
|||
provide: NG_VALIDATORS, |
|||
useExisting: forwardRef(() => HtmlComponent), |
|||
multi: true, |
|||
} |
|||
], |
|||
encapsulation: ViewEncapsulation.None |
|||
}) |
|||
export class HtmlComponent implements OnInit, OnDestroy, ControlValueAccessor, Validator { |
|||
|
|||
@ViewChild('htmlEditor', {static: true}) |
|||
htmlEditorElmRef: ElementRef; |
|||
|
|||
private htmlEditor: Ace.Editor; |
|||
private editorsResizeCaf: CancelAnimationFrame; |
|||
private editorResize$: ResizeObserver; |
|||
private ignoreChange = false; |
|||
|
|||
@Input() label: string; |
|||
|
|||
@Input() disabled: boolean; |
|||
|
|||
@Input() fillHeight: boolean; |
|||
|
|||
private requiredValue: boolean; |
|||
get required(): boolean { |
|||
return this.requiredValue; |
|||
} |
|||
@Input() |
|||
set required(value: boolean) { |
|||
this.requiredValue = coerceBooleanProperty(value); |
|||
} |
|||
|
|||
fullscreen = false; |
|||
|
|||
modelValue: string; |
|||
|
|||
hasErrors = false; |
|||
|
|||
private propagateChange = null; |
|||
|
|||
constructor(public elementRef: ElementRef, |
|||
private utils: UtilsService, |
|||
private translate: TranslateService, |
|||
protected store: Store<AppState>, |
|||
private raf: RafService, |
|||
private cd: ChangeDetectorRef) { |
|||
} |
|||
|
|||
ngOnInit(): void { |
|||
const editorElement = this.htmlEditorElmRef.nativeElement; |
|||
let editorOptions: Partial<Ace.EditorOptions> = { |
|||
mode: 'ace/mode/html', |
|||
showGutter: true, |
|||
showPrintMargin: true, |
|||
readOnly: this.disabled |
|||
}; |
|||
|
|||
const advancedOptions = { |
|||
enableSnippets: true, |
|||
enableBasicAutocompletion: true, |
|||
enableLiveAutocompletion: true |
|||
}; |
|||
|
|||
editorOptions = {...editorOptions, ...advancedOptions}; |
|||
getAce().subscribe( |
|||
(ace) => { |
|||
this.htmlEditor = ace.edit(editorElement, editorOptions); |
|||
this.htmlEditor.session.setUseWrapMode(true); |
|||
this.htmlEditor.setValue(this.modelValue ? this.modelValue : '', -1); |
|||
this.htmlEditor.setReadOnly(this.disabled); |
|||
this.htmlEditor.on('change', () => { |
|||
if (!this.ignoreChange) { |
|||
this.updateView(); |
|||
} |
|||
}); |
|||
// @ts-ignore
|
|||
this.htmlEditor.session.on('changeAnnotation', () => { |
|||
const annotations = this.htmlEditor.session.getAnnotations(); |
|||
const hasErrors = annotations.filter(annotation => annotation.type === 'error').length > 0; |
|||
if (this.hasErrors !== hasErrors) { |
|||
this.hasErrors = hasErrors; |
|||
this.propagateChange(this.modelValue); |
|||
this.cd.markForCheck(); |
|||
} |
|||
}); |
|||
this.editorResize$ = new ResizeObserver(() => { |
|||
this.onAceEditorResize(); |
|||
}); |
|||
this.editorResize$.observe(editorElement); |
|||
} |
|||
); |
|||
} |
|||
|
|||
ngOnDestroy(): void { |
|||
if (this.editorResize$) { |
|||
this.editorResize$.disconnect(); |
|||
} |
|||
} |
|||
|
|||
private onAceEditorResize() { |
|||
if (this.editorsResizeCaf) { |
|||
this.editorsResizeCaf(); |
|||
this.editorsResizeCaf = null; |
|||
} |
|||
this.editorsResizeCaf = this.raf.raf(() => { |
|||
this.htmlEditor.resize(); |
|||
this.htmlEditor.renderer.updateFull(); |
|||
}); |
|||
} |
|||
|
|||
registerOnChange(fn: any): void { |
|||
this.propagateChange = fn; |
|||
} |
|||
|
|||
registerOnTouched(fn: any): void { |
|||
} |
|||
|
|||
setDisabledState(isDisabled: boolean): void { |
|||
this.disabled = isDisabled; |
|||
if (this.htmlEditor) { |
|||
this.htmlEditor.setReadOnly(this.disabled); |
|||
} |
|||
} |
|||
|
|||
public validate(c: FormControl) { |
|||
return (!this.hasErrors) ? null : { |
|||
html: { |
|||
valid: false, |
|||
}, |
|||
}; |
|||
} |
|||
|
|||
beautifyHtml() { |
|||
beautifyHtml(this.modelValue, {indent_size: 4}).subscribe( |
|||
(res) => { |
|||
if (this.modelValue !== res) { |
|||
this.htmlEditor.setValue(res ? res : '', -1); |
|||
this.updateView(); |
|||
} |
|||
} |
|||
); |
|||
} |
|||
|
|||
writeValue(value: string): void { |
|||
this.modelValue = value; |
|||
if (this.htmlEditor) { |
|||
this.ignoreChange = true; |
|||
this.htmlEditor.setValue(this.modelValue ? this.modelValue : '', -1); |
|||
this.ignoreChange = false; |
|||
} |
|||
} |
|||
|
|||
updateView() { |
|||
const editorValue = this.htmlEditor.getValue(); |
|||
if (this.modelValue !== editorValue) { |
|||
this.modelValue = editorValue; |
|||
this.propagateChange(this.modelValue); |
|||
this.cd.markForCheck(); |
|||
} |
|||
} |
|||
} |
|||
Loading…
Reference in new issue