20181105 贝塞尔曲线绘制

看到一个数学公式,坐在那里干干看着,一直也不明白,有什么意思,所以就画一下试试。♪(*)

二次贝塞尔曲线

    // 公式 B(t)=(1-t)^2*P0 + 2*t*(1-t)P1 + t^2*P2; t∈[0,1]
    // P0:startPoint, P1:controlPoint, P2:endPoint
    let factory = new CanvasCtx('#canvas2');
    const {
      ctx, width, height
    } = factory;

    let quadraticBezierCreator = (p0, c1, p1) => {
      return (t = 1) => (1 - t) * (1 - t) * p0 + 2 * t * (1 - t) * c1 + t * t * p1
    };

    let p0, c1, p1;
    ctx.lineWidth = 1;
    ctx.strokeStyle = '#000';
    ctx.fillStyle = '#000';

    let t = 0, x, y;

    function render() {

      ctx.save();
      ctx.beginPath();
      ctx.fillStyle = '#0f0';
      ctx.arc(...p0, 5, 0, Math.PI * 2);
      ctx.fill();
      ctx.beginPath();
      ctx.arc(...p1, 5, 0, Math.PI * 2);
      ctx.fill();
      ctx.restore();

      ctx.save();
      ctx.beginPath();
      ctx.fillStyle = '#f00';
      ctx.arc(...c1, 5, 0, Math.PI * 2);
      ctx.fill();
      ctx.restore();

      ctx.save();
      ctx.fillStyle = '#000';
      ctx.beginPath();
      ctx.arc(x(t), y(t), 2, 0, Math.PI * 2);
      ctx.fill();
      ctx.restore();

      t += 0.01;
      if (t <= 1) {
        requestAnimationFrame(render)
      } else {
        t = 0;

        setTimeout(function () {
          ctx.save();
          ctx.strokeStyle = '#f00';
          ctx.lineWidth = 5;
          ctx.beginPath();
          ctx.moveTo(...p0);
          ctx.quadraticCurveTo(...c1, ...p1);
          ctx.stroke();
          ctx.restore();
        }, 100);
      }
    }


    select('#render1').onsubmit = e => {
      e.preventDefault();

      p0 = select('#sp1').value.split(',').map(d => parseFloat(d));
      c1 = select('#cp1').value.split(',').map(d => parseFloat(d));
      p1 = select('#ep1').value.split(',').map(d => parseFloat(d));

      x = quadraticBezierCreator(p0[0], c1[0], p1[0]);
      y = quadraticBezierCreator(p0[1], c1[1], p1[1]);

      ctx.fillStyle = 'rgba(255,255,255,0.3)';
      ctx.clearRect(0, 0, width, height);

      render();
    }
二次贝塞尔.gif
二次贝塞尔曲线_20181105185358.png

三次贝塞尔曲线

    // B(t) = P0*(1-t)^3 + 3*P1*t*(1-t)^2 + 3*P2*t^2*(1-t) + P3*t^3; t∈[0,1]
    // P0:startPoint, P1:controlPoint1, P2:controlPoint2, P3:endPoint
    const pow = Math.pow;
    const bezierCurveCreator = (p0, p1, p2, p3) => t => {
      return p0 * pow((1 - t), 3) + 3 * p1 * t * pow((1 - t), 2) + 3 * p2 * t * t * (1 - t) + p3 * pow(t, 3)
    };

    const {ctx, width, height} = new CanvasCtx('#canvas3');

    let p0, p1, cp1, cp2, x, y, t = 0;
    ctx.fillStyle = '#000';

    function render2() {

      // start end
      ctx.save();
      ctx.beginPath();
      ctx.fillStyle = '#0f0';
      ctx.arc(...p0, 5, 0, Math.PI * 2);
      ctx.arc(...p1, 5, 0, Math.PI * 2);
      ctx.fill();
      ctx.restore();

      // control
      ctx.save();
      ctx.beginPath();
      ctx.fillStyle = '#f00';
      ctx.arc(...cp1, 5, 0, Math.PI * 2);
      ctx.arc(...cp2, 5, 0, Math.PI * 2);
      ctx.fill();
      ctx.restore();

      ctx.save();
      ctx.beginPath();
      ctx.fillStyle = '#000';
      ctx.arc(x(t), y(t), 2, 0, Math.PI * 2);
      ctx.fill();
      ctx.restore();
      t += 0.01;
      if (t <= 1) {
        requestAnimationFrame(render2);
      } else {
        t = 0;
        setTimeout(function () {
          ctx.save();
          ctx.strokeStyle = '#f00';
          ctx.lineWidth = 5;
          ctx.beginPath();
          ctx.moveTo(...p0);
          ctx.bezierCurveTo(...cp1, ...cp2, ...p1);
          ctx.stroke();
          ctx.restore();
        }, 100);
      }

    }

    function getValues(selector) {
      return select(selector).value.split(',').map(d => parseFloat(d))
    }

    select('#render2').onsubmit = e => {
      e.preventDefault();
      p0 = getValues('#sp2');
      p1 = getValues('#ep2');
      cp1 = getValues('#cp2-0');
      cp2 = getValues('#cp2-1');

      x = bezierCurveCreator(p0[0], cp1[0], cp2[0], p1[0]);
      y = bezierCurveCreator(p0[1], cp1[1], cp2[1], p1[1]);

      ctx.clearRect(0, 0, width, height);
      render2();
    }
三次贝塞尔.gif
三次贝塞尔曲线_20181105185730.png

代码终究是一个实践的艺术,所以还是得动起来。 O(∩_∩)O哈哈~
白日欺人,难逃清夜之鬼报;红颜失志,空贻皓首之悲伤。 ------ 《菜根谭》

====
https://www.cnblogs.com/hyb1/p/3875468.html 动画图片来源
====

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,233评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,013评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,030评论 0 241
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,827评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,221评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,542评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,814评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,513评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,225评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,497评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,998评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,342评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,986评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,055评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,812评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,560评论 2 271
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,461评论 2 266

推荐阅读更多精彩内容