Tinymce富文本编辑器二次开发电子病历时解决的bug

news2025/1/11 20:08:42

前言

本文是在Tinymce富文本编辑器添加自定义toolbar,二级菜单,自定义表单,签名的基础之上进行一些bug记录,功能添加,以及模版的应用和打印
项目描述

  1. 建立电子病历模版—录入(电子病历模版和电子病历打印模版)—查看电子病历和打印病历模版
    ![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/9fd8573225e94f8e9fc30b48975fd424.png
  2. 建立电子病历----添加一个电子病历可以添加多个电子病历模版----并输入设置的值提交以key和value的组成的数组结构,提交,并在编辑时返显电子病历模版和数据----点击打印按钮,调出打印模版,赋值进行打印。
    在这里插入图片描述

一、模版应用及打印

  1. 模版的应用
    点击新增电子病历,选择用户,选择电子模版,展示。此时电子病历只填值,此处需要优化。
    选择完顾客,顾客的信息需要返显在电子模版上,
  • 解决的问题:

    模版返显—拿到模版的key和value组成的数组—然后赋值—更新模版—在页面中展示

  • 引出的问题:

    1)输入后每次光标会到最左侧,输入出错。
    原因封装的编辑器每次更新之后都会通过模版的key和value组成的数组,然后重新设置模版,所以光标返回最左侧
    解决方法,定义变量只有一次拿到模版的key和value组成的数组,其他变更不触发emit返回新的数组,不重新给模版赋值

    ![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/ae8287df6a2e4b009fc5886cfb99915c.png
    编辑组件的应用

    <Editor
        ref="editorTinymce"
         height="900"
         v-model="templateData"  // 模版的html
         :templatevalue="templatevalue" // 模版的key和value组成的数组
         :isHide="true" // 是否展示菜单栏和toolbar栏
         v-if="!!templateData" // 模版展示条件
         :templateData="templateData" // 模版的html
         :getParams="getParams" // 是否需要获取key和value组成的数组
         @templateparams="handleTemplateParams" // 模版返回的key和value组成的数组
       />
    
  1. 模版的赋值
    点击左侧已添加的病历,返显模版以及将模版之前输入的值进行返显
    在这里插入图片描述
  • 引出的问题:

    1)切换模版,赋值有时不更新
    赋值分2种,1.初始化赋值,2.变更赋值。
    解决方法,第一次点击为初始化赋值,之后的点击为变更赋值,切换已添加的模版并没有销毁编辑器,所以是变更。

  1. 模版的打印
    点击编辑页打印—调出打印模版—将录入模版的值赋值给打印模版
    首先:输入模版和打印模版对应的值需要设置对应的key,如果不一样会赋值失败,此处属于初化始化赋值。
    在这里插入图片描述
    点击列表操作列的打印—弹出所有关联的模版—选中模版,给打印模版赋值,进行打印预览
    在这里插入图片描述
  2. 模版的销毁
    关闭弹窗之后,需要销毁模版,否者再次打开时之前打开的模版还在。

封装组件代码

<template>
  <div :class="prefixCls" :style="{ width: containerWidth }">
    <ImgUpload
      :fullscreen="fullscreen"
      @uploading="handleImageUploading"
      @done="handleDone"
      v-if="showImageUpload && !props.isHide"
      v-show="editorRef"
      :disabled="disabled"
      :uploadParams="props.uploadParams"
    />
    <textarea :id="tinymceId" ref="elRef" :style="{ visibility: 'hidden' }" v-if="!initOptions.inline"></textarea>
    <slot v-else></slot>
    <signModal @register="signatureModal" @success="handleSignature" @exportSign="getSign" />
    <recordModal @register="recorderModal" @success="handleGetText" />
  </div>
</template>

<script lang="ts">
import sign from '/@/assets/svg/sign.svg';
import recordSvg from '/@/assets/svg/record.svg';
import type { Editor, RawEditorSettings, BodyComponentSpec } from 'tinymce';
import { useMessage } from '/@/hooks/web/useMessage';
import tinymce from 'tinymce/tinymce';
import 'tinymce/themes/silver';
import 'tinymce/icons/default/icons';
import 'tinymce/plugins/advlist';
import 'tinymce/plugins/anchor';
import 'tinymce/plugins/autolink';
import 'tinymce/plugins/autosave';
import 'tinymce/plugins/code';
import 'tinymce/plugins/codesample';
import 'tinymce/plugins/directionality';
import 'tinymce/plugins/fullscreen';
import 'tinymce/plugins/hr';
import 'tinymce/plugins/insertdatetime';
import 'tinymce/plugins/link';
import 'tinymce/plugins/lists';
import 'tinymce/plugins/media';
import 'tinymce/plugins/nonbreaking';
import 'tinymce/plugins/noneditable';
import 'tinymce/plugins/pagebreak';
import 'tinymce/plugins/paste';
import 'tinymce/plugins/preview';
import 'tinymce/plugins/print';
import 'tinymce/plugins/save';
import 'tinymce/plugins/searchreplace';
import 'tinymce/plugins/spellchecker';
import 'tinymce/plugins/tabfocus';
import 'tinymce/plugins/table';
import 'tinymce/plugins/template';
import 'tinymce/plugins/textpattern';
import 'tinymce/plugins/visualblocks';
import 'tinymce/plugins/visualchars';
import 'tinymce/plugins/wordcount';
// import '/@/components/MedicalTinymce/plugins/control/index.js';

