mirror of
https://github.com/JefferyHcool/BiliNote.git
synced 2026-06-28 03:01:28 +08:00
之前插件 popup / options 的笔记选项跟 web 端 NoteForm 不齐,存在三处差距:
1. style 字段实质 broken
· backend prompt_builder.get_style_format 是 enum 映射(minimal/detailed/
academic/tutorial/xiaohongshu/life_journal/task_oriented/business/
meeting_minutes 共 9 个),不命中直接 return ''
· 插件原来给的是自由文本框,用户填什么都对不上 enum,等于没传
· 改:popup + options 都换成 9 个预设的下拉框,与 backend 严格对齐
2. format 字段缺一半
· backend 支持 toc / link / screenshot / summary 四个
· 插件只暴露了 screenshot / link 两个 checkbox
· 改:types.ts 新增 NOTE_FORMATS 常量,UI 渲染完整 4 个 checkbox。
生成请求时 format 数组、screenshot/link 单布尔由 settings.formats 派生,单一真相源
3. 缺 extras 字段
· backend VideoRequest.extras 直接拼到 prompt 末尾给 LLM
· 改:popup 折叠的"高级"区 + options 默认生成选项区都加 textarea
Settings 默认值:style='minimal'、formats=['toc','summary']、extras=''。
旧 settings 里若 style 是无效字符串,下拉会显示空白,用户重选一次即可。
logic/types.ts:
- 新增 NoteStyle / NoteFormat type alias 与 NOTE_STYLES / NOTE_FORMATS 常量
- Settings 接口加 formats: NoteFormat[] / extras: string,style 改为 NoteStyle
- 老的 screenshot / link 布尔保留(向后兼容旧 storage),但 UI 不再绑定,submit 时也由 formats 派生
popup / background / options 三处提交 generate_note 的逻辑同步收口。
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
170 lines
5.8 KiB
Vue
170 lines
5.8 KiB
Vue
<script setup lang="ts">
|
||
import { onMounted, ref } from 'vue'
|
||
import { getProviders, ping } from '~/logic/api'
|
||
import { settings, settingsReady } from '~/logic/storage'
|
||
import { getModelsByProvider } from '~/logic/api'
|
||
import { NOTE_FORMATS, NOTE_STYLES, type Model, type NoteFormat, type Provider } from '~/logic/types'
|
||
import { watch } from 'vue'
|
||
|
||
function toggleFormat(value: NoteFormat, checked: boolean) {
|
||
const cur = settings.value.formats || []
|
||
settings.value.formats = checked
|
||
? Array.from(new Set([...cur, value]))
|
||
: cur.filter(v => v !== value)
|
||
}
|
||
|
||
const providers = ref<Provider[]>([])
|
||
const models = ref<Model[]>([])
|
||
const status = ref<{ kind: 'idle' | 'ok' | 'err', text: string }>({ kind: 'idle', text: '' })
|
||
const loading = ref(false)
|
||
|
||
async function refresh() {
|
||
loading.value = true
|
||
status.value = { kind: 'idle', text: '' }
|
||
try {
|
||
providers.value = (await getProviders()).filter(p => p.enabled === 1)
|
||
if (settings.value.providerId)
|
||
await refreshModels(settings.value.providerId)
|
||
status.value = { kind: 'ok', text: `已加载 ${providers.value.length} 个供应商` }
|
||
}
|
||
catch (e) {
|
||
status.value = { kind: 'err', text: `加载失败:${(e as Error).message}` }
|
||
providers.value = []
|
||
models.value = []
|
||
}
|
||
finally {
|
||
loading.value = false
|
||
}
|
||
}
|
||
|
||
async function refreshModels(providerId: string) {
|
||
if (!providerId) {
|
||
models.value = []
|
||
return
|
||
}
|
||
try {
|
||
models.value = await getModelsByProvider(providerId)
|
||
}
|
||
catch {
|
||
models.value = []
|
||
}
|
||
}
|
||
|
||
async function testConnection() {
|
||
status.value = { kind: 'idle', text: '正在测试…' }
|
||
const ok = await ping()
|
||
status.value = ok
|
||
? { kind: 'ok', text: '后端连通 ✓' }
|
||
: { kind: 'err', text: '无法连接后端,请检查地址、端口与 CORS' }
|
||
}
|
||
|
||
watch(() => settings.value?.providerId, (id) => {
|
||
if (id)
|
||
refreshModels(id)
|
||
})
|
||
|
||
onMounted(async () => {
|
||
await settingsReady
|
||
if (settings.value.backendUrl)
|
||
await refresh()
|
||
})
|
||
</script>
|
||
|
||
<template>
|
||
<div class="p-6 max-w-2xl">
|
||
<h1 class="text-xl font-bold mb-4">通用</h1>
|
||
|
||
<section class="section-card">
|
||
<h2 class="font-semibold">后端地址</h2>
|
||
<div class="flex gap-2">
|
||
<input v-model="settings.backendUrl" class="input flex-1" placeholder="http://localhost:8483">
|
||
<button class="btn-secondary" @click="testConnection">测试连通</button>
|
||
<button class="btn-secondary" :disabled="loading" @click="refresh">
|
||
{{ loading ? '加载中…' : '刷新' }}
|
||
</button>
|
||
</div>
|
||
<div
|
||
v-if="status.text"
|
||
class="text-xs"
|
||
:class="{
|
||
'text-green-700': status.kind === 'ok',
|
||
'text-red-600': status.kind === 'err',
|
||
'text-gray-500': status.kind === 'idle',
|
||
}"
|
||
>
|
||
{{ status.text }}
|
||
</div>
|
||
<p class="text-xs text-gray-500">
|
||
默认 http://localhost:8483 — 需要在该地址先跑起 BiliNote 后端
|
||
</p>
|
||
</section>
|
||
|
||
<section class="section-card">
|
||
<h2 class="font-semibold">默认供应商与模型</h2>
|
||
<label class="flex flex-col gap-1 text-sm">
|
||
<span class="text-gray-600">供应商</span>
|
||
<select v-model="settings.providerId" class="input">
|
||
<option value="">— 选择供应商 —</option>
|
||
<option v-for="p in providers" :key="p.id" :value="p.id">
|
||
{{ p.name }} <span v-if="p.type === 'built-in'">(内置)</span>
|
||
</option>
|
||
</select>
|
||
</label>
|
||
<label class="flex flex-col gap-1 text-sm">
|
||
<span class="text-gray-600">模型</span>
|
||
<select v-model="settings.modelName" class="input" :disabled="!settings.providerId">
|
||
<option value="">— 选择模型 —</option>
|
||
<option v-for="m in models" :key="m.id" :value="m.model_name">{{ m.model_name }}</option>
|
||
</select>
|
||
<span v-if="settings.providerId && models.length === 0" class="text-xs text-amber-700">
|
||
该供应商还没添加可用模型,去「模型供应商」页编辑
|
||
</span>
|
||
</label>
|
||
</section>
|
||
|
||
<section class="section-card">
|
||
<h2 class="font-semibold">默认生成选项</h2>
|
||
<div class="grid grid-cols-2 gap-3 text-sm">
|
||
<label class="flex flex-col gap-1">
|
||
<span class="text-gray-600">画质</span>
|
||
<select v-model="settings.quality" class="input">
|
||
<option value="fast">快速 (32k)</option>
|
||
<option value="medium">中等 (64k)</option>
|
||
<option value="slow">高质 (128k)</option>
|
||
</select>
|
||
</label>
|
||
<label class="flex flex-col gap-1">
|
||
<span class="text-gray-600">笔记风格</span>
|
||
<select v-model="settings.style" class="input">
|
||
<option v-for="s in NOTE_STYLES" :key="s.value" :value="s.value">{{ s.label }}</option>
|
||
</select>
|
||
</label>
|
||
</div>
|
||
|
||
<div class="flex flex-col gap-1 text-sm">
|
||
<span class="text-gray-600">输出形式(与 web 端 NoteForm 对齐)</span>
|
||
<div class="flex flex-wrap gap-x-4 gap-y-2">
|
||
<label v-for="f in NOTE_FORMATS" :key="f.value" class="flex items-center gap-2">
|
||
<input
|
||
type="checkbox"
|
||
:checked="(settings.formats || []).includes(f.value)"
|
||
@change="toggleFormat(f.value, ($event.target as HTMLInputElement).checked)"
|
||
>
|
||
{{ f.label }}
|
||
</label>
|
||
</div>
|
||
</div>
|
||
|
||
<label class="flex flex-col gap-1 text-sm">
|
||
<span class="text-gray-600">额外提示词(追加到 prompt 末尾)</span>
|
||
<textarea
|
||
v-model="settings.extras"
|
||
class="input resize-y"
|
||
rows="3"
|
||
placeholder="例如:重点关注游戏开发部分;保留所有专业术语原文"
|
||
/>
|
||
</label>
|
||
</section>
|
||
</div>
|
||
</template>
|