diff --git a/packages/desktop-client/src/components/common/Search.tsx b/packages/desktop-client/src/components/common/Search.tsx index d07f5e515fd8627391aa6f31d5ae1d5adfe5ac3c..c88cb76eb298a742c3394d85fe688d363404638c 100644 --- a/packages/desktop-client/src/components/common/Search.tsx +++ b/packages/desktop-client/src/components/common/Search.tsx @@ -8,11 +8,11 @@ import Button from './Button'; import InputWithContent from './InputWithContent'; type SearchProps = { - inputRef: Ref<HTMLInputElement>; + inputRef?: Ref<HTMLInputElement>; value: string; onChange: (value: string) => unknown; placeholder: string; - isInModal: boolean; + isInModal?: boolean; width?: number; }; @@ -21,7 +21,7 @@ export default function Search({ value, onChange, placeholder, - isInModal, + isInModal = false, width = 250, }: SearchProps) { return ( diff --git a/packages/desktop-client/src/components/schedules/index.js b/packages/desktop-client/src/components/schedules/index.tsx similarity index 78% rename from packages/desktop-client/src/components/schedules/index.js rename to packages/desktop-client/src/components/schedules/index.tsx index eec15cc59841f58beb0f8ebaa717d1a2bb1d5dd4..87f762df6e506d8bf97a1c64e84f7e3377d2566b 100644 --- a/packages/desktop-client/src/components/schedules/index.js +++ b/packages/desktop-client/src/components/schedules/index.tsx @@ -2,6 +2,7 @@ import React, { useState } from 'react'; import { useSchedules } from 'loot-core/src/client/data-hooks/schedules'; import { send } from 'loot-core/src/platform/client/fetch'; +import { type ScheduleEntity } from 'loot-core/src/types/models'; import { useActions } from '../../hooks/useActions'; import Button from '../common/Button'; @@ -12,18 +13,18 @@ import { Page } from '../Page'; import { SchedulesTable, ROW_HEIGHT } from './SchedulesTable'; export default function Schedules() { - let { pushModal } = useActions(); - let [filter, setFilter] = useState(''); + const { pushModal } = useActions(); + const [filter, setFilter] = useState(''); - let scheduleData = useSchedules(); + const scheduleData = useSchedules(); if (scheduleData == null) { return null; } - let { schedules, statuses } = scheduleData; + const { schedules, statuses } = scheduleData; - function onEdit(id) { + function onEdit(id: ScheduleEntity['id']) { pushModal('schedule-edit', { id }); } @@ -35,7 +36,8 @@ export default function Schedules() { pushModal('schedules-discover'); } - async function onAction(name, id) { + // @todo: replace name: string with enum + async function onAction(name: string, id: ScheduleEntity['id']) { switch (name) { case 'post-transaction': await send('schedule/post-transaction', { id }); @@ -44,7 +46,9 @@ export default function Schedules() { await send('schedule/skip-next-date', { id }); break; case 'complete': - await send('schedule/update', { schedule: { id, completed: true } }); + await send('schedule/update', { + schedule: { id, completed: true }, + }); break; case 'restart': await send('schedule/update', { @@ -83,6 +87,9 @@ export default function Schedules() { onSelect={onEdit} onAction={onAction} style={{ backgroundColor: 'white' }} + // @todo: Remove following props after typing SchedulesTable + minimal={undefined} + tableStyle={undefined} /> </View> diff --git a/packages/desktop-client/src/components/settings/Encryption.tsx b/packages/desktop-client/src/components/settings/Encryption.tsx index 6df29b85a43cc1800d0672f47494f81f831ea8f8..3eed3bc95481ac37f1cde0d2b21c687cbe0403e3 100644 --- a/packages/desktop-client/src/components/settings/Encryption.tsx +++ b/packages/desktop-client/src/components/settings/Encryption.tsx @@ -18,7 +18,6 @@ export default function EncryptionSettings() { const missingCryptoAPI = !(window.crypto && crypto.subtle); function onChangeKey() { - // @ts-expect-error useActions() type does not properly handle overloads pushModal('create-encryption-key', { recreate: true }); } diff --git a/packages/loot-core/src/client/actions/budgets.ts b/packages/loot-core/src/client/actions/budgets.ts index 97e2673ca675973b1f4adea269e2823c7b6b4ff8..ccac2680d9fb0066e0580d07ee488d754930e359 100644 --- a/packages/loot-core/src/client/actions/budgets.ts +++ b/packages/loot-core/src/client/actions/budgets.ts @@ -76,7 +76,6 @@ export function loadBudget(id: string, loadingText = '', options = {}) { ); if (showBackups) { - // @ts-expect-error manager modals are not yet typed dispatch(pushModal('load-backup', { budgetId: id })); } } else { diff --git a/packages/loot-core/src/client/actions/modals.ts b/packages/loot-core/src/client/actions/modals.ts index efc1a0ae124d05d6787f1d2721b8c76da4830673..7e50ba6523e7042765d911724af067f0fc77f8b4 100644 --- a/packages/loot-core/src/client/actions/modals.ts +++ b/packages/loot-core/src/client/actions/modals.ts @@ -15,12 +15,15 @@ export function pushModal<M extends keyof ModalWithOptions>( options: ModalWithOptions[M], ): PushModalAction; export function pushModal(name: OptionlessModal): PushModalAction; +export function pushModal<M extends ModalType>( + name: M, + options?: FinanceModals[M], +): PushModalAction; export function pushModal<M extends ModalType>( name: M, options?: FinanceModals[M], ): PushModalAction { - // @ts-expect-error TS is unable to determine that `name` and `options` match - let modal: M = { name, options }; + const modal = { name, options }; return { type: constants.PUSH_MODAL, modal }; } diff --git a/packages/loot-core/src/client/data-hooks/schedules.tsx b/packages/loot-core/src/client/data-hooks/schedules.tsx index 3603561f85438f76e067dd3b299ea83c892efe4b..9131d3c96b1b55175a69ea449f3d5501f51e307d 100644 --- a/packages/loot-core/src/client/data-hooks/schedules.tsx +++ b/packages/loot-core/src/client/data-hooks/schedules.tsx @@ -2,9 +2,10 @@ import React, { createContext, useEffect, useState, useContext } from 'react'; import { type Query } from '../../shared/query'; import { getStatus, getHasTransactionsQuery } from '../../shared/schedules'; +import { type ScheduleEntity } from '../../types/models'; import q, { liveQuery } from '../query-helpers'; -function loadStatuses(schedules, onData) { +function loadStatuses(schedules: ScheduleEntity[], onData) { return liveQuery(getHasTransactionsQuery(schedules), onData, { mapper: data => { let hasTrans = new Set(data.filter(Boolean).map(row => row.schedule)); @@ -20,8 +21,12 @@ function loadStatuses(schedules, onData) { } type UseSchedulesArgs = { transform?: (q: Query) => Query }; +type UseSchedulesReturnType = { + schedules: ScheduleEntity[]; + statuses: Record<string, ReturnType<typeof getStatus>>; +} | null; export function useSchedules({ transform }: UseSchedulesArgs = {}) { - let [data, setData] = useState(null); + let [data, setData] = useState<UseSchedulesReturnType | null>(null); useEffect(() => { let query = q('schedules').select('*'); @@ -29,14 +34,16 @@ export function useSchedules({ transform }: UseSchedulesArgs = {}) { scheduleQuery = liveQuery( transform ? transform(query) : query, - async schedules => { + async (schedules: ScheduleEntity[]) => { if (scheduleQuery) { if (statusQuery) { statusQuery.unsubscribe(); } - statusQuery = loadStatuses(schedules, statuses => - setData({ schedules, statuses }), + statusQuery = loadStatuses( + schedules, + (statuses: Record<string, ReturnType<typeof getStatus>>) => + setData({ schedules, statuses }), ); } }, diff --git a/upcoming-release-notes/1644.md b/upcoming-release-notes/1644.md new file mode 100644 index 0000000000000000000000000000000000000000..bebaa2739dd3ae96eff4628e9ef54523c0bdab12 --- /dev/null +++ b/upcoming-release-notes/1644.md @@ -0,0 +1,6 @@ +--- +category: Maintenance +authors: [muhsinkamil] +--- + +Refactor Schedules to tsx.