mirror of
https://github.com/zadam/trilium.git
synced 2025-12-13 02:44:24 +01:00
241 lines
8.0 KiB
TypeScript
241 lines
8.0 KiB
TypeScript
import { t } from "i18next";
|
|
import FNote from "../../entities/fnote";
|
|
import attributes from "../../services/attributes";
|
|
import NoteContextAwareWidget from "../note_context_aware_widget";
|
|
import { DEFAULT_MAP_LAYER_NAME, MAP_LAYERS, type MapLayer } from "../collections/geomap/map_layer";
|
|
import { ViewTypeOptions } from "../collections/interface";
|
|
import { FilterLabelsByType } from "@triliumnext/commons";
|
|
import { DEFAULT_THEME, getPresentationThemes } from "../collections/presentation/themes";
|
|
import { VNode } from "preact";
|
|
import { useNoteLabel } from "../react/hooks";
|
|
import { FormDropdownDivider, FormListItem } from "../react/FormList";
|
|
import Component from "../../components/component";
|
|
|
|
interface BookConfig {
|
|
properties: BookProperty[];
|
|
}
|
|
|
|
export interface CheckBoxProperty {
|
|
type: "checkbox",
|
|
label: string;
|
|
bindToLabel: FilterLabelsByType<boolean>;
|
|
icon?: string;
|
|
}
|
|
|
|
export interface ButtonProperty {
|
|
type: "button",
|
|
label: string;
|
|
title?: string;
|
|
icon?: string;
|
|
onClick(context: BookContext): void;
|
|
}
|
|
|
|
export interface SplitButtonProperty extends Omit<ButtonProperty, "type"> {
|
|
type: "split-button";
|
|
items({ note, parentComponent }: { note: FNote, parentComponent: Component }): VNode;
|
|
}
|
|
|
|
export interface NumberProperty {
|
|
type: "number",
|
|
label: string;
|
|
bindToLabel: FilterLabelsByType<number>;
|
|
width?: number;
|
|
min?: number;
|
|
icon?: string;
|
|
disabled?: (note: FNote) => boolean;
|
|
}
|
|
|
|
export interface ComboBoxItem {
|
|
value: string;
|
|
label: string;
|
|
}
|
|
|
|
interface ComboBoxGroup {
|
|
title: string;
|
|
items: ComboBoxItem[];
|
|
}
|
|
|
|
export interface ComboBoxProperty {
|
|
type: "combobox",
|
|
label: string;
|
|
icon?: string;
|
|
bindToLabel: FilterLabelsByType<string>;
|
|
/**
|
|
* The default value is used when the label is not set.
|
|
*/
|
|
defaultValue?: string;
|
|
options: (ComboBoxItem | ComboBoxGroup)[];
|
|
}
|
|
|
|
export type BookProperty = CheckBoxProperty | ButtonProperty | NumberProperty | ComboBoxProperty | SplitButtonProperty;
|
|
|
|
interface BookContext {
|
|
note: FNote;
|
|
triggerCommand: NoteContextAwareWidget["triggerCommand"];
|
|
}
|
|
|
|
export const bookPropertiesConfig: Record<ViewTypeOptions, BookConfig> = {
|
|
grid: {
|
|
properties: []
|
|
},
|
|
list: {
|
|
properties: [
|
|
{
|
|
label: t("book_properties.collapse"),
|
|
title: t("book_properties.collapse_all_notes"),
|
|
type: "button",
|
|
icon: "bx bx-layer-minus",
|
|
async onClick({ note, triggerCommand }) {
|
|
const { noteId } = note;
|
|
|
|
// owned is important - we shouldn't remove inherited expanded labels
|
|
for (const expandedAttr of note.getOwnedLabels("expanded")) {
|
|
await attributes.removeAttributeById(noteId, expandedAttr.attributeId);
|
|
}
|
|
|
|
triggerCommand("refreshNoteList", { noteId });
|
|
},
|
|
},
|
|
{
|
|
label: t("book_properties.expand"),
|
|
title: t("book_properties.expand_tooltip"),
|
|
type: "split-button",
|
|
icon: "bx bx-move-vertical",
|
|
onClick: buildExpandListHandler(1),
|
|
items: ListExpandDepth
|
|
}
|
|
]
|
|
},
|
|
calendar: {
|
|
properties: [
|
|
{
|
|
label: t("book_properties_config.hide-weekends"),
|
|
icon: "bx bx-calendar-week",
|
|
type: "checkbox",
|
|
bindToLabel: "calendar:hideWeekends"
|
|
},
|
|
{
|
|
label: t("book_properties_config.display-week-numbers"),
|
|
icon: "bx bx-hash",
|
|
type: "checkbox",
|
|
bindToLabel: "calendar:weekNumbers"
|
|
}
|
|
]
|
|
},
|
|
geoMap: {
|
|
properties: [
|
|
{
|
|
label: t("book_properties_config.map-style"),
|
|
icon: "bx bx-palette",
|
|
type: "combobox",
|
|
bindToLabel: "map:style",
|
|
defaultValue: DEFAULT_MAP_LAYER_NAME,
|
|
options: [
|
|
{
|
|
title: t("book_properties_config.raster"),
|
|
items: Object.entries(MAP_LAYERS)
|
|
.filter(([_, layer]) => layer.type === "raster")
|
|
.map(buildMapLayer)
|
|
},
|
|
{
|
|
title: t("book_properties_config.vector_light"),
|
|
items: Object.entries(MAP_LAYERS)
|
|
.filter(([_, layer]) => layer.type === "vector" && !layer.isDarkTheme)
|
|
.map(buildMapLayer)
|
|
},
|
|
{
|
|
title: t("book_properties_config.vector_dark"),
|
|
items: Object.entries(MAP_LAYERS)
|
|
.filter(([_, layer]) => layer.type === "vector" && layer.isDarkTheme)
|
|
.map(buildMapLayer)
|
|
}
|
|
]
|
|
},
|
|
{
|
|
label: t("book_properties_config.show-scale"),
|
|
icon: "bx bx-ruler",
|
|
type: "checkbox",
|
|
bindToLabel: "map:scale"
|
|
}
|
|
]
|
|
},
|
|
table: {
|
|
properties: [
|
|
{
|
|
label: t("book_properties_config.max-nesting-depth"),
|
|
icon: "bx bx-subdirectory-right",
|
|
type: "number",
|
|
bindToLabel: "maxNestingDepth",
|
|
width: 65,
|
|
disabled: (note) => note.type === "search"
|
|
}
|
|
]
|
|
},
|
|
board: {
|
|
properties: []
|
|
},
|
|
presentation: {
|
|
properties: [
|
|
{
|
|
label: "Theme",
|
|
type: "combobox",
|
|
icon: "bx bx-palette",
|
|
bindToLabel: "presentation:theme",
|
|
defaultValue: DEFAULT_THEME,
|
|
options: getPresentationThemes().map(theme => ({
|
|
value: theme.id,
|
|
label: theme.name
|
|
}))
|
|
}
|
|
]
|
|
}
|
|
};
|
|
|
|
function buildMapLayer([ id, layer ]: [ string, MapLayer ]): ComboBoxItem {
|
|
return {
|
|
value: id,
|
|
label: layer.name
|
|
};
|
|
}
|
|
|
|
function ListExpandDepth(context: { note: FNote, parentComponent: Component }) {
|
|
const [ currentDepth ] = useNoteLabel(context.note, "expanded");
|
|
|
|
return (
|
|
<>
|
|
<ListExpandDepthButton label={t("book_properties.expand_first_level")} depth={1} {...context} checked={currentDepth === ""} />
|
|
<FormDropdownDivider />
|
|
{Array.from({ length: 4 }, (_, i) => i + 2).map(depth => [
|
|
<ListExpandDepthButton label={t("book_properties.expand_nth_level", { depth })} depth={depth} {...context} checked={!!currentDepth && parseInt(currentDepth, 10) === depth} />
|
|
])}
|
|
<FormDropdownDivider />
|
|
<ListExpandDepthButton label={t("book_properties.expand_all_levels")} depth="all" checked={currentDepth === "all"} {...context} />
|
|
</>
|
|
)
|
|
}
|
|
|
|
function ListExpandDepthButton({ label, depth, note, parentComponent, checked }: { label: string, depth: number | "all", note: FNote, parentComponent: Component, checked?: boolean }) {
|
|
const handler = buildExpandListHandler(depth);
|
|
|
|
return (
|
|
<FormListItem
|
|
onClick={() => handler({ note, triggerCommand: parentComponent.triggerCommand.bind(parentComponent) })}
|
|
checked={checked}
|
|
>{label}</FormListItem>
|
|
);
|
|
}
|
|
|
|
function buildExpandListHandler(depth: number | "all") {
|
|
return async ({ note, triggerCommand }: BookContext) => {
|
|
const { noteId } = note;
|
|
|
|
const existingValue = note.getLabelValue("expanded");
|
|
let newValue: string | undefined = typeof depth === "number" ? depth.toString() : depth;
|
|
if (depth === 1) newValue = undefined; // maintain existing behaviour
|
|
if (newValue === existingValue) return;
|
|
|
|
await attributes.setLabel(noteId, "expanded", newValue);
|
|
triggerCommand("refreshNoteList", { noteId });
|
|
}
|
|
}
|