fix updating ui

This commit is contained in:
Julian Freeman
2026-03-14 21:47:04 -04:00
parent 1552f1cad8
commit 1c4587e945
5 changed files with 103 additions and 51 deletions

View File

@@ -3,7 +3,8 @@
class="software-card" class="software-card"
:class="{ :class="{
'installed-mode': software.status === 'installed', 'installed-mode': software.status === 'installed',
'is-selected': isSelected && software.status !== 'installed' 'is-selected': isSelected && software.status !== 'installed',
'is-busy': software.status === 'pending' || software.status === 'installing'
}" }"
@click="handleCardClick" @click="handleCardClick"
> >
@@ -13,7 +14,7 @@
class="checkbox" class="checkbox"
:class="{ :class="{
'checked': isSelected, 'checked': isSelected,
'disabled': software.status === 'installed' 'disabled': software.status === 'installed' || software.status === 'pending' || software.status === 'installing'
}" }"
> >
<span v-if="isSelected"></span> <span v-if="isSelected"></span>
@@ -60,15 +61,16 @@
已安装 已安装
</button> </button>
<div v-else-if="software.status === 'installing' || software.status === 'pending'" class="progress-status"> <!-- 等待中状态 -->
<div class="progress-ring"> <div v-else-if="software.status === 'pending'" class="status-pending">
<svg viewBox="0 0 32 32"> <span class="wait-icon"></span>
<circle class="bg" cx="16" cy="16" r="14" fill="none" stroke-width="4" /> <span class="wait-text">等待中</span>
<circle class="fg" cx="16" cy="16" r="14" fill="none" stroke-width="4"
:style="{ strokeDasharray: 88, strokeDashoffset: 88 - (88 * (software.progress || 0.5)) }"
/>
</svg>
</div> </div>
<!-- 安装中状态旋转 Loading -->
<div v-else-if="software.status === 'installing'" class="progress-status">
<div class="spinning-loader"></div>
<span class="loading-text">正在安装</span>
</div> </div>
<div v-else-if="software.status === 'success'" class="status-success"> <div v-else-if="software.status === 'success'" class="status-success">
@@ -114,6 +116,9 @@ const placeholderColor = computed(() => {
}); });
const handleCardClick = () => { const handleCardClick = () => {
// 安装或等待中禁止修改勾选
if (props.software.status === 'pending' || props.software.status === 'installing') return;
if (props.selectable && props.software.status !== 'installed') { if (props.selectable && props.software.status !== 'installed') {
emit('toggleSelect', props.software.id); emit('toggleSelect', props.software.id);
} }
@@ -135,7 +140,7 @@ const handleCardClick = () => {
cursor: default; cursor: default;
} }
.software-card:not(.installed-mode):hover { .software-card:not(.installed-mode):not(.is-busy):hover {
border-color: rgba(0, 122, 255, 0.3); border-color: rgba(0, 122, 255, 0.3);
box-shadow: 0 8px 24px rgba(0, 0, 0, 0.06); box-shadow: 0 8px 24px rgba(0, 0, 0, 0.06);
} }
@@ -145,6 +150,10 @@ const handleCardClick = () => {
border-color: rgba(0, 122, 255, 0.4); border-color: rgba(0, 122, 255, 0.4);
} }
.software-card.is-busy {
opacity: 0.8;
}
/* 勾选框样式 */ /* 勾选框样式 */
.selection-area { .selection-area {
margin-right: 16px; margin-right: 16px;
@@ -302,31 +311,46 @@ const handleCardClick = () => {
cursor: not-allowed; cursor: not-allowed;
} }
.status-pending {
display: flex;
align-items: center;
gap: 6px;
width: 90px;
justify-content: center;
color: #AEAEB2;
}
.wait-icon {
font-size: 14px;
}
.wait-text {
font-size: 12px;
font-weight: 600;
}
.progress-status { .progress-status {
display: flex; display: flex;
align-items: center; align-items: center;
gap: 10px; flex-direction: column;
gap: 4px;
width: 90px; width: 90px;
justify-content: center; justify-content: center;
} }
.progress-ring { .spinning-loader {
width: 18px; width: 18px;
height: 18px; height: 18px;
border: 2px solid rgba(0, 122, 255, 0.1);
border-top-color: var(--primary-color);
border-radius: 50%;
animation: spin 1s linear infinite;
} }
.progress-ring svg { .loading-text {
transform: rotate(-90deg); font-size: 10px;
} font-weight: 700;
color: var(--primary-color);
.progress-ring .bg {
stroke: var(--border-color);
}
.progress-ring .fg {
stroke: var(--primary-color);
stroke-linecap: round;
transition: stroke-dashoffset 0.3s ease;
} }
.status-success, .status-error { .status-success, .status-error {
@@ -343,4 +367,8 @@ const handleCardClick = () => {
.status-error { .status-error {
color: #FF3B30; color: #FF3B30;
} }
@keyframes spin {
to { transform: rotate(360deg); }
}
</style> </style>

View File

@@ -46,24 +46,27 @@ export const useSoftwareStore = defineStore('software', {
}); });
}, },
sortedUpdates: (state) => [...state.updates].sort(sortByName), sortedUpdates: (state) => [...state.updates].sort(sortByName),
sortedAllSoftware: (state) => [...state.allSoftware].sort(sortByName) sortedAllSoftware: (state) => [...state.allSoftware].sort(sortByName),
// 全局繁忙状态:只要有任何软件在等待或安装中,就锁定关键操作
isBusy: (state) => {
const allItems = [...state.essentials, ...state.updates, ...state.allSoftware];
return allItems.some(item => item.status === 'pending' || item.status === 'installing');
}
}, },
actions: { actions: {
async initializeApp() { async initializeApp() {
if (this.isInitialized) return; if (this.isInitialized) return;
this.initStatus = '正在同步 Winget 模块...'; this.initStatus = '正在同步 Winget 模块...';
try { try {
await invoke('initialize_app'); await invoke('initialize_app');
this.isInitialized = true; this.isInitialized = true;
} catch (err) { } catch (err) {
this.initStatus = '环境配置失败,请检查运行日志'; this.initStatus = '环境配置失败,请检查运行日志';
console.error('Init failed:', err);
// 即使失败也允许进入,让用户看日志
setTimeout(() => { this.isInitialized = true; }, 2000); setTimeout(() => { this.isInitialized = true; }, 2000);
} }
}, },
toggleSelection(id: string, type: 'essential' | 'update') { toggleSelection(id: string, type: 'essential' | 'update') {
if (this.isBusy) return; // 繁忙时禁止修改勾选
const list = type === 'essential' ? this.selectedEssentialIds : this.selectedUpdateIds; const list = type === 'essential' ? this.selectedEssentialIds : this.selectedUpdateIds;
const index = list.indexOf(id); const index = list.indexOf(id);
if (index === -1) list.push(id); if (index === -1) list.push(id);
@@ -95,6 +98,7 @@ export const useSoftwareStore = defineStore('software', {
this.essentials = await invoke('get_essentials') this.essentials = await invoke('get_essentials')
}, },
async fetchUpdates() { async fetchUpdates() {
if (this.isBusy) return;
this.loading = true this.loading = true
try { try {
const res = await invoke('get_updates') const res = await invoke('get_updates')
@@ -105,6 +109,7 @@ export const useSoftwareStore = defineStore('software', {
} }
}, },
async fetchAll() { async fetchAll() {
if (this.isBusy) return;
this.loading = true this.loading = true
try { try {
const res = await invoke('get_all_software') const res = await invoke('get_all_software')
@@ -114,6 +119,7 @@ export const useSoftwareStore = defineStore('software', {
} }
}, },
async syncDataIfNeeded(force = false) { async syncDataIfNeeded(force = false) {
if (this.isBusy) return;
const now = Date.now(); const now = Date.now();
const CACHE_TIMEOUT = 5 * 60 * 1000; const CACHE_TIMEOUT = 5 * 60 * 1000;
if (!force && this.allSoftware.length > 0 && (now - this.lastFetched < CACHE_TIMEOUT)) { if (!force && this.allSoftware.length > 0 && (now - this.lastFetched < CACHE_TIMEOUT)) {
@@ -140,8 +146,12 @@ export const useSoftwareStore = defineStore('software', {
} }
}, },
async install(id: string) { async install(id: string) {
// 这里的 logic 同时负责单个安装和批量安装的任务入队
const software = this.findSoftware(id) const software = this.findSoftware(id)
if (software) software.status = 'pending' if (software) {
// 进入队列前统一标记为等待中
software.status = 'pending';
}
await invoke('install_software', { id }) await invoke('install_software', { id })
}, },
findSoftware(id: string) { findSoftware(id: string) {

View File

@@ -9,7 +9,7 @@
<button <button
@click="store.fetchAll" @click="store.fetchAll"
class="secondary-btn action-btn" class="secondary-btn action-btn"
:disabled="store.loading" :disabled="store.loading || store.isBusy"
> >
<span class="icon" :class="{ 'spinning': store.loading }"> <span class="icon" :class="{ 'spinning': store.loading }">
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"> <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round">
@@ -27,6 +27,7 @@
v-model="searchQuery" v-model="searchQuery"
placeholder="搜索已安装的软件..." placeholder="搜索已安装的软件..."
class="search-input" class="search-input"
:disabled="store.isBusy"
/> />
</div> </div>
</div> </div>
@@ -118,7 +119,7 @@ const filteredSoftware = computed(() => {
white-space: nowrap; white-space: nowrap;
} }
.action-btn:hover { .action-btn:hover:not(:disabled) {
background-color: white; background-color: white;
border-color: var(--primary-color); border-color: var(--primary-color);
color: var(--primary-color); color: var(--primary-color);
@@ -154,6 +155,11 @@ const filteredSoftware = computed(() => {
box-shadow: 0 0 0 3px rgba(0, 122, 255, 0.1); box-shadow: 0 0 0 3px rgba(0, 122, 255, 0.1);
} }
.search-input:disabled {
background-color: #F2F2F7;
cursor: not-allowed;
}
.software-list { .software-list {
display: flex; display: flex;
flex-direction: column; flex-direction: column;

View File

@@ -8,7 +8,7 @@
<button <button
@click="store.syncDataIfNeeded(true)" @click="store.syncDataIfNeeded(true)"
class="secondary-btn action-btn" class="secondary-btn action-btn"
:disabled="store.loading" :disabled="store.loading || store.isBusy"
> >
<span class="icon" :class="{ 'spinning': store.loading }"> <span class="icon" :class="{ 'spinning': store.loading }">
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"> <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round">
@@ -23,7 +23,7 @@
<button <button
@click="installSelected" @click="installSelected"
class="primary-btn action-btn" class="primary-btn action-btn"
:disabled="store.loading || store.selectedEssentialIds.length === 0" :disabled="store.loading || store.isBusy || store.selectedEssentialIds.length === 0"
> >
安装/更新所选 ({{ store.selectedEssentialIds.length }}) 安装/更新所选 ({{ store.selectedEssentialIds.length }})
</button> </button>
@@ -36,11 +36,11 @@
<span class="selection-count">已选 {{ store.selectedEssentialIds.length }} / {{ selectableItems.length }} </span> <span class="selection-count">已选 {{ store.selectedEssentialIds.length }} / {{ selectableItems.length }} </span>
</div> </div>
<div class="toolbar-actions"> <div class="toolbar-actions">
<button @click="store.selectAll('essential')" class="text-btn">全选</button> <button @click="store.selectAll('essential')" class="text-btn" :disabled="store.isBusy">全选</button>
<div class="divider"></div> <div class="divider"></div>
<button @click="store.deselectAll('essential')" class="text-btn">取消</button> <button @click="store.deselectAll('essential')" class="text-btn" :disabled="store.isBusy">取消</button>
<div class="divider"></div> <div class="divider"></div>
<button @click="store.invertSelection('essential')" class="text-btn">反选</button> <button @click="store.invertSelection('essential')" class="text-btn" :disabled="store.isBusy">反选</button>
</div> </div>
</div> </div>
@@ -150,10 +150,15 @@ onMounted(() => {
transition: all 0.2s ease; transition: all 0.2s ease;
} }
.text-btn:hover { .text-btn:hover:not(:disabled) {
background-color: rgba(0, 122, 255, 0.08); background-color: rgba(0, 122, 255, 0.08);
} }
.text-btn:disabled {
color: #AEAEB2;
cursor: not-allowed;
}
.divider { .divider {
width: 1px; width: 1px;
height: 12px; height: 12px;
@@ -180,7 +185,7 @@ onMounted(() => {
box-shadow: var(--btn-shadow); box-shadow: var(--btn-shadow);
} }
.primary-btn:hover { .primary-btn:hover:not(:disabled) {
background-color: var(--primary-hover); background-color: var(--primary-hover);
} }
@@ -190,7 +195,7 @@ onMounted(() => {
border: 1px solid var(--border-color); border: 1px solid var(--border-color);
} }
.secondary-btn:hover { .secondary-btn:hover:not(:disabled) {
background-color: white; background-color: white;
border-color: var(--primary-color); border-color: var(--primary-color);
color: var(--primary-color); color: var(--primary-color);
@@ -199,7 +204,6 @@ onMounted(() => {
.action-btn:disabled { .action-btn:disabled {
opacity: 0.5; opacity: 0.5;
cursor: not-allowed; cursor: not-allowed;
transform: none !important;
box-shadow: none; box-shadow: none;
} }

View File

@@ -8,7 +8,7 @@
<button <button
@click="store.fetchUpdates" @click="store.fetchUpdates"
class="secondary-btn action-btn" class="secondary-btn action-btn"
:disabled="store.loading" :disabled="store.loading || store.isBusy"
> >
<span class="icon" :class="{ 'spinning': store.loading }"> <span class="icon" :class="{ 'spinning': store.loading }">
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"> <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round">
@@ -23,7 +23,7 @@
<button <button
@click="updateSelected" @click="updateSelected"
class="primary-btn action-btn" class="primary-btn action-btn"
:disabled="store.selectedUpdateIds.length === 0 || store.loading" :disabled="store.selectedUpdateIds.length === 0 || store.loading || store.isBusy"
> >
更新所选 ({{ store.selectedUpdateIds.length }}) 更新所选 ({{ store.selectedUpdateIds.length }})
</button> </button>
@@ -36,11 +36,11 @@
<span class="selection-count">已选 {{ store.selectedUpdateIds.length }} / {{ store.sortedUpdates.length }} </span> <span class="selection-count">已选 {{ store.selectedUpdateIds.length }} / {{ store.sortedUpdates.length }} </span>
</div> </div>
<div class="toolbar-actions"> <div class="toolbar-actions">
<button @click="store.selectAll('update')" class="text-btn">全选</button> <button @click="store.selectAll('update')" class="text-btn" :disabled="store.isBusy">全选</button>
<div class="divider"></div> <div class="divider"></div>
<button @click="store.deselectAll('update')" class="text-btn">取消</button> <button @click="store.deselectAll('update')" class="text-btn" :disabled="store.isBusy">取消</button>
<div class="divider"></div> <div class="divider"></div>
<button @click="store.invertSelection('update')" class="text-btn">反选</button> <button @click="store.invertSelection('update')" class="text-btn" :disabled="store.isBusy">反选</button>
</div> </div>
</div> </div>
@@ -151,10 +151,15 @@ onMounted(() => {
transition: all 0.2s ease; transition: all 0.2s ease;
} }
.text-btn:hover { .text-btn:hover:not(:disabled) {
background-color: rgba(0, 122, 255, 0.08); background-color: rgba(0, 122, 255, 0.08);
} }
.text-btn:disabled {
color: #AEAEB2;
cursor: not-allowed;
}
.divider { .divider {
width: 1px; width: 1px;
height: 12px; height: 12px;
@@ -181,7 +186,7 @@ onMounted(() => {
box-shadow: var(--btn-shadow); box-shadow: var(--btn-shadow);
} }
.primary-btn:hover { .primary-btn:hover:not(:disabled) {
background-color: var(--primary-hover); background-color: var(--primary-hover);
} }
@@ -191,7 +196,7 @@ onMounted(() => {
border: 1px solid var(--border-color); border: 1px solid var(--border-color);
} }
.secondary-btn:hover { .secondary-btn:hover:not(:disabled) {
background-color: white; background-color: white;
border-color: var(--primary-color); border-color: var(--primary-color);
color: var(--primary-color); color: var(--primary-color);
@@ -200,7 +205,6 @@ onMounted(() => {
.action-btn:disabled { .action-btn:disabled {
opacity: 0.5; opacity: 0.5;
cursor: not-allowed; cursor: not-allowed;
transform: none !important;
} }
.icon { .icon {