hexabot/frontend/src/hooks/crud/useUpdate.tsx
2025-06-21 10:37:09 +01:00

71 lines
2.3 KiB
TypeScript

/*
* Copyright © 2025 Hexastack. All rights reserved.
*
* Licensed under the GNU Affero General Public License v3.0 (AGPLv3) with the following additional terms:
* 1. The name "Hexabot" is a trademark of Hexastack. You may not use this name in derivative works without express written permission.
* 2. All derivative works must include clear attribution to the original creator and software, Hexastack and Hexabot, in a prominent location (e.g., in the software's "About" section, documentation, and README file).
*/
import { useMutation, useQueryClient } from "react-query";
import { QueryType, TMutationOptions, TSetCacheProps } from "@/services/types";
import { IBaseSchema, IEntityMapTypes, THook, TType } from "@/types/base.types";
import { merge } from "@/utils/object";
import { useEntityApiClient } from "../useApiClient";
import { useNormalizeAndCache } from "./helpers";
import { useGetFromCache } from "./useGet";
export const useUpdate = <
TE extends THook["entity"],
TAttr = THook<{ entity: TE }>["attributes"],
TBasic extends IBaseSchema = THook<{ entity: TE }>["basic"],
TFull extends IBaseSchema = THook<{ entity: TE }>["full"],
>(
entity: TE,
options?: Omit<
TMutationOptions<
TBasic,
Error,
{ id: string; params: Partial<TAttr> },
TBasic
>,
"mutationFn" | "mutationKey"
>,
) => {
const api = useEntityApiClient<TAttr, TBasic, TFull>(entity);
const normalizeAndCache = useNormalizeAndCache<TBasic, string>(entity);
return useMutation({
mutationFn: async ({ id, params }) => {
const data = await api.update(id, params);
const { entities, result } = normalizeAndCache(data);
return entities[entity]?.[result] as unknown as TBasic;
},
...options,
});
};
export const useUpdateCache = <E extends keyof IEntityMapTypes>(entity: E) => {
const getFromCache = useGetFromCache(entity);
const queryClient = useQueryClient();
return ({
id,
payload = {},
preprocess = (data) => data,
strategy = "merge",
}: TSetCacheProps<TType<E>["basic"]>) => {
const oldData = getFromCache(id);
const newData =
strategy === "merge" && oldData
? merge(preprocess(oldData), payload)
: payload;
const [qEntity] = entity.split("/");
queryClient.setQueryData([QueryType.item, qEntity, id], newData);
};
};