2022-12-01 17:38:05 +08:00

17 KiB
Raw Blame History

Editor组件 props

modelValue/v-model

  • 详情:

    页面初始值

  • 默认值: {}

  • 类型: MApp[]

  • 示例:

<template>
  <m-editor v-model="dsl"></m-editor>
</template>

<script setup>
import { ref } from 'Vue';

const dsl = ref({
  type: 'app',
  id: 'app_1',
  items: [
    {
      type: 'page',
      id: 'page_1',
      items: [
        {
          type: 'text',
          id: 'text_1',
          text: '文本'
        },
      ],
    },
  ],
});
</script>

componentGroupList

  • 详情:

    左侧面板中的组件列表

  • 默认值: []

  • 类型: ComponentGroup

::: tip icon使用的是element-plus icon

也可直接使用url例如

{
  icon: 'https://vfiles.gtimg.cn/vupload/20220614/9cc3091655207317835.png'
}

:::

  • 示例:
<template>
  <m-editor :component-group-list="componentGroupList"></m-editor>
</template>

<script setup>
import { ref, markRaw } from 'Vue';
import { FolderOpened, SwitchButton, Tickets } from '@element-plus/icons-vue';

const componentGroupList = ref([
  {
    title: '容器',
    items: [
      {
        icon: markRaw(FolderOpened),
        text: '组',
        type: 'container',
      },
    ],
  },
  {
    title: '基础组件',
    items: [
      {
        icon: markRaw(Tickets),
        text: '文本',
        type: 'text',
      },
      {
        icon: markRaw(SwitchButton),
        text: '按钮',
        type: 'button',
      },
    ],
  },
]);
</script>

::: warning 此配置仅在sidebar中配置了'component-list'时有效 :::

sidebar

  • 详情:

    左侧面板目前只支持type: 'tabs';

  • 默认值:

{
  type: 'tabs',
  status: '组件',
  items: ['component-list', 'layer', 'code-block'],
}
<template>
  <m-editor :sidebar="sidebar"></m-editor>
</template>

<script setup>
import { ref, markRaw } from 'Vue';
import { List } from '@element-plus/icons-vue';
import ModListPanel from '../components/sidebars/ModListPanel.vue';

