schlechtenburg/packages/layout/lib/edit.tsx

203 lines
5.1 KiB
TypeScript
Raw Normal View History

2020-05-20 14:21:08 +00:00
import {
2020-12-27 21:32:43 +00:00
defineComponent,
2020-05-20 14:21:08 +00:00
reactive,
2020-05-24 15:33:25 +00:00
computed,
watch,
2020-05-24 20:00:14 +00:00
PropType,
2020-12-27 21:32:43 +00:00
} from 'vue';
2020-05-20 14:21:08 +00:00
import {
model,
IBlockData,
2022-03-12 16:16:24 +00:00
OnUpdateSelfCb,
2020-12-30 13:34:23 +00:00
useActivation,
2020-05-25 18:07:34 +00:00
2020-12-30 13:34:23 +00:00
SbBlock,
SbButton,
SbToolbar,
SbBlockPlaceholder,
SbBlockOrdering,
} from '@schlechtenburg/core';
2020-05-25 18:07:34 +00:00
2020-05-24 20:00:14 +00:00
import {
2022-03-11 17:23:14 +00:00
ILayoutData,
2020-05-24 20:00:14 +00:00
getDefaultData,
} from './util';
2020-05-20 14:21:08 +00:00
2020-05-24 20:00:14 +00:00
import './style.scss';
2020-05-20 14:21:08 +00:00
export default defineComponent({
2020-05-24 20:00:14 +00:00
name: 'sb-layout-edit',
2020-05-20 14:21:08 +00:00
model,
props: {
2022-03-12 16:16:24 +00:00
onUpdate: {
type: (null as unknown) as PropType<OnUpdateSelfCb<ILayoutData>>,
default: () => {},
},
2020-05-24 20:00:14 +00:00
data: {
2022-03-11 17:23:14 +00:00
type: (null as unknown) as PropType<ILayoutData>,
2020-05-24 20:00:14 +00:00
default: getDefaultData,
},
2020-05-20 14:21:08 +00:00
},
2021-03-08 15:29:35 +00:00
setup(props) {
const { activate } = useActivation();
2020-05-20 14:21:08 +00:00
2022-03-11 17:23:14 +00:00
const localData: ILayoutData = reactive({
2020-05-24 15:33:25 +00:00
orientation: props.data.orientation,
children: [...props.data.children],
2020-05-20 14:21:08 +00:00
});
2020-05-24 15:33:25 +00:00
watch(() => props.data, () => {
localData.orientation = props.data.orientation;
localData.children = [...props.data.children];
2020-05-20 14:21:08 +00:00
});
2020-05-24 15:33:25 +00:00
const classes = computed(() => ({
2020-05-20 14:21:08 +00:00
'sb-layout': true,
[`sb-layout_${localData.orientation}`]: true,
2020-05-24 15:33:25 +00:00
}));
2020-05-20 14:21:08 +00:00
const toggleOrientation = () => {
2021-02-22 18:13:37 +00:00
props.onUpdate({
2020-05-20 14:21:08 +00:00
orientation: localData.orientation === 'vertical' ? 'horizontal' : 'vertical',
});
};
const onChildUpdate = (child: IBlockData<any>, updated: IBlockData<any>) => {
2020-05-20 14:21:08 +00:00
const index = localData.children.indexOf(child);
2020-05-27 18:36:46 +00:00
if (index === -1) {
return;
}
2021-02-22 18:13:37 +00:00
props.onUpdate({
2020-05-20 14:21:08 +00:00
children: [
...localData.children.slice(0, index),
2020-05-24 15:33:25 +00:00
{
...child,
...updated,
},
2020-05-20 14:21:08 +00:00
...localData.children.slice(index + 1),
],
});
};
const appendBlock = (block: IBlockData<any>) => {
2020-05-27 18:36:46 +00:00
localData.children = [
...localData.children,
block,
];
2021-02-22 18:13:37 +00:00
props.onUpdate({ children: [...localData.children] });
2021-03-07 17:47:28 +00:00
activate(block.id);
2020-05-20 14:21:08 +00:00
};
const insertBlock = (index: number, block: IBlockData<any>) => {
2020-05-27 18:36:46 +00:00
localData.children = [
...localData.children.slice(0, index + 1),
block,
...localData.children.slice(index + 1),
];
2021-02-22 18:13:37 +00:00
props.onUpdate({ children: [...localData.children] });
2021-03-07 17:47:28 +00:00
activate(block.id);
2020-05-24 15:33:25 +00:00
};
2020-05-27 18:36:46 +00:00
const removeBlock = (index: number) => {
localData.children = [
...localData.children.slice(0, index),
...localData.children.slice(index + 1),
];
2021-02-22 18:13:37 +00:00
props.onUpdate({ children: [...localData.children] });
2020-05-27 18:36:46 +00:00
const newActiveIndex = Math.max(index - 1, 0);
2021-03-07 17:47:28 +00:00
activate(localData.children[newActiveIndex].id);
2020-05-27 18:36:46 +00:00
};
2021-02-22 18:13:37 +00:00
const activateBlock = (index: number) => {
const safeIndex =
Math.max(
Math.min(
localData.children.length - 1,
index,
),
0,
);
2021-03-07 17:47:28 +00:00
activate(localData.children[safeIndex].id);
2021-02-22 18:13:37 +00:00
};
const moveBackward = (index: number) => {
2020-05-28 20:16:35 +00:00
if (index === 0) {
return;
}
const curr = localData.children[index];
const prev = localData.children[index - 1];
localData.children = [
...localData.children.slice(0, index - 1),
curr,
prev,
...localData.children.slice(index + 1),
];
2021-02-22 18:13:37 +00:00
props.onUpdate({ children: [...localData.children] });
2020-05-28 20:16:35 +00:00
};
2021-02-22 18:13:37 +00:00
const moveForward = (index: number) => {
2020-05-28 20:16:35 +00:00
if (index === localData.children.length - 1) {
return;
}
const curr = localData.children[index];
const next = localData.children[index + 1];
localData.children = [
...localData.children.slice(0, index),
next,
curr,
...localData.children.slice(index + 2),
];
2021-02-22 18:13:37 +00:00
props.onUpdate({ children: [...localData.children] });
2020-05-28 20:16:35 +00:00
};
2020-05-25 21:10:21 +00:00
return () => (
<div class={classes.value}>
2020-12-30 01:32:46 +00:00
<SbToolbar>
2020-05-27 13:57:57 +00:00
<SbButton
2021-03-08 15:29:35 +00:00
{...{
type: 'button',
onClick: toggleOrientation,
}}
2020-05-27 13:57:57 +00:00
>{localData.orientation}</SbButton>
2020-05-20 14:21:08 +00:00
</SbToolbar>
2022-03-17 17:59:51 +00:00
<>
{...localData.children.map((child, index) => (
<SbBlock
{...{ key: child.id }}
data-order={index}
block={child}
onUpdate={(updated: IBlockData<any>) => onChildUpdate(child, updated)}
onRemoveSelf={() => removeBlock(index)}
onPrependBlock={(block: IBlockData<any>) => insertBlock(index - 1, block)}
onAppendBlock={(block: IBlockData<any>) => insertBlock(index, block)}
onActivatePrevious={() => activateBlock(index - 1,)}
onActivateNext={() => activateBlock(index + 1,)}
>
{{
'context-toolbar': () =>
<SbBlockOrdering
onMoveBackward={() => moveBackward(index)}
onMoveForward={() => moveForward(index)}
onRemove={() => removeBlock(index)}
orientation={localData.orientation}
/>,
}}
</SbBlock>
))}
</>
<SbBlockPlaceholder onInsertBlock={appendBlock}></SbBlockPlaceholder>
2020-05-24 15:33:25 +00:00
</div>
2020-05-20 14:21:08 +00:00
);
},
});