Vue 3 打印模板设计器 (print-canvas-designer)

我用 Vue3 做了一个可扩展的打印模板设计器

在业务系统里,标签打印存在一个常见弊端:它往往不是一次性的输出功能,而是一套需要持续维护的编辑能力。

例如:鞋盒标可能需要展示款号、颜色、尺码、品牌图片、条形码和二维码;不同客户的布局不同,同一个客户后续也可能调整模板。仅靠开发人员在页面里写死位置,不但迭代慢,而且每一次排版变化都要重新发版。

我希望解决的是这样一个问题:

让业务人员通过拖拽搭建打印模板,同时让开发人员可以控制数据、组件和接入方式。

基于这个目标,我做了一个 Vue 3 打印模板画布 :print-canvas-designer,并另外制作了一个真实接入示例项目:print-canvas-examples演示参考文档

它解决什么问题

print-canvas-designer 主要面向标签、鞋盒标、物流面单、商品贴纸等需要自由排版的打印场景。

当前版本支持:

  • 文本、图片、矩形、横线、二维码、条形码等基础组件。
  • A4、100 x 60 和自定义纸张。
  • 元素选中、拖拽、缩放、旋转、复制、删除、锁定、隐藏和层级调整。
  • 标尺、网格、参考线、页边距、安全区和缩放。
  • 文本字段渲染和换行策略。
  • 图片地址设置,以及通过业务上传方法写入图片地址。
  • 打印和导出 PDF。
  • 自定义业务组件及对应的属性编辑区域。
  • 完整编辑器接入,或只接入画布、自行搭建外围 UI。

注:打印设计器中最重要的不是固定的一套左侧面板或右侧表单,而是画布本身。不同业务对组件和属性的要求并不相同,因此画布提供基础编辑能力,业务决定需要出现什么组件。

三种接入方式

为了演示 npm 包在真实 Vue3 项目中的接入方式,我创建了 print-canvas-examples。其中包括三类场景:

  1. 完整编辑器:直接使用默认工具栏、组件面板、画布与属性面板。
  2. 自定义业务组件:将鞋盒标信息块作为业务组件注册到画布中。
  3. 只接入画布:左侧组件区、顶部工具栏、右侧属性区都由业务项目自己实现。

