hippofish/packages/client/src/components/MkWidgets.vue

225 lines
4.6 KiB
Vue
Raw Normal View History

2021-02-16 14:17:13 +01:00
<template>
2023-04-08 02:01:42 +02:00
<div class="vjoppmmu">
<template v-if="edit">
2023-09-02 01:27:33 +02:00
<header v-focus tabindex="-1">
2023-04-08 02:01:42 +02:00
<MkSelect
v-model="widgetAdderSelected"
style="margin-bottom: var(--margin)"
class="mk-widget-select"
>
<template #label>{{ i18n.ts.selectWidget }}</template>
<option
v-for="widget in widgetDefs"
:key="widget"
:value="widget"
>
{{ i18n.t(`_widgets.${widget}`) }}
</option>
</MkSelect>
<MkButton
inline
primary
class="mk-widget-add"
@click="addWidget"
><i :class="icon('ph-plus')"></i>
2023-04-08 02:01:42 +02:00
{{ i18n.ts.add }}</MkButton
>
<MkButton inline @click="$emit('exit')">{{
i18n.ts.close
}}</MkButton>
</header>
<VueDraggable v-model="widgets_" handle=".handle" :animation="150">
2023-07-12 04:21:52 +02:00
<div v-for="element in widgets_" :key="element.id">
2023-04-08 02:01:42 +02:00
<div class="customize-container">
<button
class="config _button"
@click.prevent.stop="configWidget(element.id)"
>
<i :class="icon('ph-gear-six')"></i>
2023-04-08 02:01:42 +02:00
</button>
<button
class="remove _button"
:aria-label="i18n.ts.close"
2023-09-02 01:27:33 +02:00
@click.prevent.stop="removeWidget(element)"
2023-04-08 02:01:42 +02:00
>
<i :class="icon('ph-x')"></i>
2023-04-08 02:01:42 +02:00
</button>
<div class="handle">
<component
:is="`mkw-${element.name}`"
:ref="(el) => (widgetRefs[element.id] = el)"
class="widget"
:widget="element"
@updateProps="updateWidget(element.id, $event)"
/>
</div>
2022-06-30 12:19:54 +02:00
</div>
2023-07-12 04:20:58 +02:00
</div>
</VueDraggable>
2023-04-08 02:01:42 +02:00
</template>
<component
:is="`mkw-${widget.name}`"
v-for="widget in widgets"
v-else
:key="widget.id"
:ref="(el) => (widgetRefs[widget.id] = el)"
class="widget"
:widget="widget"
@updateProps="updateWidget(widget.id, $event)"
@contextmenu.stop="onContextmenu(widget, $event)"
/>
</div>
2021-02-16 14:17:13 +01:00
</template>
<script lang="ts" setup>
2023-09-02 01:27:33 +02:00
import { computed, ref } from "vue";
2023-04-08 02:01:42 +02:00
import { v4 as uuid } from "uuid";
import { VueDraggable } from "vue-draggable-plus";
2023-04-08 02:01:42 +02:00
import MkSelect from "@/components/form/select.vue";
import MkButton from "@/components/MkButton.vue";
import { widgets as widgetDefs } from "@/widgets";
import * as os from "@/os";
import { i18n } from "@/i18n";
import icon from "@/scripts/icon";
2023-09-02 01:27:33 +02:00
interface Widget {
name: string;
id: string;
2024-04-12 16:02:03 +02:00
data: Record<string, unknown>;
2023-09-02 01:27:33 +02:00
}
const props = defineProps<{
widgets: Widget[];
edit: boolean;
}>();
const emit = defineEmits<{
2023-04-08 02:01:42 +02:00
(ev: "updateWidgets", widgets: Widget[]): void;
(ev: "addWidget", widget: Widget): void;
(ev: "removeWidget", widget: Widget): void;
(ev: "updateWidget", widget: Partial<Widget>): void;
(ev: "exit"): void;
}>();
const widgetRefs = {};
const configWidget = (id: string) => {
widgetRefs[id].configure();
};
const widgetAdderSelected = ref(null);
const addWidget = () => {
if (widgetAdderSelected.value == null) return;
2023-04-08 02:01:42 +02:00
emit("addWidget", {
name: widgetAdderSelected.value,
id: uuid(),
data: {},
});
widgetAdderSelected.value = null;
};
const removeWidget = (widget) => {
2023-04-08 02:01:42 +02:00
emit("removeWidget", widget);
};
const updateWidget = (id, data) => {
2023-04-08 02:01:42 +02:00
emit("updateWidget", { id, data });
};
const widgets_ = computed({
get: () => props.widgets,
set: (value) => {
2023-04-08 02:01:42 +02:00
emit("updateWidgets", value);
2021-02-16 14:17:13 +01:00
},
});
2021-02-16 14:17:13 +01:00
function onContextmenu(widget: Widget, ev: MouseEvent) {
const isLink = (el: HTMLElement) => {
2023-04-08 02:01:42 +02:00
if (el.tagName === "A") return true;
if (el.parentElement) {
return isLink(el.parentElement);
}
};
2024-04-12 16:02:03 +02:00
if (isLink(ev.target as HTMLElement)) return;
2023-04-08 02:01:42 +02:00
if (
["INPUT", "TEXTAREA", "IMG", "VIDEO", "CANVAS"].includes(
2024-04-12 16:02:03 +02:00
(ev.target as HTMLElement).tagName,
2023-04-08 02:01:42 +02:00
) ||
2024-04-12 16:02:03 +02:00
(ev.target as HTMLElement).getAttribute("contentEditable")
2023-04-08 02:01:42 +02:00
)
return;
if (window.getSelection()?.toString() !== "") return;
os.contextMenu(
[
{
type: "label",
text: i18n.t(`_widgets.${widget.name}`),
},
{
icon: `${icon("ph-gear-six")}`,
2023-04-08 02:01:42 +02:00
text: i18n.ts.settings,
action: () => {
configWidget(widget.id);
},
},
],
2023-07-06 03:28:27 +02:00
ev,
2023-04-08 02:01:42 +02:00
);
}
2021-02-16 14:17:13 +01:00
</script>
<style lang="scss" scoped>
.vjoppmmu {
2023-06-26 00:19:37 +02:00
display: flex;
flex-direction: column;
flex-grow: 1;
2021-02-16 14:17:13 +01:00
> header {
margin: 16px 0;
> * {
width: 100%;
padding: 4px;
}
}
2023-04-08 02:01:42 +02:00
> .widget,
.customize-container {
contain: content;
2023-06-26 00:19:37 +02:00
margin-bottom: var(--margin);
2021-02-16 14:17:13 +01:00
&:first-of-type {
margin-top: 0;
}
}
.customize-container {
position: relative;
cursor: move;
> .config,
> .remove {
position: absolute;
z-index: 10000;
top: 8px;
width: 32px;
height: 32px;
color: #fff;
background: rgba(#000, 0.7);
border-radius: 4px;
}
> .config {
right: 8px + 8px + 32px;
}
> .remove {
right: 8px;
}
2022-06-30 12:19:54 +02:00
> .handle {
> .widget {
pointer-events: none;
}
}
2021-02-16 14:17:13 +01:00
}
}
</style>