const sidebar = ref({
  type: 'tabs',
  status: '组件',
  items: [
    'component-list',
    'layer',
    {
      type: 'component',
      icon: markRaw(List),
      component: markRaw(ModListPanel),
      text: '模块',
    },
});
</script>

::: tip icon使用的是element-plus icon

也可直接使用url例如

{
  icon: 'https://vfiles.gtimg.cn/vupload/20220614/9cc3091655207317835.png'
}

:::

menu

  • 详情:

    顶部工具栏

    系统提供了几个常用功能: '/' | 'delete' | 'undo' | 'redo' | 'zoom-in' | 'zoom-out' | 'zoom' | 'guides' | 'rule' | 'scale-to-original' | 'scale-to-fit'

    '/': 分隔符

    'delete': 删除按钮

    'undo': 撤销按钮

    'redo': 恢复按钮

    'zoom-in': 放大按钮

    'zoom-out': 缩小按钮

    'zoom': 缩放(zoom-in', 'zoom-out', 'scale-to-original', 'scale-to-fit' 的集合)

    'guides': 显示隐藏参考线

    'rule': 显示隐藏标尺

    'scale-to-original': 缩放到实际大小

    'scale-to-fit': 缩放以适应

  • 默认值:

{ left: [], center: [], right: [] }
<template>
  <m-editor :menu="menu"></m-editor>
</template>

<script setup>
import { ref, markRaw, toRaw } from 'Vue';
import { ArrowLeft, Coin } from '@element-plus/icons-vue';

const menu = ref({
  left: [
    {
      type: 'button',
      icon: markRaw(ArrowLeft),
      tooltip: '返回',
    },
    '/',
    {
      type: 'text',
      text: '.temp',
    },
  ],
  center: ['delete', 'undo', 'redo', 'zoom'],
  right: [
    {
      type: 'button',
      text: '保存',
      icon: markRaw(Coin),
      disabled: true,
      handler: ({ editorService }) => {
        console.log(toRaw(editorService.get('root'));
      }),
    },
  ],
})
</script>

layerContentMenu

  • 详情: 扩展已选组件(组件树)右键菜单

  • 默认值: []

  • 类型: (MenuButton | MenuComponent)[]

  • 示例:

<template>
  <m-editor :layer-content-menu="layerContentMenu"></m-editor>
</template>

<script setup>
import { ref, toRaw } from 'Vue';
import { Upload } from '@element-plus/icons-vue';

const layerContentMenu = ref([
  {
    {
      type: 'button',
      icon: toRow(Upload),
      text: '导出',
      handler: () => {
        console.log('export')
      },
    },
  },
]);
</script>

stageContentMenu

  • 详情: 扩展stage画布区域右键菜单

  • 默认值: []

  • 类型: (MenuButton | MenuComponent)[]

  • 示例:

<template>
  <m-editor :stage-content-menu="stageContentMenu"></m-editor>
</template>

<script setup>
import { ref, toRaw } from 'Vue';
import { Upload } from '@element-plus/icons-vue';

const stageContentMenu = ref([
  {
    {
      type: 'button',
      icon: toRow(Upload),
      text: '导出',
      handler: () => {
        console.log('export')
      },
    },
  },
]);
</script>

runtimeUrl

  • 详情:

    runtime的HTML地址

    在编辑器画布中渲染组件是使用iframe实现runtimeUrl就是配置在iframe上的src属性

  • 默认值: undefined

  • 类型: string

  • 示例:

<template>
  <m-editor
    runtime-url="https://tencent.github.io/tmagic-editor/playground/runtime/vue3/playground/index.html"
  ></m-editor>
</template>

render

  • 详情:

    中间工作区域中画布渲染的内容,通常是通过解析modelValue来渲染出DOMreturn的DOM结构需要有一个根节点。

    :::tip runtimeUrl与render有且只需要配置一个 :::

  • 默认值: undefined

  • 类型: (stage: StageCore) => HTMLDivElement | Promise<HTMLDivElement>

  • 示例:

<template>
  <m-editor
    :render="renderFunction"
  ></m-editor>
</template>

<script setup>
const renderFunction = async (stage) => {
  const { iframe } = stage.renderer;

  if (!iframe) throw new Error('iframe 未创建');

  return iframe.contentDocument.createElement('div');
}
</script>

autoScrollIntoView

  • 详情:

    选中组件时,是否自动滚动该组件到可视区域

  • 默认值: undefined

  • 类型: boolean

  • 示例:

<template>
  <m-editor :auto-scroll-intoView="true"></m-editor>
</template>

propsConfigs

<template>
  <m-editor :props-configs="propsConfigs"></m-editor>
</template>

<script setup>
const propsConfigs = {
  text: [
    {
      name: 'text',
      text: '文本',
    },
    {
      name: 'multiple',
      text: '多行文本',
      type: 'switch',
    },
  ],
  button: [
    {
      name: 'text',
      text: '文本',
    },
  ]
}
</script>

propsValues

  • 详情:

    添加组件时的默认值

    添加组件时组件节点DSL的生成过程为先从componentGroupList中items中item的data得到一个基础配置然后再通过propsService.getPropsValue()方法获取到propsValues中对应type的默认值二者合并生成新增节点的DSL配置

    :::tip 该属性最终会设置到propsService中,所以也可直接调用propsService.setPropsValues()方法来配置 :::

  • 默认值: {}

  • 类型: Record<string, Object>

  • 示例:

<template>
  <m-editor :props-values="propsValues"></m-editor>
</template>

<script setup>
const propsValues = {
  text: {
    text: '文本',
    // 多行文本
    multiple: true,
  },
  button: {
    text: '按钮',
  },
};
</script>

eventMethodList

<template>
  <m-editor :event-method-list="eventMethodList"></m-editor>
</template>

<script setup>
const eventMethodList = {
  page: {
    methods: [
      {
        label: '刷新',
        value: 'refresh',
      },
    ],
    events: [
      {
        label: '加载',
        value: 'load',
      },
    ],
  },
};
</script>

moveableOptions

<template>
  <m-editor :moveable-options="moveableOptions"></m-editor>
</template>

<script setup>
const moveableOptions = ({ targetId }) => {
  const options = {};
  const node = editorService.getNodeById(id);

  if (!node) return options;

  const isPage = node.type === 'page';

  // 页面不允许拖动
  options.draggable = !isPage;
  options.resizable = !isPage

  return options;
};
</script>

defaultSelected

  • 详情:

    编辑器初始化后默认选中的组件节点id

  • 默认值: undefined

  • 类型: string | number

  • 示例:

<template>
  <m-editor :default-selected="defaultSelected"></m-editor>
</template>

<script setup>
import { ref } from 'vue';

const defaultSelected = ref('');

// 加载dsl后设置默认选中节点
getDsl().then(() => {
  defaultSelected.value = 'xxx';
});
</script>

canSelect

  • 详情:

    鼠标在画布点击时当前坐标下的dom节点是否可以选中当前坐标下的dom可能不止只有一个当有多个时会遍历这些节点并调用canSelect方法进行判断第一个返回true的dom节点将被选中

    :::tip 获取坐标下的节点是通过document.elementsFromPoint方法 :::

  • 默认值: (el: HTMLElement) => Boolean(el.id)

  • 类型: (el: HTMLElement) => boolean | Promise<boolean>

  • 示例:

<template>
  <m-editor :can-select="canSelect"></m-editor>
</template>

<script setup>
// 只有dom id为纯数字时才可以被选中
const canSelect = (el) => /^\d+$/.test(el.id);
</script>

isContainer

  • 详情:

    当组件拖动过程中停留在画布上超过 containerHighlightDuration 时长时,识别当前是否有容器

    当停留超过containerHighlightDuration 时长时会通过停留的坐标获取当前坐标下所有dom节点然后遍历这些节点并通过isContainer方法判断第一个返回true的节点将视为容器

    :::tip 获取坐标下的节点是通过document.elementsFromPoint方法 :::

  • 默认值: (el: HTMLElement) => el.classList.contains('magic-ui-container')

  • 类型: (el: HTMLDivElement) => boolean | Promise<boolean>;

  • 示例:

<template>
  <m-editor :is-container="isContainer"></m-editor>
</template>

<script setup>
const canSelect = (el) => /^\d+$/.test(el.id);

const isContainer = (el) =>
  canSelect(el) &&
  (el.classList.contains('magic-ui-container') ||
    el.classList.contains('magic-ui-pc-container') ||
    el.classList.contains('magic-ui-page') ||
    el.classList.contains('magic-ui-pop-content'));
</script>

containerHighlightClassName

  • 详情:

识别到容器后会给其dom上添加的class

  • 默认值: 'tmagic-stage-container-highlight'

  • 类型: string

containerHighlightDuration

  • 详情:

当组件拖动过程中停留在画布上超过 containerHighlightDuration 时长时,识别当前是否有容器

  • 默认值: 800 单位为ms

  • 类型: number

containerHighlightType

  • 详情:

在画布中,将组件拖入其他容器的方式

启动方式

default: 停留在画布上启动识别

alt: 按住alt键启动识别

其他值:不启动

  • 默认值: 'default'

  • 类型: 'default' | 'alt' | ''

stageRect

{
  width: 375,
  height: 817,
}
  • 类型:
interface StageRect {
  width: number;
  height: number;
}

codeOptions

  • 详情:

    编辑器中的代码编辑器配置

    :::tip tmagic-editor中的所有代码编辑器都使用monaco-editor,详细配置请前往monaco-editor官网查看 :::

  • 默认值: ``{}

  • 类型: Object

  • 示例:

<template>
  <m-editor :code-options="codeOptions"></m-editor>
</template>

<script setup>
const codeOptions = {
  tabSize: 2,
  fontSize: 16,
  formatOnPaste: true,
}
</script>

updateDragEl

  • 详情:

当选中框与组件不贴合时,可以通过此方法进行调整

:::tip 由于画布中组件是渲染在iframe中而选中框是渲染在编辑器中所以会导致两者的坐标系有差异为了解决这个问题canSelect为true后会在编辑中创建一个位置大小与组件target一致的domel :::

  • 类型: (el: HTMLElement | SVGElement, target: HTMLElement | SVGElement) => void

  • 默认值: undefined

  • 示例:

<template>
  <m-editor :update-drag-el="updateDragEl"></m-editor>
</template>

<script setup>
const updateDragEl = (el, target) => {
  const node = editorStore.pop;
  if (!node || !isPop(node)) {
    return;
  }

  const { top, left } = target.getBoundingClientRect();

  el.style.transform = 'none';
  el.style.top = `${top}px`;
  el.style.left = `${left}px`;
};