Files
bknd/app/src/ui/routes/settings/components/Setting.tsx

355 lines
12 KiB
TypeScript

import { useHotkeys } from "@mantine/hooks";
import { type TObject, ucFirst } from "core/utils";
import { omit } from "lodash-es";
import { type ReactNode, useMemo, useRef, useState } from "react";
import { TbSettings } from "react-icons/tb";
import { useBknd } from "ui/client/bknd";
import { Button } from "ui/components/buttons/Button";
import { IconButton } from "ui/components/buttons/IconButton";
import { Alert } from "ui/components/display/Alert";
import { Empty } from "ui/components/display/Empty";
import { JsonSchemaForm, type JsonSchemaFormRef } from "ui/components/form/json-schema";
import { Dropdown } from "ui/components/overlay/Dropdown";
import { DataTable } from "ui/components/table/DataTable";
import { useEvent } from "ui/hooks/use-event";
import * as AppShell from "ui/layouts/AppShell/AppShell";
import { Breadcrumbs } from "ui/layouts/AppShell/Breadcrumbs";
import { Link, Route, useLocation } from "wouter";
import { extractSchema } from "../utils/schema";
import { SettingNewModal, type SettingsNewModalProps } from "./SettingNewModal";
import { SettingSchemaModal, type SettingsSchemaModalRef } from "./SettingSchemaModal";
export type SettingProps<
Schema extends TObject = TObject,
Props = Schema extends TObject<infer TProperties> ? TProperties : any
> = {
schema: Schema;
config: any;
prefix?: string;
path?: string[];
uiSchema?: any;
options?: {
allowDelete?: (config: any) => boolean;
allowEdit?: (config: any) => boolean;
showAlert?: (config: any) => string | ReactNode | undefined;
reloadOnSave?: boolean;
};
properties?: {
[key in keyof Partial<Props>]: {
extract?: boolean;
hide?: boolean;
new?: Pick<SettingsNewModalProps, "schema" | "generateKey" | "uiSchema" | "anyOfValues">;
tableValues?: (config: any) => Record<string, any>[];
};
};
};
export function Setting<Schema extends TObject = any>({
schema,
uiSchema,
config,
prefix = "/",
options,
path = [],
properties
}: SettingProps<Schema>) {
const [submitting, setSubmitting] = useState(false);
const { actions } = useBknd();
const formRef = useRef<JsonSchemaFormRef>(null);
const schemaLocalModalRef = useRef<SettingsSchemaModalRef>(null);
const schemaModalRef = useRef<SettingsSchemaModalRef>(null);
const [editing, setEditing] = useState(false);
useHotkeys([
[
"mod+s",
(e) => {
e.preventDefault();
onSave();
return false;
}
],
[
"e",
() => {
if (!editing) {
onToggleEdit();
}
}
],
[
"Escape",
() => {
if (editing) {
onToggleEdit();
}
}
]
]);
const exclude = useMemo(
() =>
properties
? Object.entries(properties)
.filter(([, value]) => value.hide || value.extract)
.map(([key]) => key)
: [],
[properties]
);
const goBack = useEvent((replace?: boolean) => {
const backHref = window.location.href.split("/").slice(0, -1).join("/");
if (replace) {
window.location.replace(backHref);
} else {
window.history.back();
}
return;
});
const deleteAllowed = options?.allowDelete?.(config) ?? true;
const editAllowed = options?.allowEdit?.(config) ?? true;
const showAlert = options?.showAlert?.(config) ?? undefined;
console.log("--setting", { schema, config, prefix, path, exclude });
const [reducedSchema, reducedConfig, extracted] = extractSchema(schema, config, exclude);
console.log({ reducedSchema, reducedConfig, extracted });
const extractedKeys = Object.keys(extracted);
const selectedSubKey =
extractedKeys.find((key) => window.location.pathname.endsWith(key)) ?? extractedKeys[0];
const onToggleEdit = useEvent(() => {
if (!editAllowed) return;
setEditing((prev) => !prev);
//formRef.current?.cancel();
});
const onSave = useEvent(async () => {
if (!editAllowed || !editing) return;
if (formRef.current?.validateForm()) {
setSubmitting(true);
const data = formRef.current?.formData();
const [module, ...restOfPath] = path;
let success: boolean;
console.log("save:data", { module, restOfPath }, data);
if (restOfPath.length > 0) {
// patch
console.log("-> patch", { module, path: restOfPath.join(".") }, data);
success = await actions.patch(module as any, restOfPath.join("."), data);
} else {
// set
console.log("-> set", { module }, data);
success = await actions.set(module as any, data, true);
}
console.log("save:success", success);
if (success) {
if (options?.reloadOnSave) {
window.location.reload();
//await actions.reload();
}
} else {
setSubmitting(false);
}
}
});
const handleDelete = useEvent(async () => {
if (!deleteAllowed) return;
const [module, ...restOfPath] = path;
if (window.confirm(`Are you sure you want to delete ${path.join(".")}`)) {
if (await actions.remove(module as any, restOfPath.join("."))) {
goBack(true);
}
}
});
if (!config) {
return (
<Empty
title="Not found"
description={`Configuration at path ${path.join(".")} doesn't exist.`}
buttonText="Go back"
/>
);
}
return (
<>
<AppShell.SectionHeader
className={path.length > 1 ? "pl-3" : ""}
scrollable
right={
<>
<Dropdown
items={[
{
label: "Inspect local schema",
onClick: () => {
schemaLocalModalRef.current?.open();
}
},
{
label: "Inspect schema",
onClick: () => {
schemaModalRef.current?.open();
}
},
deleteAllowed && {
label: "Delete",
destructive: true,
onClick: handleDelete
}
]}
position="bottom-end"
>
<IconButton Icon={TbSettings} />
</Dropdown>
<Button onClick={onToggleEdit} disabled={!editAllowed}>
{editing ? "Cancel" : "Edit"}
</Button>
{editing && (
<Button
variant="primary"
onClick={onSave}
disabled={submitting || !editAllowed}
>
{submitting ? "Save..." : "Save"}
</Button>
)}
</>
}
>
<Breadcrumbs path={path} />
</AppShell.SectionHeader>
<AppShell.Scrollable key={path.join("-")}>
{typeof showAlert !== "undefined" && <Alert.Warning message={showAlert} />}
<div className="flex flex-col flex-grow p-3 gap-3">
<JsonSchemaForm
ref={formRef}
readonly={!editing}
schema={omit(reducedSchema, ["title"])}
formData={reducedConfig}
uiSchema={uiSchema}
onChange={console.log}
className="legacy hide-required-mark fieldset-alternative mute-root"
/>
</div>
{extractedKeys.length > 0 && (
<div className="flex flex-col max-w-full">
<AppShell.SectionHeaderTabs
items={extractedKeys.map((sub) => ({
as: Link,
label: ucFirst(sub),
href: `${prefix}/${sub}`.replace(/\/+/g, "/"),
active: selectedSubKey === sub,
badge: Object.keys(extracted[sub]?.config ?? {}).length
}))}
/>
<div className="flex flex-grow flex-col gap-3 p-3">
<Route
path="/:prop?"
component={({ params }) => {
const [, navigate] = useLocation();
const key = (params.prop ?? selectedSubKey) as string;
const localConfig = extracted[key]?.config ?? {};
const values = properties?.[key]?.tableValues
? properties?.[key]?.tableValues(localConfig)
: Object.entries(localConfig).map(([key, value]) => {
if (!value || typeof value !== "object") {
return {
key,
value
};
}
const fistValueKey = Object.keys(value)[0]!;
const firstValueKeyValue = value[fistValueKey];
const _value = omit(value as any, [fistValueKey]);
return {
key,
[fistValueKey]: firstValueKeyValue,
value: _value
};
});
const newSetting = properties?.[key]?.new;
if (!key) {
return (
<div className="h-80 flex justify-center align-center">
<Empty
title="No sub-setting selected"
description="Select one from the tab bar"
/>
</div>
);
}
return (
<>
{newSetting && (
<SettingNewModal
schema={newSetting.schema}
uiSchema={newSetting.uiSchema}
anyOfValues={newSetting.anyOfValues}
prefixPath={`/${key}/`}
path={[...path, key]}
generateKey={newSetting.generateKey}
/>
)}
<DataTable
data={values}
onClickRow={(row) => {
const firstKeyValue = Object.values(row)[0];
navigate(`/${key}/${firstKeyValue}`);
}}
/>
</>
);
}}
/>
</div>
</div>
)}
</AppShell.Scrollable>
<SettingSchemaModal
ref={schemaLocalModalRef}
title={path.join(".")}
tabs={[
{
title: "Schema",
json: reducedSchema
},
{
title: "Config",
json: reducedConfig
}
]}
/>
<SettingSchemaModal
ref={schemaModalRef}
title={path.join(".")}
tabs={[
{
title: "Schema",
json: schema
},
{
title: "Config",
json: config
}
]}
/>
</>
);
}