aboutsummaryrefslogtreecommitdiff
path: root/src/tools/sql-prettify/sql-prettify.vue
blob: 8e05158d8db37e203db5c62864521767a99e6668 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
<template>
  <div style="flex: 0 0 100%">
    <div style="margin: 0 auto; width: 600px">
      <n-space n-space item-style="flex: 1 1 0">
        <div>
          <n-form-item label="Dialect">
            <n-select
              v-model:value="config.language"
              :options="[
                { label: 'GCP BigQuery', value: 'bigquery' },
                { label: 'IBM DB2', value: 'db2' },
                { label: 'Apache Hive', value: 'hive' },
                { label: 'MariaDB', value: 'mariadb' },
                { label: 'MySQL', value: 'mysql' },
                { label: 'Couchbase N1QL', value: 'n1ql' },
                { label: 'Oracle PL/SQL', value: 'plsql' },
                { label: 'PostgreSQL', value: 'postgresql' },
                { label: 'Amazon Redshift', value: 'redshift' },
                { label: 'Spark', value: 'spark' },
                { label: 'Standard SQL', value: 'sql' },
                { label: 'sqlite', value: 'sqlite' },
                { label: 'SQL Server Transact-SQL', value: 'tsql' },
              ]"
            />
          </n-form-item>
        </div>
        <div>
          <n-form-item label="Keyword case">
            <n-select
              v-model:value="config.keywordCase"
              :options="[
                { label: 'UPPERCASE', value: 'upper' },
                { label: 'lowercase', value: 'lower' },
                { label: 'Preserve', value: 'preserve' },
              ]"
            />
          </n-form-item>
        </div>
        <div>
          <n-form-item label="Indent style">
            <n-select
              v-model:value="config.indentStyle"
              :options="[
                { label: 'Standard', value: 'standard' },
                { label: 'Tabular left', value: 'tabularLeft' },
                { label: 'Tabular right', value: 'tabularRight' },
              ]"
            />
          </n-form-item>
        </div>
      </n-space>
    </div>
  </div>

  <n-form-item label="Your SQL query">
    <n-input
      ref="inputElement"
      v-model:value="rawSQL"
      placeholder="Put your SQL query here..."
      type="textarea"
      rows="20"
      autocomplete="off"
      autocorrect="off"
      autocapitalize="off"
      spellcheck="false"
    />
  </n-form-item>
  <n-form-item label="Prettify version of your query">
    <n-card class="result-card" :style="`min-height: ${inputElementHeight ?? 400}px`">
      <n-config-provider :hljs="hljs">
        <n-code :code="prettySQL" language="sql" :trim="false" />
      </n-config-provider>
      <n-button v-if="prettySQL" class="copy-button" secondary @click="copy">Copy</n-button>
    </n-card>
  </n-form-item>
</template>

<script setup lang="ts">
import { useCopy } from '@/composable/copy';
import { useElementSize } from '@vueuse/core';
import hljs from 'highlight.js/lib/core';
import sqlHljs from 'highlight.js/lib/languages/sql';
import { format as formatSQL, type FormatFnOptions } from 'sql-formatter';
import { computed, reactive, ref } from 'vue';
hljs.registerLanguage('sql', sqlHljs);

const inputElement = ref<HTMLElement>();
const { height: inputElementHeight } = useElementSize(inputElement);

const config = reactive<Partial<FormatFnOptions>>({
  keywordCase: 'upper',
  useTabs: false,
  language: 'sql',
  indentStyle: 'standard',
  tabulateAlias: true,
});

const rawSQL = ref('select field1,field2,field3 from my_table where my_condition;');
const prettySQL = computed(() => formatSQL(rawSQL.value, config));
const { copy } = useCopy({ source: prettySQL });
</script>

<style lang="less" scoped>
.result-card {
  position: relative;
  .copy-button {
    position: absolute;
    top: 10px;
    right: 10px;
  }
}
</style>