在线示例(点我

最快接入:使用完整编辑器

安装依赖:

bash 复制代码
npm install print-canvas-designer

在入口文件引入样式:

ts 复制代码
import 'print-canvas-designer/style.css'

页面中使用完整编辑器:

vue 复制代码
<template>
  <PrintDesigner
    v-model="document"
    :data="printData"
    :upload-image="uploadImage"
    @save="handleSave"
    @change="handleChange"
  />
</template>

<script setup lang="ts">
import { ref } from 'vue'
import {
  PrintDesigner,
  createDefaultDocument,
  type PrintDocument
} from 'print-canvas-designer'
import 'print-canvas-designer/style.css'

const document = ref<PrintDocument>(createDefaultDocument())

const printData = {
  styleColorSize: 'RUNNER-01 / BLACK / 42',
  barcode: '6901234567890'
}

const uploadImage = async (file: File) => {
  const form = new FormData()
  form.append('file', file)
  const response = await fetch('/api/upload', {
    method: 'POST',
    body: form
  })
  const result = await response.json()
  return result.url
}

const handleSave = (value: PrintDocument) => {
  // 将模板 JSON 保存到业务服务端
  console.log('save document', value)
}

const handleChange = (value: PrintDocument) => {
  console.log('document changed', value)
}
</script>

v-model 对应的是模板数据。业务系统可以把这份 JSON 保存到数据库,之后重新传给组件即可回显模板。

为什么支持只接入画布

完整编辑器适合快速开始,但在真实项目里,已有系统通常有自己的页面结构和交互方式:

  • 左侧可能不是基础组件列表,而是"商品字段""订单字段""客户 Logo"等业务物料。
  • 右侧可能需要结合权限、表单校验、字段绑定和业务规则。
  • 顶部操作区可能需要和系统已有的保存、审核、发布流程结合。

这时可以只接入 PrintCanvas,由业务自行组织页面。

vue 复制代码
<template>
  <div class="designer-page">
    <aside>
      <button @click="addText">添加文本</button>
      <button @click="addBarcode">添加条形码</button>
    </aside>

    <PrintCanvas :designer="designer" :data="printData" />

    <aside>
      <!-- 根据 designer.activeElement.value 渲染自己的属性表单 -->
    </aside>
  </div>
</template>

<script setup lang="ts">
import {
  PrintCanvas,
  createDefaultDocument,
  createPrintDesigner,
  providePrintDesigner
} from 'print-canvas-designer'
import 'print-canvas-designer/style.css'

const printData = {
  styleColorSize: 'RUNNER-01 / BLACK / 42'
}

const designer = createPrintDesigner({
  modelValue: createDefaultDocument(),
  data: printData,
  onChange(value) {
    console.log('template changed', value)
  },
  onSave(value) {
    console.log('save template', value)
  }
})

providePrintDesigner(designer)

const addText = () => {
  designer.addElement('text', { x: 24, y: 24 })
}

const addBarcode = () => {
  designer.addElement('barcode', { x: 24, y: 80 })
}
</script>

面板和画布之间通过同一个 designer 通信。比如选中元素后,业务属性表单可以调用:

ts 复制代码
designer.updateElement(activeId, { field: 'styleColorSize' })
designer.updateElementStyle(activeId, { width: 200, height: 42 })
designer.removeElement(activeId)
designer.undo()
designer.redo()
designer.save()

这样,画布负责编辑交互,业务系统负责 UI 和数据规则。

自定义组件:以鞋盒标信息块为例

基础文本组件已经可以通过字段渲染业务内容,但一些重复出现、结构固定的区域,更适合封装成业务组件。

例如鞋盒标中的商品信息区域,可能固定包含:

  • 标题,例如 SPORT SERIES
  • 主内容,例如 RUNNER-01 / BLACK / 42
  • 副内容,例如 STYLE / COLOR / SIZE
  • 强调颜色或品牌样式。

业务可以定义一个组件,在画布中负责展示结构,同时为它提供自己的属性编辑 UI。组件内容、业务字段和表单交互由业务实现,画布仍然提供选中、移动、缩放、旋转、删除和保存能力。

ts 复制代码
import {
  defaultPrintComponents,
  type PrintComponentDefinition
} from 'print-canvas-designer'
import ShoeInfoBlockRender from './ShoeInfoBlockRender.vue'
import ShoeInfoBlockInspector from './ShoeInfoBlockInspector.vue'

const shoeInfoBlock: PrintComponentDefinition = {
  type: 'shoe-info-block',
  label: '鞋盒标信息块',
  icon: 'i-lucide-tag',
  render: ShoeInfoBlockRender,
  inspector: ShoeInfoBlockInspector,
  createElement: (point) => ({
    id: `shoe_${Date.now()}`,
    type: 'shoe-info-block',
    name: '鞋盒标信息块',
    props: {
      title: 'SPORT SERIES',
      mainText: 'RUNNER-01 / BLACK / 42',
      subText: 'STYLE / COLOR / SIZE',
      accentColor: '#2563eb'
    },
    style: {
      position: 'absolute',
      left: point.x,
      top: point.y,
      width: 260,
      height: 92,
      rotate: 0
    }
  })
}

export const components = [
  ...defaultPrintComponents,
  shoeInfoBlock
]

将它传给完整编辑器即可出现在物料列表和画布中:

vue 复制代码
<PrintDesigner
  v-model="document"
  :components="components"
/>

在只接入画布的模式下,也可以把同样的 components 传给 createPrintDesigner。因此自定义组件不是固定编辑器才有的能力,而是画布 SDK 提供给业务的扩展机制。

模板数据与业务数据如何结合

模板本身保存布局与元素配置,实际打印数据在运行时传入。

以文本为例,模板中可以保存字段名:

json 复制代码
{
  "type": "text",
  "name": "款色码",
  "field": "styleColorSize",
  "style": {
    "left": 24,
    "top": 32,
    "width": 220,
    "height": 42
  }
}

业务在打印前把多个字段拼接为需要展示的内容:

ts 复制代码
const printData = {
  styleColorSize: [product.style, product.color, product.size].join(' / ')
}

这样可以让画布继续保持通用,不必为每一种业务字段组合设计专门的布局规则。

图片、打印与导出

图片组件既可以直接填写图片地址,也可以将上传过程交给业务系统:

ts 复制代码
const uploadImage = async (file: File) => {
  const url = await uploadToObjectStorage(file)
  return url
}

输出方面,完整编辑器提供打印和导出 PDF 的交互。只接入画布时,也可以通过 designer.print()designer.exportPdf() 接入自己的操作入口和输出流程。

当前阶段与后续计划

当前版本主要聚焦于打印模板画布的核心能力,以及业务扩展所需要的组件机制。它已经可以用于搭建标签类模板并验证实际接入方式。

后续我计划继续完善:

  • 更丰富的自定义业务组件示例。
  • 模板管理、保存与复用场景的参考实现。
  • 打印与导出流程在真实业务中的接入示例。

相关地址

相关推荐
名字都不重要何况昵称5 小时前
canvas 分层渲染思路和脏矩形处理
前端·canvas
布列瑟农的星空5 小时前
前端是否需要架构
前端
子云zy5 小时前
JS 对象与包装类:new 做了什么?字符串为什么有 length?
前端·javascript
还有多久拿退休金5 小时前
LLM应用开发二:让AI学会"翻书"——RAG检索增强从踩坑到跑通
前端·llm
weiggle5 小时前
第二篇:搭建你的第一个 Compose 项目——开发环境与项目结构
android·前端
Simon523146 小时前
Spring AOP 五大通知类型
java·前端·spring
Asmewill6 小时前
LangGraph学习笔记八(SubGraph)
前端
叶落阁主6 小时前
AntV npm 投毒复盘:一次公司私服缓存恶意包引发的账号封禁事件
前端·安全·npm
vaexu6 小时前
Android 定时提醒的终极防线:我是如何用“双保险机制”攻克后台保活的?
前端