From 2a285a0520d9ef9e2caaefc971da45258f3713d1 Mon Sep 17 00:00:00 2001 From: Patrick Browne Date: Tue, 20 Feb 2024 11:26:53 +0100 Subject: [PATCH 1/7] fix: Gap between tables was too small --- app/login/components/profile-content-tabs.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/login/components/profile-content-tabs.tsx b/app/login/components/profile-content-tabs.tsx index 775444f05..b348a9d95 100644 --- a/app/login/components/profile-content-tabs.tsx +++ b/app/login/components/profile-content-tabs.tsx @@ -90,7 +90,7 @@ export const ProfileContentTabs = (props: ProfileContentTabsProps) => { Date: Tue, 20 Feb 2024 16:19:56 +0100 Subject: [PATCH 2/7] feat: Remove publish/turn into draft action (design not yet decided) --- app/login/components/profile-tables.tsx | 67 ++++++++++++------------- 1 file changed, 33 insertions(+), 34 deletions(-) diff --git a/app/login/components/profile-tables.tsx b/app/login/components/profile-tables.tsx index e4d7fd5e2..d528539b5 100644 --- a/app/login/components/profile-tables.tsx +++ b/app/login/components/profile-tables.tsx @@ -336,7 +336,6 @@ const ProfileVisualizationsRow = (props: ProfileVisualizationsRowProps) => { const { invalidate: invalidateUserConfigs } = useUserConfigs(); - const updateConfigMut = useMutate(updateConfig); const removeConfigMut = useMutate(removeConfig); const { @@ -375,40 +374,40 @@ const ProfileVisualizationsRow = (props: ProfileVisualizationsRowProps) => { label: t({ id: "login.chart.share", message: "Share" }), iconName: "linkExternal", }, - { - type: "button", - label: - config.published_state === PUBLISHED_STATE.DRAFT - ? t({ - id: "login.chart.actions.publish", - message: `Publish`, - }) - : t({ - id: "login.chart.actions.turn-into-draft", - message: "Turn into draft", - }), - iconName: - updateConfigMut.status === "fetching" ? "loading" : "linkExternal", + // { + // type: "button", + // label: + // config.published_state === PUBLISHED_STATE.DRAFT + // ? t({ + // id: "login.chart.actions.publish", + // message: `Publish`, + // }) + // : t({ + // id: "login.chart.actions.turn-into-draft", + // message: "Turn into draft", + // }), + // iconName: + // updateConfigMut.status === "fetching" ? "loading" : "linkExternal", - onClick: async () => { - await updateConfigMut.mutate({ - key: config.key, - user_id: userId, - data: { - ...config.data, - state: "PUBLISHING", - }, - published_state: - config.published_state === PUBLISHED_STATE.DRAFT - ? PUBLISHED_STATE.PUBLISHED - : PUBLISHED_STATE.DRAFT, - }); - invalidateUserConfigs(); - }, - onSuccess: () => { - invalidateUserConfigs(); - }, - }, + // onClick: async () => { + // await updateConfigMut.mutate({ + // key: config.key, + // user_id: userId, + // data: { + // ...config.data, + // state: "PUBLISHING", + // }, + // published_state: + // config.published_state === PUBLISHED_STATE.DRAFT + // ? PUBLISHED_STATE.PUBLISHED + // : PUBLISHED_STATE.DRAFT, + // }); + // invalidateUserConfigs(); + // }, + // onSuccess: () => { + // invalidateUserConfigs(); + // }, + // }, { type: "button", label: t({ id: "login.chart.rename", message: "Rename" }), From 029de00fe527fc710807b50778e2935b3585d32b Mon Sep 17 00:00:00 2001 From: Patrick Browne Date: Tue, 20 Feb 2024 16:19:56 +0100 Subject: [PATCH 3/7] refactor: Move related API config methods together --- app/pages/api/config-create.ts | 22 ++----------- app/pages/api/config-remove.ts | 19 ++--------- app/pages/api/config-update.ts | 23 ++----------- app/server/config-controller.ts | 57 +++++++++++++++++++++++++++++++++ app/server/nextkit.ts | 11 ++++++- 5 files changed, 74 insertions(+), 58 deletions(-) create mode 100644 app/server/config-controller.ts diff --git a/app/pages/api/config-create.ts b/app/pages/api/config-create.ts index 75da64b09..87709c303 100644 --- a/app/pages/api/config-create.ts +++ b/app/pages/api/config-create.ts @@ -1,24 +1,8 @@ -import { getServerSession } from "next-auth"; - -import { createConfig } from "@/db/config"; - -import { api } from "../../server/nextkit"; - -import { nextAuthOptions } from "./auth/[...nextauth]"; +import ConfigController from "@/server/config-controller"; +import { api } from "@/server/nextkit"; const route = api({ - POST: async ({ req, res }) => { - const session = await getServerSession(req, res, nextAuthOptions); - const userId = session?.user?.id; - const { data, publishedState } = req.body; - - return await createConfig({ - key: data.key, - data, - userId, - publishedState: publishedState, - }); - }, + POST: ConfigController.create, }); export default route; diff --git a/app/pages/api/config-remove.ts b/app/pages/api/config-remove.ts index b53f1e711..d0d7c2b2f 100644 --- a/app/pages/api/config-remove.ts +++ b/app/pages/api/config-remove.ts @@ -1,24 +1,9 @@ -import { getServerSession } from "next-auth"; - -import { getConfig, removeConfig } from "@/db/config"; +import ConfigController from "@/server/config-controller"; import { api } from "../../server/nextkit"; -import { nextAuthOptions } from "./auth/[...nextauth]"; - const route = api({ - POST: async ({ req, res }) => { - const session = await getServerSession(req, res, nextAuthOptions); - const sessionUserId = session?.user?.id; - const { key } = req.body; - - const config = await getConfig(key); - if (sessionUserId !== config?.user_id) { - throw new Error("Unauthorized!"); - } - - return await removeConfig({ key }); - }, + POST: ConfigController.remove, }); export default route; diff --git a/app/pages/api/config-update.ts b/app/pages/api/config-update.ts index 9caac853a..69e779417 100644 --- a/app/pages/api/config-update.ts +++ b/app/pages/api/config-update.ts @@ -1,28 +1,9 @@ -import { getServerSession } from "next-auth"; - -import { updateConfig } from "@/db/config"; +import ConfigController from "@/server/config-controller"; import { api } from "../../server/nextkit"; -import { nextAuthOptions } from "./auth/[...nextauth]"; - const route = api({ - POST: async ({ req, res }) => { - const session = await getServerSession(req, res, nextAuthOptions); - const serverUserId = session?.user?.id; - const { key, user_id, data, published_state } = req.body; - - if (serverUserId !== user_id) { - throw new Error("Unauthorized!"); - } - - return await updateConfig({ - key, - data, - user_id, - published_state, - }); - }, + POST: ConfigController.update, }); export default route; diff --git a/app/server/config-controller.ts b/app/server/config-controller.ts new file mode 100644 index 000000000..18792438f --- /dev/null +++ b/app/server/config-controller.ts @@ -0,0 +1,57 @@ +import { getServerSession } from "next-auth"; + +import { + createConfig, + getConfig, + removeConfig, + updateConfig, +} from "@/db/config"; +import { nextAuthOptions } from "@/pages/api/auth/[...nextauth]"; +import { controller } from "@/server/nextkit"; + +const ConfigController = controller({ + create: async ({ req, res }) => { + const session = await getServerSession(req, res, nextAuthOptions); + const userId = session?.user?.id; + const { data, publishedState } = req.body; + + return await createConfig({ + key: data.key, + data, + userId, + publishedState: publishedState, + }); + }, + + remove: async ({ req, res }) => { + const session = await getServerSession(req, res, nextAuthOptions); + const sessionUserId = session?.user?.id; + const { key } = req.body; + + const config = await getConfig(key); + if (sessionUserId !== config?.user_id) { + throw new Error("Unauthorized!"); + } + + return await removeConfig({ key }); + }, + + update: async ({ req, res }) => { + const session = await getServerSession(req, res, nextAuthOptions); + const serverUserId = session?.user?.id; + const { key, user_id, data, published_state } = req.body; + + if (serverUserId !== user_id) { + throw new Error("Unauthorized!"); + } + + return await updateConfig({ + key, + data, + user_id, + published_state, + }); + }, +}); + +export default ConfigController; diff --git a/app/server/nextkit.ts b/app/server/nextkit.ts index a483c5eac..e7af7362a 100644 --- a/app/server/nextkit.ts +++ b/app/server/nextkit.ts @@ -1,4 +1,13 @@ -import createAPI from "nextkit"; +import createAPI, { NextkitHandler } from "nextkit"; + +/** Provides type hints */ +export const controller = < + THandlers extends Record> +>( + methods: THandlers +) => { + return methods; +}; export const api = createAPI({ async onError(_req, _res, error) { From 25cf1cbe5bf9fb5220c262610194784e0bcf7573 Mon Sep 17 00:00:00 2001 From: Patrick Browne Date: Tue, 20 Feb 2024 16:58:41 +0100 Subject: [PATCH 4/7] fix: Correctly save published state published state was not saved correctly before the 1st time we would save in draft. I had not seen this because I had first put @default=DRAFT in Prisma locally, then had changed it to @default=PUBLISHED without running the migration. Then on test, the default was PUBLISHED and I could see that saving in draft would not work for the 1st time --- app/db/config.ts | 6 +++--- app/server/config-controller.ts | 4 ++-- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/app/db/config.ts b/app/db/config.ts index 5f226490c..d0b4ea229 100644 --- a/app/db/config.ts +++ b/app/db/config.ts @@ -20,19 +20,19 @@ export const createConfig = async ({ key, data, userId, - publishedState, + published_state, }: { key: string; data: Prisma.ConfigCreateInput["data"]; userId?: User["id"] | undefined; - publishedState: PUBLISHED_STATE; + published_state: PUBLISHED_STATE; }): Promise<{ key: string }> => { return await prisma.config.create({ data: { key, data, user_id: userId, - published_state: publishedState, + published_state, }, }); }; diff --git a/app/server/config-controller.ts b/app/server/config-controller.ts index 18792438f..ad0879f8f 100644 --- a/app/server/config-controller.ts +++ b/app/server/config-controller.ts @@ -13,13 +13,13 @@ const ConfigController = controller({ create: async ({ req, res }) => { const session = await getServerSession(req, res, nextAuthOptions); const userId = session?.user?.id; - const { data, publishedState } = req.body; + const { data, published_state } = req.body; return await createConfig({ key: data.key, data, userId, - publishedState: publishedState, + published_state: published_state, }); }, From 81177e178e9c89a7b0a9961bfc50ca8556149a3e Mon Sep 17 00:00:00 2001 From: Patrick Browne Date: Tue, 20 Feb 2024 17:14:37 +0100 Subject: [PATCH 5/7] refactor: Extract saveChartLocally --- app/configurator/configurator-state.tsx | 32 ++++++++++++------------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/app/configurator/configurator-state.tsx b/app/configurator/configurator-state.tsx index fb48f49f5..b7d721655 100644 --- a/app/configurator/configurator-state.tsx +++ b/app/configurator/configurator-state.tsx @@ -1532,6 +1532,14 @@ export const initChartStateFromLocalStorage = async ( window.localStorage.removeItem(getLocalStorageKey(chartId)); }; +export const saveChartLocally = (chartId: string, state: ConfiguratorState) => { + window.localStorage.setItem( + getLocalStorageKey(chartId), + JSON.stringify(state) + ); + return; +}; + const ConfiguratorStateProviderInternal = ( props: ConfiguratorStateProviderProps ) => { @@ -1598,24 +1606,16 @@ const ConfiguratorStateProviderInternal = ( switch (state.state) { case "CONFIGURING_CHART": case "LAYOUTING": - if (chartId === "new") { - const chartId = - query.edit && typeof query.edit === "string" + const savedChartId = + chartId === "new" + ? query.edit && typeof query.edit === "string" ? query.edit - : createChartId(); - replace(`/create/${chartId}`); - window.localStorage.setItem( - getLocalStorageKey(chartId), - JSON.stringify(state) - ); - } else { - // Store current state in localstorage - window.localStorage.setItem( - getLocalStorageKey(chartId), - JSON.stringify(state) - ); + : createChartId() + : chartId; + if (chartId === "new") { + replace(`/create/${savedChartId}`); } - + saveChartLocally(savedChartId, state); return; case "PUBLISHING": From e9ab3ed6de2cb0f626716c1464134f0a036b7915 Mon Sep 17 00:00:00 2001 From: Patrick Browne Date: Tue, 20 Feb 2024 17:15:07 +0100 Subject: [PATCH 6/7] refactor: Removed unnecessary --- app/login/components/profile-tables.tsx | 3 --- 1 file changed, 3 deletions(-) diff --git a/app/login/components/profile-tables.tsx b/app/login/components/profile-tables.tsx index d528539b5..b39cf18d3 100644 --- a/app/login/components/profile-tables.tsx +++ b/app/login/components/profile-tables.tsx @@ -442,14 +442,11 @@ const ProfileVisualizationsRow = (props: ProfileVisualizationsRowProps) => { return sortBy(actions, (x) => x.priority); }, [ - config.data, config.key, config.published_state, invalidateUserConfigs, openRename, removeConfigMut, - updateConfigMut, - userId, ]); const chartTitle = React.useMemo(() => { From 5247f005a41604046ac5062b1ddcfdb45bc4f294 Mon Sep 17 00:00:00 2001 From: Patrick Browne Date: Tue, 20 Feb 2024 17:16:50 +0100 Subject: [PATCH 7/7] fix: Reduce flickering when saving for first time in draft After saving in draft, save locally instead of switching page --- app/components/chart-selection-tabs.tsx | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/app/components/chart-selection-tabs.tsx b/app/components/chart-selection-tabs.tsx index 56ab11d83..3d71ba62f 100644 --- a/app/components/chart-selection-tabs.tsx +++ b/app/components/chart-selection-tabs.tsx @@ -29,10 +29,12 @@ import { enableLayouting, getChartConfig, hasChartConfigs, + initChartStateFromChartEdit, isConfiguring, isLayouting, isPublished, isPublishing, + saveChartLocally, useConfiguratorState, } from "@/configurator"; import { ChartTypeSelector } from "@/configurator/components/chart-type-selector"; @@ -329,7 +331,7 @@ export const SaveDraftButton = ({ const { data: config, invalidate: invalidateConfig } = useUserConfig(chartId); const session = useSession(); - const [state] = useConfiguratorState(); + const [state, dispatch] = useConfiguratorState(); const [snack, enqueueSnackbar, dismissSnack] = useLocalSnack(); const [debouncedSnack] = useDebounce(snack, 500); @@ -370,7 +372,15 @@ export const SaveDraftButton = ({ }), variant: "success", }); - replace(`/create/new?edit=${saved.key}`); + const config = await initChartStateFromChartEdit(saved.key); + if (!config) { + return; + } + dispatch({ type: "INITIALIZED", value: config }); + saveChartLocally(saved.key, config); + replace(`/create/${saved.key}`, undefined, { + shallow: true, + }); } else { throw new Error("Could not save draft"); }