营销:优化装修编辑器样式

(cherry picked from commit 0b0ba1f313)
pull/420/head
owen 2023-11-05 10:08:03 +08:00 committed by shizhong
parent f526675293
commit cf9691f993
4 changed files with 174 additions and 165 deletions

View File

@ -1,8 +1,7 @@
<template> <template>
<div <div
:style="{ :style="{
...style, ...style
background: property.bgType === 'color' ? property.bgColor : `url(${property.bgImg})`
}" }"
> >
<slot></slot> <slot></slot>
@ -18,7 +17,7 @@ import { ComponentStyle } from '@/components/DiyEditor/util'
*/ */
defineOptions({ name: 'ComponentContainer' }) defineOptions({ name: 'ComponentContainer' })
const props = defineProps<{ property: ComponentStyle }>() const props = defineProps<{ property: ComponentStyle | undefined }>()
const style = computed(() => { const style = computed(() => {
if (!props.property) { if (!props.property) {
@ -37,7 +36,9 @@ const style = computed(() => {
borderTopRightRadius: `${props.property.borderTopRightRadius || 0}px`, borderTopRightRadius: `${props.property.borderTopRightRadius || 0}px`,
borderBottomRightRadius: `${props.property.borderBottomRightRadius || 0}px`, borderBottomRightRadius: `${props.property.borderBottomRightRadius || 0}px`,
borderBottomLeftRadius: `${props.property.borderBottomLeftRadius || 0}px`, borderBottomLeftRadius: `${props.property.borderBottomLeftRadius || 0}px`,
overflow: 'hidden' overflow: 'hidden',
background:
props.property.bgType === 'color' ? props.property.bgColor : `url(${props.property.bgImg})`
} }
}) })
</script> </script>

View File

@ -11,6 +11,7 @@
<draggable <draggable
class="component-container" class="component-container"
ghost-class="draggable-ghost" ghost-class="draggable-ghost"
item-key="index"
:list="group.components" :list="group.components"
:sort="false" :sort="false"
:group="{ name: 'component', pull: 'clone', put: false }" :group="{ name: 'component', pull: 'clone', put: false }"

View File

@ -42,7 +42,6 @@ defineProps<{ property: SearchProperty }>()
<style scoped lang="scss"> <style scoped lang="scss">
.search-bar { .search-bar {
width: 375px;
/* 搜索框 */ /* 搜索框 */
.inner { .inner {
position: relative; position: relative;

View File

@ -33,111 +33,111 @@
<ComponentLibrary ref="componentLibrary" :list="libs" v-if="libs && libs.length > 0" /> <ComponentLibrary ref="componentLibrary" :list="libs" v-if="libs && libs.length > 0" />
<!-- 中心设计区域 --> <!-- 中心设计区域 -->
<div class="editor-center page-prop-area" @click="handlePageSelected"> <div class="editor-center page-prop-area" @click="handlePageSelected">
<div class="editor-design"> <!-- 手机顶部 -->
<!-- 手机顶部 --> <div class="editor-design-top">
<div class="editor-design-top"> <!-- 手机顶部状态栏 -->
<!-- 手机顶部状态栏 --> <img src="@/assets/imgs/diy/statusBar.png" alt="" class="status-bar" />
<img src="@/assets/imgs/diy/statusBar.png" alt="" class="status-bar" /> <!-- 手机顶部导航栏 -->
<!-- 手机顶部导航栏 --> <NavigationBar
<NavigationBar v-if="showNavigationBar"
v-if="showNavigationBar" :property="navigationBarComponent.property"
:property="navigationBarComponent.property" @click="handleNavigationBarSelected"
@click="handleNavigationBarSelected"
:class="[
'component',
'cursor-pointer!',
{ active: selectedComponent?.id === navigationBarComponent.id }
]"
/>
</div>
<!-- 手机页面编辑区域 -->
<el-scrollbar class="editor-design-center" height="100%" view-class="page-prop-area">
<div
class="phone-container"
:style="{
backgroundColor: pageConfigComponent.property.backgroundColor,
backgroundImage: `url(${pageConfigComponent.property.backgroundImage})`
}"
>
<draggable
class="page-prop-area drag-area"
v-model="pageComponents"
item-key="index"
:animation="200"
filter=".component-toolbar"
ghost-class="draggable-ghost"
:force-fallback="true"
group="component"
@change="handleComponentChange"
>
<template #item="{ element, index }">
<div class="component-container" @click="handleComponentSelected(element, index)">
<!-- 左侧组件名 -->
<div
:class="['component-name', { active: selectedComponentIndex === index }]"
v-if="element.name"
>
{{ element.name }}
</div>
<!-- 组件内容区 -->
<div :class="['component', { active: selectedComponentIndex === index }]">
<component
:is="element.id"
:property="element.property"
:data-type="element.id"
/>
</div>
<!-- 左侧组件操作工具栏 -->
<div
class="component-toolbar"
v-if="element.name && selectedComponentIndex === index"
>
<el-button-group type="primary">
<el-tooltip content="上移" placement="right">
<el-button
:disabled="index === 0"
@click.stop="handleMoveComponent(index, -1)"
>
<Icon icon="ep:arrow-up" />
</el-button>
</el-tooltip>
<el-tooltip content="下移" placement="right">
<el-button
:disabled="index === pageComponents.length - 1"
@click.stop="handleMoveComponent(index, 1)"
>
<Icon icon="ep:arrow-down" />
</el-button>
</el-tooltip>
<el-tooltip content="复制" placement="right">
<el-button @click.stop="handleCopyComponent(index)">
<Icon icon="ep:copy-document" />
</el-button>
</el-tooltip>
<el-tooltip content="删除" placement="right">
<el-button @click.stop="handleDeleteComponent(index)">
<Icon icon="ep:delete" />
</el-button>
</el-tooltip>
</el-button-group>
</div>
</div>
</template>
</draggable>
</div>
</el-scrollbar>
<!-- 手机底部导航 -->
<div
v-if="showTabBar"
:class="[ :class="[
'editor-design-bottom',
'component', 'component',
'cursor-pointer!', 'cursor-pointer!',
{ active: selectedComponent?.id === tabBarComponent.id } { active: selectedComponent?.id === navigationBarComponent.id }
]" ]"
/>
</div>
<!-- 手机页面编辑区域 -->
<el-scrollbar
height="100%"
wrap-class="editor-design-center page-prop-area"
view-class="phone-container"
:view-style="{
backgroundColor: pageConfigComponent.property.backgroundColor,
backgroundImage: `url(${pageConfigComponent.property.backgroundImage})`
}"
>
<draggable
class="page-prop-area drag-area"
v-model="pageComponents"
item-key="index"
:animation="200"
filter=".component-toolbar"
ghost-class="draggable-ghost"
:force-fallback="true"
group="component"
@change="handleComponentChange"
> >
<TabBar :property="tabBarComponent.property" @click="handleTabBarSelected" /> <template #item="{ element, index }">
</div> <div class="component" @click="handleComponentSelected(element, index)">
<!-- 组件内容区 -->
<ComponentContainer :property="element.property.style">
<component
:is="element.id"
:property="element.property"
:data-type="element.id"
/>
</ComponentContainer>
<div :class="['component-wrap', { active: selectedComponentIndex === index }]">
<!-- 左侧组件名 -->
<div
:class="['component-name', { active: selectedComponentIndex === index }]"
v-if="element.name"
>
{{ element.name }}
</div>
<!-- 左侧组件操作工具栏 -->
<div
class="component-toolbar"
v-if="element.name && selectedComponentIndex === index"
>
<el-button-group type="primary">
<el-tooltip content="上移" placement="right">
<el-button
:disabled="index === 0"
@click.stop="handleMoveComponent(index, -1)"
>
<Icon icon="ep:arrow-up" />
</el-button>
</el-tooltip>
<el-tooltip content="下移" placement="right">
<el-button
:disabled="index === pageComponents.length - 1"
@click.stop="handleMoveComponent(index, 1)"
>
<Icon icon="ep:arrow-down" />
</el-button>
</el-tooltip>
<el-tooltip content="复制" placement="right">
<el-button @click.stop="handleCopyComponent(index)">
<Icon icon="ep:copy-document" />
</el-button>
</el-tooltip>
<el-tooltip content="删除" placement="right">
<el-button @click.stop="handleDeleteComponent(index)">
<Icon icon="ep:delete" />
</el-button>
</el-tooltip>
</el-button-group>
</div>
</div>
</div>
</template>
</draggable>
</el-scrollbar>
<!-- 手机底部导航 -->
<div
v-if="showTabBar"
:class="[
'editor-design-bottom',
'component',
'cursor-pointer!',
{ active: selectedComponent?.id === tabBarComponent.id }
]"
>
<TabBar :property="tabBarComponent.property" @click="handleTabBarSelected" />
</div> </div>
</div> </div>
<!-- 右侧属性面板 --> <!-- 右侧属性面板 -->
@ -383,6 +383,7 @@ onMounted(() => setDefaultSelectedComponent())
<style lang="scss" scoped> <style lang="scss" scoped>
/* 手机宽度 */ /* 手机宽度 */
$phone-width: 375px; $phone-width: 375px;
$toolbar-height: 42px;
/* 根节点样式 */ /* 根节点样式 */
.editor { .editor {
height: 100%; height: 100%;
@ -394,7 +395,7 @@ $phone-width: 375px;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: space-between; justify-content: space-between;
height: auto; height: $toolbar-height;
padding: 0; padding: 0;
border-bottom: solid 1px var(--el-border-color); border-bottom: solid 1px var(--el-border-color);
background-color: var(--el-bg-color); background-color: var(--el-bg-color);
@ -416,12 +417,14 @@ $phone-width: 375px;
/* 中心操作区 */ /* 中心操作区 */
.editor-container { .editor-container {
height: calc( height: calc(
100vh - var(--top-tool-height) - var(--tags-view-height) - var(--app-footer-height) - 42px 100vh - var(--top-tool-height) - var(--tags-view-height) - var(--app-footer-height) -
$toolbar-height
); );
/* 右侧属性面板 */ /* 右侧属性面板 */
.editor-right { .editor-right {
flex-shrink: 0; flex-shrink: 0;
box-shadow: -8px 0 8px -8px rgba(0, 0, 0, 0.12); box-shadow: -8px 0 8px -8px rgba(0, 0, 0, 0.12);
overflow: hidden;
/* 属性面板顶部:减少内边距 */ /* 属性面板顶部:减少内边距 */
:deep(.el-card__header) { :deep(.el-card__header) {
padding: 8px 16px; padding: 8px 16px;
@ -440,75 +443,80 @@ $phone-width: 375px;
/* 中心区域 */ /* 中心区域 */
.editor-center { .editor-center {
position: relative;
flex: 1 1 0; flex: 1 1 0;
padding: 16px 0;
background-color: var(--app-content-bg-color); background-color: var(--app-content-bg-color);
display: flex; display: flex;
flex-direction: column;
justify-content: center; justify-content: center;
/* 中心设计区域 */ margin: 16px 0 0 0;
.editor-design { overflow: hidden;
position: relative; width: 100%;
height: 100%;
/* 组件 */
.component {
width: $phone-width;
cursor: move;
/* 鼠标放到组件上时 */
&:hover {
border: 1px dashed var(--el-color-primary);
box-shadow: 0 0 5px 0 rgba(24, 144, 255, 0.3);
}
}
/* 组件选中 */
.component.active {
border: 2px solid var(--el-color-primary);
}
/* 手机顶部 */
.editor-design-top {
width: $phone-width;
margin: 0 auto;
/* 手机顶部状态栏 */
.status-bar {
height: 20px;
width: $phone-width;
background-color: #fff;
}
}
/* 手机底部导航 */
.editor-design-bottom {
width: $phone-width;
margin: 0 auto;
}
/* 手机页面编辑区域 */
:deep(.editor-design-center) {
width: 100%; width: 100%;
display: flex;
flex-direction: column;
align-items: center;
overflow: hidden;
/* 组件 */ /* 主体内容 */
.component { .phone-container {
border: 1px solid #fff; position: relative;
background-repeat: no-repeat;
background-size: 100% 100%;
height: 100%;
width: $phone-width; width: $phone-width;
cursor: move; margin: 0 auto;
/* 鼠标放到组件上时 */ .drag-area {
&:hover {
border: 1px dashed var(--el-color-primary);
}
}
/* 组件选中 */
.component.active {
border: 2px solid var(--el-color-primary);
}
/* 手机顶部 */
.editor-design-top {
width: $phone-width;
/* 手机顶部状态栏 */
.status-bar {
height: 20px;
width: $phone-width;
background-color: #fff;
}
}
/* 手机底部导航 */
.editor-design-bottom {
width: $phone-width;
}
/* 手机页面编辑区域 */
.editor-design-center {
width: 100%;
flex: 1 1 0;
:deep(.el-scrollbar__view) {
height: 100%; height: 100%;
width: 100%;
} }
/* 主体内容 */ .component {
.phone-container {
height: 100%;
box-sizing: border-box;
position: relative; position: relative;
background-repeat: no-repeat; cursor: move;
background-size: 100% 100%;
width: $phone-width;
margin: 0 auto;
.drag-area {
height: 100%;
}
/* 组件容器(左侧:组件名称,中间:组件,右侧:操作工具栏) */ .component-wrap {
.component-container { display: none;
position: absolute;
left: -2px;
top: 0;
width: 100%; width: 100%;
position: relative; height: 100%;
&.active {
display: block;
border: 2px solid var(--el-color-primary);
box-shadow: 0 0 10px 0 rgba(24, 144, 255, 0.3);
}
/* 左侧:组件名称 */ /* 左侧:组件名称 */
.component-name { .component-name {
position: absolute; position: absolute;
@ -518,7 +526,7 @@ $phone-width: 375px;
height: 25px; height: 25px;
background: #fff; background: #fff;
font-size: 12px; font-size: 12px;
left: -85px; left: -88px;
top: 0; top: 0;
box-shadow: box-shadow:
0 0 4px #00000014, 0 0 4px #00000014,
@ -548,7 +556,7 @@ $phone-width: 375px;
.component-toolbar { .component-toolbar {
position: absolute; position: absolute;
top: 0; top: 0;
right: -57px; right: -55px;
/* 左侧小三角 */ /* 左侧小三角 */
&:before { &:before {
position: absolute; position: absolute;