import { defineComponent, computed, nextTick, ref, unref, watch, onDeactivated, onBeforeUnmount, toRaw } from 'vue';
import ImgUpload from './ImgUpload.vue';
import { toolbar, plugins } from './tinymce';
import { buildShortUUID } from '/@/utils/uuid';
import { bindHandlers } from './helper';
import { useModal } from '/@/components/Modal';
import { ActionEnum, VALIDATE_API } from '/@/enums/commonEnum';
import { onMountedOrActivated } from '/@/hooks/core/onMountedOrActivated';
import { useDesign } from '/@/hooks/web/useDesign';
import { isNumber } from '/@/utils/is';
import { useLocale } from '/@/locales/useLocale';
import { useAppStore } from '/@/store/modules/app';
import { asyncFindDefUrlById, asyncFindUrlById } from '/@/api/lamp/file/upload';
import signModal from '/@/components/Signature/components/signModal/index.vue';
import recordModal from '/@/components/CustomRecorder/index.vue';
const tinymceProps = {
  options: {
    type: Object as PropType<Partial<RawEditorSettings>>,
    default: () => ({}),
  },
  value: {
    type: String,
  },

  toolbar: {
    type: Array as PropType<string[]>,
    default: toolbar,
  },
  plugins: {
    type: Array as PropType<string[]>,
    default: plugins,
  },
  modelValue: {
    type: String,
  },
  height: {
    type: [Number, String] as PropType<string | number>,
    required: false,
    default: 400,
  },
  width: {
    type: [Number, String] as PropType<string | number>,
    required: false,
    default: 'auto',
  },
  showImageUpload: {
    type: Boolean,
    default: true,
  },
  isDef: {
    type: Boolean,
    default: false,
  },
  uploadParams: {
    type: Object as PropType<any>,
    default: {},
  },
  isHide: {
    type: Boolean,
    default: false,
  },
  isPrint: {
    type: Boolean,
    default: false,
  },
  templatevalue: {
    type: Array,
  },
  getParams: {
    type: Boolean,
    default: false,
  },
};
export default defineComponent({
  name: 'Tinymce',
  components: { ImgUpload, signModal, recordModal },
  inheritAttrs: false,
  props: tinymceProps,
  emits: ['change', 'update:modelValue', 'inited', 'init-error', 'templateparams'],
  setup(props, { emit, attrs }) {
    const { createMessage } = useMessage();
    const editorRef = ref<Nullable<Editor>>(null);
    const fullscreen = ref(false);
    const tinymceId = ref<string>(buildShortUUID('tiny-vue'));
    const elRef = ref<Nullable<HTMLElement>>(null);
    let dialogConfig = ref(null);
    const { prefixCls } = useDesign('tinymce-container');
    const [signatureModal, { openModal: openSignModal }] = useModal();
    const [recorderModal, { openModal: openRecord }] = useModal();
    const appStore = useAppStore();
    const appEnv = import.meta.env.MODE;
    let currentBookMark = ref<any>('');

    const tinymceContent = computed(() => props.modelValue);
    const childBtn = {
      type: 'grid', // component type
      columns: 1, // number of columns
      items: [
        {
          type: 'button',
          name: 'add',
          text: '添加子项',
        },
        {
          type: 'button',
          name: 'del',
          text: '删除子项',
        },
        {
          type: 'collection', // component type
          name: 'collection', // identifier
          label: '',
        },
        {
          type: 'collection', // component type
          name: 'collection1', // identifier
          label: '',
        },
      ], // array of panel components
    };
    let childItem = {
      type: 'grid', // component type
      columns: 1, // number of columns
      items: [
        {
          type: 'grid',
          columns: 2,
          items: [
            {
              type: 'input',
              name: 'label1',
              label: '标签1',
            },
            {
              type: 'input',
              name: 'value1',
              label: '值1',
            },
          ],
        },
      ], // array of panel components
    };
    const containerWidth = computed(() => {
      const width = props.width;
      if (isNumber(width)) {
        return `${width}px`;
      }
      return width;
    });

    const skinName = computed(() => {
      return appStore.getDarkMode === 'light' ? 'oxide' : 'oxide-dark';
    });

    const langName = computed(() => {
      const lang = useLocale().getLocale.value;
      return ['zh_CN', 'en'].includes(lang) ? lang : 'zh_CN';
    });

    const initOptions = computed((): RawEditorSettings => {
      const { height, options, toolbar, plugins } = props;
      const publicPath = import.meta.env.VITE_PUBLIC_PATH || '/';

      return {
        selector: `#${unref(tinymceId)}`,
        height,
        // toolbar: appEnv === 'development' ? [...toolbar, 'HtmlBtn'] : toolbar,
        toolbar: !!props.isHide ? false : !!props.isPrint ? false : toolbar,
        menubar: !!props.isHide ? false : !!props.isPrint ? 'print' : 'file edit insert view format table',
        menu: {
          print: {
            title: '打印',
            items: 'print',
          },
        },
        plugins,
        fontsize_formats: '8pt 10pt 12pt 14pt 16pt 18pt 20pt 22pt 24pt 36pt',
        font_formats: `微软雅黑='微软雅黑';宋体='宋体';黑体='黑体';仿宋='仿宋';楷体='楷体';隶书='隶书';幼圆='幼圆';
        Andale Mono=andale mono,times;Arial=arial,helvetica,sans-serif;Arial Black=arial black,avant garde;
        Book Antiqua=book antiqua,palatino;Comic Sans MS=comic sans ms,sans-serif;Courier New=courier new,
        courier;Georgia=georgia,palatino;Helvetica=helvetica;Impact=impact,chicago;Symbol=symbol;Tahoma=tahoma,
        arial,helvetica,sans-serif;Terminal=terminal,monaco;Times New Roman=times new roman,times;Trebuchet MS=trebuchet
        ms,geneva;Verdana=verdana,geneva;Webdings=webdings;Wingdings=wingdings`,
        language_url: publicPath + 'resource/tinymce/langs/' + langName.value + '.js',
        language: langName.value,
        branding: false,
        default_link_target: '_blank',
        link_title: false,
        object_resizing: false,
        auto_focus: true,
        skin: skinName.value,
        skin_url: publicPath + 'resource/tinymce/skins/ui/' + skinName.value,
        content_css: publicPath + 'resource/tinymce/skins/ui/' + skinName.value + '/content.min.css',
        ...options,
        extended_valid_elements: 'a[class|target|href|onclick],div[class|onclick|id|style],link[rel|href]',
        setup: (editor: Editor) => {
          console.log(editor, 'editoreditoreditoreditoreditor');
          editorRef.value = editor;
          editor.on('init', (e) => initSetup(e));
          // 注册一个icon
          editor.ui.registry.addIcon(
            'shopping-cart',
            `<svg viewBox="0 0 1024 1024" data-icon="shopping-cart" width="1.5em" height="1.5em" fill="currentColor" aria-hidden="true" focusable="false" class=""><path d="M922.9 701.9H327.4l29.9-60.9 496.8-.9c16.8 0 31.2-12 34.2-28.6l68.8-385.1c1.8-10.1-.9-20.5-7.5-28.4a34.99 34.99 0 0 0-26.6-12.5l-632-2.1-5.4-25.4c-3.4-16.2-18-28-34.6-28H96.5a35.3 35.3 0 1 0 0 70.6h125.9L246 312.8l58.1 281.3-74.8 122.1a34.96 34.96 0 0 0-3 36.8c6 11.9 18.1 19.4 31.5 19.4h62.8a102.43 102.43 0 0 0-20.6 61.7c0 56.6 46 102.6 102.6 102.6s102.6-46 102.6-102.6c0-22.3-7.4-44-20.6-61.7h161.1a102.43 102.43 0 0 0-20.6 61.7c0 56.6 46 102.6 102.6 102.6s102.6-46 102.6-102.6c0-22.3-7.4-44-20.6-61.7H923c19.4 0 35.3-15.8 35.3-35.3a35.42 35.42 0 0 0-35.4-35.2zM305.7 253l575.8 1.9-56.4 315.8-452.3.8L305.7 253zm96.9 612.7c-17.4 0-31.6-14.2-31.6-31.6 0-17.4 14.2-31.6 31.6-31.6s31.6 14.2 31.6 31.6a31.6 31.6 0 0 1-31.6 31.6zm325.1 0c-17.4 0-31.6-14.2-31.6-31.6 0-17.4 14.2-31.6 31.6-31.6s31.6 14.2 31.6 31.6a31.6 31.6 0 0 1-31.6 31.6z"></path></svg>`,
          );
          // 注册获取html以及数据的按钮
          registerSignBtn(editor);
        },
        // 生命周期:挂载后回调
        init_instance_callback: (editor: Editor) => {
          // 修改编辑器默认字体和字号
          editor.getBody().style.fontSize = '16pt';
          editor.getBody().style.fontFamily = '宋体';
        },
      };
    });

    // 注册获取html以及数据的按钮
    function registerSignBtn(editor: Editor) {
      editor.ui.registry.addButton('CardBtn', {
        type: 'button',
        // icon: `shopping-cart`,
        text: '获取并保存html',
        onAction: function (_) {
          //按钮事件:组装 html + data数据
          getControlValue();
          saveTemplate(editor.getContent(), getControlValue());
        },
      });
    }

    // 获取控件数据值
    function getControlValue() {
      let dom = tinymce.activeEditor.dom;
      let controls = dom.select('.control');
      let data = controls.map((item) => {
        // console.log('item', item);
        let dataControl = JSON.parse(item.getAttribute('data-control'));
        let controlValue = item.getAttribute('data-value');
        //文本框 没有data-value
        console.log(controlValue, item.firstElementChild.innerHTML, 'item.firstElementChild.innerHTML');
        if (!controlValue) {
          if (dataControl.initialData.select == 'input') {
            if (!!item.firstElementChild.innerHTML) {
              controlValue = item.firstElementChild.innerHTML;
            } else {
              controlValue = '';
            }
          }
          // controlValue = item.firstElementChild.innerHTML;
        }
        return {
          controlType: dataControl.initialData.select,
          fieldName: dataControl.initialData.name,
          fieldKey: dataControl.initialData.fieldKey,
          controlValue,
        };
      });
      console.log(data);
      return data;
    }
    // 保存模板
    async function saveTemplate(doc: string, data: any) {
      try {
        const params: any = { doc, data };

        emit('templateparams', params);
      } finally {
      }
    }

    const disabled = computed(() => {
      const { options } = props;
      const getdDisabled = options && Reflect.get(options, 'readonly');
      const editor = unref(editorRef);
      if (editor) {
        editor.setMode(getdDisabled ? 'readonly' : 'design');
      }
      return getdDisabled ?? false;
    });
    watch(
      () => attrs.disabled,
      () => {
        const editor = unref(editorRef);
        if (!editor) {
          return;
        }
        editor.setMode(attrs.disabled ? 'readonly' : 'design');
      },
    );

    onMountedOrActivated(() => {
      if (!initOptions.value.inline) {
        tinymceId.value = buildShortUUID('tiny-vue');
      }
      nextTick(() => {
        setTimeout(() => {
          initEditor();
        }, 30);
      });
    });

    onBeforeUnmount(() => {
      destory();
    });

    onDeactivated(() => {
      destory();
    });

    function destory() {
      if (tinymce !== null) {
        tinymce?.remove?.(unref(initOptions).selector!);
      }
    }
    // 弹框配置
    let Dialog = (editor) => {
      return {
        title: '添加控件', // The dialog's title - displayed in the dialog header
        body: {
          type: 'panel', // The root body type - a Panel or TabPanel
          items: [
            // A list of panel components
            {
              type: 'selectbox',
              name: 'select',
              label: '控件',
              items: [
                { value: 'input', text: '输入框' },
                { value: 'date', text: '日期' },
                { value: 'radio', text: '单选框' },
                { value: 'checkbox', text: '多选框' },
                { value: 'select', text: '下拉' },
                { value: 'sign', text: '签名' },
                { value: 'record', text: '语音转换' },
                // { value: 'textarea', text: 'textarea' },
              ],
            },
            {
              type: 'input', //类型可以是 checkbox, input, selectbox, textarea and urlinput
              name: 'name',
              label: '字段名称',
            },
            {
              type: 'input', //类型可以是 checkbox, input, selectbox, textarea and urlinput
              name: 'fieldKey',
              label: '字段key值',
            },
            {
              type: 'selectbox', //类型可以是 checkbox, input, selectbox, textarea and urlinput
              name: 'isShow',
              label: '字段名称是否展示',
              items: [
                { value: '1', text: '展示' },
                { value: '2', text: '不展示' },
              ],
            },
            // {
            //     type: 'checkbox',
            //     name:'checkbox',
            //     label:'checkbox'
            // },
            // {
            //     type: 'button',
            //     name:'tianjia',
            //     text:'添加子项',
            //     disabled:false
            // },
            // {
            //     type: 'htmlpanel', // A HTML panel component
            //     html: '11'
            // },
          ],
        },
        //初始值
        // initialData: {
        //   name: '2'
        // },
        buttons: [
          // A list of footer buttons
          {
            type: 'cancel',
            name: 'closeButton',
            text: '取消',
          },
          {
            type: 'submit',
            primary: true,
            text: '确认',
          },
        ],
        // radio select checkbox 添加子项
        onAction: dialogFn.onAction(editor),
        // 切换表单控件
        onChange: dialogFn.onChange(editor),
        onSubmit: dialogFn.onSubmit(editor),
      };
    };

    function initEditor() {
      const el = unref(elRef);
      if (el) {
        el.style.visibility = '';
      }
      tinymce
        .init(unref(initOptions))
        .then((editor) => {
          emit('inited', editor);
        })
        .catch((err) => {
          emit('init-error', err);
        });
      tinymce.PluginManager.add('control', function (editor) {
        dialogConfig.value = Dialog(editor);
        const dialogOpener = () => {
          return editor.windowManager.open(dialogConfig.value);
        };
        editor.ui.registry.addButton('control', {
          icon: 'non-breaking', //图标
          tooltip: '插入控件', //提示
          text: '控件库',
          onAction: function () {
            dialogOpener();
          },
        });
        setupButtons(editor);
        if (!props.isHide) {
          addToEditor(editor);
        }
        initBindEvent(editor);
      });
    }
    // 为初始化内容中控件绑定事件 和 控件值回填
    let initBindEvent = (editor) => {
      console.log(props.templatevalue, 'propspropspropsprops');
      let templateValue = toRaw(props.templatevalue);
      editor.on('init', () => {
        //绑定事件
        bindEvent(editor);
        //控件值回填
        let dom = tinymce.activeEditor.dom;
        let controls = dom.select('.control');
        controls.forEach((item) => {
          let dataControl = JSON.parse(item.getAttribute('data-control'));
          let controlValue = item.getAttribute('data-value');
          if (!!templateValue && templateValue.length > 0) {
            templateValue.forEach((i) => {
              if (i.fieldKey == dataControl.initialData.fieldKey) {
                controlValue = i.controlValue;
                switch (dataControl.initialData.select) {
                  case 'input':
                    if (!!controlValue) {
                      console.log(item, item.firstElementChild, 'itemitemitemitemitem');
                      if (!!item.firstElementChild) {
                        item.firstElementChild.innerText = controlValue;
                      }
                      // item.setAttribute('data-value', controlValue);
                    }
                    break;
                  case 'radio':
                    if (!!controlValue) {
                      item.querySelector('[value="' + controlValue + '"]').checked = true;
                    }
                    break;
                  case 'checkbox':
                    let checkboxs = item.querySelectorAll('input');
                    checkboxs.forEach((it) => {
                      if (controlValue.split(',').includes(it.value)) {
                        it.checked = true;
                      }
                    });
                    break;
                  case 'select':
                    item.querySelector('[value=' + controlValue + ']').selected = true;
                    break;
                }
              }
            });
          } else {
            switch (dataControl.initialData.select) {
              case 'radio':
                if (!!controlValue) {
                  item.querySelector('[value="' + controlValue + '"]').checked = true;
                }
                break;
              case 'checkbox':
                let checkboxs = item.querySelectorAll('input');
                checkboxs.forEach((it) => {
                  if (controlValue.split(',').includes(it.value)) {
                    it.checked = true;
                  }
                });
                break;
              case 'select':
                item.querySelector('[value=' + controlValue + ']').selected = true;
                break;
            }
          }
          console.log(controlValue);
        });
      });
    };

    let addToEditor = (editor) => {
      // 添加悬浮 上下文工具栏
      editor.ui.registry.addContextToolbar('editcontrol', {
        //触发条件
        predicate: function (node) {
          console.log(node, props.isHide, 'isHideisHideisHide');
          // alert(node);
          // if (node.className == 'c-sign') {
          //   openSignModal(true, {
          //     type: ActionEnum.ADD,
          //   });
          // }
          return !props.isHide && node.className === 'control';
          // return !props.isHide && node.className === 'control' && node.nodeName.toLowerCase() === 'span';
        },
        items: 'changecontrol removecontrol', //显示的工具列表
        position: 'selection', //工具栏放置位置  selection node line
        // scope: 'node',
      });
    };
    // 弹框中的方法
    let dialogFn = {
      // radio select checkbox 添加子项编辑控件
      onAction: (edntor) => (dialogApi, details) => {
        let data = dialogApi.getData();

        if (details.name == 'add') {
          addChildItem(dialogApi);
        }
        if (details.name == 'del') {
          let items = dialogConfig.value.body.items;
          items[items.length - 1].items.pop();
        }
        dialogConfig.value.initialData = data;
        dialogApi.redial(dialogConfig.value);
      },
      // 控件弹窗选择控件Change事件
      onChange: (editor) => (dialogApi, details) => {
        console.log(dialogConfig.value.body, details, dialogApi);
        if (dialogConfig.value.title == '编辑控件') return;

        let data = dialogApi.getData();
        // dialogConfig.body.items[4].html = formControl[data.select]

        if (data.select == 'input' || data.select == 'date' || data.select == 'textarea') {
          dialogConfig.value.body.items.splice(4);
          // dialogApi.redial(dialogConfig.value);
          // dialogApi.setData(data);
        }
        if (
          data.select != 'input' &&
          data.select != 'date' &&
          data.select != 'textarea' &&
          data.select != 'sign' &&
          data.select != 'record' &&
          !dialogConfig.value.body.items[4]
        ) {
          let btns = JSON.parse(JSON.stringify(childBtn));

          let items = JSON.parse(JSON.stringify(childItem));
          dialogConfig.value.body.items.splice(4, 0, btns, items);
          dialogApi.redial(dialogConfig.value);
          dialogApi.setData(data);
        }
        // dialogApi.redial(dialogConfig.value);
        // if (details.name == 'select') dialogApi.redial(dialogConfig.value); //重新渲染dialog
        // dialogApi.setData(data);
        // dialogApi.focus(); // 聚焦
        console.log('dataTdataTdataT', data);
      },
      // 控件弹窗确认回调事件
      onSubmit: (editor) => (api) => {
        let control = '';
        let data = api.getData();
        let controlName = data.select + getId();
        // 输入框
        if (data.select == 'input') {
          control = `<span contenteditable="true" style="display:inline-block;
        min-width:100px;border-bottom:1px solid black;outline: none;padding: 0" name=${controlName} type="${data.select}"> </span>`;
        } else if (data.select == 'date') {
          // 日期
          control = `<input class="c-form" contenteditable="true" style='border: none;border-bottom: 1px solid' name=${controlName} type="datetime-local" value="${data.name}" />`;
        } else if (data.select == 'textarea') {
          //文本御
          control = `<textarea class="c-form" contenteditable="true" rows="2" cols="30" style="display:inline-block;
        min-width:100px;border-bottom:1px solid;"  name=${controlName}></textarea>`;
        }
        // if(data.select == 'input' || data.select == 'date'){
        //   control = `<input contenteditable="true" name=${controlName} type="${data.select}" value="${data.name}" />`
        // }
        else if (data.select == 'radio' || data.select == 'checkbox') {
          let l = dialogConfig.value.body.items[dialogConfig.value.body.items.length - 1].items.length;
          for (let i = 1; i <= l; i++) {
            control =
              control +
              `<label contenteditable="true"><input class="c-form" name=${controlName} type=${data.select} value=${data['value' + i]} /> ${
                data['label' + i]
              }</label>`;
          }
        } else if (data.select == 'select') {
          // 下拉
          let l = dialogConfig.value.body.items[dialogConfig.value.body.items.length - 1].items.length;
          for (let i = 1; i <= l; i++) {
            control = control + `<option value=${data['value' + i]} label=${data['label' + i]}></option>`;
          }
          control = `<select class="c-form" contenteditable="true" style='border: none;border-bottom: 1px solid;padding: 5px 0px 5px 5px' name=${controlName}>${control}</select>`;
        } else if (data.select === 'sign') {
          control = `<img style="width: 32px;height: 32px;" class='c-sign' src='${sign}'></img>`;
        } else if (data.select === 'record') {
          control = `<img style="width: 32px;height: 32px;" class='c-record' src='${recordSvg}'></img>`;
        }
        // 通用 dom 结构和样式
        control = `${
          !!data.name && data.isShow == '1' && dialogConfig.value.title != '编辑控件' ? `<span>${data.name}</span>:` : ``
        }<span class="control" id="span1"
        style="display:inline-block;margin:0 5px;background-color: #f1f1f1;"
        contenteditable="false" data-control=${JSON.stringify({ body: dialogConfig.value.body, initialData: data })} data-value="">
        ${control}
        <span class="c-menu" style="padding:0 5px;"></span>
        </span></span>`;
        // console.log(editor.selection.getNode())
        if (editor.selection.getContent()) {
          //编辑控件
          // editor.selection.getNode().parentNode.removeChild(editor.selection.getNode());
          editor.selection.setContent(control);
          // editor.insertContent(control)
        } else {
          //添加控件
          editor.insertContent(control);
        }

        bindEvent(editor);
        api.close();
      },
    };
    // 为控件绑定事件
    let bindEvent = (editor) => {
      console.log(navigator.userAgent, 'editoreditoreditor');

      let dom = tinymce.activeEditor.dom;
      // console.log(dom.select('#span1'))
      setTimeout(() => {
        // dom.bind(dom.select('.c-menu'), 'click', (e) => {
        //   // 显示指定的上下文菜单
        //   editor.dispatch('contexttoolbar-show', { toolbarKey: 'editcontrol' });
        //   // 隐藏指定的上下文菜单
        //   editor.dispatch('contexttoolbar-hide', { toolbarKey: 'editcontrol' });

        //   e.stopPropagation();
        // });
        dom.bind(dom.select('.c-form'), 'change', (e) => {
          if (e.target.type == 'date') {
            e.target.parentNode.setAttribute('data-value', e.target.value);
            e.target.setAttribute('value', e.target.value);
          }
          if (e.target.type == 'radio') {
            e.target.parentNode.parentNode.querySelectorAll('label').forEach((item) => {
              item.querySelector('input').removeAttribute('checked');
            });
            e.target.parentNode.parentNode.setAttribute('data-value', e.target.value);
            e.target.setAttribute('checked', 'checked');
          }
          if (e.target.type == 'checkbox') {
            let checkedArr = [];
            let parentSpan = e.target.parentNode.parentNode;
            // ---------------------- 值响应有问题
            parentSpan.querySelectorAll('label').forEach((item) => {
              let checkbox = item.querySelector('input');
              // checkbox.removeAttribute('checked')
              if (checkbox.checked) {
                checkedArr.push(checkbox);
                // checkbox.setAttribute('checked','checked')
              }
            });
            parentSpan.setAttribute('data-value', checkedArr.toString());
          }
          if (e.target.localName == 'select') {
            e.target.querySelectorAll('option').forEach((item) => {
              item.removeAttribute('selected');
              if (item.selected) {
                item.setAttribute('selected', 'selected');
              }
            });
            e.target.parentNode.setAttribute('data-value', e.target.value);
          }
        });

        // 单独为 日期 控件绑定失焦事件,change不好使
        dom.bind(dom.select('.c-form'), 'blur', (e) => {
          if (e.target.type == 'datetime-local') {
            const time = new Date(e.target.value).toLocaleString();
            e.target.parentNode.setAttribute('data-value', time);
            e.target.setAttribute('value', e.target.value);
          }
        });

        //签名点击事件
        dom.bind(dom.select('.c-sign'), `click`, () => {
          currentBookMark.value = editor.selection.getBookmark(); //记录点击的位置,弹框结束后插入签名
          openSignModal(true, {
            type: ActionEnum.ADD,
          });
          // editor.selection.setContent(sign);  //把签名插入编辑器
        });
        // 签名ipad touch事件
        dom.bind(dom.select('.c-sign'), `touchstart`, (e) => {
          console.log(e);
          currentBookMark.value = editor.selection.getBookmark(); //记录点击的位置,弹框结束后插入签名
          openSignModal(true, {
            type: ActionEnum.ADD,
          });
          // editor.selection.setContent(sign);  //把签名插入编辑器
        });

        // 语音转换事件
        dom.bind(dom.select('.c-record'), `click`, () => {
          currentBookMark.value = editor.selection.getBookmark(); //记录点击的位置,弹框结束后插入文字
          openRecord(true, {
            type: ActionEnum.ADD,
          });
          // editor.selection.setContent(sign);  //把签名插入编辑器
        });
        // ipad touch事件
        dom.bind(dom.select('.c-record'), `touchstart`, (e) => {
          console.log(e);
          currentBookMark.value = editor.selection.getBookmark();
          openRecord(true, {
            type: ActionEnum.ADD,
          });
          // editor.selection.setContent(sign);  //把签名插入编辑器
        });
      }, 100);
    };
    // 添加子项 key value 方法
    function addChildItem(dialogApi) {
      console.log(dialogApi, 'dialogApidialogApidialogApidialogApi');
      let childItems = dialogConfig.value.body.items[dialogConfig.value.body.items.length - 1].items;
      childItems.push({
        type: 'grid', // component type
        columns: 2, // number of columns
        items: [
          {
            type: 'input',
            name: 'label' + (childItems.length + 1),
            label: '标签' + (childItems.length + 1),
          },
          {
            type: 'input',
            name: 'value' + (childItems.length + 1),
            label: '值' + (childItems.length + 1),
          },
        ], // array of panel components
      });
    }
    // 添加上下文工具栏
    let setupButtons = (editor) => {
      editor.ui.registry.addButton('changecontrol', {
        icon: 'edit-block',
        tooltip: '编辑控件',
        onAction: () => {
          console.log(editor);
          // editor.windowManager.open(dialogConfig)
          // editor.setContent('1213')
          // editor.selection.select(123)
          // editor.selection.setContent('<span>123</span>')
          // editor.selection.getNode()
          console.log(editor.selection.getNode());
          let data = editor.selection.getNode().getAttribute('data-control');
          // let name = editor.selection.getNode().querySelector('input').getAttribute('name')
          // let val = editor.selection.getNode().querySelectorAll('[name='+name+']').value
          let dataJson = JSON.parse(data);
          dialogConfig.value = Dialog(editor);
          // dialogConfig.value.body = data.body;
          dialogConfig.value.title = '编辑控件';
          dialogConfig.value.initialData = dataJson.initialData;
          if (dataJson.initialData.select == 'radio' || dataJson.initialData.select == 'checkbox') {
            let btns = JSON.parse(JSON.stringify(childBtn));
            var keyArr: string[] = [];
            for (var key in dataJson.initialData) {
              if (key.indexOf('label') > -1) {
                keyArr.push(key);
              }
            }
            console.log(keyArr);
            let obj = {
              type: 'grid', // component type
              columns: 1, // number of columns
              items: [],
            };
            keyArr.forEach((i) => {
              let index = i.indexOf('label');
              let strI = i.substring(5);
              console.log(i, index, strI, 'strIstrI');
              let itemObj = {
                type: 'grid',
                columns: 2,
                items: [
                  {
                    type: 'input',
                    name: i,
                    label: '标签' + strI,
                  },
                  {
                    type: 'input',
                    name: 'value' + strI,
                    label: '值' + strI,
                  },
                ],
              };
              obj.items.push(itemObj);
            });
            let items = JSON.parse(JSON.stringify(obj));
            dialogConfig.value.body.items.splice(4, 0, btns, items);
          }
          console.log(dialogConfig.value, 'dialogConfig.valuedialogConfig.valuedialogConfig.value');
          editor.windowManager.open(dialogConfig.value);
        },
      });
      editor.ui.registry.addButton('removecontrol', {
        icon: 'remove',
        tooltip: '删除控件',
        onAction: (editor) => {
          editor.selection.setContent('');
          editor.dispatch('contexttoolbar-hide', { toolbarKey: 'editcontrol' });
          console.log(arguments);
        },
      });
    };
    // 获取id name
    function getId() {
      return '_' + new Date().getTime();
    }
    function initSetup(e) {
      const editor = unref(editorRef);
      console.log(editor, 'editoreditoreditoreditor');
      if (!editor) {
        return;
      }
      const value = props.modelValue || '';
      editor.setContent(value);

      bindModelHandlers(editor);
      bindHandlers(e, attrs, unref(editorRef));
    }

    function setValue(editor: Recordable, val: string, prevVal?: string) {
      // console.log(editor, val, props.templatevalue, 'valvalvalval1111111111111111111111111');

      // console.log(controls, 'controlscontrolscontrols');
      if (editor && typeof val === 'string' && val !== prevVal && val !== editor.getContent({ format: attrs.outputFormat })) {
        editor.setContent(val);
      }
      if (!!props.getParams) {
        saveTemplate(editor.getContent(), getControlValue());
      }
    }

    function bindModelHandlers(editor: any) {
      const modelEvents = attrs.modelEvents ? attrs.modelEvents : null;
      const normalizedEvents = Array.isArray(modelEvents) ? modelEvents.join(' ') : modelEvents;

      watch(
        () => props.modelValue,
        (val: string, prevVal: string) => {
          console.log(val, '00000000000000000000');
          setValue(editor, val, prevVal);
        },
      );

      watch(
        () => props.value,
        (val, prevVal) => {
          console.log(val, prevVal, 'vvvvvvvvvvvvvvvvvvvv');
          setValue(editor, val, prevVal);
        },
        {
          immediate: true,
        },
      );
      watch(
        () => props.templatevalue,
        (val, prevVal) => {
          console.log(val, prevVal, '1111111111111111');
          // setTemplateData(editor, val);
        },
        {
          immediate: true,
        },
      );

      editor.on(normalizedEvents ? normalizedEvents : 'change keyup undo redo', () => {
        const content = editor.getContent({ format: attrs.outputFormat });
        emit('update:modelValue', content);
        emit('change', content);
      });

      editor.on('FullscreenStateChanged', (e) => {
        fullscreen.value = e.state;
      });
    }
    function setTemplateData(arr) {
      let templateValue = toRaw(arr);
      let dom = tinymce.activeEditor.dom;
      let controls = dom.select('.control');
      controls.forEach((item) => {
        let dataControl = JSON.parse(item.getAttribute('data-control'));
        let controlValue = item.getAttribute('data-value');
        if (!!templateValue && templateValue.length > 0) {
          templateValue.forEach((i) => {
            if (i.fieldKey == dataControl.initialData.fieldKey) {
              controlValue = i.controlValue;
              switch (dataControl.initialData.select) {
                case 'input':
                  if (!!controlValue) {
                    console.log(item, item.firstElementChild, 'itemitemitemitemitem');
                    if (!!item.firstElementChild) {
                      item.firstElementChild.innerText = controlValue;
                    }
                    // item.setAttribute('data-value', controlValue);
                  }
                  break;
                case 'radio':
                  if (!!controlValue) {
                    item.querySelector('[value="' + controlValue + '"]').checked = true;
                  }
                  break;
                case 'checkbox':
                  let checkboxs = item.querySelectorAll('input');
                  checkboxs.forEach((it) => {
                    if (controlValue.split(',').includes(it.value)) {
                      it.checked = true;
                    }
                  });
                  break;
                case 'select':
                  item.querySelector('[value=' + controlValue + ']').selected = true;
                  break;
              }
            }
          });
        } else {
          switch (dataControl.initialData.select) {
            case 'radio':
              if (!!controlValue) {
                item.querySelector('[value="' + controlValue + '"]').checked = true;
              }
              break;
            case 'checkbox':
              let checkboxs = item.querySelectorAll('input');
              checkboxs.forEach((it) => {
                if (controlValue.split(',').includes(it.value)) {
                  it.checked = true;
                }
              });
              break;
            case 'select':
              item.querySelector('[value=' + controlValue + ']').selected = true;
              break;
          }
        }
        console.log(controlValue);
      });
      // console.log(data, '000000000000001111111111111111111');
      // editor.setContent(content);
      // console.log(dom, '99999999999999999');
      console.log(controls[0], '666666666666666666666666');
    }
    function handleImageUploading(name: string) {
      const editor = unref(editorRef);
      if (!editor) {
        return;
      }
      editor.execCommand('mceInsertContent', false, getUploadingImgName(name));
      const content = editor?.getContent() ?? '';
      setValue(editor, content);
    }

    function handleDone(name: string, fileId: string) {
      const editor = unref(editorRef);
      if (!editor) {
        return;
      }
      const content = editor?.getContent() ?? '';

      if (fileId) {
        const api = props.isDef ? asyncFindDefUrlById : asyncFindUrlById;
        api(fileId).then((res) => {
          // bug: 这里返回的图片链接必须是永久有效的,否则会出现图片过期无法访问的情况。 暂时没好的解决方案
          if (res.code === 0) {
            const val = content?.replace(getUploadingImgName(name), `<img data-id="${fileId}" src="${res?.data}"/>`) ?? '';
            setValue(editor, val);
          }
        });
      } else {
        const val = content?.replace(getUploadingImgName(name), `<img data-path="${fileId}" src="${fileId}" alt="上传失败"/>`) ?? '';
        setValue(editor, val);
      }
    }

    function getUploadingImgName(name: string) {
      return `[uploading:${name}]`;
    }

    function clickSetValue(val) {
      const editor = unref(editorRef);
      setValue(editor, val);
    }
    function handleSignature(data, base64Data) {
      editorRef.value?.selection.moveToBookmark(currentBookMark.value); // 把光标位置摆正
      editorRef.value?.focus(); // 聚焦
      editorRef.value.execCommand(
        // 填充!搞定!
        'mceInsertContent',
        false,
        `<img style='height: 80px;width: 250px;' data-id="${data.id}" src="${data}" alt=""/>`,
      );
      // editorRef.value?.selection.setContent(`<img style='height: 80px;width: 250px' data-id="${data.id}" src="${data}" alt=""/>`);
    }
    // 获取语音转换后的文字
    function handleGetText(data?) {
      console.log(data);
    }
    function getSign(data: any) {
      console.log(data);
    }
    function getSubmitData() {
      const editor = unref(editorRef);
      let controlValue = getControlValue();

      let content = editor?.getContent() ?? '';
      let data = {
        doc: content,
        data: controlValue,
      };
      return data;
    }
    function destroyTiny() {
      tinymce.editors[unref(tinymceId)].destroy();
    }
    return {
      prefixCls,
      containerWidth,
      initOptions,
      signatureModal,
      tinymceContent,
      elRef,
      tinymceId,
      handleImageUploading,
      handleDone,
      editorRef,
      fullscreen,
      disabled,
      props,
      setValue,
      clickSetValue,
      handleSignature,
      getSign,
      recorderModal,
      handleGetText,
      getSubmitData,
      setTemplateData,
      destroyTiny,
    };
  },
});
</script>

