2017-09-02 01:50:31 +00:00
|
|
|
<template>
|
|
|
|
<div class="bbcodeEditorContainer">
|
|
|
|
<slot></slot>
|
|
|
|
<div class="btn-group" role="toolbar">
|
|
|
|
<div class="bbcodeEditorButton btn" v-for="button in buttons" :title="button.title" @click.prevent.stop="apply(button)">
|
|
|
|
<span :class="'fa ' + button.icon"></span>
|
|
|
|
</div>
|
|
|
|
<div @click="previewBBCode" class="bbcodeEditorButton btn" :class="preview ? 'active' : ''"
|
|
|
|
:title="preview ? 'Close Preview' : 'Preview'">
|
|
|
|
<span class="fa fa-eye"></span>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div class="bbcodeEditorTextarea">
|
|
|
|
<textarea ref="input" :value="text" @input="$emit('input', $event.target.value)" v-show="!preview" :maxlength="maxlength"
|
|
|
|
:class="'bbcodeTextAreaTextArea ' + classes" @keyup="onKeyUp" :disabled="disabled" @paste="onPaste"
|
|
|
|
:placeholder="placeholder" @keypress="$emit('keypress', $event)" @keydown="onKeyDown"></textarea>
|
|
|
|
<div class="bbcodePreviewArea" v-show="preview">
|
|
|
|
<div class="bbcodePreviewHeader">
|
|
|
|
<ul class="bbcodePreviewWarnings" v-show="previewWarnings.length">
|
|
|
|
<li v-for="warning in previewWarnings">{{warning}}</li>
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
<div class="bbcode" ref="preview-element"></div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
|
|
|
import Vue from 'vue';
|
|
|
|
import Component from 'vue-class-component';
|
|
|
|
import {Prop, Watch} from 'vue-property-decorator';
|
|
|
|
import {BBCodeElement} from '../chat/bbcode';
|
|
|
|
import {getKey} from '../chat/common';
|
|
|
|
import {CoreBBCodeParser, urlRegex} from './core';
|
|
|
|
import {defaultButtons, EditorButton, EditorSelection} from './editor';
|
2017-12-05 01:47:27 +00:00
|
|
|
import {BBCodeParser} from './parser';
|
2017-09-02 01:50:31 +00:00
|
|
|
|
|
|
|
@Component
|
|
|
|
export default class Editor extends Vue {
|
|
|
|
@Prop()
|
|
|
|
readonly extras?: EditorButton[];
|
|
|
|
@Prop({default: 1000})
|
|
|
|
readonly maxlength: number;
|
|
|
|
@Prop()
|
|
|
|
readonly classes?: string;
|
|
|
|
@Prop()
|
|
|
|
readonly value?: string;
|
|
|
|
@Prop()
|
|
|
|
readonly disabled?: boolean;
|
|
|
|
@Prop()
|
|
|
|
readonly placeholder?: string;
|
|
|
|
preview = false;
|
|
|
|
previewWarnings: ReadonlyArray<string> = [];
|
|
|
|
previewResult = '';
|
|
|
|
text = this.value !== undefined ? this.value : '';
|
|
|
|
element: HTMLTextAreaElement;
|
|
|
|
maxHeight: number;
|
|
|
|
minHeight: number;
|
2017-12-05 01:47:27 +00:00
|
|
|
protected parser: BBCodeParser;
|
2017-09-02 01:50:31 +00:00
|
|
|
protected defaultButtons = defaultButtons;
|
|
|
|
private isShiftPressed = false;
|
|
|
|
|
2017-12-05 01:47:27 +00:00
|
|
|
created(): void {
|
|
|
|
this.parser = new CoreBBCodeParser();
|
|
|
|
}
|
|
|
|
|
2017-09-02 01:50:31 +00:00
|
|
|
mounted(): void {
|
|
|
|
this.element = <HTMLTextAreaElement>this.$refs['input'];
|
|
|
|
const $element = $(this.element);
|
|
|
|
this.maxHeight = parseInt($element.css('max-height'), 10);
|
|
|
|
//tslint:disable-next-line:strict-boolean-expressions
|
|
|
|
this.minHeight = parseInt($element.css('min-height'), 10) || $element.outerHeight() || 50;
|
|
|
|
}
|
|
|
|
|
|
|
|
get buttons(): EditorButton[] {
|
|
|
|
const buttons = this.defaultButtons.slice();
|
|
|
|
if(this.extras !== undefined)
|
|
|
|
for(let i = 0, l = this.extras.length; i < l; i++)
|
|
|
|
buttons.push(this.extras[i]);
|
|
|
|
return buttons;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Watch('value')
|
|
|
|
watchValue(newValue: string): void {
|
|
|
|
this.text = newValue;
|
|
|
|
this.$nextTick(() => this.resize());
|
|
|
|
}
|
|
|
|
|
|
|
|
getSelection(): EditorSelection {
|
|
|
|
const length = this.element.selectionEnd - this.element.selectionStart;
|
|
|
|
return {
|
|
|
|
start: this.element.selectionStart,
|
|
|
|
end: this.element.selectionEnd,
|
|
|
|
length,
|
|
|
|
text: this.element.value.substr(this.element.selectionStart, length)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
replaceSelection(replacement: string): string {
|
|
|
|
const selection = this.getSelection();
|
|
|
|
const start = this.element.value.substr(0, selection.start) + replacement;
|
|
|
|
const end = this.element.value.substr(selection.end);
|
|
|
|
this.element.value = start + end;
|
|
|
|
this.element.dispatchEvent(new Event('input'));
|
|
|
|
return start + end;
|
|
|
|
}
|
|
|
|
|
|
|
|
setSelection(start: number, end?: number): void {
|
|
|
|
if(end === undefined)
|
|
|
|
end = start;
|
|
|
|
this.element.focus();
|
|
|
|
this.element.setSelectionRange(start, end);
|
|
|
|
}
|
|
|
|
|
|
|
|
applyText(startText: string, endText: string): void {
|
|
|
|
const selection = this.getSelection();
|
|
|
|
if(selection.length > 0) {
|
|
|
|
const replacement = startText + selection.text + endText;
|
|
|
|
this.text = this.replaceSelection(replacement);
|
|
|
|
this.setSelection(selection.start, selection.start + replacement.length);
|
|
|
|
} else {
|
|
|
|
const start = this.text.substr(0, selection.start) + startText;
|
|
|
|
const end = endText + this.text.substr(selection.start);
|
|
|
|
this.text = start + end;
|
|
|
|
this.$nextTick(() => this.setSelection(start.length));
|
|
|
|
}
|
|
|
|
this.$emit('input', this.text);
|
|
|
|
}
|
|
|
|
|
|
|
|
apply(button: EditorButton): void {
|
|
|
|
// Allow emitted variations for custom buttons.
|
|
|
|
this.$once('insert', (startText: string, endText: string) => this.applyText(startText, endText));
|
|
|
|
if(button.handler !== undefined)
|
|
|
|
return <void>button.handler.call(this, this);
|
|
|
|
if(button.startText === undefined)
|
|
|
|
button.startText = `[${button.tag}]`;
|
|
|
|
if(button.endText === undefined)
|
|
|
|
button.endText = `[/${button.tag}]`;
|
|
|
|
this.applyText(button.startText, button.endText);
|
|
|
|
}
|
|
|
|
|
|
|
|
onKeyDown(e: KeyboardEvent): void {
|
|
|
|
const key = getKey(e);
|
2017-10-18 23:29:28 +00:00
|
|
|
if((e.metaKey || e.ctrlKey) && !e.shiftKey && key !== 'Control' && key !== 'Meta') { //tslint:disable-line:curly
|
2017-09-02 01:50:31 +00:00
|
|
|
for(const button of this.buttons)
|
|
|
|
if(button.key === key) {
|
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
|
|
|
this.apply(button);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if(key === 'Shift') this.isShiftPressed = true;
|
|
|
|
this.$emit('keydown', e);
|
|
|
|
}
|
|
|
|
|
|
|
|
onKeyUp(e: KeyboardEvent): void {
|
|
|
|
if(getKey(e) === 'Shift') this.isShiftPressed = false;
|
|
|
|
this.$emit('keyup', e);
|
|
|
|
}
|
|
|
|
|
|
|
|
resize(): void {
|
|
|
|
if(this.maxHeight > 0) {
|
|
|
|
this.element.style.height = 'auto';
|
|
|
|
this.element.style.height = `${Math.max(Math.min(this.element.scrollHeight + 5, this.maxHeight), this.minHeight)}px`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onPaste(e: ClipboardEvent): void {
|
|
|
|
const data = e.clipboardData.getData('text/plain');
|
|
|
|
if(!this.isShiftPressed && urlRegex.test(data)) {
|
|
|
|
e.preventDefault();
|
|
|
|
this.applyText(`[url=${data}]`, '[/url]');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
focus(): void {
|
|
|
|
this.element.focus();
|
|
|
|
}
|
|
|
|
|
|
|
|
previewBBCode(): void {
|
|
|
|
this.doPreview();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected doPreview(): void {
|
|
|
|
const targetElement = <HTMLElement>this.$refs['preview-element'];
|
|
|
|
if(this.preview) {
|
|
|
|
this.preview = false;
|
|
|
|
this.previewWarnings = [];
|
|
|
|
this.previewResult = '';
|
|
|
|
const previewElement = (<BBCodeElement>targetElement.firstChild);
|
|
|
|
if(previewElement.cleanup !== undefined) previewElement.cleanup();
|
|
|
|
if(targetElement.firstChild !== null) targetElement.removeChild(targetElement.firstChild);
|
|
|
|
} else {
|
|
|
|
this.preview = true;
|
|
|
|
this.parser.storeWarnings = true;
|
|
|
|
targetElement.appendChild(this.parser.parseEverything(this.text));
|
|
|
|
this.previewWarnings = this.parser.warnings;
|
|
|
|
this.parser.storeWarnings = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|