Vue3仿PostMan组件

204 阅读1分钟

近期项目里有一个仿PostMan功能的需求,要求能够在系统内部调试接口,找了一下网上没有合适的组件用,所以自己就写了一个,如果大家也有类似的需求,就直接拷贝代码用去吧,会节省很多时间。

image.png

<template>
    <el-card shadow="never">
        <el-input v-model="url">
            <template #prepend>
                <el-select v-model="method" placeholder="Select" style="width: 115px">
                    <el-option label="GET" value="GET" />
                    <el-option label="POST" value="POST" />
                    <el-option label="PUT" value="PUT" />
                    <el-option label="PATCH" value="PATCH" />
                    <el-option label="DELETE" value="DELETE" />
                    <el-option label="HEAN" value="HEAN" />
                    <el-option label="OPTIONS" value="OPTIONS" />
                </el-select>
            </template>
            <template #append>
                <el-button type="primary" @click="onSubmit">发 送</el-button>
            </template>
        </el-input>
        <el-tabs v-model="activeName" type="card" class="mt-5">
            <el-tab-pane label="Body" name="body">
                <el-radio-group v-model="sendType">
                    <el-radio label="form-data">form-data</el-radio>
                    <el-radio label="x-www-form-urlencoded">x-www-form-urlencoded</el-radio>
                    <el-radio label="raw">raw</el-radio>
                </el-radio-group>
                <div class="my-2">
                    <!-- form-data -->
                    <div v-if="sendType === 'form-data'">
                        <el-table :data="formData" style="width: 100%" border>
                            <el-table-column label="Key">
                                <template #default="scope">
                                    <el-input v-model="scope.row.key">
                                        <template #append>
                                            <el-select v-model="scope.row.type" style="width: 80px">
                                                <el-option label="Text" value="Text" />
                                                <el-option label="File" value="File" />
                                            </el-select>
                                        </template>
                                    </el-input>
                                </template>
                            </el-table-column>
                            <el-table-column label="Value">
                                <template #default="scope">
                                    <el-input v-model="scope.row.value" v-if="scope.row.type === 'Text'" />
                                    <el-upload v-else action :limit="1" :auto-upload="true" :show-file-list="true"
                                        :http-request="(parmas) => httpRequest(parmas, scope.$index)">
                                        <el-button type="primary">上传文件</el-button>
                                    </el-upload>
                                </template>
                            </el-table-column>
                            <el-table-column label="操作" width="100" align="center">
                                <template #default="scope">
                                    <el-button type="danger" @click="deleteObj(scope.$index, 1)">删除</el-button>
                                </template>
                            </el-table-column>
                        </el-table>
                        <div class="mt-3">
                            <el-button type="primary" @click="addObj(1)">添加属性</el-button>
                        </div>
                    </div>
                    <!-- x-www-form-urlencoded -->
                    <div v-if="sendType === 'x-www-form-urlencoded'">
                        <el-table :data="xFormData" style="width: 100%" border>
                            <el-table-column label="Key">
                                <template #default="scope">
                                    <el-input v-model="scope.row.key" />
                                </template>
                            </el-table-column>
                            <el-table-column label="Value">
                                <template #default="scope">
                                    <el-input v-model="scope.row.value" />
                                </template>
                            </el-table-column>
                            <el-table-column label="操作" width="100" align="center">
                                <template #default="scope">
                                    <el-button type="danger" @click="deleteObj(scope.$index, 2)">删除</el-button>
                                </template>
                            </el-table-column>
                        </el-table>
                        <div class="mt-3">
                            <el-button type="primary" @click="addObj(2)">添加属性</el-button>
                        </div>
                    </div>
                    <!-- raw -->
                    <div v-if="sendType === 'raw'">
                        <el-input v-model="raw" :rows="15" type="textarea" />
                    </div>
                </div>
            </el-tab-pane>
            <el-tab-pane label="Headers" name="headers">
                <el-table :data="headerData" style="width: 100%" border>
                    <el-table-column label="Key">
                        <template #default="scope">
                            <el-input v-model="scope.row.key" />
                        </template>
                    </el-table-column>
                    <el-table-column label="Value">
                        <template #default="scope">
                            <el-input v-model="scope.row.value" />
                        </template>
                    </el-table-column>
                    <el-table-column label="操作" width="100" align="center">
                        <template #default="scope">
                            <el-button type="danger" @click="deleteObj(scope.$index, 3)">删除</el-button>
                        </template>
                    </el-table-column>
                </el-table>
                <div class="mt-3">
                    <el-button type="primary" @click="addObj(3)">添加属性</el-button>
                </div>
            </el-tab-pane>
        </el-tabs>
        <div class="mt-10">
            <div class="mb-2">响应结果</div>
            <v-ace-editor v-model:value="response" lang="json" theme="chrome" :readonly="true" :printMargin="false"
                style="height: 500px" />
        </div>
    </el-card>
