From db07d7a73d2824a6ec5c94e0fe98f0fa1578aace Mon Sep 17 00:00:00 2001
From: Mohamed Muhsin <62111075+muhsinkamil@users.noreply.github.com>
Date: Mon, 4 Sep 2023 18:41:02 +0200
Subject: [PATCH] refactor: make Schedules component to tsx (#1644)

---
 .../src/components/common/Search.tsx          |  6 +++---
 .../schedules/{index.js => index.tsx}         | 21 ++++++++++++-------
 .../src/components/settings/Encryption.tsx    |  1 -
 .../loot-core/src/client/actions/budgets.ts   |  1 -
 .../loot-core/src/client/actions/modals.ts    |  7 +++++--
 .../src/client/data-hooks/schedules.tsx       | 17 ++++++++++-----
 upcoming-release-notes/1644.md                |  6 ++++++
 7 files changed, 40 insertions(+), 19 deletions(-)
 rename packages/desktop-client/src/components/schedules/{index.js => index.tsx} (78%)
 create mode 100644 upcoming-release-notes/1644.md

diff --git a/packages/desktop-client/src/components/common/Search.tsx b/packages/desktop-client/src/components/common/Search.tsx
index d07f5e515..c88cb76eb 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 eec15cc59..87f762df6 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 6df29b85a..3eed3bc95 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 97e2673ca..ccac2680d 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 efc1a0ae1..7e50ba652 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 3603561f8..9131d3c96 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 000000000..bebaa2739
--- /dev/null
+++ b/upcoming-release-notes/1644.md
@@ -0,0 +1,6 @@
+---
+category: Maintenance
+authors: [muhsinkamil]
+---
+
+Refactor Schedules to tsx.
-- 
GitLab