119 lines
2.2 KiB
Vue
119 lines
2.2 KiB
Vue
<template>
|
|
<div class="show-if-initialized" :class="{ initialized }">
|
|
<pre ref="editor"></pre>
|
|
</div>
|
|
</template>
|
|
|
|
<style lang="scss">
|
|
.show-if-initialized {
|
|
opacity: 0;
|
|
|
|
&.initialized {
|
|
opacity: 1;
|
|
}
|
|
|
|
& > * {
|
|
transition: none;
|
|
}
|
|
}
|
|
</style>
|
|
|
|
<script>
|
|
const DEFAULT_THEME = "twilight";
|
|
|
|
import ace from "ace-builds";
|
|
import "ace-builds/webpack-resolver";
|
|
|
|
export default {
|
|
props: {
|
|
value: {
|
|
type: String,
|
|
default: ""
|
|
},
|
|
theme: {
|
|
type: String,
|
|
required: false
|
|
},
|
|
lang: {
|
|
type: String,
|
|
default: "json"
|
|
},
|
|
options: {
|
|
type: Object,
|
|
default: {}
|
|
}
|
|
},
|
|
|
|
data() {
|
|
return {
|
|
initialized: false,
|
|
editor: null,
|
|
cacheValue: ""
|
|
};
|
|
},
|
|
|
|
watch: {
|
|
value(value) {
|
|
if (value !== this.cacheValue) {
|
|
this.editor.session.setValue(value, 1);
|
|
this.cacheValue = value;
|
|
}
|
|
},
|
|
theme() {
|
|
this.initialized = false;
|
|
this.editor.setTheme(`ace/theme/${this.defineTheme()}`, () => {
|
|
this.$nextTick().then(() => {
|
|
this.initialized = true;
|
|
});
|
|
});
|
|
},
|
|
lang(value) {
|
|
this.editor.getSession().setMode("ace/mode/" + value);
|
|
},
|
|
options(value) {
|
|
this.editor.setOptions(value);
|
|
}
|
|
},
|
|
|
|
mounted() {
|
|
const editor = ace.edit(this.$refs.editor, {
|
|
mode: `ace/mode/${this.lang}`,
|
|
...this.options
|
|
});
|
|
|
|
// Set the theme and show the editor only after it's been set to prevent FOUC.
|
|
editor.setTheme(`ace/theme/${this.defineTheme()}`, () => {
|
|
this.$nextTick().then(() => {
|
|
this.initialized = true;
|
|
});
|
|
});
|
|
|
|
if (this.value) editor.setValue(this.value, 1);
|
|
|
|
this.editor = editor;
|
|
this.cacheValue = this.value;
|
|
|
|
editor.on("change", () => {
|
|
const content = editor.getValue();
|
|
this.$emit("input", content);
|
|
this.cacheValue = content;
|
|
});
|
|
},
|
|
|
|
methods: {
|
|
defineTheme() {
|
|
if (this.theme) {
|
|
return this.theme;
|
|
}
|
|
return (
|
|
this.$store.state.postwoman.settings.THEME_ACE_EDITOR || DEFAULT_THEME
|
|
);
|
|
}
|
|
},
|
|
|
|
destroyed() {
|
|
this.editor.destroy();
|
|
}
|
|
};
|
|
</script>
|