<style lang="less" scoped></style>

<style lang="less">
@prefix-cls: ~'@{namespace}-tinymce-container';

.@{prefix-cls} {
  position: relative;
  line-height: normal;

  textarea {
    z-index: -1;
    visibility: hidden;
  }
}
</style>

总结

  1. 需要优化模版应用的时候,达到的效果,只输入,不能对其他的模版设置进行修改,目前可以对模版进行增删改。
  2. 模版打印的存在同1的问题
  3. tinymce的中文文档
  4. tinymce的英文Api

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1566013.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

运筹学基础(三):求解整数规划的切平面法(cutting plane method)

文章目录 算法思想一个例子参考文档 算法思想 先将整数规划问题松弛为线性规划问题&#xff0c;然后割掉线性规划问题可行域的一部分&#xff08;只包含非整数解&#xff09;&#xff0c;使得线性规划问题的最优解在原整数规划问题的可行域某顶点上取得。 因此&#xff0c;割平…

Spring之BeanFactoryPostProcessor详解

目录 功能与作用 使用案例 spring提供的常见BeanFactoryPostProcessor 1.EventListenerMethodProcessor 2.BeanDefinitionRegistryPostProcessor 功能与作用 使用案例 spring提供的唯一BeanDefinitionRegistryPostProcessor 总结 功能与作用 参考BeanFactoryPostProce…

