使用canvas制作“学写一个字”

image.png

主要就是这样一个程序,选择一种颜色,然后可以用鼠标或者手指在米字格中绘制想要的文字,点击“清除”按钮可以清除米字格中的内容。笔画的粗细与书写的速度成反比,滑动速度越快,则笔画效果越细。
不多解释,直接贴代码:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8"
                content="width=device-width,
                                height=device-height,
                                initial-scale=1.0,
                                minimum-scale=1.0,
                                maximum-scale=1.0,
                                user-scalable=no
                            ">
    <title>学写一个字</title>
    <script src="https://cdn.bootcss.com/jquery/3.3.1/jquery.min.js"></script>
    <style>
        #canvas{
            display:block;
            margin:0 auto;
            border: 1px solid #ccc;
        }
        #controller{
            margin:0 auto;
        }
        .op_btn{
            float: right;
            margin:10px 0 0 10px;
            border:2px solid #aaa;
            width:80px;
            height:40px;
            line-height:40px;
            font-size:20px;
            text-align:center;
            border-radius: 5px 5px;
            cursor:pointer;
            background-color: white;
            font-weight:bold;
            font-family: Microsoft Yahei, Arial;
        }
        .op_btn:hover{
            background-color:#def;
        }
        .clearfix{
            clear:both;
        }

        .color_btn{
            float: left;
            margin: 10px 10px 0 0;
            border:5px solid white;
            width:40px;
            height:40px;
            border-radius: 5px 5px;
            cursor:pointer;
        }
        .color_btn:hover{
            border: 5px solid violet;
        }
        .color_btn_selected{
            border: 5px solid blueviolet;
        }
        #black_btn{
            background-color: black;
        }
        #blue_btn{
            background-color: blue;
        }
        #green_btn{
            background-color: green;
        }
        #red_btn{
            background-color: red;
        }
        #orange_btn{
            background-color: orange;
        }
        #yellow_btn{
            background-color: yellow;
        }
    </style>
