2022-03-04 03:39:48 +00:00
|
|
|
import Col from "components/layout/Column.vue";
|
|
|
|
import Row from "components/layout/Row.vue";
|
2022-06-27 00:17:22 +00:00
|
|
|
import type { CoercableComponent, GenericComponent, JSXFunction } from "features/feature";
|
2023-02-16 02:00:36 +00:00
|
|
|
import {
|
|
|
|
Component as ComponentKey,
|
|
|
|
GatherProps,
|
2024-03-18 04:28:57 +00:00
|
|
|
Visibility,
|
2023-02-16 02:00:36 +00:00
|
|
|
isVisible,
|
2024-03-18 04:28:57 +00:00
|
|
|
jsx
|
2023-02-16 02:00:36 +00:00
|
|
|
} from "features/feature";
|
2022-06-27 00:17:22 +00:00
|
|
|
import type { ProcessedComputable } from "util/computed";
|
|
|
|
import { DoNotCache } from "util/computed";
|
2024-10-20 10:47:59 +00:00
|
|
|
import type { Component, DefineComponent, Ref, ShallowRef, UnwrapRef } from "vue";
|
2022-01-14 04:25:47 +00:00
|
|
|
import {
|
2022-01-25 04:25:34 +00:00
|
|
|
computed,
|
|
|
|
defineComponent,
|
2022-02-27 19:49:34 +00:00
|
|
|
isRef,
|
2022-03-11 23:01:22 +00:00
|
|
|
onUnmounted,
|
2022-01-25 04:25:34 +00:00
|
|
|
ref,
|
2022-02-27 19:49:34 +00:00
|
|
|
shallowRef,
|
2022-01-25 04:25:34 +00:00
|
|
|
unref,
|
2022-02-27 19:49:34 +00:00
|
|
|
watchEffect
|
2022-01-25 04:25:34 +00:00
|
|
|
} from "vue";
|
2024-03-18 04:28:57 +00:00
|
|
|
import { JSX } from "vue/jsx-runtime";
|
2023-05-21 22:27:04 +00:00
|
|
|
import { camelToKebab } from "./common";
|
2022-01-14 04:25:47 +00:00
|
|
|
|
2022-02-27 19:49:34 +00:00
|
|
|
export function coerceComponent(
|
|
|
|
component: CoercableComponent,
|
|
|
|
defaultWrapper = "span"
|
|
|
|
): DefineComponent {
|
|
|
|
if (typeof component === "function") {
|
|
|
|
return defineComponent({ render: component });
|
|
|
|
}
|
2022-01-14 04:25:47 +00:00
|
|
|
if (typeof component === "string") {
|
2022-02-27 19:49:34 +00:00
|
|
|
if (component.length > 0) {
|
|
|
|
component = component.trim();
|
|
|
|
if (component.charAt(0) !== "<") {
|
|
|
|
component = `<${defaultWrapper}>${component}</${defaultWrapper}>`;
|
|
|
|
}
|
2022-01-14 04:25:47 +00:00
|
|
|
|
2022-02-27 19:49:34 +00:00
|
|
|
return defineComponent({ template: component });
|
|
|
|
}
|
|
|
|
return defineComponent({ render: () => ({}) });
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
|
|
|
return component;
|
|
|
|
}
|
|
|
|
|
2022-08-14 03:28:11 +00:00
|
|
|
export interface VueFeature {
|
2022-02-27 19:49:34 +00:00
|
|
|
[ComponentKey]: GenericComponent;
|
|
|
|
[GatherProps]: () => Record<string, unknown>;
|
2022-08-14 03:28:11 +00:00
|
|
|
}
|
2022-02-27 19:49:34 +00:00
|
|
|
|
|
|
|
export function render(object: VueFeature | CoercableComponent): JSX.Element | DefineComponent {
|
|
|
|
if (isCoercableComponent(object)) {
|
|
|
|
if (typeof object === "function") {
|
|
|
|
return (object as JSXFunction)();
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
2022-02-27 19:49:34 +00:00
|
|
|
return coerceComponent(object);
|
|
|
|
}
|
|
|
|
const Component = object[ComponentKey];
|
|
|
|
return <Component {...object[GatherProps]()} />;
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
|
|
|
|
2022-02-27 19:49:34 +00:00
|
|
|
export function renderRow(...objects: (VueFeature | CoercableComponent)[]): JSX.Element {
|
2022-02-27 22:41:39 +00:00
|
|
|
return <Row>{objects.map(render)}</Row>;
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
|
|
|
|
2022-02-27 19:49:34 +00:00
|
|
|
export function renderCol(...objects: (VueFeature | CoercableComponent)[]): JSX.Element {
|
2022-02-27 22:41:39 +00:00
|
|
|
return <Col>{objects.map(render)}</Col>;
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
|
|
|
|
2022-04-16 05:55:35 +00:00
|
|
|
export function renderJSX(object: VueFeature | CoercableComponent): JSX.Element {
|
|
|
|
if (isCoercableComponent(object)) {
|
|
|
|
if (typeof object === "function") {
|
|
|
|
return (object as JSXFunction)();
|
|
|
|
}
|
|
|
|
if (typeof object === "string") {
|
|
|
|
return <>{object}</>;
|
|
|
|
}
|
|
|
|
// TODO why is object typed as never?
|
|
|
|
const Comp = object as DefineComponent;
|
|
|
|
return <Comp />;
|
|
|
|
}
|
|
|
|
const Component = object[ComponentKey];
|
|
|
|
return <Component {...object[GatherProps]()} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function renderRowJSX(...objects: (VueFeature | CoercableComponent)[]): JSX.Element {
|
|
|
|
return <Row>{objects.map(renderJSX)}</Row>;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function renderColJSX(...objects: (VueFeature | CoercableComponent)[]): JSX.Element {
|
|
|
|
return <Col>{objects.map(renderJSX)}</Col>;
|
|
|
|
}
|
|
|
|
|
2022-08-14 03:28:11 +00:00
|
|
|
export function joinJSX(objects: JSX.Element[], joiner: JSX.Element): JSX.Element {
|
|
|
|
return objects.reduce((acc, curr) => (
|
|
|
|
<>
|
|
|
|
{acc}
|
|
|
|
{joiner}
|
|
|
|
{curr}
|
|
|
|
</>
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2022-01-14 04:25:47 +00:00
|
|
|
export function isCoercableComponent(component: unknown): component is CoercableComponent {
|
|
|
|
if (typeof component === "string") {
|
|
|
|
return true;
|
|
|
|
} else if (typeof component === "object") {
|
|
|
|
if (component == null) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return "render" in component || "component" in component;
|
2022-02-27 19:49:34 +00:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
} else if (typeof component === "function" && (component as any)[DoNotCache] === true) {
|
|
|
|
return true;
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function setupHoldToClick(
|
2022-03-27 05:14:35 +00:00
|
|
|
onClick?: Ref<((e?: MouseEvent | TouchEvent) => void) | undefined>,
|
2022-01-14 04:25:47 +00:00
|
|
|
onHold?: Ref<VoidFunction | undefined>
|
|
|
|
): {
|
2022-03-27 05:14:35 +00:00
|
|
|
start: (e: MouseEvent | TouchEvent) => void;
|
2022-01-14 04:25:47 +00:00
|
|
|
stop: VoidFunction;
|
|
|
|
handleHolding: VoidFunction;
|
|
|
|
} {
|
2024-10-18 14:26:10 +00:00
|
|
|
const interval = ref<NodeJS.Timeout | null>(null);
|
2022-03-27 05:14:35 +00:00
|
|
|
const event = ref<MouseEvent | TouchEvent | undefined>(undefined);
|
2022-01-14 04:25:47 +00:00
|
|
|
|
2022-03-27 05:14:35 +00:00
|
|
|
function start(e: MouseEvent | TouchEvent) {
|
2024-10-18 14:26:10 +00:00
|
|
|
if (interval.value == null) {
|
2022-01-25 04:25:34 +00:00
|
|
|
interval.value = setInterval(handleHolding, 250);
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
2022-03-27 05:14:35 +00:00
|
|
|
event.value = e;
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
|
|
|
function stop() {
|
2024-10-18 14:26:10 +00:00
|
|
|
if (interval.value != null) {
|
2022-01-25 04:25:34 +00:00
|
|
|
clearInterval(interval.value);
|
|
|
|
interval.value = null;
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
function handleHolding() {
|
|
|
|
if (onHold && onHold.value) {
|
|
|
|
onHold.value();
|
|
|
|
} else if (onClick && onClick.value) {
|
2022-03-27 05:14:35 +00:00
|
|
|
onClick.value(event.value);
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-11 23:01:22 +00:00
|
|
|
onUnmounted(stop);
|
|
|
|
|
2022-01-14 04:25:47 +00:00
|
|
|
return { start, stop, handleHolding };
|
|
|
|
}
|
2022-01-25 04:25:34 +00:00
|
|
|
|
2022-05-22 17:13:47 +00:00
|
|
|
export function getFirstFeature<
|
2023-02-16 02:00:36 +00:00
|
|
|
T extends VueFeature & { visibility: ProcessedComputable<Visibility | boolean> }
|
2022-05-22 17:13:47 +00:00
|
|
|
>(
|
2022-03-27 18:47:36 +00:00
|
|
|
features: T[],
|
|
|
|
filter: (feature: T) => boolean
|
2022-05-22 17:13:47 +00:00
|
|
|
): {
|
|
|
|
firstFeature: Ref<T | undefined>;
|
|
|
|
collapsedContent: JSXFunction;
|
|
|
|
hasCollapsedContent: Ref<boolean>;
|
|
|
|
} {
|
2022-03-27 18:47:36 +00:00
|
|
|
const filteredFeatures = computed(() =>
|
2023-02-16 02:00:36 +00:00
|
|
|
features.filter(feature => isVisible(feature.visibility) && filter(feature))
|
2022-03-27 18:47:36 +00:00
|
|
|
);
|
|
|
|
return {
|
|
|
|
firstFeature: computed(() => filteredFeatures.value[0]),
|
2022-05-22 17:13:47 +00:00
|
|
|
collapsedContent: jsx(() => renderCol(...filteredFeatures.value.slice(1))),
|
|
|
|
hasCollapsedContent: computed(() => filteredFeatures.value.length > 1)
|
2022-03-27 18:47:36 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-01-25 04:25:34 +00:00
|
|
|
export function computeComponent(
|
2024-10-20 10:47:59 +00:00
|
|
|
component: Ref<CoercableComponent>,
|
2022-02-27 19:49:34 +00:00
|
|
|
defaultWrapper = "div"
|
2022-04-23 20:23:38 +00:00
|
|
|
): ShallowRef<Component | ""> {
|
|
|
|
const comp = shallowRef<Component | "">();
|
2022-02-27 19:49:34 +00:00
|
|
|
watchEffect(() => {
|
2024-10-20 10:47:59 +00:00
|
|
|
comp.value = coerceComponent(unref(component), defaultWrapper);
|
2022-01-25 04:25:34 +00:00
|
|
|
});
|
2022-04-23 19:19:39 +00:00
|
|
|
return comp as ShallowRef<Component | "">;
|
2022-01-25 04:25:34 +00:00
|
|
|
}
|
|
|
|
export function computeOptionalComponent(
|
2024-10-20 10:47:59 +00:00
|
|
|
component: Ref<CoercableComponent | undefined>,
|
2022-02-27 19:49:34 +00:00
|
|
|
defaultWrapper = "div"
|
2022-04-23 20:23:38 +00:00
|
|
|
): ShallowRef<Component | "" | null> {
|
|
|
|
const comp = shallowRef<Component | "" | null>(null);
|
2022-02-27 19:49:34 +00:00
|
|
|
watchEffect(() => {
|
2024-10-20 10:47:59 +00:00
|
|
|
const currComponent = unref(component);
|
2022-12-21 03:26:25 +00:00
|
|
|
comp.value =
|
2024-03-29 05:19:57 +00:00
|
|
|
currComponent === "" || currComponent == null
|
2022-12-21 03:26:25 +00:00
|
|
|
? null
|
|
|
|
: coerceComponent(currComponent, defaultWrapper);
|
2022-01-25 04:25:34 +00:00
|
|
|
});
|
2022-02-27 19:49:34 +00:00
|
|
|
return comp;
|
2022-01-25 04:25:34 +00:00
|
|
|
}
|
|
|
|
|
2024-10-20 10:47:59 +00:00
|
|
|
export function deepUnref<T extends object>(refObject: T): { [K in keyof T]: UnwrapRef<T[K]> } {
|
|
|
|
return (Object.keys(refObject) as (keyof T)[]).reduce(
|
|
|
|
(acc, curr) => {
|
|
|
|
acc[curr] = unref(refObject[curr]) as UnwrapRef<T[keyof T]>;
|
|
|
|
return acc;
|
|
|
|
},
|
|
|
|
{} as { [K in keyof T]: UnwrapRef<T[K]> }
|
|
|
|
);
|
2022-02-27 19:49:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function setRefValue<T>(ref: Ref<T | Ref<T>>, value: T) {
|
|
|
|
if (isRef(ref.value)) {
|
|
|
|
ref.value.value = value;
|
|
|
|
} else {
|
|
|
|
ref.value = value;
|
|
|
|
}
|
2022-01-25 04:25:34 +00:00
|
|
|
}
|
|
|
|
|
2022-03-09 01:40:51 +00:00
|
|
|
export type PropTypes =
|
2022-01-25 04:25:34 +00:00
|
|
|
| typeof Boolean
|
|
|
|
| typeof String
|
|
|
|
| typeof Number
|
|
|
|
| typeof Function
|
|
|
|
| typeof Object
|
|
|
|
| typeof Array;
|
2022-07-26 02:29:56 +00:00
|
|
|
|
|
|
|
export function trackHover(element: VueFeature): Ref<boolean> {
|
|
|
|
const isHovered = ref(false);
|
|
|
|
|
|
|
|
const elementGatherProps = element[GatherProps].bind(element);
|
|
|
|
element[GatherProps] = () => ({
|
2022-07-28 03:21:20 +00:00
|
|
|
...elementGatherProps(),
|
|
|
|
onPointerenter: () => (isHovered.value = true),
|
|
|
|
onPointerleave: () => (isHovered.value = false)
|
2022-07-26 02:29:56 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
return isHovered;
|
|
|
|
}
|
2023-05-21 22:27:04 +00:00
|
|
|
|
|
|
|
export function kebabifyObject(object: Record<string, unknown>) {
|
2024-10-20 10:47:59 +00:00
|
|
|
return Object.keys(object).reduce(
|
|
|
|
(acc, curr) => {
|
|
|
|
acc[camelToKebab(curr)] = object[curr];
|
|
|
|
return acc;
|
|
|
|
},
|
|
|
|
{} as Record<string, unknown>
|
|
|
|
);
|
2023-05-21 22:27:04 +00:00
|
|
|
}
|