feat(editor,form,schema): 组件样式配置可视化

This commit is contained in:
moonszhang 2024-12-30 12:17:43 +00:00 committed by roymondchen
parent ec94eed787
commit 246e6941d5
56 changed files with 1491 additions and 312 deletions

View File

@ -1,6 +1,7 @@
import { describe, expect, test } from 'vitest';
import path from 'node:path';
import { describe, expect, test } from 'vitest';
import Core from '../src/Core';
describe('Core', () => {

View File

@ -1,10 +1,12 @@
<template>
<component class="tmagic-design-icon" :is="uiComponent">
<component class="tmagic-design-icon" :is="uiComponent" v-bind="uiProps">
<slot></slot>
</component>
</template>
<script setup lang="ts">
import { computed } from 'vue';
import { getDesignConfig } from './config';
import { IconProps } from './types';
@ -14,5 +16,6 @@ defineOptions({
const ui = getDesignConfig('components')?.icon;
const uiComponent = ui?.component || 'el-icon';
defineProps<IconProps>();
const props = defineProps<IconProps>();
const uiProps = computed(() => ui?.props(props) || props);
</script>

View File

@ -0,0 +1,69 @@
<template>
<div class="m-fields-style-setter">
<TMagicCollapse :model-value="collapseValue">
<template v-for="(item, index) in list" :key="index">
<TMagicCollapseItem :name="`${index}`">
<template #title><MIcon :icon="Grid"></MIcon>{{ item.title }}</template>
<component v-if="item.component" :is="item.component" :values="model[name]" @change="change"></component>
</TMagicCollapseItem>
</template>
</TMagicCollapse>
</div>
</template>
<script setup lang="ts">
import { shallowRef } from 'vue';
import { Grid } from '@element-plus/icons-vue';
import { TMagicCollapse, TMagicCollapseItem } from '@tmagic/design';
import type { ContainerChangeEventData, FieldProps } from '@tmagic/form';
import type { StyleSchema } from '@tmagic/schema';
import MIcon from '@editor/components/Icon.vue';
import { Background, Border, Font, Layout, Position } from './pro/';
defineOptions({
name: 'MFieldsStyleSetter',
});
defineProps<FieldProps<StyleSchema>>();
const emit = defineEmits<{
change: [v: any, eventData: ContainerChangeEventData];
}>();
const list = [
{
name: 'font',
title: '布局',
component: Layout,
},
{
title: '文字',
component: Font,
},
{
title: '背景',
component: Background,
},
{
title: '位置',
component: Position,
},
{
title: '边框与圆角',
component: Border,
},
];
const collapseValue = shallowRef(
Array(list.length)
.fill(1)
.map((x, i) => `${i}`),
);
const change = (v: any, eventData: ContainerChangeEventData) => {
emit('change', v, eventData);
};
</script>

View File

@ -0,0 +1,71 @@
<template>
<div class="background-position-container">
<div class="presets-value-list">
<TMagicButton
v-for="(item, index) in list"
:key="index"
link
:class="model[name] === item.value && 'btn-active'"
@click="changeHandler(item.value)"
>
<div :class="['position-icon', item.class, model[name] === item.value && 'active']"></div>
</TMagicButton>
</div>
<div class="custom-value">
<TMagicInput v-model="model[name]" size="small" placeholder="自定义背景位置" clearable @change="changeHandler">
</TMagicInput>
</div>
</div>
</template>
<script lang="ts" setup>
import { TMagicButton, TMagicInput } from '@tmagic/design';
import type { FieldProps, FormItem } from '@tmagic/form';
const emit = defineEmits(['change']);
defineProps<FieldProps<{ type: 'style-setter' } & FormItem>>();
const horizontalList = [
{
value: 'left',
text: '左',
},
{
value: 'center',
text: '中',
},
{
value: 'right',
text: '右',
},
];
const verticalList = [
{
value: 'top',
text: '上',
},
{
value: 'center',
text: '中',
},
{
value: 'bottom',
text: '下',
},
];
const list = verticalList
.map((vertical) =>
horizontalList.map((horizontal) => ({
value: `${horizontal.value} ${vertical.value}`,
tips: `${horizontal.text}${vertical.text}`,
class: `${horizontal.value}-${vertical.value}`,
})),
)
.flat();
const changeHandler = (v: string) => {
emit('change', v);
};
</script>

View File

@ -0,0 +1,104 @@
<template>
<div class="border-box-container">
<div class="border-icon-container">
<div class="border-icon-container-row">
<div
class="border-icon border-icon-top"
:class="{ active: direction === 'Top' }"
@click="selectDirection('Top')"
></div>
</div>
<div class="border-icon-container-row">
<div
class="border-icon border-icon-left"
:class="{ active: direction === 'Left' }"
@click="selectDirection('Left')"
></div>
<div class="border-icon" :class="{ active: direction === '' }" @click="selectDirection()"></div>
<div
class="border-icon border-icon-right"
:class="{ active: direction === 'Right' }"
@click="selectDirection('Right')"
></div>
</div>
<div class="border-icon-container-row">
<div
class="border-icon border-icon-bottom"
:class="{ active: direction === 'Bottom' }"
@click="selectDirection('Bottom')"
></div>
</div>
</div>
<div class="border-value-container">
<MContainer :config="config" :model="model" @change="change"></MContainer>
</div>
</div>
</template>
<script lang="ts" setup>
import { computed, ref } from 'vue';
import type { ContainerChangeEventData, FormValue } from '@tmagic/form';
import { MContainer } from '@tmagic/form';
import type { StyleSchema } from '@tmagic/schema';
const direction = ref('');
const config = computed(() => ({
items: [
{
name: `border${direction.value}Width`,
text: '边框宽度',
labelWidth: '68px',
type: 'data-source-field-select',
fieldConfig: {
type: 'text',
},
},
{
name: `border${direction.value}Color`,
text: '边框颜色',
labelWidth: '68px',
type: 'data-source-field-select',
fieldConfig: {
type: 'colorPicker',
},
},
{
name: `border${direction.value}Style`,
text: '边框样式',
labelWidth: '68px',
type: 'data-source-field-select',
fieldConfig: {
type: 'select',
options: ['solid', 'dashed', 'dotted'].map((item) => ({
value: item,
text: item,
})),
},
},
],
}));
const selectDirection = (d?: string) => (direction.value = d || '');
const emit = defineEmits<{
change: [v: StyleSchema, eventData: ContainerChangeEventData];
}>();
withDefaults(
defineProps<{
model: FormValue;
}>(),
{},
);
const change = (value: StyleSchema, eventData: ContainerChangeEventData) => {
eventData.changeRecords?.forEach((record) => {
emit('change', record.value, {
modifyKey: record.propPath,
});
});
};
</script>

View File

@ -0,0 +1,68 @@
<template>
<div class="layout-box-container">
<div v-for="(item, index) in list" :key="index" :class="item.class">
<span class="help-txt" v-if="item.text">{{ item.text }}</span>
<span class="next-input">
<input v-model="model[item.name]" @change="change($event, item.name)" placeholder="0" />
</span>
</div>
</div>
</template>
<script lang="ts" setup>
import type { ContainerChangeEventData, FormValue } from '@tmagic/form';
const list = [
{
name: 'marginTop',
class: 'outer-top-border',
},
{
name: 'marginRight',
class: 'outer-right-border',
},
{
name: 'marginBottom',
text: 'MARGIN',
class: 'outer-bottom-border',
},
{
name: 'marginLeft',
class: 'outer-left-border',
},
{
name: 'paddingTop',
class: 'inner-top-border',
},
{
name: 'paddingRight',
class: 'inner-right-border',
},
{
name: 'paddingBottom',
text: 'PADDING',
class: 'inner-bottom-border',
},
{
name: 'paddingLeft',
class: 'inner-left-border',
},
];
const emit = defineEmits<{
change: [v: string, eventData: ContainerChangeEventData];
}>();
withDefaults(
defineProps<{
model: FormValue;
}>(),
{},
);
const change = (event: Event, name: string) => {
emit('change', (event.target as HTMLInputElement).value, {
modifyKey: name,
});
};
</script>

View File

@ -0,0 +1,49 @@
<template>
<div class="layout-box-container">
<div v-for="(item, index) in list" :key="index" :class="item.class">
<span class="next-input">
<input v-model="model[item.name]" @change="change($event, item.name)" placeholder="0" />
</span>
</div>
</div>
</template>
<script lang="ts" setup>
import type { ContainerChangeEventData, FormValue } from '@tmagic/form';
const list = [
{
name: 'top',
class: 'outer-top-border',
},
{
name: 'right',
class: 'outer-right-border',
},
{
name: 'bottom',
class: 'outer-bottom-border',
},
{
name: 'left',
class: 'outer-left-border',
},
];
const emit = defineEmits<{
change: [v: string, eventData: ContainerChangeEventData];
}>();
withDefaults(
defineProps<{
model: FormValue;
}>(),
{},
);
const change = (event: Event, name: string) => {
emit('change', (event.target as HTMLInputElement).value, {
modifyKey: name,
});
};
</script>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M960 128a64 64 0 0 1 64 64v640a64 64 0 0 1-64 64H64a64 64 0 0 1-64-64V192a64 64 0 0 1 64-64h896z m0 64H64v640h896V192z m-96 64a32 32 0 0 1 32 32v448a32 32 0 0 1-32 32H160a32 32 0 0 1-32-32V288a32 32 0 0 1 32-32h704z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M960 128a64 64 0 0 1 64 64v640a64 64 0 0 1-64 64H64a64 64 0 0 1-64-64V192a64 64 0 0 1 64-64h896z m0 64H64v640h896V192z m-96 64a32 32 0 0 1 32 32v448a32 32 0 0 1-32 32H160a32 32 0 0 1-32-32V288a32 32 0 0 1 32-32h704z"
></path>
</svg>
</template>

View File

@ -0,0 +1,12 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M1024 0v1024H0V0h1024z m-73.142857 73.142857H73.142857v877.714286h877.714286V73.142857z"
fill="#1D1F24"
></path>
<path
d="M512 219.428571v292.571429H219.428571V219.428571h292.571429zM438.857143 292.571429H292.571429v146.285714h146.285714V292.571429z"
fill="#1D1F24"
></path>
</svg>
</template>

View File

@ -0,0 +1,3 @@
export { default as BackgroundPositionLeftTop } from './LeftTop.vue';
export { default as DisplayFlex } from './Flex.vue';
export { default as DisplayInline } from './Inline.vue';

View File

@ -0,0 +1,8 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M912.526651 867.741144 555.540144 510.712681l356.986507-357.000833c11.171434-11.18576 11.171434-29.257348 0-40.443108-11.20111-11.18576-29.272697-11.18576-40.444131 0L515.096013 470.267527 158.096203 113.267716c-11.187807-11.159154-29.258371-11.159154-40.444131 0-11.186783 11.186783-11.186783 29.286 0 40.47176L474.623229 510.712681 117.623419 867.741144c-11.159154 11.172457-11.159154 29.216415 0 40.443108 11.18576 11.17348 29.284977 11.17348 40.47176 0l357.000833-357.027439 356.985484 357.027439c11.171434 11.17348 29.243021 11.17348 40.444131 0C923.698085 896.957559 923.725714 878.913601 912.526651 867.741144z"
fill="#5D5D5D"
></path>
</svg>
</template>

View File

@ -0,0 +1,31 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M884.736 102.4l-147.456 0c-20.48 0-36.864 16.384-36.864 36.864l0 147.456C696.32 311.296 712.704 327.68 733.184 327.68l147.456 0c20.48 0 36.864-16.384 36.864-36.864L917.504 139.264C921.6 118.784 905.216 102.4 884.736 102.4zM884.736 290.816l-147.456 0L737.28 139.264l147.456 0L884.736 290.816z"
></path>
<path
d="M884.736 696.32l-147.456 0c-20.48 0-36.864 16.384-36.864 36.864l0 147.456c0 20.48 16.384 36.864 36.864 36.864l147.456 0c20.48 0 36.864-16.384 36.864-36.864l0-147.456C921.6 712.704 905.216 696.32 884.736 696.32zM884.736 884.736l-147.456 0 0-147.456 147.456 0L884.736 884.736z"
></path>
<path
d="M884.736 401.408l-147.456 0c-20.48 0-36.864 16.384-36.864 36.864l0 147.456c0 20.48 16.384 36.864 36.864 36.864l147.456 0c20.48 0 36.864-16.384 36.864-36.864l0-147.456C921.6 417.792 905.216 401.408 884.736 401.408zM884.736 585.728l-147.456 0 0-147.456 147.456 0L884.736 585.728z"
></path>
<path
d="M585.728 401.408l-147.456 0c-20.48 0-36.864 16.384-36.864 36.864l0 147.456c0 20.48 16.384 36.864 36.864 36.864l147.456 0c20.48 0 36.864-16.384 36.864-36.864l0-147.456C622.592 417.792 606.208 401.408 585.728 401.408zM585.728 585.728l-147.456 0 0-147.456 147.456 0L585.728 585.728z"
></path>
<path
d="M585.728 102.4l-147.456 0c-20.48 0-36.864 16.384-36.864 36.864l0 147.456c0 20.48 16.384 36.864 36.864 36.864l147.456 0c20.48 0 36.864-16.384 36.864-36.864L622.592 139.264C622.592 118.784 606.208 102.4 585.728 102.4zM585.728 290.816l-147.456 0L438.272 139.264l147.456 0L585.728 290.816z"
></path>
<path
d="M585.728 696.32l-147.456 0c-20.48 0-36.864 16.384-36.864 36.864l0 147.456c0 20.48 16.384 36.864 36.864 36.864l147.456 0c20.48 0 36.864-16.384 36.864-36.864l0-147.456C622.592 712.704 606.208 696.32 585.728 696.32zM585.728 884.736l-147.456 0 0-147.456 147.456 0L585.728 884.736z"
></path>
<path
d="M290.816 696.32 139.264 696.32c-20.48 0-36.864 16.384-36.864 36.864l0 147.456c0 20.48 16.384 36.864 36.864 36.864l147.456 0c20.48 0 36.864-16.384 36.864-36.864l0-147.456C327.68 712.704 311.296 696.32 290.816 696.32zM290.816 884.736 139.264 884.736l0-147.456 147.456 0L286.72 884.736z"
></path>
<path
d="M290.816 401.408 139.264 401.408c-20.48 0-36.864 16.384-36.864 36.864l0 147.456c0 20.48 16.384 36.864 36.864 36.864l147.456 0c20.48 0 36.864-16.384 36.864-36.864l0-147.456C327.68 417.792 311.296 401.408 290.816 401.408zM290.816 585.728 139.264 585.728l0-147.456 147.456 0L286.72 585.728z"
></path>
<path
d="M290.816 102.4 139.264 102.4c-20.48 0-36.864 16.384-36.864 36.864l0 147.456C102.4 311.296 118.784 327.68 139.264 327.68l147.456 0C311.296 327.68 327.68 311.296 327.68 290.816L327.68 139.264C327.68 118.784 311.296 102.4 290.816 102.4zM290.816 290.816 139.264 290.816 139.264 139.264l147.456 0L286.72 290.816z"
></path>
</svg>
</template>

View File

@ -0,0 +1,8 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M796.444444 170.666667h-227.555555v682.666666h227.555555V170.666667z m-56.888888 625.777777h-113.777778V227.555556h113.777778v568.888888zM455.111111 170.666667H227.555556v682.666666h227.555555V170.666667zM398.222222 796.444444H284.444444V227.555556h113.777778v568.888888zM910.222222 56.888889h56.888889v910.222222h-56.888889zM56.888889 56.888889h56.888889v910.222222H56.888889z"
fill="#333333"
></path>
</svg>
</template>

View File

@ -0,0 +1,8 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M170.666667 227.555556v227.555555h682.666666V227.555556H170.666667z m625.777777 56.888888v113.777778H227.555556V284.444444h568.888888zM170.666667 568.888889v227.555555h682.666666v-227.555555H170.666667z m625.777777 56.888889v113.777778H227.555556v-113.777778h568.888888zM56.888889 56.888889h910.222222v56.888889H56.888889zM56.888889 910.222222h910.222222v56.888889H56.888889z"
fill="#333333"
></path>
</svg>
</template>

View File

@ -0,0 +1,4 @@
export { default as BackgroundRepeat } from './Repeat.vue';
export { default as BackgroundRepeatX } from './RepeatX.vue';
export { default as BackgroundRepeatY } from './RepeatY.vue';
export { default as BackgroundNoRepeat } from './NoRepeat.vue';

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M960 128a64 64 0 0 1 64 64v640a64 64 0 0 1-64 64H64a64 64 0 0 1-64-64V192a64 64 0 0 1 64-64h896z m0 64H64v640h896V192z m-96 64a32 32 0 0 1 32 32v448a32 32 0 0 1-32 32H160a32 32 0 0 1-32-32V288a32 32 0 0 1 32-32h704z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M960 128a64 64 0 0 1 64 64v640a64 64 0 0 1-64 64H64a64 64 0 0 1-64-64V192a64 64 0 0 1 64-64h896z m0 64H64v640h896V192zM352 256v512H160V256h192z m256 0v512h-192V256h192z m256 0v512h-192V256h192z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M787.616 651.36a142.72 142.72 0 0 0 52.864 31.904 205.504 205.504 0 0 0 67.264 10.848 245.76 245.76 0 0 0 38.784-2.4c9.504-1.6 19.36-4 30.208-7.04 8.16-2.24 15.968-5.12 23.36-8.32l23.392-10.816V587.52h-8.736c-3.968 3.424-8.96 7.648-15.104 12.544a190.272 190.272 0 0 1-19.936 13.504 120.32 120.32 0 0 1-27.616 12.544 102.528 102.528 0 0 1-62.72-0.576 80 80 0 0 1-29.824-17.088 89.28 89.28 0 0 1-21.664-32.512c-5.696-13.664-8.64-30.784-8.64-51.296 0-19.744 2.72-36.384 7.968-50.08 5.248-13.664 12.384-24.896 20.992-33.056 9.12-8.832 18.816-14.816 29.632-18.816a101.824 101.824 0 0 1 63.68-0.736c9.408 3.136 18.08 7.04 26.112 11.68 13.024 7.808 25.28 16.768 36.672 26.784H1024v-78.72c-6.08-2.88-13.12-6.08-21.12-9.6a205.984 205.984 0 0 0-57.152-15.36 270.016 270.016 0 0 0-37.056-2.304c-24.864 0-47.744 4-68.704 11.84-20.992 7.776-38.432 18.784-52.64 33.6a144.256 144.256 0 0 0-33.632 54.176 207.68 207.68 0 0 0-12.48 70.176c0 27.84 3.968 52.352 12.16 73.568a146.08 146.08 0 0 0 34.176 53.184l0.064 0.416z m-186.112 34.656a122.912 122.912 0 0 0 42.784-21.504c11.2-8.32 20.224-19.296 26.24-31.936 6.56-13.664 9.888-28.704 9.664-43.904 0-22.784-6.272-41.024-18.912-54.72-12.544-13.92-29.664-23.04-51.104-27.36v-2.304c14.4-6.24 26.688-16.448 35.52-29.408 8.384-12.544 12.544-27.008 12.544-43.52 0-14.24-2.944-26.784-9.12-37.6a65.696 65.696 0 0 0-27.936-26.24 117.44 117.44 0 0 0-36.48-12.096A437.312 437.312 0 0 0 526.656 352h-127.552v340.384h143.392c23.84 0 43.52-2.112 59.008-6.4z m-118.112-271.584h7.968c16.544 0 29.472 0.128 38.656 0.352 9.248 0.576 16.864 1.728 23.04 4a26.24 26.24 0 0 1 14.4 13.12c2.656 5.28 4.064 11.072 4.128 16.96 0.16 6.656-1.056 13.248-3.52 19.392a28.064 28.064 0 0 1-15.04 13.696c-7.04 2.912-14.56 4.48-22.208 4.576-9.472 0.32-20.864 0.544-34.24 0.544h-13.184v-72.64z m6.144 215.52h-6.144v-84.8h18.464c12.64 0 24.736 0 36.384 0.256 9.216-0.16 18.432 0.896 27.36 3.2 9.696 2.848 16.544 7.232 20.512 13.12 4 5.92 6.08 14.24 6.08 25.088 0 8.192-1.728 15.488-4.992 21.824-3.232 6.336-9.696 11.392-18.24 15.392-8.544 4-18.24 5.696-28.48 6.272-10.304 0.224-27.392 0.224-51.328 0.224l0.384-0.576z m-378.56-6.016h124.672l23.36 68.416H348.8L224.32 353.728H124.416L0 692.384h87.52l23.488-68.416-0.064-0.032z m62.336-183.552l41.472 121.44h-82.88l41.408-121.44z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M960 128a64 64 0 0 1 64 64v640a64 64 0 0 1-64 64H64a64 64 0 0 1-64-64V192a64 64 0 0 1 64-64h896z m0 64H64v640h896V192z m-131.936 158.496c65.984 1.824 109.024 32.512 129.184 92.064l-65.952 15.136-1.856-5.664c-9.984-27.52-30.432-42.112-61.376-43.84-46.72 2.784-71.456 35.296-74.208 97.6 1.856 63.2 26.56 96.64 74.24 100.32 37.536-0.928 59.52-23.36 65.92-67.328L960 559.36l-1.792 7.232c-16.96 63.872-60.352 95.84-130.144 95.84-93.44-4.608-142.464-56.8-147.04-156.672 4.576-98.048 53.6-149.792 147.04-155.296z m-568.928 4.128l114.08 302.336H297.6l-22.016-65.984H160.192L138.24 656.96H64l112.672-302.336h82.464z m278.976 0l7.68 0.32c62.816 3.36 96 27.104 99.52 71.136 0 35.744-19.712 59.552-59.104 71.456 42.144 9.184 65.952 32.544 71.456 70.08 0 59.552-37.568 89.344-112.672 89.344h-148.416v-302.336h141.536z m-4.128 173.152H465.28v81.088h67.328c33.92 0 51.296-13.76 52.224-41.248-2.752-25.632-19.68-38.912-50.848-39.84zM217.92 408.224h-1.376l-0.96 7.36c-1.184 7.232-3.168 13.92-5.92 20.128l-34.336 103.04h87.936l-38.464-104.416-2.624-9.024c-1.6-5.888-3.04-11.584-4.256-17.088z m309.184-2.752H465.28v71.456h61.824c29.344-0.928 44.448-13.76 45.376-38.464-0.96-22.016-16.032-32.992-45.376-32.992z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M931.37 477.14c-43.94-69.49-93.39-125.03-147.99-166.35L663.31 441.46c8.79 21.36 13.63 44.91 13.63 69.62 0 96.27-73.84 174.32-164.95 174.32-21.63 0-42.28-4.39-61.32-12.47l-94.09 102.52a455.097 455.097 0 0 0 155.41 26.2c172.62 0 312.43-85.54 419.59-256.63 12.83-20.43 12.72-47.45-0.21-67.88zM795.52 127.99L682.24 252.82C627.73 231 570.1 220.04 512.11 220.5c-171.15 0-310.99 85.66-419.48 256.75-12.93 20.43-13.04 47.33-0.2 67.78 46.95 75.15 100.3 133.68 160.01 175.82L131.9 853.65l38.9 42.36 663.6-725.76-38.88-42.26zM519.68 438.37c-3.72-0.59-7.46-0.81-11.27-0.81-45.63 0-82.43 39.03-82.43 87.16 0 4.97 0.42 9.81 1.14 14.54l-57.15 60.14-5.92 8.33c-13.15-24.94-20.5-53.68-20.5-84.28 0-96.3 73.84-174.45 164.95-174.45 27.2 0 52.76 6.93 75.4 19.29l-64.22 70.08z m52.05 135.53c21.74-23.26 27.17-55.48 16.31-81.39l-96.42 103.16c26.59 9.08 58.38 1.48 80.11-21.77z"
></path>
</svg>
</template>

View File

@ -0,0 +1,5 @@
export { default as DisplayBlock } from './Block.vue';
export { default as DisplayFlex } from './Flex.vue';
export { default as DisplayInline } from './Inline.vue';
export { default as DisplayInlineBlock } from './InlineBlock.vue';
export { default as DisplayNone } from './None.vue';

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M896 320H128V0h768v320z m0 320H128v-256h768v256z m-128 192l-256 192-256-192 192-0.032V704h128v128h192z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M896 704H128v320h768V704z m0-320H128v256h768v-256z m-128-192l-256-192-256 192 192 0.032V320h128V192h192z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M320 128v768H0V128h320z m320 0v768h-256V128h256z m192 128l192 256-192 256-0.032-192H704v-128h128V256z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M704 128v768h320V128H704zM384 128v768h256V128h-256zM192 256l-192 256 192 256 0.032-192H320v-128H192V256z"
></path>
</svg>
</template>

View File

@ -0,0 +1,4 @@
export { default as FlexDirectionColumn } from './Column.vue';
export { default as FlexDirectionColumnReverse } from './ColumnReverse.vue';
export { default as FlexDirectionRow } from './Row.vue';
export { default as FlexDirectionRowReverse } from './RowReverse.vue';

View File

@ -0,0 +1,5 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path d="M480 1024V0h64v1024h-64z m128-64V64h320v896H608zM96 960V64h320v896H96z"></path>
</svg>
</template>

View File

@ -0,0 +1,5 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path d="M416 160H96v704h320V160z m384 0H480v704h320V160z m128-160h-64v1024h64V0z"></path>
</svg>
</template>

View File

@ -0,0 +1,5 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path d="M608 160h320v704H608V160zM224 160h320v704H224V160zM96 0h64v1024H96V0z"></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M320 864H192v160H128v-160H0V160h128V0h64v160h128v704z m704 0h-128v160h-64v-160h-128V160h128V0h64v160h128v704z"
></path>
</svg>
</template>

View File

@ -0,0 +1,5 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path d="M64 1024H0V0h64v1024z m384-160H128V160h320v704z m448 0H576V160h320v704z m128 160h-64V0h64v1024z"></path>
</svg>
</template>

View File

@ -0,0 +1,5 @@
export { default as JustifyContentCenter } from './Center.vue';
export { default as JustifyContentFlexEnd } from './FlexEnd.vue';
export { default as JustifyContentFlexStart } from './FlexStart.vue';
export { default as JustifyContentSpaceAround } from './SpaceAround.vue';
export { default as JustifyContentSpaceBetween } from './SpaceBetween.vue';

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M128 810.666667h768v85.333333H128z m128-170.666667h512v85.333333H256z m-128-170.666667h768v85.333334H128z m0-341.333333h768v85.333333H128z m128 170.666667h512v85.333333H256z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M128 810.666667h768v85.333333H128z m0-170.666667h512v85.333333H128z m0-170.666667h768v85.333334H128z m0-341.333333h768v85.333333H128z m0 170.666667h512v85.333333H128z"
></path>
</svg>
</template>

View File

@ -0,0 +1,7 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024">
<path
d="M128 128h768v85.333333H128z m0 682.666667h768v85.333333H128z m213.333333-170.666667h554.666667v85.333333H341.333333z m-213.333333-170.666667h768v85.333334H128z m213.333333-170.666666h554.666667v85.333333H341.333333z"
></path>
</svg>
</template>

View File

@ -0,0 +1,3 @@
export { default as AlignLeft } from './Left.vue';
export { default as AlignCenter } from './Center.vue';
export { default as AlignRight } from './Right.vue';

View File

@ -0,0 +1,78 @@
<template>
<MContainer :config="config" :model="values" @change="change"></MContainer>
</template>
<script lang="ts" setup>
import { markRaw } from 'vue';
import { ContainerChangeEventData, MContainer } from '@tmagic/form';
import type { StyleSchema } from '@tmagic/schema';
import BackgroundPosition from '../components/BackgroundPosition.vue';
import { BackgroundNoRepeat, BackgroundRepeat, BackgroundRepeatX, BackgroundRepeatY } from '../icons/background-repeat';
defineProps<{ values: Partial<StyleSchema> }>();
const emit = defineEmits<{
change: [v: StyleSchema, eventData: ContainerChangeEventData];
}>();
const config = {
items: [
{
name: 'backgroundColor',
text: '背景色',
labelWidth: '68px',
type: 'data-source-field-select',
fieldConfig: {
type: 'colorPicker',
},
},
{
name: 'backgroundImage',
text: '背景图',
labelWidth: '68px',
type: 'data-source-field-select',
fieldConfig: {
type: 'img-upload',
},
},
{
name: 'backgroundSize',
text: '背景尺寸',
type: 'radioGroup',
childType: 'button',
labelWidth: '68px',
options: [
{ value: 'auto', text: '默认', tooltip: '默认 auto' },
{ value: 'contain', text: '等比填充', tooltip: '等比填充 contain' },
{ value: 'cover', text: '等比覆盖', tooltip: '等比覆盖 cover' },
],
},
{
name: 'backgroundRepeat',
text: '重复显示',
type: 'radioGroup',
childType: 'button',
labelWidth: '68px',
options: [
{ value: 'repeat', icon: markRaw(BackgroundRepeat), tooltip: '垂直和水平方向重复 repeat' },
{ value: 'repeat-x', icon: markRaw(BackgroundRepeatX), tooltip: '水平方向重复 repeat-x' },
{ value: 'repeat-y', icon: markRaw(BackgroundRepeatY), tooltip: '垂直方向重复 repeat-y' },
{ value: 'no-repeat', icon: markRaw(BackgroundNoRepeat), tooltip: '不重复 no-repeat' },
],
},
{
name: 'backgroundPosition',
text: '背景定位',
type: 'component',
component: BackgroundPosition,
labelWidth: '68px',
},
],
};
const change = (value: StyleSchema, eventData: ContainerChangeEventData) => {
emit('change', value, eventData);
};
</script>

View File

@ -0,0 +1,35 @@
<template>
<MContainer :config="config" :model="values" @change="change"></MContainer>
<Border :model="values" @change="change"></Border>
</template>
<script lang="ts" setup>
import { type ContainerChangeEventData, MContainer } from '@tmagic/form';
import type { StyleSchema } from '@tmagic/schema';
import Border from '../components/Border.vue';
defineProps<{ values: Partial<StyleSchema> }>();
const emit = defineEmits<{
change: [v: StyleSchema, eventData: ContainerChangeEventData];
}>();
const config = {
items: [
{
labelWidth: '68px',
name: 'borderRadius',
text: '圆角',
type: 'data-source-field-select',
fieldConfig: {
type: 'text',
},
},
],
};
const change = (value: StyleSchema, eventData: ContainerChangeEventData) => {
emit('change', value, eventData);
};
</script>

View File

@ -0,0 +1,90 @@
<template>
<MContainer :config="config" :model="values" @change="change"></MContainer>
</template>
<script lang="ts" setup>
import { markRaw } from 'vue';
import { ContainerChangeEventData, MContainer } from '@tmagic/form';
import type { StyleSchema } from '@tmagic/schema';
import { AlignCenter, AlignLeft, AlignRight } from '../icons/text-align';
defineProps<{ values: Partial<StyleSchema> }>();
const emit = defineEmits<{
change: [v: StyleSchema, eventData: ContainerChangeEventData];
}>();
const config = {
items: [
{
type: 'row',
items: [
{
labelWidth: '68px',
name: 'fontSize',
text: '字号',
type: 'data-source-field-select',
fieldConfig: {
type: 'text',
},
},
{
labelWidth: '68px',
name: 'lineHeight',
text: '行高',
type: 'data-source-field-select',
fieldConfig: {
type: 'text',
},
},
],
},
{
name: 'fontWeight',
text: '字重',
labelWidth: '68px',
type: 'data-source-field-select',
fieldConfig: {
type: 'select',
options: ['normal', 'bold']
.concat(
Array(7)
.fill(1)
.map((x, i) => `${i + 1}00`),
)
.map((item) => ({
value: item,
text: item,
})),
},
},
{
labelWidth: '68px',
name: 'color',
text: '颜色',
type: 'data-source-field-select',
fieldConfig: {
type: 'colorPicker',
},
},
{
name: 'textAlign',
text: '对齐',
type: 'radioGroup',
childType: 'button',
labelWidth: '68px',
options: [
{ value: 'left', icon: markRaw(AlignLeft), tooltip: '左对齐 row' },
{ value: 'center', icon: markRaw(AlignCenter), tooltip: '居中对齐 center' },
{ value: 'right', icon: markRaw(AlignRight), tooltip: '右对齐 right' },
],
},
],
};
const change = (value: StyleSchema, eventData: ContainerChangeEventData) => {
emit('change', value, eventData);
};
</script>

View File

@ -0,0 +1,164 @@
<template>
<MContainer :config="config" :model="values" @change="change"></MContainer>
<Box :model="values" @change="change"></Box>
</template>
<script lang="ts" setup>
import { markRaw } from 'vue';
import type { ContainerChangeEventData, FormState } from '@tmagic/form';
import { MContainer } from '@tmagic/form';
import type { StyleSchema } from '@tmagic/schema';
import Box from '../components/Box.vue';
import { DisplayBlock, DisplayFlex, DisplayInline, DisplayInlineBlock, DisplayNone } from '../icons/display';
import {
FlexDirectionColumn,
FlexDirectionColumnReverse,
FlexDirectionRow,
FlexDirectionRowReverse,
} from '../icons/flex-direction';
import {
JustifyContentCenter,
JustifyContentFlexEnd,
JustifyContentFlexStart,
JustifyContentSpaceAround,
JustifyContentSpaceBetween,
} from '../icons/justify-content';
defineProps<{
values: Partial<StyleSchema>;
}>();
const emit = defineEmits<{
change: [v: string | StyleSchema, eventData: ContainerChangeEventData];
}>();
const config = {
items: [
{
name: 'display',
text: '模式',
type: 'radioGroup',
childType: 'button',
labelWidth: '68px',
options: [
{ value: 'inline', icon: markRaw(DisplayInline), tooltip: '内联布局 inline' },
{ value: 'flex', icon: markRaw(DisplayFlex), tooltip: '弹性布局 flex' },
{ value: 'block', icon: markRaw(DisplayBlock), tooltip: '块级布局 block' },
{ value: 'inline-block', icon: markRaw(DisplayInlineBlock), tooltip: '内联块布局 inline-block' },
{ value: 'none', icon: markRaw(DisplayNone), tooltip: '隐藏 none' },
],
},
{
name: 'flexDirection',
text: '主轴方向',
type: 'radioGroup',
childType: 'button',
labelWidth: '68px',
options: [
{ value: 'row', icon: markRaw(FlexDirectionRow), tooltip: '水平方向 起点在左侧 row' },
{ value: 'row-reverse', icon: markRaw(FlexDirectionRowReverse), tooltip: '水平方向 起点在右侧 row-reverse' },
{ value: 'column', icon: markRaw(FlexDirectionColumn), tooltip: '垂直方向 起点在上沿 column' },
{
value: 'column-reverse',
icon: markRaw(FlexDirectionColumnReverse),
tooltip: '垂直方向 起点在下沿 column-reverse',
},
],
display: (mForm: FormState, { model }: { model: Record<any, any> }) => model.display === 'flex',
},
{
name: 'justifyContent',
text: '主轴对齐',
type: 'radioGroup',
childType: 'button',
labelWidth: '68px',
options: [
{ value: 'flex-start', icon: markRaw(JustifyContentFlexStart), tooltip: '左对齐 flex-start' },
{ value: 'flex-end', icon: markRaw(JustifyContentFlexEnd), tooltip: '右对齐 flex-end' },
{ value: 'center', icon: markRaw(JustifyContentCenter), tooltip: '居中 center' },
{ value: 'space-between', icon: markRaw(JustifyContentSpaceBetween), tooltip: '两端对齐 space-between' },
{ value: 'space-around', icon: markRaw(JustifyContentSpaceAround), tooltip: '横向平分 space-around' },
],
display: (mForm: FormState, { model }: { model: Record<any, any> }) => model.display === 'flex',
},
{
name: 'alignItems',
text: '辅轴对齐',
type: 'radioGroup',
childType: 'button',
labelWidth: '68px',
options: [
{ value: 'flex-start', icon: markRaw(JustifyContentFlexStart), tooltip: '左对齐 flex-start' },
{ value: 'flex-end', icon: markRaw(JustifyContentFlexEnd), tooltip: '右对齐 flex-end' },
{ value: 'center', icon: markRaw(JustifyContentCenter), tooltip: '居中 center' },
{ value: 'space-between', icon: markRaw(JustifyContentSpaceBetween), tooltip: '两端对齐 space-between' },
{ value: 'space-around', icon: markRaw(JustifyContentSpaceAround), tooltip: '横向平分 space-around' },
],
display: (mForm: FormState, { model }: { model: Record<any, any> }) => model.display === 'flex',
},
{
name: 'flexWrap',
text: '换行',
type: 'radioGroup',
childType: 'button',
labelWidth: '68px',
options: [
{ value: 'nowrap', text: '不换行', tooltip: '不换行 nowrap' },
{ value: 'wrap', text: '正换行', tooltip: '第一行在上方 wrap' },
{ value: 'wrap-reverse', text: '逆换行', tooltip: '第一行在下方 wrap-reverse' },
],
display: (mForm: FormState, { model }: { model: Record<any, any> }) => model.display === 'flex',
},
{
type: 'row',
items: [
{
name: 'width',
text: '宽度',
labelWidth: '68px',
type: 'data-source-field-select',
fieldConfig: {
type: 'text',
},
},
{
name: 'height',
text: '宽度',
labelWidth: '68px',
type: 'data-source-field-select',
fieldConfig: {
type: 'text',
},
},
],
},
{
type: 'data-source-field-select',
text: 'overflow',
name: 'overflow',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'select',
clearable: true,
allowCreate: true,
options: [
{ text: 'visible', value: 'visible' },
{ text: 'hidden', value: 'hidden' },
{ text: 'clip', value: 'clip' },
{ text: 'scroll', value: 'scroll' },
{ text: 'auto', value: 'auto' },
{ text: 'overlay', value: 'overlay' },
{ text: 'initial', value: 'initial' },
],
},
},
],
};
const change = (value: string | StyleSchema, eventData: ContainerChangeEventData) => {
emit('change', value, eventData);
};
</script>

View File

@ -0,0 +1,48 @@
<template>
<MContainer :config="config" :model="values" @change="change"></MContainer>
<Position v-show="values['position'] !== 'static'" :model="values" @change="change"></Position>
</template>
<script lang="ts" setup>
import { ContainerChangeEventData, MContainer } from '@tmagic/form';
import type { StyleSchema } from '@tmagic/schema';
import Position from '../components/Position.vue';
defineProps<{ values: Partial<StyleSchema> }>();
const emit = defineEmits<{
change: [v: string | StyleSchema, eventData: ContainerChangeEventData];
}>();
const config = {
items: [
{
name: 'position',
text: '定位',
labelWidth: '68px',
type: 'data-source-field-select',
fieldConfig: {
type: 'select',
options: ['static', 'relative', 'absolute', 'fixed', 'sticky'].map((item) => ({
value: item,
text: item,
})),
},
},
{
labelWidth: '68px',
name: 'zIndex',
text: 'zIndex',
type: 'data-source-field-select',
fieldConfig: {
type: 'text',
},
},
],
};
const change = (value: string | StyleSchema, eventData: ContainerChangeEventData) => {
emit('change', value, eventData);
};
</script>

View File

@ -0,0 +1,5 @@
export { default as Background } from './Background.vue';
export { default as Font } from './Font.vue';
export { default as Layout } from './Layout.vue';
export { default as Position } from './Position.vue';
export { default as Border } from './Border.vue';

View File

@ -39,6 +39,7 @@ import DisplayConds from './fields/DisplayConds.vue';
import EventSelect from './fields/EventSelect.vue';
import KeyValue from './fields/KeyValue.vue';
import PageFragmentSelect from './fields/PageFragmentSelect.vue';
import StyleSetter from './fields/StyleSetter/Index.vue';
import uiSelect from './fields/UISelect.vue';
import CodeEditor from './layouts/CodeEditor.vue';
import { setEditorConfig } from './utils/config';
@ -106,6 +107,7 @@ export { default as TreeNode } from './components/TreeNode.vue';
export { default as PageFragmentSelect } from './fields/PageFragmentSelect.vue';
export { default as DisplayConds } from './fields/DisplayConds.vue';
export { default as CondOpSelect } from './fields/CondOpSelect.vue';
export { default as StyleSetter } from './fields/StyleSetter/Index.vue';
const defaultInstallOpt: EditorInstallOptions = {
// eslint-disable-next-line no-eval
@ -143,5 +145,6 @@ export default {
app.component('m-fields-page-fragment-select', PageFragmentSelect);
app.component('m-fields-display-conds', DisplayConds);
app.component('m-fields-cond-op-select', CondOpSelect);
app.component('m-form-style-setter', StyleSetter);
},
};

View File

@ -0,0 +1,113 @@
.background-position-container {
display: flex;
width: 100%;
.presets-value-list {
display: flex;
flex-wrap: wrap;
width: 80px;
height: auto;
.el-button {
& + .el-button {
margin-left: 2px;
}
&:nth-child(3n + 1) {
margin-left: 0 !important;
}
}
.position-icon {
position: relative;
width: 14px;
height: 14px;
border: 1px solid #1d1f24;
&.active {
background-color: var(--el-color-primary);
&::after {
border: 1px solid #fff;
}
}
&::after {
position: absolute;
content: "";
border: 1px solid #1d1f24;
box-sizing: border-box;
}
&.left-top {
&::after {
top: 1px;
left: 1px;
width: 6px;
height: 6px;
}
}
&.center-top {
&::after {
top: 1px;
left: 1px;
width: 12px;
height: 6px;
}
}
&.right-top {
&::after {
top: 1px;
right: 1px;
width: 6px;
height: 6px;
}
}
&.left-center {
&::after {
top: 1px;
left: 1px;
width: 6px;
height: 12px;
}
}
&.center-center {
&::after {
top: 1px;
left: 1px;
width: 12px;
height: 12px;
}
}
&.right-center {
&::after {
top: 1px;
right: 1px;
width: 6px;
height: 12px;
}
}
&.left-bottom {
&::after {
bottom: 1px;
left: 1px;
width: 6px;
height: 6px;
}
}
&.center-bottom {
&::after {
bottom: 1px;
left: 1px;
width: 12px;
height: 6px;
}
}
&.right-bottom {
&::after {
bottom: 1px;
right: 1px;
width: 6px;
height: 6px;
}
}
}
}
.custom-value {
position: relative;
flex: 1;
}
}

View File

@ -0,0 +1,53 @@
.border-box-container {
display: flex;
.border-icon-container {
display: flex;
flex-direction: column;
justify-content: center;
width: 88px;
&-row {
display: flex;
justify-content: center;
align-items: center;
& + .border-icon-container-row {
margin-top: 8px;
}
}
.border-icon {
box-sizing: border-box;
width: 16px;
height: 16px;
border-width: 1px;
border-color: #111;
border-style: solid;
cursor: pointer;
& + .border-icon {
margin-left: 8px;
}
&.active {
border-width: 1px;
border-color: var(--el-color-primary);
}
&.border-icon-top {
border-top-width: 2px;
border-style: solid dashed dashed dashed;
}
&.border-icon-right {
border-right-width: 2px;
border-style: dashed solid dashed dashed;
}
&.border-icon-bottom {
border-bottom-width: 2px;
border-style: dashed dashed solid dashed;
}
&.border-icon-left {
border-left-width: 2px;
border-style: dashed dashed dashed solid;
}
}
}
.border-value-container {
margin-left: 16px;
flex: 1;
}
}

View File

@ -0,0 +1,16 @@
@use "./border.scss";
@use "./layout.scss";
@use "./background.scss";
.m-fields-style-setter {
width: 100%;
}
.text-align-list {
display: flex;
height: 100%;
.btn-active {
color: var(--el-color-primary) !important;
}
}

View File

@ -0,0 +1,200 @@
// 盒子模型
.layout-box-container {
position: relative;
margin: 0 0 16px 16px;
width: 270px;
height: 150px;
.help-txt {
float: left;
margin-left: -10px;
transform: scale(0.75);
}
.outer-top-border,
.inner-top-border,
.outer-right-border,
.inner-right-border,
.outer-bottom-border,
.inner-bottom-border,
.outer-left-border,
.inner-left-border {
position: absolute;
transition: all 0.3s ease;
.next-input {
position: absolute;
height: 20px;
input {
padding: 0;
width: 100%;
border: none;
outline: none;
margin: 0;
font-weight: 400;
vertical-align: top;
background-color: transparent;
color: #333;
text-align: center;
line-height: 20px;
height: 20px;
}
}
}
// top
.outer-top-border,
.inner-top-border {
height: 0;
border-left: 20px solid transparent;
border-right: 20px solid transparent;
border-top: 20px solid #d6e4ff;
.next-input {
left: 0;
right: 0;
top: -20px;
}
}
.outer-top-border {
top: 0;
left: 0;
right: 0;
}
.inner-top-border {
top: 25px;
left: 25px;
right: 25px;
}
.outer-top-border,
.inner-top-border {
&:hover {
border-top: 20px solid #bfd4fb;
}
}
// right
.outer-right-border,
.inner-right-border {
width: 0;
border-top: 20px solid transparent;
border-bottom: 20px solid transparent;
border-right: 20px solid #d6e4ff;
.next-input {
top: 0;
bottom: 0;
right: -20px;
width: 20px;
margin: auto;
input {
position: absolute;
top: 0;
bottom: 0;
left: 0;
margin: auto;
width: 20px;
}
}
}
.outer-right-border {
top: 5px;
bottom: 5px;
right: 0;
}
.inner-right-border {
top: 30px;
bottom: 30px;
right: 25px;
}
.outer-right-border,
.inner-right-border {
&:hover {
border-right: 20px solid #bfd4fb;
}
}
// bottom
.outer-bottom-border,
.inner-bottom-border {
height: 0;
border-left: 20px solid transparent;
border-right: 20px solid transparent;
border-bottom: 20px solid #d6e4ff;
.next-input {
position: absolute;
left: 0;
right: 0;
bottom: -20px;
}
}
.outer-bottom-border {
bottom: 0;
left: 0;
right: 0;
}
.inner-bottom-border {
bottom: 25px;
left: 25px;
right: 25px;
}
.outer-bottom-border,
.inner-bottom-border {
&:hover {
border-bottom: 20px solid #bfd4fb;
}
}
// left
.outer-left-border,
.inner-left-border {
width: 0;
border-top: 20px solid transparent;
border-bottom: 20px solid transparent;
border-left: 20px solid #d6e4ff;
.next-input {
position: absolute;
top: 0;
bottom: 0;
left: -20px;
width: 20px;
margin: auto;
input {
position: absolute;
top: 0;
bottom: 0;
right: 0;
margin: auto;
width: 20px;
}
}
}
.outer-left-border {
top: 5px;
bottom: 5px;
left: 0;
}
.inner-left-border {
top: 30px;
bottom: 30px;
left: 25px;
}
.outer-left-border,
.inner-left-border {
&:hover {
border-left: 20px solid #bfd4fb;
}
}
}

View File

@ -26,3 +26,4 @@
@use "./page-fragment-select.scss";
@use "./data-source-field.scss";
@use "./data-source-field-select.scss";
@use "./style-setter/index.scss";

View File

@ -46,305 +46,60 @@ export const styleTabConfig: TabPaneConfig = {
items: [
{
name: 'style',
labelWidth: '100px',
type: 'style-setter',
items: [
{
type: 'fieldset',
legend: '位置',
items: [
{
type: 'data-source-field-select',
name: 'position',
text: '固定定位',
labelPosition: 'left',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'checkbox',
activeValue: 'fixed',
inactiveValue: 'absolute',
defaultValue: 'absolute',
},
},
{
type: 'data-source-field-select',
name: 'left',
text: 'left',
checkStrictly: false,
dataSourceFieldType: ['string', 'number'],
fieldConfig: {
type: 'text',
},
},
{
type: 'data-source-field-select',
name: 'top',
text: 'top',
checkStrictly: false,
dataSourceFieldType: ['string', 'number'],
fieldConfig: {
type: 'text',
},
disabled: (_vm: FormState, { model }: any) =>
model.position === 'fixed' && model._magic_position === 'fixedBottom',
},
{
type: 'data-source-field-select',
name: 'right',
text: 'right',
checkStrictly: false,
dataSourceFieldType: ['string', 'number'],
fieldConfig: {
type: 'text',
},
},
{
type: 'data-source-field-select',
name: 'bottom',
text: 'bottom',
checkStrictly: false,
dataSourceFieldType: ['string', 'number'],
fieldConfig: {
type: 'text',
},
disabled: (_vm: FormState, { model }: any) =>
model.position === 'fixed' && model._magic_position === 'fixedTop',
},
],
},
{
type: 'fieldset',
legend: '盒子',
items: [
{
type: 'data-source-field-select',
name: 'display',
text: 'display',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'select',
clearable: true,
allowCreate: true,
options: [
{ text: 'block', value: 'block' },
{ text: 'flex', value: 'flex' },
{ text: 'none', value: 'none' },
{ text: 'inline-block', value: 'inline-block' },
{ text: 'grid', value: 'grid' },
{ text: 'inline', value: 'inline' },
{ text: 'initial', value: 'initial' },
],
},
},
{
type: 'data-source-field-select',
name: 'width',
text: '宽度',
checkStrictly: false,
dataSourceFieldType: ['string', 'number'],
fieldConfig: {
type: 'text',
},
},
{
type: 'data-source-field-select',
name: 'height',
text: '高度',
checkStrictly: false,
dataSourceFieldType: ['string', 'number'],
fieldConfig: {
type: 'text',
},
},
{
type: 'data-source-field-select',
text: 'overflow',
name: 'overflow',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'select',
clearable: true,
allowCreate: true,
options: [
{ text: 'visible', value: 'visible' },
{ text: 'hidden', value: 'hidden' },
{ text: 'clip', value: 'clip' },
{ text: 'scroll', value: 'scroll' },
{ text: 'auto', value: 'auto' },
{ text: 'overlay', value: 'overlay' },
{ text: 'initial', value: 'initial' },
],
},
},
],
},
{
type: 'fieldset',
legend: '边框',
items: [
{
type: 'data-source-field-select',
name: 'borderWidth',
text: '宽度',
defaultValue: '0',
checkStrictly: false,
dataSourceFieldType: ['string', 'number'],
fieldConfig: {
type: 'text',
},
},
{
type: 'data-source-field-select',
name: 'borderColor',
text: '颜色',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'text',
},
},
{
type: 'data-source-field-select',
name: 'borderStyle',
text: '样式',
defaultValue: 'none',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'select',
options: [
{ text: 'none', value: 'none' },
{ text: 'hidden', value: 'hidden' },
{ text: 'dotted', value: 'dotted' },
{ text: 'dashed', value: 'dashed' },
{ text: 'solid', value: 'solid' },
{ text: 'double', value: 'double' },
{ text: 'groove', value: 'groove' },
{ text: 'ridge', value: 'ridge' },
{ text: 'inset', value: 'inset' },
{ text: 'outset', value: 'outset' },
],
},
},
],
},
{
type: 'fieldset',
legend: '背景',
items: [
{
type: 'data-source-field-select',
name: 'backgroundImage',
text: '背景图',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'img-upload',
},
},
{
type: 'data-source-field-select',
name: 'backgroundColor',
text: '背景颜色',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'colorPicker',
},
},
{
type: 'data-source-field-select',
name: 'backgroundRepeat',
text: '背景图重复',
defaultValue: 'no-repeat',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'select',
options: [
{ text: 'repeat', value: 'repeat' },
{ text: 'repeat-x', value: 'repeat-x' },
{ text: 'repeat-y', value: 'repeat-y' },
{ text: 'no-repeat', value: 'no-repeat' },
{ text: 'inherit', value: 'inherit' },
],
},
},
{
type: 'data-source-field-select',
name: 'backgroundSize',
text: '背景图大小',
defaultValue: '100% 100%',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'text',
},
},
],
},
{
type: 'fieldset',
legend: '字体',
items: [
{
type: 'data-source-field-select',
name: 'color',
text: '颜色',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'colorPicker',
},
},
{
type: 'data-source-field-select',
name: 'fontSize',
text: '大小',
checkStrictly: false,
dataSourceFieldType: ['string', 'number'],
fieldConfig: {
type: 'text',
},
},
{
type: 'data-source-field-select',
name: 'fontWeight',
text: '粗细',
checkStrictly: false,
dataSourceFieldType: ['string', 'number'],
fieldConfig: {
type: 'text',
},
},
],
},
{
type: 'fieldset',
legend: '变形',
name: 'transform',
items: [
{
type: 'data-source-field-select',
name: 'rotate',
text: '旋转角度',
checkStrictly: false,
dataSourceFieldType: ['string'],
fieldConfig: {
type: 'text',
},
},
{
type: 'data-source-field-select',
name: 'scale',
text: '缩放',
checkStrictly: false,
dataSourceFieldType: ['number', 'string'],
fieldConfig: {
type: 'text',
},
},
names: [
'display',
'flexDirection',
'justifyContent',
'alignItems',
'flexWrap',
'marginTop',
'marginRight',
'marginBottom',
'marginLeft',
'paddingTop',
'paddingRight',
'paddingBottom',
'paddingLeft',
'width',
'height',
'overflow',
'fontSize',
'lineHeight',
'fontWeight',
'color',
'textAlign',
'backgroundColor',
'backgroundImage',
'backgroundSize',
'backgroundPosition',
'backgroundRepeat',
'position',
'zIndex',
'top',
'right',
'bottom',
'left',
'borderRadius',
'borderTopWidth',
'borderTopStyle',
'borderTopColor',
'borderRightColor',
'borderRightWidth',
'borderRightStyle',
'borderRightColor',
'borderBottomWidth',
'borderBottomStyle',
'borderBottomColor',
'borderLeftStyle',
'borderLeftWidth',
'borderLeftColor',
'borderWidth',
'borderStyle',
'borderColor',
],
},
],

