6 Commits

Author SHA1 Message Date
922e3d449f 更新 Dockerfile 中的 npm 安装命令,添加 --legacy-peer-deps 选项以解决依赖问题
All checks were successful
构建并运行 Docker / build (push) Successful in 1h22m28s
2025-07-31 20:23:41 +08:00
07b2c5e3a6 重构股票信息表单,替换复盘卡片相关逻辑,更新底部导航,移除不再使用的复盘卡片存储,优化 Dockerfile 中的 npm 安装命令
Some checks failed
构建并运行 Docker / build (push) Failing after 2m30s
2025-07-31 20:18:44 +08:00
1b41d10574 优化 Docker 构建流程,移除构建缓存选项
Some checks failed
构建并运行 Docker / build (push) Failing after 12m9s
2025-07-31 18:33:46 +08:00
51ef838add 添加 lucide-svelte 和 svelte-masonry 依赖,更新 svelte.config.js 以支持新组件路径,优化 Docker 构建流程,增加 BottomBar 组件到布局中
Some checks failed
构建并运行 Docker / build (push) Failing after 52s
2025-07-31 18:31:26 +08:00
f96e141b17 修复缓存
Some checks failed
构建并运行 Docker / build (push) Failing after 12m17s
2025-07-29 14:25:03 +08:00
cfdc5e5d9a 增加构建缓存
Some checks failed
构建并运行 Docker / build (push) Failing after 23s
2025-07-29 14:22:54 +08:00
13 changed files with 486 additions and 4 deletions

View File

@ -15,4 +15,7 @@ dist
.idea
*.swp
*.swo
*~
*~
bun.lock
.git
.gitea

View File

@ -9,7 +9,8 @@ WORKDIR /app
# 安装依赖
COPY package.json ./
RUN npm install
# 使用 npm 缓存来加速构建
RUN npm install --prefer-offline --no-audit --no-fund --production=false --legacy-peer-deps
# 构建阶段
FROM base AS builder

18
build-docker.sh Normal file
View File

@ -0,0 +1,18 @@
#!/bin/bash
# Gitea Actions 专用构建脚本
set -e
echo "🚀 开始构建 Docker 镜像..."
# 使用标准 docker build 命令
docker build -t watch-stock-list:latest .
echo "✅ 构建完成!"
# 启动容器
echo "🐳 启动容器..."
docker-compose down || true
docker-compose up -d --force-recreate
echo "🎉 部署完成!应用运行在 http://localhost:7799"

23
build.sh Normal file
View File

@ -0,0 +1,23 @@
#!/bin/bash
# 本地构建脚本 - 包含缓存优化
set -e
echo "🚀 开始构建 Docker 镜像..."
# 清理旧镜像(可选)
# docker system prune -f
# 使用 Buildx 构建
docker buildx build \
--tag watch-stock-list:latest \
--load \
.
echo "✅ 构建完成!"
# 启动容器(可选)
echo "🐳 启动容器..."
docker-compose up -d
echo "🎉 部署完成!应用运行在 http://localhost:7799"

View File