如何插入LinK3D、CSF、BALM来直接插入各个SLAM框架中

0. 简介 LinK3D、CSF、BALM这几个都是非常方便去插入到激光SLAM框架的。这里我们会分别从多个角度来介绍如何将每个框架插入到SLAM框架中 1. LinK3D:三维LiDAR点云的线性关键点表示 LinK3D的核心思想和基于我们的LinK3D的两个LiDAR扫描的匹配结果。绿色线是有效匹配。当前关…

C++ 中的 vector 的模拟实现【代码纯享】

文章目录 C 中的 vector 模拟实现1. vector 的基本概念2. vector 的基本操作3. vector 的模拟实现4.代码纯享5. 总结 C 中的 vector 模拟实现 在 C 中&#xff0c;vector 是一个非常重要的容器&#xff0c;它提供了动态数组的功能。在本篇博客中&#xff0c;我们将尝试模拟实现…

搭建电商网站外贸网站用API接口可以实现哪些功能(天猫API接口|京东API接口)

在电商领域&#xff0c;API接口可以实现多种功能&#xff0c;起到连接内外部系统及优化电商业务流程等多种作用&#xff0c;从而来提高电商企业的运营效率。 具体来看&#xff0c;API接口接入可以用来&#xff1a; 商品管理&#xff1a; API接口能够用来获取商品详情等&#…

