Compare commits
3 Commits
36085f225f
...
9378cabfe5
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9378cabfe5 | ||
|
|
c735659567 | ||
|
|
c5c05ee651 |
24
CHANGELOG.md
24
CHANGELOG.md
@@ -6,6 +6,27 @@
|
|||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
## [0.5.4] — 2026-05-19
|
||||||
|
|
||||||
|
Обновление приложения теперь по-настоящему фоновое + почти моментальный
|
||||||
|
рестарт в новую версию.
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- **Скачивание апдейта — фоновое.** Раньше клик «Скачать» блокировал
|
||||||
|
кнопку (`busy=true`) до конца download'а (минуты на медленной сети).
|
||||||
|
Теперь IPC `updaterDownload` — fire-and-forget, прогресс приходит
|
||||||
|
через события. Пользователь сразу может уйти на Dashboard и
|
||||||
|
продолжать упражнения, апдейт качается в фоне.
|
||||||
|
- **«Рестарт» — почти моментальный.** `quitAndInstall(true, true)`:
|
||||||
|
isSilent=true — NSIS без UI установщика (~1-2 сек вместо ~5-10),
|
||||||
|
isForceRunAfter=true — гарантия что приложение откроется после.
|
||||||
|
Раньше показывался диалог установщика с прогрессом, теперь —
|
||||||
|
только мгновение между закрытием и появлением новой версии.
|
||||||
|
- Подсказка на экране скачивания: «можно закрыть это окно, продолжится
|
||||||
|
в фоне». На downloaded-экране: «нажми Рестарт — приложение
|
||||||
|
моментально откроется в новой версии».
|
||||||
|
|
||||||
## [0.5.3] — 2026-05-19
|
## [0.5.3] — 2026-05-19
|
||||||
|
|
||||||
Полировка кастомного тайтлбара и размера окна.
|
Полировка кастомного тайтлбара и размера окна.
|
||||||
@@ -201,7 +222,8 @@
|
|||||||
иконки), системный трей, автозапуск с Windows, native-уведомления,
|
иконки), системный трей, автозапуск с Windows, native-уведомления,
|
||||||
NSIS-инсталлятор, auto-update через electron-updater.
|
NSIS-инсталлятор, auto-update через electron-updater.
|
||||||
|
|
||||||
[Unreleased]: https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/compare/v0.5.3...HEAD
|
[Unreleased]: https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/compare/v0.5.4...HEAD
|
||||||
|
[0.5.4]: https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/releases/tag/v0.5.4
|
||||||
[0.5.3]: https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/releases/tag/v0.5.3
|
[0.5.3]: https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/releases/tag/v0.5.3
|
||||||
[0.5.2]: https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/releases/tag/v0.5.2
|
[0.5.2]: https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/releases/tag/v0.5.2
|
||||||
[0.5.1]: https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/releases/tag/v0.5.1
|
[0.5.1]: https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/releases/tag/v0.5.1
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
## TL;DR
|
## TL;DR
|
||||||
|
|
||||||
**Laude / Exercise Reminder** — Windows desktop приложение на Electron 33, которое напоминает делать упражнения и опционально парсит статистику матчей Dota 2 (через GSI) в количество повторений. Текущая версия — **0.5.3**. Один разработчик (AnRil), один remote — self-hosted Gitea.
|
**Laude / Exercise Reminder** — Windows desktop приложение на Electron 33, которое напоминает делать упражнения и опционально парсит статистику матчей Dota 2 (через GSI) в количество повторений. Текущая версия — **0.5.4**. Один разработчик (AnRil), один remote — self-hosted Gitea.
|
||||||
|
|
||||||
## Стек
|
## Стек
|
||||||
|
|
||||||
|
|||||||
@@ -2,8 +2,8 @@
|
|||||||
|
|
||||||
Windows desktop приложение, которое напоминает делать упражнения во время работы за компьютером. Опционально подключается к Dota 2 и после каждого матча превращает статистику (смерти, убийства, ассисты) в количество повторений.
|
Windows desktop приложение, которое напоминает делать упражнения во время работы за компьютером. Опционально подключается к Dota 2 и после каждого матча превращает статистику (смерти, убийства, ассисты) в количество повторений.
|
||||||
|
|
||||||
[](https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/releases/latest)
|
[](https://xn--90adajar8af4h.xn--p1ai/git/AnRil/laude/releases/latest)
|
||||||
[]()
|
[]()
|
||||||
[]()
|
[]()
|
||||||
|
|
||||||
## Что внутри
|
## Что внутри
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "laude",
|
"name": "laude",
|
||||||
"version": "0.5.3",
|
"version": "0.5.4",
|
||||||
"description": "Exercise reminder — Windows desktop app",
|
"description": "Exercise reminder — Windows desktop app",
|
||||||
"main": "out/main/index.js",
|
"main": "out/main/index.js",
|
||||||
"author": "AnRil",
|
"author": "AnRil",
|
||||||
|
|||||||
@@ -285,8 +285,13 @@ export function registerIpc(): void {
|
|||||||
// Auto-updater
|
// Auto-updater
|
||||||
ipcMain.handle(IPC.updaterStatus, () => getUpdaterStatus())
|
ipcMain.handle(IPC.updaterStatus, () => getUpdaterStatus())
|
||||||
ipcMain.handle(IPC.updaterCheck, () => checkForUpdates())
|
ipcMain.handle(IPC.updaterCheck, () => checkForUpdates())
|
||||||
ipcMain.handle(IPC.updaterDownload, () => downloadUpdate())
|
// download/install — fire-and-forget. Прогресс и завершение приходят в
|
||||||
ipcMain.handle(IPC.updaterInstall, () => quitAndInstall())
|
// renderer через evtUpdaterStatus, ждать promise бессмысленно — renderer
|
||||||
|
// только зря держал бы `busy=true` весь download (минуты на медленной сети).
|
||||||
|
ipcMain.on(IPC.updaterDownload, () => {
|
||||||
|
void downloadUpdate()
|
||||||
|
})
|
||||||
|
ipcMain.on(IPC.updaterInstall, () => quitAndInstall())
|
||||||
|
|
||||||
// History
|
// History
|
||||||
ipcMain.handle(IPC.getHistory, (_e, sinceMs?: number) => getHistory(sinceMs))
|
ipcMain.handle(IPC.getHistory, (_e, sinceMs?: number) => getHistory(sinceMs))
|
||||||
|
|||||||
@@ -172,5 +172,12 @@ export async function downloadUpdate(): Promise<void> {
|
|||||||
|
|
||||||
export function quitAndInstall(): void {
|
export function quitAndInstall(): void {
|
||||||
if (!app.isPackaged) return
|
if (!app.isPackaged) return
|
||||||
autoUpdater.quitAndInstall()
|
// (isSilent=true, isForceRunAfter=true):
|
||||||
|
// - isSilent: NSIS работает без UI-диалогов установки → restart занимает
|
||||||
|
// ~1-2 сек вместо ~5-10 (без чёрного окна установщика на половину экрана).
|
||||||
|
// - isForceRunAfter: гарантируем что после установки приложение запустится
|
||||||
|
// автоматически, даже если в NSIS-конфиге runAfterFinish был выключен
|
||||||
|
// для этого сценария. Без этого пользователь нажал «Рестарт» — и остался
|
||||||
|
// без открытого приложения.
|
||||||
|
autoUpdater.quitAndInstall(true, true)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -105,8 +105,10 @@ const api = {
|
|||||||
ipcRenderer.invoke(IPC.updaterStatus),
|
ipcRenderer.invoke(IPC.updaterStatus),
|
||||||
updaterCheck: (): Promise<UpdaterStatus> =>
|
updaterCheck: (): Promise<UpdaterStatus> =>
|
||||||
ipcRenderer.invoke(IPC.updaterCheck),
|
ipcRenderer.invoke(IPC.updaterCheck),
|
||||||
updaterDownload: (): Promise<void> => ipcRenderer.invoke(IPC.updaterDownload),
|
// Fire-and-forget. Прогресс и завершение прилетают через onUpdaterStatus —
|
||||||
updaterInstall: (): Promise<void> => ipcRenderer.invoke(IPC.updaterInstall),
|
// renderer не должен `await`'ить, иначе busy-state висит весь download.
|
||||||
|
updaterDownload: (): void => ipcRenderer.send(IPC.updaterDownload),
|
||||||
|
updaterInstall: (): void => ipcRenderer.send(IPC.updaterInstall),
|
||||||
|
|
||||||
// History
|
// History
|
||||||
getHistory: (sinceMs?: number): Promise<HistoryEntry[]> =>
|
getHistory: (sinceMs?: number): Promise<HistoryEntry[]> =>
|
||||||
|
|||||||
@@ -24,6 +24,9 @@ function formatChecked(ts: number, t: TFn): string {
|
|||||||
|
|
||||||
export function UpdaterCard(): JSX.Element {
|
export function UpdaterCard(): JSX.Element {
|
||||||
const [status, setStatus] = useState<UpdaterStatus>({ kind: 'idle' })
|
const [status, setStatus] = useState<UpdaterStatus>({ kind: 'idle' })
|
||||||
|
// busy используется только для синхронного `check()` — для асинхронного
|
||||||
|
// download/install статус сам переключится через события (downloading →
|
||||||
|
// downloaded), отдельный busy-флаг будет только дублировать визуально.
|
||||||
const [busy, setBusy] = useState(false)
|
const [busy, setBusy] = useState(false)
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -39,16 +42,15 @@ export function UpdaterCard(): JSX.Element {
|
|||||||
setBusy(false)
|
setBusy(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
async function download(): Promise<void> {
|
function download(): void {
|
||||||
setBusy(true)
|
// Fire-and-forget — UI моментально перейдёт в kind:'downloading' через
|
||||||
try {
|
// первое же event'ное обновление статуса. Никакого `await` — пользователь
|
||||||
await window.api.updaterDownload()
|
// должен иметь возможность уйти на Dashboard, продолжать упражнения,
|
||||||
} finally {
|
// пока обновление качается в фоне.
|
||||||
setBusy(false)
|
window.api.updaterDownload()
|
||||||
}
|
|
||||||
}
|
}
|
||||||
function install(): void {
|
function install(): void {
|
||||||
void window.api.updaterInstall()
|
window.api.updaterInstall()
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -180,6 +182,10 @@ function Body({
|
|||||||
transition={{ duration: 0.3, ease: 'linear' }}
|
transition={{ duration: 0.3, ease: 'linear' }}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
{/* Подсказка: download идёт в фоне, не нужно сидеть на этом экране. */}
|
||||||
|
<div className="text-[12px] text-text/55 mt-3 font-medium">
|
||||||
|
{t('updater.downloading.hint')}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -194,8 +194,9 @@ export const ru: Dict = {
|
|||||||
'updater.available.title': 'Доступна v{v}',
|
'updater.available.title': 'Доступна v{v}',
|
||||||
'updater.downloading.title': 'Загружаем обновление',
|
'updater.downloading.title': 'Загружаем обновление',
|
||||||
'updater.downloading.subtitle': '{got} / {total} МБ · {speed} МБ/с',
|
'updater.downloading.subtitle': '{got} / {total} МБ · {speed} МБ/с',
|
||||||
|
'updater.downloading.hint': 'Можно закрыть это окно — скачивание продолжится в фоне.',
|
||||||
'updater.downloaded.title': 'Готово · v{v}',
|
'updater.downloaded.title': 'Готово · v{v}',
|
||||||
'updater.downloaded.subtitle': 'Перезапусти для применения',
|
'updater.downloaded.subtitle': 'Нажми «Рестарт» — приложение моментально откроется в новой версии.',
|
||||||
'updater.error.title': 'Ошибка проверки',
|
'updater.error.title': 'Ошибка проверки',
|
||||||
'updater.idle.title': 'Проверить обновления',
|
'updater.idle.title': 'Проверить обновления',
|
||||||
'updater.idle.subtitle': 'Авто-проверка раз в час',
|
'updater.idle.subtitle': 'Авто-проверка раз в час',
|
||||||
@@ -440,8 +441,9 @@ export const en: Dict = {
|
|||||||
'updater.available.title': 'v{v} available',
|
'updater.available.title': 'v{v} available',
|
||||||
'updater.downloading.title': 'Downloading update',
|
'updater.downloading.title': 'Downloading update',
|
||||||
'updater.downloading.subtitle': '{got} / {total} MB · {speed} MB/s',
|
'updater.downloading.subtitle': '{got} / {total} MB · {speed} MB/s',
|
||||||
|
'updater.downloading.hint': 'You can close this window — download continues in the background.',
|
||||||
'updater.downloaded.title': 'Ready · v{v}',
|
'updater.downloaded.title': 'Ready · v{v}',
|
||||||
'updater.downloaded.subtitle': 'Restart to apply',
|
'updater.downloaded.subtitle': 'Click Restart — the app will reopen instantly in the new version.',
|
||||||
'updater.error.title': 'Check failed',
|
'updater.error.title': 'Check failed',
|
||||||
'updater.idle.title': 'Check for updates',
|
'updater.idle.title': 'Check for updates',
|
||||||
'updater.idle.subtitle': 'Auto-check every hour',
|
'updater.idle.subtitle': 'Auto-check every hour',
|
||||||
|
|||||||
Reference in New Issue
Block a user