123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198 |
- <template>
- <SmartModal
- v-if="show"
- dialog
- :title="`${t('request.generate_code')}`"
- @close="hideModal"
- >
- <template #body>
- <div class="flex flex-col px-2">
- <label for="requestType" class="px-4 pb-4">
- {{ t("request.choose_language") }}
- </label>
- <tippy ref="options" interactive trigger="click" theme="popover" arrow>
- <template #trigger>
- <span class="select-wrapper">
- <ButtonSecondary
- :label="
- CodegenDefinitions.find((x) => x.name === codegenType).caption
- "
- outline
- class="flex-1 pr-8"
- />
- </span>
- </template>
- <div class="flex flex-col space-y-2">
- <div class="sticky top-0">
- <input
- v-model="searchQuery"
- type="search"
- autocomplete="off"
- class="flex w-full p-4 py-2 !bg-popover input"
- :placeholder="`${t('action.search')}`"
- />
- </div>
- <div class="flex flex-col" role="menu">
- <SmartItem
- v-for="codegen in filteredCodegenDefinitions"
- :key="codegen.name"
- :label="codegen.caption"
- :info-icon="codegen.name === codegenType ? 'done' : ''"
- :active-info-icon="codegen.name === codegenType"
- @click.native="
- () => {
- codegenType = codegen.name
- options.tippy().hide()
- }
- "
- />
- </div>
- </div>
- </tippy>
- <div class="flex justify-between flex-1">
- <label for="generatedCode" class="p-4">
- {{ t("request.generated_code") }}
- </label>
- </div>
- <div
- v-if="errorState"
- class="w-full px-4 py-2 overflow-auto font-mono text-red-400 whitespace-normal rounded bg-primaryLight"
- >
- {{ t("error.something_went_wrong") }}
- </div>
- <div
- v-else-if="codegenType"
- ref="generatedCode"
- class="border rounded border-dividerLight"
- ></div>
- </div>
- </template>
- <template #footer>
- <span class="flex">
- <ButtonPrimary
- :label="`${t('action.copy')}`"
- :svg="copyIcon"
- @click.native="copyRequestCode"
- />
- <ButtonSecondary
- :label="`${t('action.dismiss')}`"
- @click.native="hideModal"
- />
- </span>
- </template>
- </SmartModal>
- </template>
- <script setup lang="ts">
- import { computed, ref, watch } from "@nuxtjs/composition-api"
- import * as O from "fp-ts/Option"
- import { Environment, makeRESTRequest } from "@hoppscotch/data"
- import { refAutoReset } from "@vueuse/core"
- import { useCodemirror } from "~/helpers/editor/codemirror"
- import { copyToClipboard } from "~/helpers/utils/clipboard"
- import {
- getEffectiveRESTRequest,
- resolvesEnvsInBody,
- } from "~/helpers/utils/EffectiveURL"
- import { getAggregateEnvs } from "~/newstore/environments"
- import { getRESTRequest } from "~/newstore/RESTSession"
- import { useI18n, useToast } from "~/helpers/utils/composables"
- import {
- CodegenDefinitions,
- CodegenName,
- generateCode,
- } from "~/helpers/new-codegen"
- const t = useI18n()
- const props = defineProps<{
- show: boolean
- }>()
- const emit = defineEmits<{
- (e: "hide-modal"): void
- }>()
- const toast = useToast()
- const options = ref<any | null>(null)
- const request = ref(getRESTRequest())
- const codegenType = ref<CodegenName>("shell-curl")
- const errorState = ref(false)
- const copyIcon = refAutoReset<"copy" | "check">("copy", 1000)
- const requestCode = computed(() => {
- const aggregateEnvs = getAggregateEnvs()
- const env: Environment = {
- name: "Env",
- variables: aggregateEnvs,
- }
- const effectiveRequest = getEffectiveRESTRequest(request.value, env)
- if (!props.show) return ""
- const result = generateCode(
- codegenType.value,
- makeRESTRequest({
- ...effectiveRequest,
- body: resolvesEnvsInBody(effectiveRequest.body, env),
- headers: effectiveRequest.effectiveFinalHeaders.map((header) => ({
- ...header,
- active: true,
- })),
- params: effectiveRequest.effectiveFinalParams.map((param) => ({
- ...param,
- active: true,
- })),
- endpoint: effectiveRequest.effectiveFinalURL,
- })
- )
- if (O.isSome(result)) {
- errorState.value = false
- return result.value
- } else {
- errorState.value = true
- return ""
- }
- })
- const generatedCode = ref<any | null>(null)
- useCodemirror(generatedCode, requestCode, {
- extendedEditorConfig: {
- mode: "text/plain",
- readOnly: true,
- },
- linter: null,
- completer: null,
- environmentHighlights: false,
- })
- watch(
- () => props.show,
- (goingToShow) => {
- if (goingToShow) {
- request.value = getRESTRequest()
- }
- }
- )
- const hideModal = () => emit("hide-modal")
- const copyRequestCode = () => {
- copyToClipboard(requestCode.value)
- copyIcon.value = "check"
- toast.success(`${t("state.copied_to_clipboard")}`)
- }
- const searchQuery = ref("")
- const filteredCodegenDefinitions = computed(() => {
- return CodegenDefinitions.filter((obj) =>
- Object.values(obj).some((val) => val.includes(searchQuery.value))
- )
- })
- </script>
|