canvas 旋转图片并展示图片中的小部分

canvas 旋转图片并展示图片中的小部分

月光魔力鸭

2019-09-23 11:08 阅读 889 喜欢 0 canvas 图片旋转

需求如下:有一张大图,需要显示大图中的一小部分,目前能知道的时候小图的宽高和坐标,同时大图有一个旋转角度可以知道,目标就是把小图正确的显示出来。

实际上,如果只针对大图的显示和旋转,可以通过css就能搞定,但是这里面有个问题,旋转后实际上他原来的空间不变,会导致并不美观,最终还是决定通过canvas来实现。

大图的旋转显示

image.onload = function(){
    var img = this;
    var width = img.width;
    var height = img.height;
    
    //针对角度来算
    var canvas = document.createElement('canvas');
    var ctx = canvas.getContext('2d');
    var direction = ((parseInt(rotate,10) + 360) % 360) / 90;
    switch (direction) {
      case 1: {
        canvas.width = height;
        canvas.height = width;
        ctx.rotate(90*Math.PI/180);
        ctx.drawImage(img, 0, -height,width,height);
        break;
      }
      case 2: {
        canvas.width = width;
        canvas.height = height;
        ctx.rotate(180*Math.PI/180);
        ctx.drawImage(img, -width, -height,width,height);
        break;
      }
      case 3: {
        canvas.width = height;
        canvas.height = width;
        ctx.rotate(270*Math.PI/180);
        ctx.drawImage(img, -width, 0,width,height);
        break;
      }
      case 0:
      case 4:
      default: {
        canvas.width = width;
        canvas.height = height;
        ctx.rotate(0);
        ctx.drawImage(img, 0, 0,width,height);
        break;
      }
    }
    $item.get(0).appendChild(canvas);
}

大图旋转和小图展示旋转其实是一样的,只是在canvas drawImage 的时候多几个参数而已。

小图展示旋转

image.onload = (function(direction,filePath,div,showwidth,showheight,sshowwidth,sshowheight,x,y,imgIndex){
    return function(){
    var img = this;
    var thiz = img;
    var width = img.width;
    var height = img.height;
    var canvas = document.createElement('canvas');
    canvas.style.position='relative';
    var ctx = canvas.getContext('2d');
    console.log(direction);
    var canvasX = 0,canvasY = 0;
    switch (direction) {
      case 1: {
        canvas.width = showheight;
        canvas.height = showwidth;
        ctx.rotate(90*Math.PI/180);
        ctx.drawImage(thiz,x,y,showwidth,showheight,0,-showheight,showwidth,showheight);
        break;
      }
      case 2: {
        canvas.width = showwidth;
        canvas.height = showheight;
        ctx.rotate(180*Math.PI/180);
        ctx.drawImage(thiz,x,y,showwidth,showheight,-showwidth,-showheight,showwidth,showheight);
        break;
      }
      case 3: {
        canvas.width = showheight;
        canvas.height = showwidth;
        ctx.rotate(270*Math.PI/180);
        ctx.drawImage(thiz,x,y,showwidth,showheight,-showwidth,0,showwidth,showheight);
        break;
      }
      case 0:
      case 4:
      default: {
        canvas.width = showwidth;
        canvas.height = showheight;
        ctx.rotate(0);
        ctx.drawImage(thiz,x,y,showwidth,showheight,0,0,finalWidth,finalHeight);
        break;
      }
    }
    div.appendChild(canvas);

    div.setAttribute('index',imgIndex);//设置索引
    div.style.width = sshowwidth+'px';
    div.style.height = sshowheight+'px';
    //根据index 查找比对前后插入。
    var sd = document.getElementById('showDetail');
    var cn = sd.childNodes;
    if(cn.length == 0){
        sd.appendChild(div);
    }else{
        //循环找到
        var diff = 10,nearNode = null,isBefore = false;
        cn.forEach(function(tempNode,nodeIndexL){
            var nodeIndex = parseInt(tempNode.getAttribute('index'),10);
            //比对哪个更接近imgIndex
            var currentDiff = Math.abs(nodeIndex - imgIndex);
            if(currentDiff < diff && nodeIndex > imgIndex){
                diff = currentDiff;
                nearNode = tempNode;
                isBefore = true;
            }
        })
        if(isBefore){
            sd.insertBefore(div,nearNode);
        }else{
            sd.appendChild(div);
        }
    }
    
    }
})(direction,filePath,div,showwidth,showheight,sshowwidth,sshowheight,x,y,i);
                        

我这里参数比较多,主要就看switch 里面的那一部分就可以。 我这边的需求是需要显示多个大图的里面的小图,有小图的坐标,然后根据小图坐标展示每个小图图片,并正常显示。

最终的原理就是,旋转canvas,然后将图片渲染并显示部分坐标的内容,最终旋转回来展示。 具体可以查看drawImage的参数说明。

参数说明

转载请注明出处: https://chrunlee.cn/article/canvas-pic-rotate-show.html


感谢支持!

赞赏支持
提交评论
评论信息 (请文明评论)
暂无评论,快来快来写想法...
推荐
本文概括了递归、闭包、原型、继承,理清这些基本的概念,有助于你接纳更多的东西,我们会在下一个章节对函数进行更深入的讨论。
最大公因数,也称最大公约数、最大公因子,指两个或多个整数共有约数中最大的一个。a,b的最大公约数记为(a,b),同样的,a,b,c的最大公约数记为(a,b,c),多个整数的最大公约数也有同样的记号。求最大公约数有多种方法,常见的有质因数分解法、短除法、辗转相除法、更相减损法。与最大公约数相对应的概念是最小公倍数,a,b的最小公倍数记为[a,b]。
前端时间搞了个小转码,放在后台,但是特别占带宽,想着能不能从前台把这个事搞定呢?读取图片的二进制,然后将字节流处理后重新生成图片展示处理啊。
java 对象中有很多引用,甚至会出现循环引用,比如 user 对象中有 school 对象,school 对象中又有 user 对象,这样在对 user 对象序列化的时候,就会出现死循环,导致内存溢出。通过一定的方式,将每个对象增加ID 和 REF 引用标识最终可以解决这个问题
我们经常会有判断一个数值是素数的需求,那么我们如何来实现呢?
关于web打印,需要对页面内容进行页面样式设置,呈现出分页的样子,同时对于题目中的图片或表格尽量不分到两个页面中,因此实现了一个jquery的web打印插件,当然,这个插件目前只适用于部分情况,仅供借鉴
开发的项目中有使用到微信开发,由于之前偶尔才用一次,也是用的别人的地址和测试号,这里记录下自己的操作。防止后续遗忘。
在我们web开发过程中经常会碰到针对table的一些dom操作,这里整理一下关于这方面的知识点。当然我们可以通过jquery这样的插件来处理,或许会更简单一些,不过现在简单说下原生JS是如何操作的