fabic.js Quadratic Curve /可控制的曲线

news2024/9/25 21:28:17

需要绘制一条可控制的贝塞尔曲线,发现fabic官网中一个demo有点类似。感兴趣的可以移步官网查看demo。

官网的demo是对于html 而言的,放在vue中需要变换一下,具体代码如下:

<template>
  <div class="dashboard-container" @contextmenu.prevent>
    <canvas id="editorCanvas" ref="canvas" style="margin-top: 10px;" />
  </div>
</template>

<script>
import { mapGetters } from 'vuex'
// import fabric from 'fabric'
let canvas = ''
export default {
  name: 'Dashboard',
  computed: {
    ...mapGetters([
      'name'
    ])
  },
  data () {
    return {
      width: window.innerWidth,
      height: window.innerHeight,
      activeEl: {}// 获取当前点击元素
    }
  },
  mounted () {
    const self = this
    // canvas = new fabric.StaticCanvas('editorCanvas', {//静态画布,不可以修改
    canvas = new fabric.Canvas('editorCanvas', {
      width: self.width,
      height: self.height,

      backgroundColor: '#ffffff'
    })
    var line = new fabric.Path('M 65 0 Q 100, 100, 200, 0', { fill: '', stroke: 'black', objectCaching: false })

    line.path[0][1] = 100
    line.path[0][2] = 100

    line.path[1][1] = 200
    line.path[1][2] = 200

    line.path[1][3] = 300
    line.path[1][4] = 100

    line.selectable = false
    canvas.add(line)

    var p1 = self.makeCurvePoint(200, 200, null, line, null)
    p1.name = 'p1'
    canvas.add(p1)

    var p0 = self.makeCurveCircle(100, 100, line, p1, null)
    p0.name = 'p0'
    canvas.add(p0)

    var p2 = self.makeCurveCircle(300, 100, null, p1, line)
    p2.name = 'p2'
    canvas.add(p2)
    canvas.on('object:selected', function (opt) {
      self.onObjectSelected(opt)
    })
    canvas.on('object:moving', function (opt) {
      self.onObjectMoving(opt)
    })
    canvas.on('selection:cleared', function (opt) {
      self.onSelectionCleared(opt)
      // canvas.on({
      //   'object:selected': this.onObjectSelected(e),
      //   'object:moving': this.onObjectMoving(e),
      //   'selection:cleared': this.onSelectionCleared(e)
    })
  },
  methods: {
    makeCurveCircle (left, top, line1, line2, line3) {
      var c = new fabric.Circle({
        left: left,
        top: top,
        strokeWidth: 5,
        radius: 12,
        fill: '#fff',
        stroke: '#666',
        originX: 'center',
        originY: 'center'
      })

      c.hasBorders = c.hasControls = false

      c.line1 = line1
      c.line2 = line2
      c.line3 = line3

      return c
    },

    makeCurvePoint (left, top, line1, line2, line3) {
      var c = new fabric.Circle({
        left: left,
        top: top,
        strokeWidth: 8,
        originX: 'center',
        originY: 'center',
        radius: 14,
        fill: '#fff',
        stroke: '#666'
      })

      c.hasBorders = c.hasControls = false

      c.line1 = line1
      c.line2 = line2
      c.line3 = line3

      return c
    },
    onObjectSelected (e) {
      // console.log(e.target)
      var activeObject = e.target
      if (activeObject.name == 'p0' || activeObject.name == 'p2') {
        activeObject.line2.animate('opacity', '1', {
          duration: 200,
          onChange: canvas.renderAll.bind(canvas)
        })
        activeObject.line2.selectable = true
      }
    },
    onSelectionCleared (e) {
      var activeObject = e.target
      if (activeObject.name == 'p0' || activeObject.name == 'p2') {
        activeObject.line2.animate('opacity', '0', {
          duration: 200,
          onChange: canvas.renderAll.bind(canvas)
        })
        activeObject.line2.selectable = false
      } else if (activeObject.name == 'p1') {
        activeObject.animate('opacity', '0', {
          duration: 200,
          onChange: canvas.renderAll.bind(canvas)
        })
        activeObject.selectable = false
      }
    },
    onObjectMoving (e) {
      if (e.target.name == 'p0' || e.target.name == 'p2') {
        var p = e.target

        if (p.line1) {
          p.line1.path[0][1] = p.left
          p.line1.path[0][2] = p.top
        } else if (p.line3) {
          p.line3.path[1][3] = p.left
          p.line3.path[1][4] = p.top
        }
      } else if (e.target.name == 'p1') {
        var p = e.target

        if (p.line2) {
          p.line2.path[1][1] = p.left
          p.line2.path[1][2] = p.top
        }
      } else if (e.target.name == 'p0' || e.target.name == 'p2') {
        var p = e.target

        p.line1 && p.line1.set({ 'x2': p.left, 'y2': p.top })
        p.line2 && p.line2.set({ 'x1': p.left, 'y1': p.top })
        p.line3 && p.line3.set({ 'x1': p.left, 'y1': p.top })
        p.line4 && p.line4.set({ 'x1': p.left, 'y1': p.top })
      }
    }

  }
}
</script>

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