@ -5,6 +5,8 @@
"name": "watch-stock-list",
"dependencies": {
"@sveltejs/adapter-node": "^5.2.13",
"lucide-svelte": "^0.532.0",
"svelte-masonry": "^0.1.5",
},
"devDependencies": {
"@internationalized/date": "^3.8.1",
@ -450,6 +452,8 @@
"lodash.merge": ["lodash.merge@4.6.2", "", {}, "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ=="],
"lucide-svelte": ["lucide-svelte@0.532.0", "", { "peerDependencies": { "svelte": "^3 || ^4 || ^5.0.0-next.42" } }, "sha512-UMhuyPuzmJB0/2xwdhOW3NRK7vuVpqxIvHiH/bhQTu0vsvRw59YbLC65kS6fierPr7FckFV5Y/y/aUAVUZgGhA=="],
"magic-string": ["magic-string@0.30.17", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.0" } }, "sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA=="],
"memoize": ["memoize@10.1.0", "", { "dependencies": { "mimic-function": "^5.0.1" } }, "sha512-MMbFhJzh4Jlg/poq1si90XRlTZRDHVqdlz2mPyGJ6kqMpyHUyVpDd5gpFAvVehW64+RA1eKE9Yt8aSLY7w2Kgg=="],
@ -530,6 +534,8 @@
"svelte-check": ["svelte-check@4.3.0", "", { "dependencies": { "@jridgewell/trace-mapping": "^0.3.25", "chokidar": "^4.0.1", "fdir": "^6.2.0", "picocolors": "^1.0.0", "sade": "^1.7.4" }, "peerDependencies": { "svelte": "^4.0.0 || ^5.0.0-next.0", "typescript": ">=5.0.0" }, "bin": { "svelte-check": "bin/svelte-check" } }, "sha512-Iz8dFXzBNAM7XlEIsUjUGQhbEE+Pvv9odb9+0+ITTgFWZBGeJRRYqHUUglwe2EkLD5LIsQaAc4IUJyvtKuOO5w=="],
"svelte-masonry": ["svelte-masonry@0.1.5", "", { "peerDependencies": { "svelte": "^4.0.0" } }, "sha512-ZMtALHVbvriGcLODSdoeRaI0SvbmNEW6puG/Zs2tUP7hK0e/TLrxGTAAtNDinI/8DJhUJUCnAsP5xsTmebLwbw=="],
"svelte-sonner": ["svelte-sonner@1.0.5", "", { "dependencies": { "runed": "^0.28.0" }, "peerDependencies": { "svelte": "^5.0.0" } }, "sha512-9dpGPFqKb/QWudYqGnEz93vuY+NgCEvyNvxoCLMVGw6sDN/3oVeKV1xiEirW2E1N3vJEyj5imSBNOGltQHA7mg=="],
"svelte-toolbelt": ["svelte-toolbelt@0.9.3", "", { "dependencies": { "clsx": "^2.1.1", "runed": "^0.29.0", "style-to-object": "^1.0.8" }, "peerDependencies": { "svelte": "^5.30.2" } }, "sha512-HCSWxCtVmv+c6g1ACb8LTwHVbDqLKJvHpo6J8TaqwUme2hj9ATJCpjCPNISR1OCq2Q4U1KT41if9ON0isINQZw=="],

View File

@ -41,6 +41,8 @@
"vite": "^7.0.4"
},
"dependencies": {
"@sveltejs/adapter-node": "^5.2.13"
"@sveltejs/adapter-node": "^5.2.13",
"lucide-svelte": "^0.532.0",
"svelte-masonry": "^0.1.5"
}
}

View File