OR- M406A固态继电器SSR光耦,对标替代TLP170A/ASSR-1218等

低工作电流 低导通电阻 高隔离电压 400V , 600V 输出耐受电压 工业温度范围&#xff1a;-40 to 85℃ 特征 高输入输出隔离电压 &#xff08; Viso 3&#xff0c;750Vrms &#xff09; 采用 400V 和 600V 负载电压系列 常开信号极点信号投射继电器 低工作电流 低…

Redis安装-Docker

安装redis的docker容器 1、创建redis挂载目录 mkdir -p /liuchaoxu/redis/{data,conf}2、复制配置文件 在 /liuchaoxu/redis/conf 目录中创建文件 redis.conf&#xff0c;文件从 redis-6.2.7.tar.gz 中解压获取 修改默认配置(从上至下依次)&#xff1a; #bind 127.0.0.1 …

小明的背包-dp_python

用户登录 动态规划的思想是自底向上&#xff0c;先求局部最优解然后求全局最优解。 dp[i][j]代表的是当前状态物品的数量以及背包的容量。 N, V map(int,input().split()) dp [[0 for _ in range(V1)] for _ in range(N1)]for i in range(1,N1):v, w map(int,input().split(…

CSS样式-字体类型,文本对齐,外观修饰,文本缩进,文本行间距,外部引用css样式

