tiny-engine/packages/common/component/PluginBlockList.vue

740 lines
19 KiB
Vue

<template>
<ul
v-if="state.data.length"
:class="['block-list', 'lowcode-scrollbar', { 'is-small-list': blockStyle === 'mini' }, { isShortcutPanel }]"
@mouseleave="state.hover = false"
>
<li
v-for="(item, index) in state.data"
:key="item.blockName"
:draggable="!isBlockManage && showSettingIcon"
:class="[
'block-item',
{ 'is-active': state.activeIndex === index },
{ 'is-disabled': showBlockDetail },
{ 'block-item-small-list': blockStyle === 'mini' }
]"
:title="getTitle(item)"
@mousedown.stop.left="blockClick({ event: $event, item, index })"
@mouseover.stop="openBlockShotPanel(item, $event)"
@mouseleave="handleBlockItemLeave"
>
<slot :data="item">
<img
v-if="item.screenshot"
class="item-image"
:src="item.screenshot || defaultImg"
draggable="false"
@error="$event.target.src = defaultImg"
/>
<svg-icon v-else class="item-image item-default-img" name="block-default-img"></svg-icon>
<div class="item-text">
<div class="item-name">{{ item.name_cn || item.label || item.content?.fileName }}</div>
<div v-if="blockStyle === 'list'" class="item-description">{{ item.description }}</div>
</div>
<div v-if="item.isShowProgress" class="progress-bar">
<tiny-progress
:text-inside="true"
:stroke-width="8"
:percentage="item.publishProgress"
status="success"
></tiny-progress>
</div>
<div v-if="isBlockManage && !item.is_published" class="publish-flag">未发布</div>
<div v-if="isBlockManage" class="block-detail">
<tiny-tooltip effect="dark" :content="defaultIconTip" placement="top">
<icon-setting
class="block-detail-icon"
@mouseover.stop="iconSettingMove"
@mousedown.stop.prevent="iconClick({ event: $event, item, index })"
></icon-setting>
</tiny-tooltip>
</div>
<div
v-else-if="showSettingIcon"
:class="['block-setting', { 'is-current-visible-icon': state.hoverItemId === item.id }]"
title=" "
>
<tiny-popover
v-if="!item.isDefaultGroup"
placement="bottom-end"
width="151"
append-to-body
trigger="manual"
:modelValue="state.hoverItemId === item.id && state.currentShowMenuId === item.id"
:visible-arrow="false"
popper-class="popper-options block-setting-popover"
>
<template #reference>
<svg-button
name="ellipsis"
class="block-detail-icon"
@click="handleShowVersionMenu(item)"
@mouseover.stop="iconSettingMove"
@mousedown.stop.prevent=""
></svg-button>
</template>
<template #default>
<div class="setting-menu" @mouseover.stop="handleSettingMouseOver" @mouseleave="handleBlockItemLeave">
<ul class="list">
<li class="list-item" @click="$emit('openVersionPanel', { item, index })">
<span>版本列表</span>
</li>
<li class="list-item" @click="$emit('deleteBlock', item)">
<span>移除</span>
</li>
</ul>
</div>
</template>
</tiny-popover>
</div>
<div
v-if="item.isAnimation"
:class="[
'deploy',
{ success: item.deployStatus === taskStatus.FINISHED },
{
error: item.deployStatus === taskStatus.STOPPED
}
]"
@mouseover.stop="item.isAnimation = false"
>
{{ deployTips[item.deployStatus] }}
</div>
</slot>
</li>
<li v-if="state.showAddButton" class="block-item block-plus" @click="$emit('add')">
<span class="block-plus-icon"><icon-plus></icon-plus></span>
</li>
<div v-if="showBlockShot && state.hover && state.currentBlock.screenshot" class="block-shortcut">
<div class="block-shortcut-title">{{ state.currentBlock.label }}预览图</div>
<div v-if="state.currentBlock.description" class="block-shortcut-description">
{{ state.currentBlock.description }}
</div>
<div class="block-shortcut-image-wrapper">
<img
class="block-shortcut-image"
:src="state.currentBlock.screenshot || defaultImg"
@error="$event.target.src = defaultImg"
/>
</div>
</div>
</ul>
<search-empty :isShow="!state.data.length" />
</template>
<script>
import { computed, watch, inject, reactive } from 'vue'
import { iconSetting, iconPlus } from '@opentiny/vue-icon'
import { Tooltip, Progress, Popover } from '@opentiny/vue'
import SearchEmpty from './SearchEmpty.vue'
import SvgButton from './SvgButton.vue'
const defaultImg =
''
export default {
components: {
TinyProgress: Progress,
TinyTooltip: Tooltip,
IconSetting: iconSetting(),
IconPlus: iconPlus(),
TinyPopover: Popover,
SvgButton,
SearchEmpty
},
props: {
data: {
type: Array,
default: () => []
},
/*
列表样式,可选择为 default || list || mini 默认值为 default
*/
blockStyle: {
type: String,
default: 'default'
},
/*
用于区分是否是区块管理侧的列表
*/
isBlockManage: {
type: Boolean,
default: false
},
/*
是否显示新增按钮
*/
showAddButton: {
type: Boolean,
default: false
},
/*
是否显示区块详情弹框
*/
showBlockDetail: {
type: Boolean,
default: false
},
/*
是否显示快照
*/
showBlockShot: {
type: Boolean,
default: false
},
/*
默认 ICON 的提示文字
*/
defaultIconTip: {
type: String,
default: ''
},
// 是否显示历史备份按钮
showSettingIcon: {
type: Boolean,
default: true
},
// 外部传入的区块信息:不通过区块列表里点击展示,而是从外面直接调起区块面板展示的区块。
externalBlock: {
type: Object,
default: null
}
},
emits: ['click', 'iconClick', 'add', 'deleteBlock', 'openVersionPanel'],
setup(props, { emit }) {
const panelState = inject('panelState', {})
const state = reactive({
activeIndex: -1,
data: computed(() => props.data),
showAddButton: computed(() => props.showAddButton),
top: 0,
hover: false,
currentBlock: {},
hoverItemId: null,
currentShowMenuId: null,
timeoutId: null
})
const getParentNode = (el) => {
while (el.nodeName !== 'LI') {
el = el.parentNode
}
return el
}
const openBlockShotPanel = (item, event) => {
state.currentBlock = item
state.top = `${getParentNode(event.target).getBoundingClientRect().top}px`
state.hover = true
state.hoverItemId = item.id
if (state.currentShowMenuId === item.id) {
clearTimeout(state.timeoutId)
}
}
const blockClick = ({ event, item, index }) => {
if (props.isBlockManage) {
state.activeIndex = index
}
emit('click', item)
// 点击区块并不打开设置面板
emit('iconClick', { event, item, index, isOpen: false })
}
const iconClick = ({ event, item, index }) => {
state.activeIndex = index
emit('iconClick', { event, item, index, isOpen: true })
}
// 清除当前选择状态
const clearActive = () => {
state.activeIndex = -1
}
// 区块发布任务说明
const taskStatus = {
RUNNING: 1,
STOPPED: 2,
FINISHED: 3
}
const deployTips = {
1: '正在发布中',
2: '发布失败,请重新发布',
3: '发布完成'
}
const iconSettingMove = () => {
state.hover = false
}
const getTitle = (item) => (item.groupName ? `分组: ${item.groupName + '\n'}` : '') + (item.label || item.blockName)
const handleBlockItemLeave = () => {
state.timeoutId = setTimeout(() => {
state.hoverItemId = null
state.currentShowMenuId = null
}, 200)
}
const handleSettingMouseOver = () => {
clearTimeout(state.timeoutId)
}
const handleShowVersionMenu = (item) => {
if (state.currentShowMenuId) {
state.currentShowMenuId = null
} else {
state.currentShowMenuId = item.id
}
}
// 若是存在外部区块:即不通过区块列表里点击展示,而是从外面直接调起区块面板展示的区块,
// 那么当前高亮的所选中的区块需要切换成外部区块
const changeActiveIndex = (blockList, block) => {
const blockIndex = blockList.findIndex((item) => item.id === block.id)
state.activeIndex = blockIndex
}
watch(
() => props.data,
async (blockList) => {
if (blockList && props.externalBlock) {
changeActiveIndex(blockList, props.externalBlock)
}
}
)
watch(
() => props.externalBlock,
async (block) => {
if (block && state.data?.length) {
changeActiveIndex(state.data, block)
}
}
)
return {
isShortcutPanel: panelState.isShortcutPanel,
state,
getTitle,
blockClick,
iconClick,
clearActive,
taskStatus,
deployTips,
openBlockShotPanel,
iconSettingMove,
defaultImg,
handleBlockItemLeave,
handleSettingMouseOver,
handleShowVersionMenu
}
}
}
</script>
<style lang="less" scoped>
.block-shortcut {
position: fixed;
z-index: 9999;
top: 50px;
left: calc(var(--base-left-panel-width) + var(--base-nav-panel-width) + 10px);
max-width: 500px;
max-height: 136px;
padding: 12px;
background: var(--ti-lowcode-component-block-list-shortcut-bg);
border-radius: 5px;
border: 1px solid var(--ti-lowcode-common-border-color-4);
top: v-bind('state.top');
.block-shortcut-title {
color: var(--ti-lowcode-component-block-list-shortcut-title-color);
font-weight: 600;
margin-bottom: 8px;
}
.block-shortcut-description {
color: var(--ti-lowcode-component-block-list-item-color);
margin-bottom: 20px;
font-size: 12px;
}
.block-shortcut-image-wrapper {
height: 80px;
overflow: hidden;
}
.block-shortcut-image {
width: 100%;
object-fit: contain;
}
}
.block-list {
display: grid;
grid-template-columns: 50% 50%;
grid-template-rows: repeat(auto-fill, 96px);
position: relative;
flex: 1;
overflow-y: auto;
overflow-x: hidden;
color: var(--ti-lowcode-common-secondary-text-color);
&.is-small-list {
grid-template-columns: 100%;
grid-template-rows: repeat(auto-fill, 30px);
}
.block-item {
display: flex;
flex-direction: column;
align-items: center;
position: relative;
height: 96px;
padding: 10px;
border-right: 1px solid var(--ti-lowcode-component-block-list-border-color);
border-bottom: 1px solid var(--ti-lowcode-component-block-list-border-color);
text-align: center;
user-select: none;
&:nth-child(-n + 2) {
border-top: 1px solid var(--ti-lowcode-component-block-list-border-color);
}
&.block-item-small-list:nth-child(2) {
border-top: none;
}
.publish-flag {
position: absolute;
left: 2px;
top: 2px;
text-align: center;
display: block;
color: var(--ti-lowcode-common-secondary-text-color);
font-size: 12px;
background-color: var(--ti-lowcode-component-block-list-item-tag-bg);
padding: 2px;
border-radius: 4px 0 4px 0;
transform: scale(0.9);
}
&.block-item-small-list {
flex-direction: row;
align-items: center;
height: 30px;
padding: 4px 10px;
.item-image {
width: 30px;
height: 30px;
min-width: 30px;
}
.item-text {
text-align: left;
margin-top: 0;
margin-left: 4px;
}
.publish-flag {
position: static;
}
.block-detail,
.block-setting {
visibility: hidden;
position: static;
margin-left: 4px;
z-index: 9;
.block-detail-icon {
color: var(--ti-lowcode-component-block-list-setting-btn-color);
display: block;
&:hover {
cursor: pointer;
color: var(--ti-lowcode-component-block-list-setting-btn-hover-color);
}
}
}
}
&:nth-child(even) {
border-right: 0;
}
:deep(.tiny-progress.is-success .tiny-progress-bar__inner) {
animation-duration: 5s;
animation-name: roll;
animation-iteration-count: infinite;
animation-timing-function: linear;
}
:deep(.tiny-progress-bar__innerText) {
display: none;
}
.progress-bar {
width: 100%;
}
&.is-active {
background: var(--ti-lowcode-component-block-list-item-active-bg, --ti-lowcode-canvas-wrap-bg);
}
&.is-disabled {
& + .block-plus:hover {
background: transparent;
cursor: inherit;
}
}
&:not(.is-disabled):hover {
background-color: var(--ti-lowcode-component-block-list-item-active-bg);
cursor: pointer;
.block-detail,
.block-setting {
visibility: visible;
}
&[draggable='true'] {
cursor: move;
}
}
&.block-plus {
display: flex;
justify-content: center;
align-items: center;
.tiny-svg {
font-size: 24px;
color: var(--ti-lowcode-component-svg-button-color);
}
&:hover {
cursor: pointer;
color: var(--ti-lowcode-component-svg-button-hover-color);
}
}
.item-image {
width: 100px;
height: 48px;
overflow: hidden;
object-fit: cover;
}
.item-default-img {
width: 50px;
height: 50px;
}
.item-text {
color: var(--ti-lowcode-component-block-list-item-color);
text-align: center;
flex: 1;
margin-top: 10px;
overflow: hidden;
text-overflow: ellipsis;
max-width: 100%;
}
.item-name {
white-space: nowrap;
text-overflow: ellipsis;
overflow: hidden;
font-size: 12px;
}
.block-detail,
.block-setting {
visibility: hidden;
position: absolute;
top: 4px;
right: 4px;
z-index: 9;
&.is-current-visible-icon {
visibility: visible;
}
.block-detail-icon {
color: var(--ti-lowcode-component-block-list-setting-btn-color);
&:hover {
cursor: pointer;
color: var(--ti-lowcode-component-block-list-setting-btn-hover-color);
}
}
}
.block-setting {
right: 0px;
top: 0;
}
}
.deploy {
position: absolute;
top: 10px;
left: 10px;
width: 114px;
color: var(--ti-lowcode-toolbar-icon-color);
font-weight: bold;
vertical-align: middle;
text-align: center;
margin: -10px -11px;
padding: 40px 10px;
}
.loading {
animation-duration: 5s;
animation-name: roll;
animation-iteration-count: infinite;
animation-timing-function: linear;
background: repeating-linear-gradient(
135deg,
rgba(39, 115, 214, 0.6) 0px,
rgba(39, 115, 214, 0.6) 10px,
rgba(255, 255, 255, 0.6) 10px,
rgba(255, 255, 255, 0.6) 20px
);
}
.success {
width: 100%;
height: 100%;
position: absolute;
top: 10px;
animation: greenGrow 800ms ease-out 10 alternate;
background: rgb(55 68 58 / 60%);
}
.error {
width: 100%;
height: 100%;
position: absolute;
top: 10px;
animation: redGrow 800ms ease-out 20 alternate;
background: rgb(86 52 52 / 60%);
}
@keyframes roll {
from {
background-position: 0 0;
}
to {
background-position: 128px 0;
}
}
@keyframes greenGrow {
0% {
box-shadow: inset 0px 0px 3px green;
}
100% {
box-shadow: inset 0px 0px 7px green;
}
}
@keyframes redGrow {
0% {
box-shadow: inset 0px 0px 3px red;
}
100% {
box-shadow: inset 0px 0px 7px red;
}
}
&.isShortcutPanel {
grid-template-columns: 1fr 1fr 1fr 1fr;
height: 300px;
.block-item {
.item-text {
width: 100px;
}
}
}
&.is-small-list {
display: block;
grid-template-columns: initial;
.block-item {
flex-direction: row;
border-right: none;
}
.item-image {
padding: 0;
flex-shrink: 0;
}
.item-text {
margin-top: 0;
padding: 0 8px;
text-align: left;
.item-name {
font-size: 12px;
line-height: 16px;
}
.item-description {
color: var(--ti-lowcode-toolbar-title-color);
font-size: 12px;
}
}
}
&.is-small-list {
.block-item {
height: 38px;
}
.item-image {
font-size: 1.5em;
width: 27px;
height: 22px;
}
.item-text {
width: calc(100% - 35px);
}
}
}
.setting-menu {
font-size: 12px;
color: var(--ti-lowcode-component-block-setting-item-text-color);
.list {
margin-top: 6px;
}
.list-item {
box-sizing: border-box;
padding: 16px 18px;
height: 30px;
line-height: 30px;
cursor: pointer;
span {
margin-left: 8px;
}
display: flex;
align-items: center;
&:hover {
background-color: var(--ti-lowcode-component-block-setting-item-hover-bg);
color: var(--ti-lowcode-common-primary-text-color);
}
.list-item-icon {
font-size: 14px;
}
}
}
</style>
<style lang="less">
.tiny-popover.tiny-popper.popper-options.block-setting-popover {
background-color: var(--ti-lowcode-component-block-setting-popover-bg);
display: flex;
flex-direction: column;
padding: 0;
.popper__arrow,
.popper__arrow::after {
border-bottom-color: var(--ti-lowcode-component-block-setting-popover-bg);
}
}
</style>