@ -0,0 +1,284 @@
<script lang="ts">
import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogDescription, DialogFooter } from '$lib/components/ui/dialog';
import { Button } from '$lib/components/ui/button';
import { Input } from '$lib/components/ui/input';
import { Label } from '$lib/components/ui/label';
import { Plus, X } from 'lucide-svelte';
import type { StockInfo } from '$lib/types/review';
export let open = false;
export let onClose: () => void;
export let onAdd: (stockInfo: StockInfo) => void;
// 默认标签
const defaultLabels = [
{ label: '买入价', key: 'buy' },
{ label: '目标价', key: 'target' },
{ label: '止损价', key: 'stop' }
];
let form: StockInfo = {
name: '',
code: '',
tags: [],
priceGroups: []
};
// 保证 price 始终为 number 类型
let addedPriceGroups: { label: string; price: number }[] = [];
let customLabelInput = '';
let customPriceInput: string = '';
let customLabelError = '';
let nameError = '';
let codeError = '';
let tagInput = '';
let tagInputRef: any = null;
function validate() {
nameError = form.name.trim() ? '' : '股票名称不能为空';
codeError = form.code.trim() ? '' : '股票代码不能为空';
customLabelError = '';
if (customLabelInput && customPriceInput === '') {
customLabelError = '请输入价格';
}
return !nameError && !codeError && !customLabelError;
}
function addTag() {
const tag = tagInput.trim();
if (tag && !form.tags.includes(tag)) {
form.tags = [...form.tags, tag];
}
tagInput = '';
if (tagInputRef && typeof tagInputRef.focus === 'function') tagInputRef.focus();
}
function removeTag(idx: number) {
form.tags = form.tags.filter((_, i) => i !== idx);
}
// 添加自带标签
function addDefaultLabel(label: string) {
if (!addedPriceGroups.some(pg => pg.label === label)) {
addedPriceGroups = [...addedPriceGroups, { label, price: 0 }];
}
}
// 添加自定义标签
function addCustomPriceGroup() {
const label = customLabelInput.trim();
const price = Number(customPriceInput);
if (!label) {
customLabelError = '请输入标签名';
return;
}
if (customPriceInput === '' || isNaN(price)) {
customLabelError = '请输入价格';
return;
}
if (
defaultLabels.some(l => l.label === label) ||
addedPriceGroups.some(pg => pg.label === label)
) {
customLabelError = '标签名已存在';
return;
}
addedPriceGroups = [...addedPriceGroups, { label, price }];
customLabelInput = '';
customPriceInput = '';
customLabelError = '';
}
// 删除标签
function removePriceGroup(idx: number) {
addedPriceGroups = addedPriceGroups.filter((_, i) => i !== idx);
}
function resetForm() {
form = {
name: '',
code: '',
tags: [],
priceGroups: []
};
addedPriceGroups = [];
customLabelInput = '';
customPriceInput = '';
customLabelError = '';
nameError = '';
codeError = '';
tagInput = '';
}
function handleSubmit(e: Event) {
e.preventDefault();
if (!validate()) return;
onAdd({
name: form.name,
code: form.code,
tags: [...form.tags],
priceGroups: addedPriceGroups.filter(pg => !isNaN(pg.price))
});
resetForm();
onClose();
}
function handleCancel() {
resetForm();
onClose();
}
function handleTagKeydown(e: KeyboardEvent) {
if (e.key === 'Enter' || e.key === ',') {
e.preventDefault();
addTag();
}
}
$: if (open) {
setTimeout(() => {
document.getElementById('stock-name')?.focus();
}, 100);
}
</script>
<Dialog bind:open>
<DialogContent class="max-w-xl w-full rounded-md p-0 bg-background shadow-2xl">
<DialogHeader class="px-8 pt-8 pb-3">
<DialogTitle class="text-2xl font-bold tracking-tight mb-1">添加股票信息</DialogTitle>
<DialogDescription class="text-base text-muted-foreground mt-1 mb-2">
填写股票基本信息和价格标签(可选)。
</DialogDescription>
</DialogHeader>
<form class="space-y-5 px-8 pb-8 pt-2 font-sans" onsubmit={handleSubmit}>
<!-- 股票信息 -->
<div>
<div class="text-lg font-semibold mb-3 tracking-tight">股票信息</div>
<div class="grid grid-cols-1 sm:grid-cols-2 gap-4">
<div>
<Label for="stock-name" class="mb-1 block text-sm text-muted-foreground font-medium">名称</Label>
<Input
id="stock-name"
bind:value={form.name}
placeholder="股票名称"
class="h-10 w-full rounded-md px-4 text-sm placeholder:text-muted-foreground/70 focus-visible:ring-2 focus-visible:ring-primary focus-visible:border-primary border border-input"
onblur={() => (nameError = form.name.trim() ? '' : '股票名称不能为空')}
autocomplete="off"
/>
{#if nameError}
<span class="text-xs text-red-500 mt-1 block">{nameError}</span>
{/if}
</div>
<div>
<Label for="stock-code" class="mb-1 block text-sm text-muted-foreground font-medium">代码</Label>
<Input
id="stock-code"
bind:value={form.code}
placeholder="股票代码"
class="h-10 w-full rounded-md px-4 text-sm placeholder:text-muted-foreground/70 focus-visible:ring-2 focus-visible:ring-primary focus-visible:border-primary border border-input"
onblur={() => (codeError = form.code.trim() ? '' : '股票代码不能为空')}
autocomplete="off"
/>
{#if codeError}
<span class="text-xs text-red-500 mt-1 block">{codeError}</span>
{/if}
</div>
</div>
<div class="mt-4">
<Label for="stock-tags" class="mb-1 block text-sm text-muted-foreground font-medium">股票标签</Label>
<div class="flex flex-wrap gap-2 items-center mb-1">
{#each form.tags as tag, i}
<span class="inline-flex items-center px-2 py-0.5 bg-muted rounded-md shadow-sm text-sm border border-muted-foreground/10 transition hover:bg-primary/20 hover:text-primary-foreground cursor-pointer">
{tag}
<button type="button" class="ml-1 hover:text-destructive focus:outline-none" tabindex="-1" onclick={() => removeTag(i)}>
<X class="w-3 h-3" />
</button>
</span>
{/each}
</div>
<div class="flex items-center gap-2">
<Input
id="stock-tags"
bind:this={tagInputRef}
class="w-40 h-8 text-sm px-2 rounded-md focus-visible:ring-2 focus-visible:ring-primary border border-input"
placeholder="输入标签后回车"
bind:value={tagInput}
onkeydown={handleTagKeydown}
onblur={addTag}
autocomplete="off"
/>
<span class="text-xs text-muted-foreground">按回车或逗号添加标签</span>
</div>
</div>
</div>
<div class="border-t border-muted-foreground/10 my-3"></div>
<!-- 价格标签(可选) -->
<div>
<div class="text-lg font-semibold mb-2 tracking-tight">价格标签 <span class="text-xs text-muted-foreground">(可选)</span></div>
<div class="flex flex-wrap gap-2 mb-2 items-center">
{#each defaultLabels as l}
<Button
type="button"
size="sm"
variant={addedPriceGroups.some(pg => pg.label === l.label) ? 'secondary' : 'outline'}
class="rounded-md hover:bg-primary/10 hover:text-primary-foreground transition-colors font-medium h-8 px-3"
disabled={addedPriceGroups.some(pg => pg.label === l.label)}
onclick={() => addDefaultLabel(l.label)}
>
{#if addedPriceGroups.some(pg => pg.label === l.label)}
已添加
{:else}
<Plus class="w-4 h-4 mr-1" />{l.label}
{/if}
</Button>
{/each}
</div>
<div class="space-y-2">
{#each addedPriceGroups as pg, i}
<div class="flex items-center gap-3">
<span class="inline-flex items-center h-8 px-3 bg-muted rounded-md shadow-sm text-sm font-medium leading-none border border-muted-foreground/10 transition hover:bg-primary/20 hover:text-primary-foreground cursor-pointer">
{pg.label}
<button type="button" class="ml-1 hover:text-destructive focus:outline-none" tabindex="-1" onclick={() => removePriceGroup(i)}>
<X class="w-3 h-3" />
</button>
</span>
<Input
type="number"
min="0"
placeholder="请输入价格"
class="w-32 h-8 text-sm px-2 rounded-md focus-visible:ring-2 focus-visible:ring-primary border border-input"
value={pg.price === 0 ? '' : pg.price}
oninput={e => pg.price = Number((e.target as HTMLInputElement).value)}
autocomplete="off"
/>
</div>
{/each}
</div>
<!-- 自定义标签输入 -->
<div class="flex items-center gap-2 mt-2">
<Input
class="w-32 h-8 text-sm px-2 rounded-md focus-visible:ring-2 focus-visible:ring-primary border border-input"
placeholder="自定义标签名"
bind:value={customLabelInput}
autocomplete="off"
/>
<Input
type="number"
min="0"
class="w-32 h-8 text-sm px-2 rounded-md focus-visible:ring-2 focus-visible:ring-primary border border-input"
placeholder="请输入价格"
bind:value={customPriceInput}
autocomplete="off"
/>
<Button type="button" size="sm" class="h-8 rounded-md font-medium" onclick={addCustomPriceGroup}>
<Plus class="h-4 w-4" /> 添加
</Button>
</div>
{#if customLabelError}
<span class="text-xs text-red-500 mt-1 block">{customLabelError}</span>
{/if}
</div>
<DialogFooter class="flex flex-row justify-end gap-4 pt-6 pb-2 px-0">
<Button variant="outline" type="button" class="min-w-[90px] h-10 rounded-md font-medium" onclick={handleCancel}>取消</Button>
<Button
type="submit"
class="min-w-[120px] h-10 rounded-md text-base font-semibold"
disabled={!validate()}
>
添加股票
</Button>
</DialogFooter>
</form>
</DialogContent>
</Dialog>

View File

@ -0,0 +1,46 @@
<script lang="ts">
import { Button } from 'ui/button';
import { Home, Plus, Search, Settings, Minus } from 'lucide-svelte';
import { page } from '$app/stores';
import { get } from 'svelte/store';
import { addStockStore } from '$lib/stores/StockStore';
// 路由配置
const navs = [
{ label: '首页', icon: Home, href: '/' },
{ label: '搜索', icon: Search, href: '/search' },
{ label: '设置', icon: Settings, href: '/settings' },
{ label: '股票', icon: Minus, href: '/list' },
];
</script>
<div class="fixed bottom-4 left-1/2 -translate-x-1/2 w-[95vw] max-w-md mx-auto rounded-2xl bg-background/95 shadow-2xl border border-border/50 flex items-center gap-0 px-0 h-16 z-50 backdrop-blur-md supports-[backdrop-filter]:bg-background/80">
{#each navs as nav, i}
<div class="flex flex-1 h-full items-center justify-center">
<Button
variant="ghost"
size="sm"
href={nav.href}
class="group flex flex-col items-center justify-center gap-0.5 w-full h-full rounded-lg border border-transparent transition-all duration-200
{get(page).url.pathname === nav.href ? ' text-primary bg-accent/70 shadow-md border-primary/40 ring-2 ring-primary/20' : ' text-muted-foreground hover:bg-accent/60 hover:shadow-md hover:border-primary/40 hover:ring-2 hover:ring-primary/20 focus-visible:ring-2 focus-visible:ring-primary/30'}"
aria-label={nav.label}
>
<svelte:component this={nav.icon} class="w-4 h-4 transition-all duration-200 group-hover:-translate-y-1 group-hover:scale-110 group-hover:text-primary group-hover:drop-shadow {get(page).url.pathname === nav.href ? 'text-primary' : ''}" />
<span class="text-xs font-medium transition-all duration-200 group-hover:text-primary group-hover:scale-105 {get(page).url.pathname === nav.href ? 'text-primary' : ''}">{nav.label}</span>
</Button>
</div>
{#if i === 1}
<div class="flex flex-1 h-full items-center justify-center relative">
<Button
variant="default"
size="icon"
class="rounded-full w-16 h-16 shadow-2xl border-4 border-background bg-gradient-to-br from-primary via-primary to-primary/90 flex items-center justify-center transition-all duration-300 hover:scale-110 hover:shadow-[0_20px_40px_rgba(0,0,0,0.3)] active:scale-95 focus-visible:ring-4 focus-visible:ring-primary/20 absolute left-1/2 -translate-x-1/2 -top-6 z-10"
aria-label="添加"
onclick={() => addStockStore.set(true)}
>
<Plus class="w-8 h-8 text-white transition-transform duration-200 group-hover:rotate-90" />
</Button>
</div>
{/if}
{/each}
</div>

View File

@ -0,0 +1,16 @@
import { writable } from 'svelte/store';
import type { StockInfo } from '$lib/types/review';
function createStockStore() {
const { subscribe, set, update } = writable<StockInfo[]>([]);
return {
subscribe,
set,
reset: () => set([]),
add: (stock: StockInfo) => update(list => [...list, stock]),
remove: (idx: number) => update(list => list.filter((_, i) => i !== idx)),
};
}
export const stockStore = createStockStore();
export const addStockStore = writable(false);

13
src/lib/types/review.ts Normal file
View File

@ -0,0 +1,13 @@
// 股票信息类型和自定义价格标签类型
export type CustomPriceGroup = {
label: string; // 价格标签名
price: number; // 价格
};
export interface StockInfo {
name: string; // 股票名称
code: string; // 股票代码
tags: string[]; // 股票标签
priceGroups: CustomPriceGroup[]; // 价格标签+价格数组
}

View File

@ -1,7 +1,9 @@
<script lang="ts">
import '../app.css';
import BottomBar from '../lib/components/layout/BottomBar.svelte';
let { children } = $props();
</script>
{@render children()}
<BottomBar />

View File

@ -0,0 +1,67 @@
<script lang="ts">
// @ts-expect-error: svelte-masonry 没有类型声明
import Masonry from 'svelte-masonry';
import AddReviewCardForm from '$lib/components/AddReviewCardForm.svelte';
import BottomBar from '$lib/components/layout/BottomBar.svelte';
import { stockStore, addStockStore } from '$lib/stores/StockStore';
import type { StockInfo } from '$lib/types/review';
function handleAdd(stock: StockInfo) {
stockStore.add(stock);
}
</script>
<div class="mx-auto p-4">
<h1 class="text-2xl font-bold mb-6">股票列表</h1>
<button class="mb-4 px-4 py-2 bg-primary text-white rounded" on:click={() => addStockStore.set(true)}>添加股票</button>
<Masonry columns="2" gap="16">
{#each $stockStore as stock, idx (stock.name + stock.code)}
<div class="bg-white dark:bg-zinc-900 rounded-xl shadow-md overflow-hidden mb-4 border border-border">
<div class="p-3">
<div class="font-semibold text-base mb-1">{stock.name}{stock.code}</div>
<div class="mt-2 space-y-1">
{#each stock.priceGroups as group}
<div class="flex items-center gap-2">
<span class="inline-block px-2 py-0.5 bg-gray-100 rounded text-xs">{group.label}</span>
<span class="text-sm">{group.price}</span>
</div>
{/each}
</div>
<button class="mt-2 text-xs text-red-500" on:click={() => stockStore.remove(idx)}>删除</button>
</div>
</div>
{/each}
</Masonry>
</div>
<AddReviewCardForm open={$addStockStore} onAdd={handleAdd} onClose={() => addStockStore.set(false)} />
<BottomBar />
<style>
:global(html) {
scrollbar-width: thin;
scrollbar-color: var(--tw-prose-invert) var(--background);
}
:global(*::-webkit-scrollbar) {
height: 8px;
width: 8px;
background: transparent;
}
:global(*::-webkit-scrollbar-thumb) {
background: rgba(120,120,120,0.18);
border-radius: 8px;
transition: background 0.2s;
}
:global(*:hover::-webkit-scrollbar-thumb) {
background: rgba(120,120,120,0.32);
}
:global(*::-webkit-scrollbar-track) {
background: transparent;
}
:global(.masonry) {
width: 100% !important;
margin-left: 0 !important;
margin-right: 0 !important;
}
</style>

View File

@ -13,7 +13,8 @@ const config = {
out: 'build'
}),
alias: {
'@/*': './src/*'
'@/*': './src/*',
'ui/*': './src/lib/components/ui/*'
}
}
};