效果如图:

核心:定义 line时,设置path数组值。变换时修改path数组中的值。直接=

但是我需要绘制三阶贝塞尔曲线,即要两个控制点。这时候就需要变形一下。

const line = new fabric.Path(`M ${item1.left} ${item1.top} C ${item1.left} ${item2.top}, ${item2.left} ${item1.top}, ${item2.left}, ${item2.top}`, {
......

        /* 起点 */
        line.path[0][1] = item1.left
        line.path[0][2] = item1.top
        /* 控制点1 */
        line.path[1][1] = item1.left
        line.path[1][2] = item2.top
        /* 控制点1 */
        line.path[1][3] = item2.left
        line.path[1][4] = item1.top
        /* 结束点 */
        line.path[1][5] = item2.left
        line.path[1][6] = item2.top

修改直接进行重新赋值。

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

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

相关文章

NuttX实时操作系统介绍(最详细)

比起安卓、iOS、鸿蒙等&#xff0c;NuttX知名度不算高&#xff0c;该系统于2007年由Gregory Nutt先生正式开源。发展至今&#xff0c;NuttX以功能丰富、性能稳定、商业化成熟度高赢得了市场的认可。 NuttX是一个成熟的实时操作系统&#xff0c;于07年由Gregory Nutt先生正式开源…

爬虫 — Js 逆向案例四网易云音乐评论

目标网站&#xff1a;https://music.163.com/#/song?id2054300084 需求&#xff1a;获取评论内容&#xff0c;用户名 案例分析 1、分析网站加载方式 动态加载&#xff0c;抓包 找到目标 url&#xff1a;https://music.163.com/weapi/comment/resource/comments/get?csrf_to…

zookeeper异常 Cannot open channel to 3 at election address

zookeeper异常 Cannot open channel to 3 at election address 今天在测试服务器搭建单机zookeeper集群的时候遇到了下面的问题&#xff0c;花费了很长的时间才解决掉 场景 首先来说一下场景吧&#xff1a;最近生产环境的zookeeper集群想做一个迁移&#xff0c;所以本着万无…

【算法练习Day1】二分查找移除元素

​ ​&#x1f4dd;个人主页&#xff1a;Sherry的成长之路 &#x1f3e0;学习社区&#xff1a;Sherry的成长之路&#xff08;个人社区&#xff09; &#x1f4d6;专栏链接&#xff1a;练题 &#x1f3af;长路漫漫浩浩&#xff0c;万事皆有期待 文章目录 二分查找解决方法一&…

【pytest】 pytest拓展功能 pycharm PermissionError: [Errno 13] Permission denied:

目录 1. pytest-html 1.1 PermissionError: [Errno 13] Permission denied: D:\\software\\python3\\anconda3\\Lib\\site-packages\\pytest_html\\__pycache__\\tmp_ttoasov 1.2错误原因 2. 失败用例重试 3. 用例并行执行 pytest-parallel 1. pytest-html 管理员打开 A…

Java入门级基础教学(史上最详细的整合)

目录 一&#xff1a;基础语法 1.“Hello word” 2.Java的运行机制 3. Java基本语法 1.注释、标识符、关键字 2.数据类型&#xff08;四类八种&#xff09; 4.类型转换 1.自动转换 2.强制转换 5.常量和变量 1.常量 2.变量 3.变量的作用域 6.运算符 1.算数运算符 …

从零开始搭建成绩查询系统

在当前的数字化时代&#xff0c;教育行业借助技术手段不断推动教学效果的提升。作为教师&#xff0c;搭建一个专属的成绩查询系统可以更好地管理学生成绩&#xff0c;并即时向家长反馈。本文将详细介绍如何从零开始搭建一个成绩查询系统&#xff0c;以提升教学管理的便捷性和效…

Qt(day4)

思维导图 QT连接TCP网络通信 服务器 ser.h #ifndef SER_H #define SER_H#include <QWidget> #include<QTcpServer> //服务器头文件 #include<QTcpSocket> //客户端头文件 #include<QMessageBox> //消息对话框 #include<QList> …

已解决 ValueError: Found array with dim 3. Estimator expected <= 2.

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页: &#x1f405;&#x1f43e;猫头虎的博客&#x1f390;《面试题大全专栏》 &#x1f995; 文章图文并茂&#x1f996…

“新”心相印 | 长沙市网络代表人士培训班“破冰”联谊“湘”味十足

搜狐网湖南&#xff08;文/莫谦&#xff09;9月18日至9月22日&#xff0c;首期长沙市网络代表人士专题培训班在北京大学举行&#xff0c;培训班学员主要是长沙网络名人联盟成员&#xff0c;涵盖抖音达人、微博博主、网络作家、网络大V等。 为了加强长沙网络名人联盟组织建设&a…

exec函数族

1.execl函数 #include <unistd.h> int execl(const char *path, const char *arg, ...); 参数&#xff1a; -path:需要指定的执行文件的路径或者名称 -*arg:是可执行文件所需要的参数列表 第一个参数一般没有什么作用&#xff0c;一般写的是执行的程序的名称 从第二个参…

鉴源实验室 | 系统逻辑漏洞挖掘实践

作者 | 林海文 上海控安可信软件创新研究院汽车网络安全组 来源 | 鉴源实验室 社群 | 添加微信号“TICPShanghai”加入“上海控安51fusa安全社区” 01 前 言 当谈及安全测试时&#xff0c;逻辑漏洞挖掘一直是一个备受关注的话题&#xff0c;它与传统的安全漏洞&#xff08;如…

SSM整合(细节拉满)|将Mybatis、Spring、SpringMVC三个框架整合起来,通过一个demo来练习

环境要求 环境: IDEA MySQL 5.7.19 Tomcat 9 Maven 3.6 要求: 需要熟练掌握MySQL数据库&#xff0c;Spring&#xff0c;JavaWeb及MyBatis知识&#xff0c;简单的前端知识; 数据库环境 创建一个存放书籍数据的数据库表&#xff0c;并插入一些示例数据 SSM整合 新建一个空的M…

网安须知|什么是护网行动?什么是红蓝对抗?

01 什么是护网行动 护网行动是以公安部牵头的&#xff0c;用以评估企事业单位的网络安全的活动。 具体实践中&#xff0c;公安部会组织攻防两方&#xff0c;进攻方会在一个月内对防守方发动网络攻击&#xff0c;检测出防守方&#xff08;企事业单位&#xff09;存在的安全漏洞…

深入学习 Redis - 分布式锁底层实现原理,以及实际应用

目录 一、Redis 分布式锁 1.1、什么是分布式锁 1.2、分布式锁的基础实现 1.2.1、引入场景 1.2.2、基础实现思想 1.2.3、引入 setnx 1.3、引入过期时间 1.4、引入校验 id 1.5、引入 lua 脚本 1.5.1、引入 lua 脚本的原因 1.5.2、lua 脚本介绍 1.6、过期时间续约问题&…

OpenMMLab MMYOLO目标检测应用示例与常见问题(三)

基于MMYOLO的电离图实时目标检测基准 数据集 数字电离图是获取实时电离层信息的最重要方式。电离层结构检测对于准确提取电离层关键参数具有重要的研究意义。 本研究利用中国科学院在海南、武汉和怀来获得的4311张不同季节的电离图建立数据集。使用labelme手动注释包括 Laye…

实现高并发内存池(C++)

什么是内存池 池化技术 所谓“池化技术”&#xff0c;就是程序先向系统申请过量的资源&#xff0c;然后自己管理以备不时之需。之所以要申请过量的资源&#xff0c;是因为每次申请该资源都有较大的开销&#xff0c;不如提前申请好&#xff0c;这样使用时就会变得非常快捷&…

2023年华数杯数学建模C题母亲身心健康对婴儿成长的影响解题全过程文档及程序

2023年华数杯全国大学生数学建模 C题 母亲身心健康对婴儿成长的影响 原题再现&#xff1a; 母亲是婴儿生命中最重要的人之一&#xff0c;她不仅为婴儿提供营养物质和身体保护&#xff0c;还为婴儿提供情感支持和安全感。母亲心理健康状态的不良状况&#xff0c;如抑郁、焦虑、…

同为科技(TOWE)专业防雷滤波桌面PDU超级插座

2023年第19届杭州亚运会开幕在即&#xff0c;相较于以往&#xff0c;本届杭州亚运会的一大看点就是电竞项目将首次以正式比赛项目的身份亮相亚运赛场&#xff0c;让更多受众能够领略电竞的魅力。当前社会&#xff0c;电竞作为一种新兴的娱乐、社交方式&#xff0c;让很多年轻人…

Feign实战-Springboot集成OpenFeign Demo以及参数详解

最近整理一下微服务的文章&#xff0c;先拿一直用的OpenFeign开刀 思考&#xff1a;微服务之间如何方便优雅的实现服务间的远程调用 一、说说openFeign是什么吧&#xff1f; 说到这个&#xff0c;那不得不先说说RPC 1.什么是RPC RPC 全称是 Remote Procedure Call &#x…