Skip to content
This repository was archived by the owner on Sep 15, 2025. It is now read-only.

Commit e8959d9

Browse files
authored
feat: 트레이 아이콘 모드에 따라 변경 #patch (#61)
* changeTrayTitle 인터페이스 추가 * 타이머 모드에 따라 트레이 아이콘 변경
1 parent f34f3a2 commit e8959d9

File tree

4 files changed

+53
-4
lines changed

4 files changed

+53
-4
lines changed

src/main/main.ts

Lines changed: 12 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -56,12 +56,18 @@ const createWindow = () => {
5656
};
5757

5858
const trayIconMap: Record<string, string> = {
59-
cat: '',
59+
default:
60+
'',
6061
focus:
61-
'',
62+
'',
63+
'focus-exceed':
64+
'',
65+
rest: '',
66+
'rest-exceed':
67+
'',
6268
};
6369
const getTrayIcon = (icon: string): NativeImage => {
64-
return nativeImage.createFromDataURL(trayIconMap[icon] ?? trayIconMap['cat']);
70+
return nativeImage.createFromDataURL(trayIconMap[icon] ?? trayIconMap.default);
6571
};
6672

6773
const createTray = (mainWindow: BrowserWindow) => {
@@ -127,6 +133,9 @@ app.whenReady().then(() => {
127133
ipcMain.handle('change-tray-icon', (event, icon: string) => {
128134
tray?.setImage(getTrayIcon(icon));
129135
});
136+
ipcMain.handle('change-tray-title', (event, title: string) => {
137+
tray?.setTitle(title);
138+
});
130139
ipcMain.handle('get-always-on-top', () => {
131140
return mainWindow?.isAlwaysOnTop();
132141
});

src/preload/preload.ts

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ const electronAPI: IElectronAPI = {
88
showWindow: () => ipcRenderer.send('show-window'),
99
getMachineId: () => ipcRenderer.invoke('get-machine-id'),
1010
changeTrayIcon: (icon: string) => ipcRenderer.invoke('change-tray-icon', icon),
11+
changeTrayTitle: (title: string) => ipcRenderer.invoke('change-tray-title', title),
1112
getAlwaysOnTop: () => ipcRenderer.invoke('get-always-on-top'),
1213
setAlwaysOnTop: (isAlwaysOnTop: boolean) =>
1314
ipcRenderer.invoke('set-always-on-top', isAlwaysOnTop),

src/renderer/features/pomodoro/hooks/use-pomodoro.ts

Lines changed: 38 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,11 @@
1+
import { useEffect } from 'react';
2+
13
import { useLocalStorage } from 'usehooks-ts';
24

35
import { PomodoroCycle, PomodoroEndReason, PomodoroMode, PomodoroTime } from '@/entities/pomodoro';
46
import { LOCAL_STORAGE_KEY } from '@/shared/constants';
57
import { useInterval } from '@/shared/hooks';
8+
import { msToTime, padNumber } from '@/shared/utils';
69

710
// == usePomodoro 로직에 대한 description
811

@@ -64,6 +67,23 @@ const updateCycles = (cycles: PomodoroCycle[], nextCycle?: PomodoroCycle): Pomod
6467
].filter(isNotNil);
6568
};
6669

70+
const getFormattedTime = (goalTime: number, { elapsed, exceeded }: PomodoroTime) => {
71+
const isExceed = exceeded > 0;
72+
const { minutes, seconds } = msToTime(isExceed ? exceeded : goalTime - elapsed);
73+
const time = `${padNumber(minutes)}:${padNumber(seconds)}`;
74+
return { isExceed, time };
75+
};
76+
77+
const getTrayIcon = (mode: PomodoroMode, isExceed: boolean) => {
78+
if (mode === 'focus') {
79+
return isExceed ? 'focus-exceed' : 'focus';
80+
}
81+
if (mode === 'rest') {
82+
return isExceed ? 'rest-exceed' : 'rest';
83+
}
84+
return '';
85+
};
86+
6787
export const getPomodoroTime = (cycle: PomodoroCycle): PomodoroTime => {
6888
const now = cycle.endAt ?? Date.now();
6989
const maxElapsedTime = cycle.goalTime + cycle.exceedMaxTime;
@@ -171,6 +191,24 @@ export const usePomodoro = ({
171191
pomodoroCycles.length > 0 ? 250 : null,
172192
);
173193

194+
useEffect(() => {
195+
const currentCycle = pomodoroCycles[pomodoroCycles.length - 1];
196+
if (!currentCycle) {
197+
window.electronAPI.changeTrayIcon('');
198+
window.electronAPI.changeTrayTitle('');
199+
return;
200+
}
201+
202+
const { isExceed, time: trayTitle } = getFormattedTime(currentCycle.goalTime, pomodoroTime);
203+
const mode = currentCycle.mode;
204+
// 휴식 대기 중에는 직전의 집중 시간을 표시하기 위함
205+
if (mode !== 'rest-wait') {
206+
const trayIcon = getTrayIcon(mode, isExceed);
207+
window.electronAPI.changeTrayIcon(trayIcon);
208+
window.electronAPI.changeTrayTitle(trayTitle);
209+
}
210+
}, [pomodoroCycles, pomodoroTime]);
211+
174212
return {
175213
pomodoroCycles,
176214
pomodoroTime,

src/shared/type.ts

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,9 @@
11
// @see: https://www.electronjs.org/docs/latest/tutorial/context-isolation#usage-with-typescript
22
export interface IElectronAPI {
33
showWindow: () => void;
4-
changeTrayIcon: (icon: string) => void;
54
getMachineId: () => Promise<string>;
5+
changeTrayIcon: (icon: string) => void;
6+
changeTrayTitle: (title: string) => void;
67
getAlwaysOnTop: () => Promise<boolean>;
78
setAlwaysOnTop: (isAlwaysOnTop: boolean) => Promise<void>;
89
getMinimized: () => Promise<boolean>;

0 commit comments

Comments
 (0)