View File

@ -306,7 +306,12 @@ const itemProp = computed(() => {
return `${n}`;
});
const tagName = computed(() => `m-${items.value ? 'form' : 'fields'}-${type.value}`);
const tagName = computed(() => {
if (type.value === 'component') {
return props.config.component;
}
return `m-${items.value ? 'form' : 'fields'}-${type.value}`;
});
const disabled = computed(() => props.disabled || filterFunction(mForm, props.config.disabled, props));

View File

@ -1,13 +1,30 @@
<template>
<TMagicRadioGroup v-if="model" v-model="model[name]" :size="size" :disabled="disabled" @change="changeHandler">
<TMagicRadio v-for="option in config.options" :value="option.value" :key="`${option.value}`">{{
option.text
}}</TMagicRadio>
<TMagicRadioGroup v-if="model" v-model="model[name]" :size="size" :disabled="disabled">
<component
:is="itemComponent"
v-for="option in config.options"
:value="option.value"
:key="`${option.value}`"
@click.prevent="clickHandler(option.value)"
>
<TMagicTooltip v-if="option.tooltip" placement="top-start" :content="option.tooltip">
<div>
<TMagicIcon v-if="option.icon" :size="'16'"><component :is="option.icon"></component></TMagicIcon>
<span>{{ option.text }}</span>
</div>
</TMagicTooltip>
<div v-else>
<TMagicIcon v-if="option.icon" :size="'16'"><component :is="option.icon"></component></TMagicIcon>
<span>{{ option.text }}</span>
</div>
</component>
</TMagicRadioGroup>
</template>
<script lang="ts" setup>
import { TMagicRadio, TMagicRadioGroup } from '@tmagic/design';
import { computed } from 'vue';
import { TMagicIcon, TMagicRadio, TMagicRadioButton, TMagicRadioGroup, TMagicTooltip } from '@tmagic/design';
import type { FieldProps, RadioGroupConfig } from '../schema';
import { useAddField } from '../utils/useAddField';
@ -18,11 +35,18 @@ defineOptions({
const props = defineProps<FieldProps<RadioGroupConfig>>();
const itemComponent = computed(() => (props.config.childType === 'button' ? TMagicRadioButton : TMagicRadio));
const emit = defineEmits(['change']);
const changeHandler = (value: number) => {
emit('change', value);
};
const clickHandler = (item: any) => {
props.model[props.name] = props.model[props.name] === item ? '' : item;
changeHandler(props.model[props.name]);
};
useAddField(props.prop);
</script>

View File

@ -438,9 +438,12 @@ export interface SwitchConfig extends FormItem {
*/
export interface RadioGroupConfig extends FormItem {
type: 'radio-group';
childType?: 'default' | 'button';
options: {
value: string | number | boolean;
text: string;
icon?: any;
tooltip?: string;
}[];
}

View File

@ -126,9 +126,7 @@ export interface MComponent {
/** 显示条件中配置的数据源条件的编译结果 */
condResult?: boolean;
/** 组件根Dom的style */
style?: {
[key: string]: any;
};
style?: StyleSchema;
[NODE_CONDS_KEY]?: DisplayCond[];
[key: string]: any;
}
@ -308,3 +306,7 @@ export interface UiComponentProps<T extends MNode = MNode> {
config: T;
model?: any;
}
export interface StyleSchema {
[key: string]: any;
}

View File

@ -13,13 +13,15 @@ interface PageFragmentContainerProps {
iteratorContainerId: Id[];
}
const PageFragmentContainer: React.FC<PageFragmentContainerProps> = ({ config,
const PageFragmentContainer: React.FC<PageFragmentContainerProps> = ({
config,
id,
style,
className,
containerIndex,
iteratorIndex,
iteratorContainerId, }) => {
iteratorContainerId,
}) => {
const { app } = useApp({
config,
methods: {},

View File

@ -1,7 +1,8 @@
import alias from '@rollup/plugin-alias';
import { existsSync, readdirSync, readFileSync, writeFileSync } from 'node:fs';
import path from 'node:path';
import { fileURLToPath } from 'node:url';
import alias from '@rollup/plugin-alias';
import dts from 'rollup-plugin-dts';
if (!existsSync('temp')) {