字体类型和字体属性调整 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Css字体类型大小</title&…

浅述安防视频监控平台EasyCVR视频汇聚管理系统运维管理能力

智慧安防监控EasyCVR视频管理平台能在复杂的网络环境中&#xff0c;将前端设备统一集中接入与汇聚管理。国标GB28181协议视频监控/视频汇聚EasyCVR平台可以提供实时远程视频监控、视频录像、录像回放与存储、告警、语音对讲、云台控制、平台级联、磁盘阵列存储、视频集中存储、…

如何保证Redis的缓存和数据库中的数据的一致性?

Redis的缓存如何和数据库中的数据保持一致性&#xff1f; 我们都知道&#xff0c;Redis是一个基于内存的键值存储系统&#xff0c;数据完全存放在内存中&#xff0c;这使得它的读写速度远超传统的硬盘存储数据库。对于高访问频率、低修改率的数据&#xff0c;通过将它们缓存在…

4.6 offset指令,jmp short指令,far,dword ptr各种跳转指令

4.6 offset指令&#xff0c;jmp short指令&#xff0c;far&#xff0c;dword ptr各种跳转指令 可以修改IP&#xff0c;或同时修改CS和IP的指令统称为转移指令。概括的讲&#xff0c;转移指令就是可以控制CPU执行内存中某处代码的指令 1. 转移指令 1.1 8086CPU的转移行为有以…

