初始化2

This commit is contained in:
2026-04-08 17:10:58 +08:00
commit 4986d90eb9
532 changed files with 112617 additions and 0 deletions

524
app/pages/developer/git.vue Normal file
View File

@@ -0,0 +1,524 @@
<template>
<div class="dev-page">
<div class="page-header">
<div>
<h2 class="page-title">🐙 Git 账号绑定</h2>
<p class="page-desc">绑定你的 Gitea 账号用于申请源码仓库访问权限</p>
</div>
</div>
<div class="page-body">
<!-- 绑定说明 -->
<div class="info-banner">
<div class="info-icon">💡</div>
<div class="info-content">
<div class="info-title">什么是 Gitea 账号绑定</div>
<div class="info-desc">
平台使用 Gitea 私有 Git 服务管理源码仓库绑定账号后运营人员可将你加入对应仓库的访问组
之后你即可通过 Git 克隆完整源代码进行本地开发与私有化部署
</div>
</div>
</div>
<a-row :gutter="[20, 20]">
<!-- 绑定表单 -->
<a-col :xs="24" :lg="14">
<div class="panel">
<div class="panel-header">
<span class="panel-title">📝 填写 Git 信息</span>
<div v-if="loading">
<a-spin size="small" />
</div>
<div v-else>
<a-tag v-if="gitAccountStatus" :color="getStatusColor(gitAccountStatus.status)">
{{ getStatusText(gitAccountStatus.status) }}
</a-tag>
<a-tag v-else-if="isSaved" color="green"> 已保存</a-tag>
</div>
</div>
<div class="panel-body">
<a-form layout="vertical" :model="form">
<a-form-item label="Gitea 用户名" required>
<a-input
v-model:value="form.username"
placeholder="例如lily"
:prefix-slot="true"
size="large"
>
<template #prefix>🐙</template>
</a-input>
<div class="form-hint">你在平台 Gitea 上注册的用户名非邮箱</div>
</a-form-item>
<a-form-item label="联系邮箱(可选)">
<a-input
v-model:value="form.email"
placeholder="用于接收审核通知"
size="large"
>
<template #prefix>📧</template>
</a-input>
</a-form-item>
<a-form-item label="备注(可选)">
<a-textarea
v-model:value="form.remark"
:rows="3"
placeholder="例如:公司名称、项目名称、联系方式等"
:maxlength="200"
show-count
/>
</a-form-item>
<a-form-item>
<a-space>
<a-button
type="primary"
size="large"
:loading="saving"
:disabled="!form.username.trim()"
@click="handleSave"
>
💾 保存绑定信息
</a-button>
<a-button
size="large"
type="default"
@click="navigateTo('/developer/requests')"
>
📋 查看申请记录
</a-button>
</a-space>
</a-form-item>
</a-form>
<!-- 状态信息 -->
<div v-if="gitAccountStatus && gitAccountStatus.status !== 'not_bound'" class="status-info">
<div class="status-info-header">📋 当前状态信息</div>
<div class="status-info-content">
<div v-if="gitAccountStatus.lastUpdatedAt" class="status-item">
<span class="status-label">上次更新</span>
<span class="status-value">{{ gitAccountStatus.lastUpdatedAt }}</span>
</div>
<div v-if="gitAccountStatus.verificationNote" class="status-item">
<span class="status-label">审核备注</span>
<span class="status-value">{{ gitAccountStatus.verificationNote }}</span>
</div>
<div v-if="gitAccountStatus.status === 'rejected'" class="status-warning">
绑定被拒绝请根据备注修改信息后重新提交
</div>
<div v-if="gitAccountStatus.status === 'verified'" class="status-success">
绑定已成功现在可以提交仓库访问申请了
</div>
</div>
<!-- 绑定成功后显示申请权限按钮 -->
<div v-if="gitAccountStatus.status === 'verified'" class="action-buttons">
<a-button
type="primary"
block
size="large"
@click="navigateTo('/developer/requests')"
>
🚀 前往申请仓库权限
</a-button>
</div>
</div>
</div>
</div>
</a-col>
<!-- 操作步骤说明 -->
<a-col :xs="24" :lg="10">
<div class="panel">
<div class="panel-header">
<span class="panel-title">📌 操作步骤</span>
</div>
<div class="steps-list">
<div v-for="(step, i) in howToSteps" :key="i" class="how-step">
<div class="how-step-num">{{ i + 1 }}</div>
<div class="how-step-text">
<div class="how-step-title">{{ step.title }}</div>
<div class="how-step-desc">{{ step.desc }}</div>
</div>
</div>
</div>
</div>
<!-- Gitea 注册入口 -->
<div class="panel mt-4">
<div class="panel-header">
<span class="panel-title">🚀 还没有 Gitea 账号</span>
</div>
<div class="register-hint">
<p class="register-desc">
前往平台 Gitea 注册账号注册完成后将用户名填入上方表单
</p>
<a-button type="primary" ghost block @click="openGitea">
前往 Gitea 注册
</a-button>
</div>
</div>
</a-col>
</a-row>
</div>
</div>
</template>
<script setup lang="ts">
import { message } from 'ant-design-vue'
import { saveGitAccount, getGitAccountStatus, getGiteaServerInfo } from '@/api/developer'
definePageMeta({ layout: 'developer' })
useHead({ title: 'Git 账号绑定 - 开发者中心' })
const saving = ref(false)
const loading = ref(false)
const isSaved = ref(false)
const gitAccountStatus = ref<any>(null)
const giteaInfo = ref<any>(null)
const form = reactive({
username: '',
email: '',
remark: '',
})
// 加载Git账号绑定状态
async function loadGitAccountStatus() {
loading.value = true
try {
const res = await getGitAccountStatus()
if (res.data.code === 200) {
gitAccountStatus.value = res.data.data
if (gitAccountStatus.value.status !== 'not_bound' && gitAccountStatus.value.username) {
form.username = gitAccountStatus.value.username
form.email = gitAccountStatus.value.email || ''
form.remark = gitAccountStatus.value.remark || ''
isSaved.value = true
}
}
} catch (error) {
console.error('加载Git账号状态失败:', error)
} finally {
loading.value = false
}
}
// 加载Gitea服务器信息
async function loadGiteaInfo() {
try {
const res = await getGiteaServerInfo()
if (res.data.code === 200) {
giteaInfo.value = res.data.data
}
} catch (error) {
console.error('加载Gitea服务器信息失败:', error)
}
}
async function handleSave() {
if (!form.username.trim()) {
message.error('请填写 Gitea 用户名')
return
}
saving.value = true
try {
const res = await saveGitAccount({
username: form.username.trim(),
email: form.email.trim() || undefined,
remark: form.remark.trim() || undefined
})
if (res.data.code === 200) {
isSaved.value = true
message.success(res.data.message || 'Git 账号绑定成功')
// 重新加载状态
await loadGitAccountStatus()
} else {
message.error(res.data.message || '保存失败,请稍后重试')
}
} catch (error: any) {
console.error('保存Git账号信息失败:', error)
if (error.response?.status === 400) {
message.error('用户名格式不正确')
} else if (error.response?.status === 409) {
message.error('该用户名已被其他用户绑定')
} else {
message.error('保存失败,请检查网络连接后重试')
}
} finally {
saving.value = false
}
}
function openGitea() {
if (import.meta.client) {
const url = giteaInfo.value?.url || 'https://git.websoft.top'
window.open(url, '_blank', 'noopener,noreferrer')
}
}
// 获取状态标签颜色
function getStatusColor(status: string) {
const colors: Record<string, string> = {
pending: 'orange',
verified: 'green',
rejected: 'red',
not_bound: 'default'
}
return colors[status] || 'default'
}
// 获取状态标签文本
function getStatusText(status: string) {
const texts: Record<string, string> = {
pending: '待审核',
verified: '已通过',
rejected: '已拒绝',
not_bound: '未绑定'
}
return texts[status] || status
}
const howToSteps = [
{
title: '注册 Gitea 账号',
desc: '访问平台 Gitea使用邮箱注册一个账号。',
},
{
title: '填写用户名并保存',
desc: '在左侧表单填写你的 Gitea 用户名,点击保存。',
},
{
title: '申请仓库访问权限',
desc: '绑定成功后,前往权限申请页面提交仓库访问申请。',
},
{
title: '获得仓库访问权限',
desc: '申请通过后,通过 Gitea 即可克隆对应仓库。',
},
]
// 页面初始化
onMounted(async () => {
await Promise.all([
loadGitAccountStatus(),
loadGiteaInfo()
])
})
</script>
<style scoped>
.dev-page { min-height: 100%; }
.page-header {
display: flex;
align-items: flex-start;
justify-content: space-between;
padding: 24px 28px 16px;
border-bottom: 1px solid #f0f0f0;
gap: 16px;
}
.page-title {
font-size: 20px;
font-weight: 700;
color: rgba(0, 0, 0, 0.88);
margin: 0 0 4px;
}
.page-desc {
font-size: 14px;
color: rgba(0, 0, 0, 0.45);
margin: 0;
}
.page-body {
padding: 20px 24px 28px;
}
/* 说明横幅 */
.info-banner {
display: flex;
gap: 14px;
padding: 16px 18px;
background: linear-gradient(135deg, #eff6ff 0%, #f0f9ff 100%);
border-radius: 12px;
border: 1px solid #bfdbfe;
margin-bottom: 20px;
}
.info-icon { font-size: 22px; flex-shrink: 0; }
.info-title {
font-size: 14px;
font-weight: 600;
color: #1d4ed8;
margin-bottom: 4px;
}
.info-desc {
font-size: 13px;
color: #3b82f6;
line-height: 1.6;
}
.panel {
background: #fff;
border: 1px solid #f0f0f0;
border-radius: 12px;
overflow: hidden;
}
.panel-header {
display: flex;
align-items: center;
justify-content: space-between;
padding: 14px 18px;
border-bottom: 1px solid #f5f5f5;
}
.panel-title {
font-size: 14px;
font-weight: 600;
color: rgba(0, 0, 0, 0.85);
}
.panel-body {
padding: 20px 20px 10px;
}
.form-hint {
font-size: 12px;
color: rgba(0, 0, 0, 0.38);
margin-top: 5px;
}
/* 步骤列表 */
.steps-list {
padding: 14px 18px;
}
.how-step {
display: flex;
gap: 12px;
padding: 10px 0;
border-bottom: 1px solid #f9f9f9;
}
.how-step:last-child { border-bottom: none; }
.how-step-num {
width: 24px;
height: 24px;
flex-shrink: 0;
border-radius: 50%;
background: linear-gradient(135deg, #4f46e5, #7c3aed);
color: #fff;
font-size: 12px;
font-weight: 700;
display: flex;
align-items: center;
justify-content: center;
margin-top: 1px;
}
.how-step-title {
font-size: 13px;
font-weight: 500;
color: rgba(0, 0, 0, 0.85);
margin-bottom: 3px;
}
.how-step-desc {
font-size: 12px;
color: rgba(0, 0, 0, 0.4);
line-height: 1.5;
}
/* Gitea 注册 */
.register-hint {
padding: 16px 18px;
}
.register-desc {
font-size: 13px;
color: rgba(0, 0, 0, 0.5);
margin: 0 0 14px;
line-height: 1.6;
}
/* 状态信息 */
.status-info {
margin-top: 20px;
padding: 14px;
background: #f9fafb;
border-radius: 8px;
border: 1px solid #e5e7eb;
}
.status-info-header {
font-size: 13px;
font-weight: 600;
color: rgba(0, 0, 0, 0.85);
margin-bottom: 10px;
padding-bottom: 8px;
border-bottom: 1px solid #e5e7eb;
}
.status-info-content {
display: flex;
flex-direction: column;
gap: 8px;
}
.status-item {
display: flex;
align-items: flex-start;
gap: 8px;
font-size: 12px;
}
.status-label {
color: rgba(0, 0, 0, 0.45);
min-width: 60px;
flex-shrink: 0;
}
.status-value {
color: rgba(0, 0, 0, 0.65);
flex: 1;
word-break: break-word;
}
.status-warning {
font-size: 12px;
color: #dc2626;
background: #fef2f2;
padding: 8px 12px;
border-radius: 6px;
margin-top: 8px;
border: 1px solid #fecaca;
}
.status-success {
font-size: 12px;
color: #16a34a;
background: #f0fdf4;
padding: 8px 12px;
border-radius: 6px;
margin-top: 8px;
border: 1px solid #bbf7d0;
}
/* 操作按钮区域 */
.action-buttons {
margin-top: 16px;
padding-top: 16px;
border-top: 1px solid #e5e7eb;
}
</style>