</head>
<body>
    <canvas id="canvas">您的浏览器不支持canvas</canvas>
    <div id="controller">
        <div id="black_btn" class="color_btn color_btn_selected"></div>
      <div id="blue_btn" class="color_btn"></div>
      <div id="green_btn" class="color_btn"></div>
      <div id="red_btn" class="color_btn"></div>
      <div id="orange_btn" class="color_btn"></div>
      <div id="yellow_btn" class="color_btn"></div>
        <div id="clear_btn" class="op_btn">清除</div>
        <div class="clearfix"></div>
    </div>
    <script type="text/javascript">
        var canvasWidth=Math.min(800,$(window).width()-20) 
        var canvasHeight=canvasWidth
        var canvas=document.getElementById("canvas")
        var context=canvas.getContext("2d")

        canvas.width=canvasWidth;
        canvas.height=canvasHeight;

        var isMouseDown=false;
        var lastLoc={
            x:0,
            y:0
        }
        var lastTimestamp=0
        var lastLineWidth=-1

        var strokeColor="black"

        function drawGrid(){

        context.save()

        context.strokeStyle = "rgb(230,11,9)"

        context.beginPath()
        context.moveTo( 3 , 3 )
        context.lineTo( canvasWidth - 3 , 3 )
        context.lineTo( canvasWidth - 3 , canvasHeight - 3 )
        context.lineTo( 3 , canvasHeight - 3 )
        context.closePath()
        context.lineWidth = 6
        context.stroke()

        context.beginPath()
        context.moveTo(0,0)
        context.lineTo(canvasWidth,canvasHeight)

        context.moveTo(canvasWidth,0)
        context.lineTo(0,canvasHeight)

        context.moveTo(canvasWidth/2,0)
        context.lineTo(canvasWidth/2,canvasHeight)

        context.moveTo(0,canvasHeight/2)
        context.lineTo(canvasWidth,canvasHeight/2)

        context.lineWidth = 1
        context.stroke()

        context.restore()
        }
        //绘制米字格
        drawGrid()

        $("#controller").css("width",canvasWidth+"px")

        function beginStroke(obj){
            isMouseDown=true
        lastLoc=windowToCanvas(obj.x,obj.y)
        lastTimestamp=new Date().getTime()
        }

        function endStroke(){
            isMouseDown=false
        }

        function moveStroke(point){
            var curLoc=windowToCanvas(point.x,point.y)
      var curTimestamp=new Date().getTime()
      var s=calcDistance(curLoc,lastLoc)
      var t=curTimestamp-lastTimestamp
      var lineWidth=calcLineWidth(t,s)

      context.beginPath();
      context.moveTo(lastLoc.x,lastLoc.y)
      context.lineTo(curLoc.x,curLoc.y)
      context.lineWidth=lineWidth;
      context.lineCap="round"
      context.lineJoin="round"
      context.strokeStyle=strokeColor
      context.stroke()

      lastLoc=curLoc
      lastTimestamp=curTimestamp
      lastLineWidth=lineWidth
        }

        // 监听鼠标事件
        canvas.onmousedown = function(e){
        e.preventDefault()
        beginStroke( {x: e.clientX , y: e.clientY} )
        };
        canvas.onmouseup = function(e){
        e.preventDefault()
        endStroke()
        };
        canvas.onmouseout = function(e){
        e.preventDefault()
        endStroke()
        };
        canvas.onmousemove = function(e){
        e.preventDefault()
        if( isMouseDown ){
            moveStroke({x: e.clientX , y: e.clientY})
        }
        };


        //触控事件:touchstart,touchmove,touchend
        canvas.addEventListener('touchstart',function(e){
            e.preventDefault()
            touch=e.touches[0]
            beginStroke( {x: touch.pageX , y: touch.pageY} )
        })

        canvas.addEventListener('touchmove',function(e){
            e.preventDefault()
            if( isMouseDown ){
                touch=e.touches[0]
        moveStroke({x: touch.pageX , y: touch.pageY})
        }
        })

        canvas.addEventListener('touchend',function(e){
            e.preventDefault()
            endStroke()
        })


        //坐标系的转换----将鼠标事件的坐标系换成canvas的坐标系
        function windowToCanvas(x,y){
            var bbox=canvas.getBoundingClientRect()
            return {
                x:Math.round(x-bbox.left),
                y:Math.round(y-bbox.top)
            }
        }

        //根据位置信息计算距离
        function calcDistance(loc1,loc2){
            return Math.sqrt((loc1.x-loc2.x)*(loc1.x-loc2.x)+(loc1.y-loc2.y)*(loc1.y-loc2.y))
        }

        var maxLineWidth=30;
        var minLineWidth=1;
        var maxStrokeV=10;
        var minStrokeV=0.1;
        function calcLineWidth(t,s){
            var v=s/t

            var resultLineWidth

            if (v<minStrokeV) {
                resultLineWidth=maxLineWidth
            }else if(v>maxStrokeV){
                resultLineWidth=minLineWidth
            }else{
                resultLineWidth=maxLineWidth-(v-minStrokeV)/(maxStrokeV-minStrokeV)*(maxLineWidth-minLineWidth)
            }
            if(lastLineWidth==-1){
                return resultLineWidth
            }
            return lastLineWidth*2/3+resultLineWidth/3
        }


        $("#clear_btn").click(function(e){
            context.clearRect(0,0,canvasWidth,canvasHeight)
            drawGrid()
        })

        $(".color_btn").click(function(e){
            $(".color_btn").removeClass("color_btn_selected")
            $(this).addClass("color_btn_selected")
            strokeColor=$(this).css("background-color")
        })

    </script>
</body>
</html>

把这个代码保存成一个index.html文件,然后在浏览器中打开即可。

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,035评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,089评论 4 62
  • 尽管从AD市坐了4个小时的火车到了CC市,顺利的从CC火车站走到了西三条街449号天翼宾馆。这个地方是明天培训家庭...
    脑不二阅读 164评论 0 0
  • 1.你记得也好,最好你忘掉。 是啊,你忘掉吧,我记得就好。 2.爱对了人固然是运气,若是爱错了,那也叫青春。 3....
    北默萧丶阅读 342评论 0 0
  • 我讨厌所有隐瞒的爱情!不管善意还是恶意!你记住了!
    木安Siriusea阅读 144评论 0 0