225 lines
5.1 KiB
Vue
225 lines
5.1 KiB
Vue
<template>
|
|
<div class="opacity-0 show-if-initialized" :class="{ initialized }">
|
|
<pre ref="editor"></pre>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped lang="scss">
|
|
.show-if-initialized {
|
|
&.initialized {
|
|
@apply opacity-100;
|
|
}
|
|
|
|
& > * {
|
|
@apply transition-none;
|
|
}
|
|
}
|
|
</style>
|
|
|
|
<script>
|
|
const DEFAULT_THEME = "twilight"
|
|
|
|
import ace from "ace-builds"
|
|
import "ace-builds/webpack-resolver"
|
|
import "ace-builds/src-noconflict/ext-language_tools"
|
|
import "ace-builds/src-noconflict/mode-graphqlschema"
|
|
import { defineGQLLanguageMode } from "~/helpers/syntax/gqlQueryLangMode"
|
|
|
|
import * as gql from "graphql"
|
|
import { getAutocompleteSuggestions } from "graphql-language-service-interface"
|
|
import debounce from "~/helpers/utils/debounce"
|
|
|
|
export default {
|
|
props: {
|
|
value: {
|
|
type: String,
|
|
default: "",
|
|
},
|
|
theme: {
|
|
type: String,
|
|
required: false,
|
|
},
|
|
onRunGQLQuery: {
|
|
type: Function,
|
|
default: () => {},
|
|
},
|
|
options: {
|
|
type: Object,
|
|
default: {},
|
|
},
|
|
},
|
|
|
|
data() {
|
|
return {
|
|
initialized: false,
|
|
editor: null,
|
|
cacheValue: "",
|
|
validationSchema: null,
|
|
}
|
|
},
|
|
|
|
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
|
|
})
|
|
})
|
|
},
|
|
options(value) {
|
|
this.editor.setOptions(value)
|
|
},
|
|
},
|
|
|
|
mounted() {
|
|
defineGQLLanguageMode(ace)
|
|
|
|
let langTools = ace.require("ace/ext/language_tools")
|
|
|
|
const editor = ace.edit(this.$refs.editor, {
|
|
mode: `ace/mode/gql-query`,
|
|
enableBasicAutocompletion: true,
|
|
enableLiveAutocompletion: true,
|
|
...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
|
|
})
|
|
})
|
|
|
|
// 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
|
|
})
|
|
})
|
|
|
|
const completer = {
|
|
getCompletions: (editor, _session, { row, column }, _prefix, callback) => {
|
|
if (this.validationSchema) {
|
|
const completions = getAutocompleteSuggestions(this.validationSchema, editor.getValue(), {
|
|
line: row,
|
|
character: column,
|
|
})
|
|
|
|
callback(
|
|
null,
|
|
completions.map(({ label, detail }) => ({
|
|
name: label,
|
|
value: label,
|
|
score: 1.0,
|
|
meta: detail,
|
|
}))
|
|
)
|
|
} else {
|
|
callback(null, [])
|
|
}
|
|
},
|
|
}
|
|
|
|
langTools.setCompleters([completer])
|
|
|
|
if (this.value) editor.setValue(this.value, 1)
|
|
|
|
this.editor = editor
|
|
this.cacheValue = this.value
|
|
|
|
editor.commands.addCommand({
|
|
name: "runGQLQuery",
|
|
exec: () => this.onRunGQLQuery(this.editor.getValue()),
|
|
bindKey: {
|
|
mac: "cmd-enter",
|
|
win: "ctrl-enter",
|
|
},
|
|
})
|
|
|
|
editor.commands.addCommand({
|
|
name: "prettifyGQLQuery",
|
|
exec: () => this.prettifyQuery(),
|
|
bindKey: {
|
|
mac: "cmd-p",
|
|
win: "ctrl-p",
|
|
},
|
|
})
|
|
|
|
editor.on("change", () => {
|
|
const content = editor.getValue()
|
|
this.$emit("input", content)
|
|
this.parseContents(content)
|
|
this.cacheValue = content
|
|
})
|
|
|
|
this.parseContents(this.value)
|
|
},
|
|
|
|
methods: {
|
|
prettifyQuery() {
|
|
try {
|
|
this.value = gql.print(gql.parse(this.editor.getValue()))
|
|
} catch (e) {
|
|
this.$toast.error(`${this.$t("gql_prettify_invalid_query")}`, {
|
|
icon: "error",
|
|
})
|
|
}
|
|
},
|
|
|
|
defineTheme() {
|
|
if (this.theme) {
|
|
return this.theme
|
|
} else {
|
|
return this.$store.state.postwoman.settings.THEME_ACE_EDITOR || DEFAULT_THEME
|
|
}
|
|
},
|
|
|
|
setValidationSchema(schema) {
|
|
this.validationSchema = schema
|
|
this.parseContents(this.cacheValue)
|
|
},
|
|
|
|
parseContents: debounce(function (content) {
|
|
if (content !== "") {
|
|
try {
|
|
const doc = gql.parse(content)
|
|
|
|
if (this.validationSchema) {
|
|
this.editor.session.setAnnotations(
|
|
gql.validate(this.validationSchema, doc).map(({ locations, message }) => ({
|
|
row: locations[0].line - 1,
|
|
column: locations[0].column - 1,
|
|
text: message,
|
|
type: "error",
|
|
}))
|
|
)
|
|
}
|
|
} catch (e) {
|
|
this.editor.session.setAnnotations([
|
|
{
|
|
row: e.locations[0].line - 1,
|
|
column: e.locations[0].column - 1,
|
|
text: e.message,
|
|
type: "error",
|
|
},
|
|
])
|
|
}
|
|
} else {
|
|
this.editor.session.setAnnotations([])
|
|
}
|
|
}, 2000),
|
|
},
|
|
|
|
beforeDestroy() {
|
|
this.editor.destroy()
|
|
},
|
|
}
|
|
</script>
|