重构为 HTTP SSO 扫码方案并引入 Vue3 前端
移除 Playwright 浏览器自动化,改用 passport/SSO HTTP 接口获取二维码与轮询登录;后端模块化拆分,前端替换为 Vue3 SPA。 Co-authored-by: Cursor <cursoragent@cursor.com>
This commit is contained in:
@@ -0,0 +1,349 @@
|
||||
<template>
|
||||
<div class="layout-cloud" :class="{ 'is-nav-open': sidebarOpen }">
|
||||
<div class="nav-backdrop" aria-hidden="true" @click="sidebarOpen = false" />
|
||||
|
||||
<aside class="sidebar" role="navigation" :aria-hidden="desktopNav ? undefined : !sidebarOpen">
|
||||
<div class="sidebar-brand">
|
||||
<button type="button" class="sidebar-close" aria-label="关闭菜单" @click="sidebarOpen = false">
|
||||
×
|
||||
</button>
|
||||
<div class="brand-title">抖音 Cookie 提取</div>
|
||||
<div class="brand-sub">扫码登录 · JSON 导出</div>
|
||||
</div>
|
||||
|
||||
<nav class="sidebar-nav">
|
||||
<router-link
|
||||
v-for="item in navItems"
|
||||
:key="item.name"
|
||||
:to="{ name: item.name }"
|
||||
class="nav-item"
|
||||
exact-active-class="nav-item-active"
|
||||
@click="onNavClick"
|
||||
>
|
||||
{{ item.label }}
|
||||
</router-link>
|
||||
</nav>
|
||||
|
||||
<div class="sidebar-footer">
|
||||
<button type="button" class="btn btn-soft full-width" @click="refreshAll">刷新状态</button>
|
||||
</div>
|
||||
</aside>
|
||||
|
||||
<div class="main-wrap">
|
||||
<header class="mobile-topbar">
|
||||
<button
|
||||
type="button"
|
||||
class="nav-toggle"
|
||||
aria-label="打开菜单"
|
||||
:aria-expanded="sidebarOpen"
|
||||
@click="sidebarOpen = true"
|
||||
>
|
||||
<span class="nav-toggle-bar"></span>
|
||||
<span class="nav-toggle-bar"></span>
|
||||
<span class="nav-toggle-bar"></span>
|
||||
</button>
|
||||
<span class="mobile-title">Cookie 提取</span>
|
||||
</header>
|
||||
<main class="main-inner">
|
||||
<router-view v-slot="{ Component }">
|
||||
<transition name="fade" mode="out-in">
|
||||
<component :is="Component" />
|
||||
</transition>
|
||||
</router-view>
|
||||
</main>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, watch, onMounted, onUnmounted } from 'vue'
|
||||
import { useRoute } from 'vue-router'
|
||||
|
||||
const route = useRoute()
|
||||
const sidebarOpen = ref(false)
|
||||
const desktopNav = ref(true)
|
||||
|
||||
const navItems = [
|
||||
{ name: 'home', label: '扫码提取' },
|
||||
{ name: 'debug', label: 'SSO 调试' },
|
||||
]
|
||||
|
||||
let mq
|
||||
function syncMq() {
|
||||
desktopNav.value = typeof window !== 'undefined' && window.matchMedia('(min-width: 900px)').matches
|
||||
if (desktopNav.value) sidebarOpen.value = false
|
||||
}
|
||||
|
||||
function refreshAll() {
|
||||
window.dispatchEvent(new CustomEvent('douyin-refresh-all'))
|
||||
}
|
||||
|
||||
function onNavClick() {
|
||||
if (!desktopNav.value) sidebarOpen.value = false
|
||||
}
|
||||
|
||||
function onKeydown(e) {
|
||||
if (e.key === 'Escape' && sidebarOpen.value) sidebarOpen.value = false
|
||||
}
|
||||
|
||||
watch(
|
||||
() => route.fullPath,
|
||||
() => {
|
||||
if (!desktopNav.value) sidebarOpen.value = false
|
||||
},
|
||||
)
|
||||
|
||||
watch(sidebarOpen, (open) => {
|
||||
if (typeof document === 'undefined') return
|
||||
document.body.style.overflow = open && !desktopNav.value ? 'hidden' : ''
|
||||
})
|
||||
|
||||
onMounted(() => {
|
||||
syncMq()
|
||||
mq = window.matchMedia('(min-width: 900px)')
|
||||
mq.addEventListener('change', syncMq)
|
||||
window.addEventListener('keydown', onKeydown)
|
||||
})
|
||||
onUnmounted(() => {
|
||||
if (mq) mq.removeEventListener('change', syncMq)
|
||||
window.removeEventListener('keydown', onKeydown)
|
||||
document.body.style.overflow = ''
|
||||
})
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.layout-cloud {
|
||||
display: flex;
|
||||
min-height: 100vh;
|
||||
min-height: 100dvh;
|
||||
height: 100vh;
|
||||
height: 100dvh;
|
||||
max-height: 100vh;
|
||||
max-height: 100dvh;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.nav-backdrop {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.sidebar {
|
||||
width: var(--sidebar-width);
|
||||
flex-shrink: 0;
|
||||
align-self: stretch;
|
||||
background: var(--sidebar-bg);
|
||||
border-right: 1px solid var(--border-light);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
box-shadow: 1px 0 0 rgba(0, 0, 0, 0.04);
|
||||
min-height: 0;
|
||||
}
|
||||
|
||||
.sidebar-brand {
|
||||
position: relative;
|
||||
flex-shrink: 0;
|
||||
padding: 1rem 0.75rem 0.85rem;
|
||||
border-bottom: 1px solid var(--border-light);
|
||||
}
|
||||
|
||||
.sidebar-close {
|
||||
display: none;
|
||||
position: absolute;
|
||||
top: 0.75rem;
|
||||
right: 0.65rem;
|
||||
width: var(--tap-min, 44px);
|
||||
height: var(--tap-min, 44px);
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
border: none;
|
||||
background: #f5f7fa;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1.5rem;
|
||||
line-height: 1;
|
||||
color: var(--muted);
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.brand-title {
|
||||
font-size: 1.05rem;
|
||||
font-weight: 700;
|
||||
color: var(--text-strong);
|
||||
letter-spacing: -0.02em;
|
||||
padding-right: 2.5rem;
|
||||
}
|
||||
|
||||
.brand-sub {
|
||||
font-size: 0.75rem;
|
||||
color: var(--muted);
|
||||
margin-top: 0.25rem;
|
||||
}
|
||||
|
||||
.sidebar-nav {
|
||||
flex: 1;
|
||||
min-height: 0;
|
||||
padding: 0.55rem 0.5rem;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 0.35rem;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
.nav-item {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
min-height: var(--tap-min, 44px);
|
||||
padding: 0.45rem 0.6rem;
|
||||
border-radius: var(--radius-sm);
|
||||
color: var(--text-strong);
|
||||
font-size: 0.875rem;
|
||||
border: 1px solid #e8e8e8;
|
||||
background: #ffffff;
|
||||
text-decoration: none;
|
||||
transition: background 0.15s, color 0.15s, border-color 0.15s;
|
||||
}
|
||||
|
||||
.nav-item:hover {
|
||||
background: #f7f8fa;
|
||||
border-color: #dcdfe6;
|
||||
}
|
||||
|
||||
.nav-item-active {
|
||||
background: #eef5fe !important;
|
||||
color: #3b82f6 !important;
|
||||
font-weight: 600;
|
||||
border-color: #bfdbfe !important;
|
||||
}
|
||||
|
||||
.sidebar-footer {
|
||||
flex-shrink: 0;
|
||||
padding: 0.55rem 0.5rem;
|
||||
border-top: 1px solid var(--border-light);
|
||||
padding-bottom: max(0.55rem, env(safe-area-inset-bottom));
|
||||
}
|
||||
|
||||
.full-width {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.sidebar-footer .btn-soft {
|
||||
background: #f5f5f5;
|
||||
border-color: #e8e8e8;
|
||||
color: #303133;
|
||||
}
|
||||
|
||||
.mobile-topbar {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.main-wrap {
|
||||
flex: 1;
|
||||
min-width: 0;
|
||||
min-height: 0;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.main-inner {
|
||||
flex: 1;
|
||||
width: 100%;
|
||||
min-width: 0;
|
||||
min-height: 0;
|
||||
box-sizing: border-box;
|
||||
padding-top: var(--main-pad-y);
|
||||
padding-bottom: max(var(--main-pad-y), env(safe-area-inset-bottom));
|
||||
padding-left: var(--main-pad-start);
|
||||
padding-right: var(--main-pad-end);
|
||||
overflow: auto;
|
||||
}
|
||||
|
||||
.fade-enter-active,
|
||||
.fade-leave-active {
|
||||
transition: opacity 0.12s ease;
|
||||
}
|
||||
.fade-enter-from,
|
||||
.fade-leave-to {
|
||||
opacity: 0;
|
||||
}
|
||||
|
||||
@media (max-width: 899px) {
|
||||
.sidebar-close {
|
||||
display: block;
|
||||
}
|
||||
|
||||
.nav-backdrop {
|
||||
display: block;
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
z-index: 250;
|
||||
background: rgba(0, 0, 0, 0.45);
|
||||
opacity: 0;
|
||||
visibility: hidden;
|
||||
pointer-events: none;
|
||||
transition: opacity 0.2s ease, visibility 0.2s;
|
||||
}
|
||||
|
||||
.layout-cloud.is-nav-open .nav-backdrop {
|
||||
opacity: 1;
|
||||
visibility: visible;
|
||||
pointer-events: auto;
|
||||
}
|
||||
|
||||
.sidebar {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
bottom: 0;
|
||||
width: min(260px, 86vw);
|
||||
z-index: 300;
|
||||
transform: translateX(-100%);
|
||||
transition: transform 0.22s ease;
|
||||
}
|
||||
|
||||
.layout-cloud.is-nav-open .sidebar {
|
||||
transform: translateX(0);
|
||||
box-shadow: 4px 0 24px rgba(0, 0, 0, 0.12);
|
||||
}
|
||||
|
||||
.mobile-topbar {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 0.75rem;
|
||||
flex-shrink: 0;
|
||||
padding: 0.5rem var(--main-pad-end, var(--main-pad-x));
|
||||
padding-top: max(0.5rem, env(safe-area-inset-top));
|
||||
background: var(--surface);
|
||||
border-bottom: 1px solid var(--border-light);
|
||||
position: sticky;
|
||||
top: 0;
|
||||
z-index: 100;
|
||||
}
|
||||
|
||||
.nav-toggle {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
gap: 5px;
|
||||
width: var(--tap-min, 44px);
|
||||
height: var(--tap-min, 44px);
|
||||
padding: 0 10px;
|
||||
border: 1px solid var(--border);
|
||||
border-radius: var(--radius-sm);
|
||||
background: var(--surface);
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.nav-toggle-bar {
|
||||
display: block;
|
||||
height: 2px;
|
||||
background: var(--text-strong);
|
||||
border-radius: 1px;
|
||||
}
|
||||
|
||||
.mobile-title {
|
||||
font-weight: 600;
|
||||
font-size: 0.95rem;
|
||||
color: var(--text-strong);
|
||||
}
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user