城市治理/县域治理方案:构建基于AI视频能力的“一张网”,助力新型城市发展

TSINGSEE青犀AI算法中台是一款平台型产品&#xff0c;专注于提供各行业中小场景部署解决方案。平台具备接入广、性能强、支持跨平台、芯片国产化等特点&#xff0c;可提供丰富的视图接入能力和智能分析能力。 平台采用了多项IT高新技术&#xff0c;包括视频编解码技术、嵌入式…

宁盾身份域管与Coremail邮件系统完成兼容互认证,持续深化信创布局

在信创国产化改造的背景下&#xff0c;企业邮箱的替换是许多党政、央国企、金融、制造企业面临的重要任务。为了满足企业对国产邮箱、OA等其他应用、终端实现统一身份认证&#xff0c;宁盾国产化身份域管与 Coremail XT 安全增强电子邮件系统 V5.0、V6.0 完成了产品兼容互认证&…

从redux的基本概念渐进式理解redux/toolkit的用法

概念 Redux toolkit是帮助提高redux开发效率的一个库 React-redux 是将React和Redux toolkit绑定在一起的一个库 action 是一个对象,里面有一个type属性 action creator是一个函数,这个函数可以返回上面的action对象。 reducer 是一个函数,接受两个参数(initilastate, acti…

【Vue3源码学习】— CH2.8 Vue 3 响应式系统小结

Vue 3 响应式系统小结 1.核心概念1.1 Proxy和Reflect1.2 响应式API1.3 依赖收集与更新触发1.4 触发更新&#xff08;Triggering Updates&#xff09;&#xff1a;1.5 副作用函数&#xff08;Effect&#xff09;1.6 计算属性和观察者1.7 EffectScope1.8 性能优化&#xff1a; 2.…

基于深度学习的机场航拍小目标检测系统(网页版+YOLOv8/v7/v6/v5代码+训练数据集)

摘要&#xff1a;在本博客中介绍了基于YOLOv8/v7/v6/v5的机场航拍小目标检测系统。该系统的核心技术是采用YOLOv8&#xff0c;并整合了YOLOv7、YOLOv6、YOLOv5算法&#xff0c;从而进行性能指标的综合对比。我们详细介绍了国内外在机场航拍小目标检测领域的研究现状、数据集处理…

多类别分类器(Machine Learning研习十八)

多类别分类器 二元分类器可以区分两个类别&#xff0c;而多类别分类器&#xff08;也称为多叉分类器&#xff09;可以区分两个以上的类别。 一些 Scikit-Learn 分类器&#xff08;如 LogisticRegression、RandomForestClassifier 和 GaussianNB&#xff09;能够原生处理多个类…

硬件了解 笔记 2

CPU 内存控制器&#xff1a;负责读写数据 代理系统和平台IO&#xff1a;与主板上的芯片组通信&#xff0c;并管理PC中其他组件之间的数据流 主板&#xff1a;巨大的印刷电路板 Chipset&#xff1a;芯片组&#xff0c;位于散热器下方&#xff0c;直接连接到CPU的系统代理部分 …

HTTP/1.1、HTTP/2、HTTP/3 演变(计算机网络)

HTTP/1.1 相比 HTTP/1.0 提高了什么性能&#xff1f; HTTP/1.1 相比 HTTP/1.0 性能上的改进&#xff1a; 使用长连接改善了短连接造成的性能开销。支持管道网络传输&#xff0c;只要第一个请求发出去了&#xff0c;不必等其回来&#xff0c;就可以发第二个请求出去&#xff0c…