</template>

<script setup lang="ts">
import { ref } from 'vue'
import { axios } from '/@/api/system/postman'
import { VAceEditor } from 'vue3-ace-editor'
import 'ace-builds/src-noconflict/mode-json'
import 'ace-builds/src-noconflict/theme-chrome'
const method = ref('GET')
const url = ref('')
const activeName = ref('body')
const sendType = ref('form-data')
const headerData = ref([
    {
        key: '',
        value: ''
    }
])
const formData = ref([
    {
        key: '',
        value: '',
        type: 'Text'
    }
])
const xFormData = ref([
    {
        key: '',
        value: ''
    }
])
const raw = ref('')
const response = ref('')

const addObj = (type: number) => {
    if (type === 1) {
        formData.value.push({
            key: '',
            value: '',
            type: 'Text'
        })
    } else if (type === 2) {
        xFormData.value.push({
            key: '',
            value: ''
        })
    } else {
        headerData.value.push({
            key: '',
            value: ''
        })
    }
}

const deleteObj = (index: number, type: number) => {
    if (type === 1) {
        formData.value.splice(index, 1)
    } else if (type === 2) {
        xFormData.value.splice(index, 1)
    } else {
        headerData.value.splice(index, 1)
    }
}

interface RequestOptions {
    url: string;
    method: string;
    params?: any;
    data?: any;
    headers: any;
}

const transformData = (data: any) => {
    const newFormData = new FormData()
    data.forEach((item: any) => {
        if (item.key) {
            newFormData.append(item.key, item.type === 'Text' ? String(item.value) : item.value)
        }
    })
    return newFormData
}

const transXformData = (data: any) => {
    const transformedData: any = {}
    data.forEach((item: any) => {
        if (item.key) {
            transformedData[item.key] = item.value
        }
    })
    return transformedData
}

const transformSendData = (sendType: string, formData: any, xFormData: any, raw: string): any => {
    switch (sendType) {
        case 'form-data':
            return ['GET', 'DELETE', 'HEAD', 'OPTIONS'].includes(method.value) ? transXformData(xFormData) : transformData(formData)
        case 'x-www-form-urlencoded':
            return transXformData(xFormData)
        default:
            try {
                return JSON.parse(raw)
            } catch (error) {
                throw new Error('Invalid raw data format')
            }
    }
}

const httpRequest = (event: any, index: number) => {
    formData.value[index].value = event.file
}

const onSubmit = () => {
    const requestOptions: RequestOptions = {
        url: url.value,
        method: method.value,
        headers: {
            ...transformData(headerData.value),
            ...sendType.value === 'form-data' ? {
                'Content-Type': 'multipart/form-data'
            } : {}
        }
    }
    if (['GET', 'DELETE', 'HEAD', 'OPTIONS'].includes(method.value)) {
        requestOptions.params = transformSendData(sendType.value, formData.value, xFormData.value, raw.value)
    } else {
        requestOptions.data = transformSendData(sendType.value, formData.value, xFormData.value, raw.value)
    }
    console.log(formData.value)
    return
    axios(requestOptions).then((result) => {
        response.value = JSON.stringify(result, null, 2)
    }).catch((error) => {
        response.value = JSON.stringify({ error: error.message }, null, 2)
    })
}
</script>