mirror of
https://github.com/yangjian102621/geekai.git
synced 2025-09-19 01:36:38 +08:00
329 lines
12 KiB
Vue
329 lines
12 KiB
Vue
<template>
|
||
<div class="page-mj">
|
||
<div class="inner">
|
||
<div class="mj-box">
|
||
<h2>MidJourney 创作中心</h2>
|
||
|
||
<div class="mj-params" :style="{ height: mjBoxHeight + 'px' }">
|
||
<el-form :model="params" label-width="80px" label-position="left">
|
||
<div class="param-line pt">
|
||
<span>图片比例:</span>
|
||
<el-tooltip
|
||
effect="light"
|
||
content="生成图片的尺寸比例"
|
||
placement="right"
|
||
>
|
||
<el-icon>
|
||
<InfoFilled/>
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
|
||
<div class="param-line pt">
|
||
<el-row :gutter="10">
|
||
<el-col :span="8" v-for="item in rates" :key="item.value">
|
||
<div :class="item.value === params.rate?'grid-content active':'grid-content'"
|
||
@click="changeRate(item)">
|
||
<div :class="'shape '+item.css"></div>
|
||
<div class="text">{{ item.text }}</div>
|
||
</div>
|
||
</el-col>
|
||
</el-row>
|
||
</div>
|
||
|
||
<div class="param-line pt">
|
||
<span>模型选择:</span>
|
||
<el-tooltip
|
||
effect="light"
|
||
content="MJ: 偏真实通用模型 <br/>NIJI: 偏动漫风格、适用于二次元模型"
|
||
raw-content
|
||
placement="right"
|
||
>
|
||
<el-icon>
|
||
<InfoFilled/>
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
<div class="param-line pt">
|
||
<el-row :gutter="10">
|
||
<el-col :span="12" v-for="item in models" :key="item.value">
|
||
<div :class="item.value === params.model?'model active':'model'"
|
||
@click="changeModel(item)">
|
||
<el-image :src="item.img" fit="cover"></el-image>
|
||
<div class="text">{{ item.text }}</div>
|
||
</div>
|
||
</el-col>
|
||
</el-row>
|
||
</div>
|
||
|
||
<div class="param-line" style="padding-top: 10px">
|
||
<el-form-item label="创意度">
|
||
<template #default>
|
||
<div class="form-item-inner">
|
||
<el-input v-model="params.chaos" size="small"/>
|
||
<el-tooltip
|
||
effect="light"
|
||
content="参数用法:--chaos 或--c,取值范围: 0-100 <br/> 取值越高结果越发散,反之则稳定收敛<br /> 默认值0最为精准稳定"
|
||
raw-content
|
||
placement="right"
|
||
>
|
||
<el-icon>
|
||
<InfoFilled/>
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
</template>
|
||
</el-form-item>
|
||
</div>
|
||
|
||
<div class="param-line">
|
||
<el-form-item label="风格化">
|
||
<template #default>
|
||
<div class="form-item-inner">
|
||
<el-input v-model="params.stylize" size="small"/>
|
||
<el-tooltip
|
||
effect="light"
|
||
content="风格化:--stylize 或 --s,范围 1-1000,默认值100 <br/>高取值会产生非常艺术化但与提示关联性较低的图像"
|
||
raw-content
|
||
placement="right"
|
||
>
|
||
<el-icon>
|
||
<InfoFilled/>
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
</template>
|
||
</el-form-item>
|
||
</div>
|
||
|
||
<div class="param-line">
|
||
<el-form-item label="随机种子">
|
||
<template #default>
|
||
<div class="form-item-inner">
|
||
<el-input v-model="params.seed" size="small"/>
|
||
<el-tooltip
|
||
effect="light"
|
||
content="随机种子:--seed,默认值0表示随机产生 <br/>使用相同的种子参数和描述将产生相似的图像"
|
||
raw-content
|
||
placement="right"
|
||
>
|
||
<el-icon>
|
||
<InfoFilled/>
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
</template>
|
||
</el-form-item>
|
||
</div>
|
||
|
||
<div class="param-line">
|
||
<el-form-item label="原始模式">
|
||
<template #default>
|
||
<div class="form-item-inner">
|
||
<el-switch v-model="params.raw"/>
|
||
<el-tooltip
|
||
effect="light"
|
||
content="启用新的RAW模式,以“不带偏见”的方式生成图像。<br/> 同时也意味着您需要添加更长的提示。"
|
||
raw-content
|
||
placement="right"
|
||
>
|
||
<el-icon style="margin-top: 6px">
|
||
<InfoFilled/>
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
</template>
|
||
</el-form-item>
|
||
</div>
|
||
|
||
<div class="param-line">
|
||
<el-form-item label="图生图">
|
||
<template #default>
|
||
<div class="form-item-inner">
|
||
<el-input v-model="params.img" size="small" placeholder="请输入图片地址或者上传图片"
|
||
style="width: 160px"/>
|
||
<el-icon @click="params.img = ''" title="清空图片">
|
||
<DeleteFilled/>
|
||
</el-icon>
|
||
<el-tooltip
|
||
effect="light"
|
||
content="垫图:以某张图片为底稿参考来创作绘画 <br/> 支持 PNG 和 JPG 格式图片"
|
||
raw-content
|
||
placement="right"
|
||
>
|
||
<el-icon>
|
||
<InfoFilled/>
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
</template>
|
||
</el-form-item>
|
||
</div>
|
||
|
||
<div class="param-line">
|
||
<el-upload
|
||
class="img-uploader"
|
||
:auto-upload="true"
|
||
:show-file-list="false"
|
||
:http-request="afterRead"
|
||
>
|
||
<el-image v-if="params.img !== ''" :src="params.img" fit="cover"/>
|
||
<el-icon v-else class="uploader-icon">
|
||
<Plus/>
|
||
</el-icon>
|
||
</el-upload>
|
||
</div>
|
||
|
||
<div class="param-line" style="padding-top: 10px">
|
||
<el-form-item label="图像权重">
|
||
<template #default>
|
||
<div class="form-item-inner">
|
||
<el-slider v-model="params.weight" :max="1" :step="0.01" style="width: 180px"/>
|
||
<el-tooltip
|
||
effect="light"
|
||
content="使用图像权重参数--iw来调整图像 URL 与文本的重要性 <br/>权重较高时意味着图像提示将对完成的作业产生更大的影响"
|
||
raw-content
|
||
placement="right"
|
||
>
|
||
<el-icon style="margin-top: 6px">
|
||
<InfoFilled/>
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
</template>
|
||
</el-form-item>
|
||
</div>
|
||
|
||
<div class="param-line">
|
||
<el-input
|
||
v-model="params.prompt"
|
||
:autosize="{ minRows: 4, maxRows: 6 }"
|
||
type="textarea"
|
||
placeholder="这里输入你的咒语,例如:A chinese girl walking in the middle of a cobblestone street"
|
||
/>
|
||
</div>
|
||
|
||
</el-form>
|
||
</div>
|
||
<div class="submit-btn">
|
||
<el-button color="#47fff1" :dark="false" round @click="generate">立即生成</el-button>
|
||
</div>
|
||
</div>
|
||
<div class="task-list-box" :style="{ height: listBoxHeight + 'px' }">
|
||
<div class="task-list-inner">
|
||
<h2>任务列表</h2>
|
||
<div class="running-job-list">
|
||
<el-row :gutter="20">
|
||
<el-col :span="4" v-for="n in 10">
|
||
<div class="grid-content">
|
||
<el-image src="http://172.22.11.47:9010/chatgpt-plus/1694167591080692.png"/>
|
||
</div>
|
||
</el-col>
|
||
</el-row>
|
||
</div>
|
||
<h2>创作记录</h2>
|
||
<div class="finish-job-list">
|
||
<el-row :gutter="20">
|
||
<el-col :span="4" v-for="n in 100">
|
||
<div class="grid-content">
|
||
<el-image src="http://172.22.11.47:9010/chatgpt-plus/1694568531910050.png" fit="cover"/>
|
||
<div class="opt">
|
||
<div class="opt-line">
|
||
<ul>
|
||
<li><a @click="upscale(1)">U1</a></li>
|
||
<li><a @click="upscale(2)">U2</a></li>
|
||
<li><a @click="upscale(3)">U3</a></li>
|
||
<li><a @click="upscale(4)">U4</a></li>
|
||
</ul>
|
||
</div>
|
||
|
||
<div class="opt-line">
|
||
<ul>
|
||
<li><a @click="variation(1)">V1</a></li>
|
||
<li><a @click="variation(2)">V2</a></li>
|
||
<li><a @click="variation(3)">V3</a></li>
|
||
<li><a @click="variation(4)">V4</a></li>
|
||
</ul>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</el-col>
|
||
</el-row>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup>
|
||
import {ref} from "vue"
|
||
import {DeleteFilled, InfoFilled, Plus} from "@element-plus/icons-vue";
|
||
import Compressor from "compressorjs";
|
||
import {httpPost} from "@/utils/http";
|
||
import {ElMessage} from "element-plus";
|
||
|
||
const listBoxHeight = window.innerHeight - 20
|
||
const mjBoxHeight = window.innerHeight - 150
|
||
const rates = [
|
||
{css: "horizontal", value: "16:9", text: "横图"},
|
||
{css: "square", value: "1:1", text: "方图"},
|
||
{css: "vertical", value: "9:16", text: "竖图"},
|
||
]
|
||
const models = [
|
||
{text: "标准模型", value: "--v 5.2", img: "/images/mj-normal.png"},
|
||
{text: "动漫模型", value: "--niji 5", img: "/images/mj-niji.png"},
|
||
]
|
||
const params = ref({
|
||
rate: rates[0].value,
|
||
model: models[0].value,
|
||
chaos: 0,
|
||
stylize: 100,
|
||
seed: 0,
|
||
raw: false,
|
||
img: "",
|
||
weight: 0.25,
|
||
prompt: ""
|
||
})
|
||
|
||
// 切换图片比例
|
||
const changeRate = (item) => {
|
||
params.value.rate = item.value
|
||
}
|
||
// 切换模型
|
||
const changeModel = (item) => {
|
||
params.value.model = item.value
|
||
}
|
||
|
||
// 图片上传
|
||
const afterRead = (file) => {
|
||
// 压缩图片并上传
|
||
new Compressor(file.file, {
|
||
quality: 0.6,
|
||
success(result) {
|
||
const formData = new FormData();
|
||
formData.append('file', result, result.name);
|
||
// 执行上传操作
|
||
httpPost('/api/upload', formData).then((res) => {
|
||
params.value.img = res.data
|
||
ElMessage.success('上传成功')
|
||
}).catch((e) => {
|
||
ElMessage.error('上传失败:' + e.message)
|
||
})
|
||
},
|
||
error(err) {
|
||
console.log(err.message);
|
||
},
|
||
});
|
||
};
|
||
|
||
// 创建绘图任务
|
||
const generate = () => {
|
||
|
||
}
|
||
</script>
|
||
|
||
<style lang="stylus">
|
||
@import "@/assets/css/image-mj.styl"
|
||
</style>
|