283 lines
6.5 KiB
Vue
283 lines
6.5 KiB
Vue
<template>
|
|
<div class="show-if-initialized" :class="{ initialized }">
|
|
<pre ref="editor" :class="styles"></pre>
|
|
<div
|
|
v-if="provideOutline"
|
|
class="
|
|
bg-primaryLight
|
|
border-t border-divider
|
|
flex flex-nowrap flex-1
|
|
py-1
|
|
px-4
|
|
bottom-0
|
|
z-10
|
|
sticky
|
|
overflow-auto
|
|
hide-scrollbar
|
|
"
|
|
>
|
|
<div
|
|
v-for="(p, index) in currentPath"
|
|
:key="`p-${index}`"
|
|
class="
|
|
cursor-pointer
|
|
flex-grow-0 flex-shrink-0
|
|
text-secondaryLight
|
|
inline-flex
|
|
items-center
|
|
hover:text-secondary
|
|
"
|
|
>
|
|
<span @click="onBlockClick(index)">
|
|
{{ p }}
|
|
</span>
|
|
<i v-if="index + 1 !== currentPath.length" class="mx-2 material-icons">
|
|
chevron_right
|
|
</i>
|
|
<tippy
|
|
v-if="siblingDropDownIndex == index"
|
|
ref="options"
|
|
interactive
|
|
trigger="click"
|
|
theme="popover"
|
|
arrow
|
|
>
|
|
<SmartItem
|
|
v-for="(sibling, siblingIndex) in currentSibling"
|
|
:key="`p-${index}-sibling-${siblingIndex}`"
|
|
:label="sibling.key ? sibling.key.value : i"
|
|
@click.native="goToSibling(sibling)"
|
|
/>
|
|
</tippy>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import ace from "ace-builds"
|
|
import "ace-builds/webpack-resolver"
|
|
import { defineComponent } from "@nuxtjs/composition-api"
|
|
import jsonParse from "~/helpers/jsonParse"
|
|
import debounce from "~/helpers/utils/debounce"
|
|
import outline from "~/helpers/outline"
|
|
|
|
export default defineComponent({
|
|
props: {
|
|
provideOutline: {
|
|
type: Boolean,
|
|
default: false,
|
|
required: false,
|
|
},
|
|
value: {
|
|
type: String,
|
|
default: "",
|
|
},
|
|
theme: {
|
|
type: String,
|
|
required: false,
|
|
default: null,
|
|
},
|
|
lang: {
|
|
type: String,
|
|
default: "json",
|
|
},
|
|
lint: {
|
|
type: Boolean,
|
|
default: true,
|
|
required: false,
|
|
},
|
|
options: {
|
|
type: Object,
|
|
default: () => {},
|
|
},
|
|
styles: {
|
|
type: String,
|
|
default: "",
|
|
},
|
|
},
|
|
|
|
data() {
|
|
return {
|
|
initialized: false,
|
|
editor: null,
|
|
cacheValue: "",
|
|
outline: outline(),
|
|
currentPath: [],
|
|
currentSibling: [],
|
|
siblingDropDownIndex: null,
|
|
}
|
|
},
|
|
|
|
computed: {
|
|
appFontSize() {
|
|
return getComputedStyle(document.documentElement).getPropertyValue(
|
|
"--body-font-size"
|
|
)
|
|
},
|
|
},
|
|
|
|
watch: {
|
|
value(value) {
|
|
if (value !== this.cacheValue) {
|
|
this.editor.session.setValue(value, 1)
|
|
this.cacheValue = value
|
|
if (this.lint) this.provideLinting(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
|
|
})
|
|
})
|
|
|
|
editor.setFontSize(this.appFontSize)
|
|
|
|
if (this.value) editor.setValue(this.value, 1)
|
|
|
|
this.editor = editor
|
|
this.cacheValue = this.value
|
|
|
|
if (this.lang === "json" && this.provideOutline)
|
|
this.initOutline(this.value)
|
|
|
|
editor.on("change", () => {
|
|
const content = editor.getValue()
|
|
this.$emit("input", content)
|
|
this.cacheValue = content
|
|
|
|
if (this.provideOutline) debounce(this.initOutline(content), 500)
|
|
|
|
if (this.lint) this.provideLinting(content)
|
|
})
|
|
|
|
if (this.lang === "json" && this.provideOutline) {
|
|
editor.session.selection.on("changeCursor", () => {
|
|
const index = editor.session.doc.positionToIndex(
|
|
editor.selection.getCursor(),
|
|
0
|
|
)
|
|
const path = this.outline.genPath(index)
|
|
if (path.success) {
|
|
this.currentPath = path.res
|
|
}
|
|
})
|
|
}
|
|
|
|
// Disable linting, if lint prop is false
|
|
if (this.lint) this.provideLinting(this.value)
|
|
},
|
|
|
|
destroyed() {
|
|
this.editor.destroy()
|
|
},
|
|
|
|
methods: {
|
|
defineTheme() {
|
|
if (this.theme) {
|
|
return this.theme
|
|
}
|
|
const strip = (str) =>
|
|
str.replace(/#/g, "").replace(/ /g, "").replace(/"/g, "")
|
|
return strip(
|
|
window
|
|
.getComputedStyle(document.documentElement)
|
|
.getPropertyValue("--editor-theme")
|
|
)
|
|
},
|
|
|
|
provideLinting: debounce(function (code) {
|
|
if (this.lang === "json") {
|
|
try {
|
|
jsonParse(code)
|
|
this.editor.session.setAnnotations([])
|
|
} catch (e) {
|
|
const pos = this.editor.session
|
|
.getDocument()
|
|
.indexToPosition(e.start, 0)
|
|
this.editor.session.setAnnotations([
|
|
{
|
|
row: pos.row,
|
|
column: pos.column,
|
|
text: e.message,
|
|
type: "error",
|
|
},
|
|
])
|
|
}
|
|
}
|
|
}, 2000),
|
|
|
|
onBlockClick(index) {
|
|
if (this.siblingDropDownIndex === index) {
|
|
this.clearSiblingList()
|
|
} else {
|
|
this.currentSibling = this.outline.getSiblings(index)
|
|
if (this.currentSibling.length) this.siblingDropDownIndex = index
|
|
}
|
|
},
|
|
clearSiblingList() {
|
|
this.currentSibling = []
|
|
this.siblingDropDownIndex = null
|
|
},
|
|
goToSibling(obj) {
|
|
this.clearSiblingList()
|
|
if (obj.start) {
|
|
const pos = this.editor.session.doc.indexToPosition(obj.start, 0)
|
|
if (pos) {
|
|
this.editor.session.selection.moveCursorTo(pos.row, pos.column, true)
|
|
this.editor.session.selection.clearSelection()
|
|
this.editor.scrollToLine(pos.row, false, true, null)
|
|
}
|
|
}
|
|
},
|
|
initOutline: debounce(function (content) {
|
|
if (this.lang === "json") {
|
|
try {
|
|
this.outline.init(content)
|
|
|
|
if (content[0] === "[") this.currentPath.push("[]")
|
|
else this.currentPath.push("{}")
|
|
} catch (e) {
|
|
console.log("Outline error: ", e)
|
|
}
|
|
}
|
|
}),
|
|
},
|
|
})
|
|
</script>
|
|
|
|
<style scoped lang="scss">
|
|
.show-if-initialized {
|
|
&.initialized {
|
|
@apply opacity-100;
|
|
}
|
|
|
|
& > * {
|
|
@apply transition-none;
|
|
}
|
|
}
|
|
</style>
|