騰訊 AlloyCrop 1.0 發佈

来源:http://www.cnblogs.com/iamzhanglei/archive/2017/08/02/7272842.html
-Advertisement-
Play Games

寫在前面 "AlloyCrop" 這個項目是8個月前發佈的,作為 "AlloyFinger" 的典型案例,發佈之後被BAT等其他公司廣泛使用。但是發佈之後,有兩個問題一直沒有抽出時間去解決: 裁剪圖像的解析度太小,是否可配? pinch雙指放大的時候,放大的中心不是雙指中心,是否可以優化? 現在很高 ...


寫在前面

AlloyCrop 這個項目是8個月前發佈的,作為AlloyFinger 的典型案例,發佈之後被BAT等其他公司廣泛使用。但是發佈之後,有兩個問題一直沒有抽出時間去解決:

  • 裁剪圖像的解析度太小,是否可配?
  • pinch雙指放大的時候,放大的中心不是雙指中心,是否可以優化?

現在很高興地告訴大家,AlloyCrop 已經完全搞定了上面兩個問題,本文將解釋新版本的變化和AlloyCrop背後的原理。當然AlloyFinger的原理這裡就不再闡述,以前有分享過 超小Web手勢庫AlloyFinger原理

先看全新的API變化。

API

new AlloyCrop({
    image_src: "img src",
    circle: true, // optional parameters , the default value is false
    width: 200, // crop width
    height: 100, // crop height
    output: 2, // output resolution --> 400*200
    ok: function (base64, canvas) { },
    cancel: function () { },
    ok_text: "yes", // optional parameters , the default value is ok
    cancel_text: "no" // optional parameters , the default value is cancel
});
參數 是否必填 意義
image_src 必須 需要裁剪圖片的src地址
circle 不必須,預設是false 代表選取的是否是圓形還是矩形,預設是矩形,註意:圓形選取裁剪出來的也是正方形圖片
width 必須 選區的寬
height 必須 選區的高
output 必須 輸出的倍率。比如如果output為2,選區的寬300,選區的高100,輸出的圖像的解析度為 (2×300,2×100)
ok 必須 點擊ok按鈕的回調
cancel 必須 點擊cancel按鈕的回調
ok_text 不必須,預設是ok ok按鈕的文本
cancel_text 不必須,預設是cancel cancel按鈕的文本

與之前版本最主要的變化就是新增了 output 支持自定義倍率解析度的圖像輸出。

output原理

crop: function () {
    this.calculateRect();
    this.ctx.drawImage(this.img, this.crop_rect[0], this.crop_rect[1], this.crop_rect[2], this.crop_rect[3], 0, 0, this.canvas.width, this.canvas.height);

},

其中 this.calculateRect() 是計算選取和圖片重疊在一起的矩形區域,drawImage 是把裁剪的區域繪製到 canvas 上。註意 canvas 的寬高是多少?且看:

this.canvas.width = option.width * this.output;
this.canvas.height = option.height * this.output;

所以就達到了自定義倍率解析度的目的。當然這裡圖片的失真又或者超分辨,都取決於 drawImage 插值過程。關於插值,以前特意對比過,使用三次捲積插值完爆了其他幾個,但是三次捲積插值速度也是最慢,所以瀏覽器內核要權衡效率和插值結果去實現 drawImage。

img

calculateRect計算裁剪區域

因為我們需要把圖片的某個區域繪製到整個canvas上。所以drawImage的後四個參數為(0, 0, this.canvas.width, this.canvas.height),然後我們需要去計算圖片裁剪的區域。

pv

大概就分上面兩種情況,一種是完全包含,一種部分相交。

因為圖片會被放大或者縮小(scale),所以情況會變得稍微複雜一點點。求出相交的矩形區域後,要對圖片scale進行校正,校正回到1的比例,才能用於drawImage。具體代碼參見 https://github.com/AlloyTeam/AlloyCrop/blob/master/alloy-crop.js#L227-L255

pinch 縮放優化

使用AlloyCrop是可以放大或者縮小再進行裁剪,怎麼基於 pinch 的兩個手指的中間進行放大呢?所以的秘密都在這個multipointStart里。

  • multipointStart是AlloyFinger拋出的多手指開始碰到屏幕的回調函數,通過evt.touches拿到前兩個手指的坐標去計算中心坐標
  • 重置 originX 和 originY 到兩手指的中心
  • 再重置 translateX 和 translateY 去抹平 originX和originY變更帶來的位移
 new AlloyFinger(this.croppingBox, {
    multipointStart: function (evt) {
        //reset origin x and y
        var centerX = (evt.touches[0].pageX + evt.touches[1].pageX) / 2;
        var centerY = (evt.touches[0].pageY + evt.touches[1].pageY) / 2;
        var cr = self.img.getBoundingClientRect();
        var img_centerX = cr.left + cr.width / 2;
        var img_centerY = cr.top + cr.height / 2;
        var offX = centerX - img_centerX;
        var offY = centerY - img_centerY;
        var preOriginX = self.img.originX
        var preOriginY = self.img.originY
        self.img.originX = offX / self.img.scaleX;
        self.img.originY = offY / self.img.scaleY;
        //reset translateX and translateY
        self.img.translateX += offX - preOriginX * self.img.scaleX;
        self.img.translateY += offY - preOriginY * self.img.scaleX;
        self.initScale = self.img.scaleX;
    },
    pinch: function (evt) {
        self.img.scaleX = self.img.scaleY = self.initScale * evt.zoom;
    },
    pressMove: function (evt) {
        self.img.translateX += evt.deltaX;
        self.img.translateY += evt.deltaY;
        evt.preventDefault();
    }
});
  • 註意,translateX, translateY, translateZ, scaleX, scaleY, scaleZ, rotateX, rotateY, rotateZ, skewX, skewY, originX, originY, originZ 都是 css3transform 類庫 註入到DOM元素上的屬性。

Preview

Preview

Demo

Dependencies

License

This content is released under the MIT License.


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這一章主要介紹什麼是[BX]以及loop(迴圈)指令怎麼使用,loop和[BX]又怎麼樣相結合,段首碼又是什麼鬼,以及如何使用段首碼。 1、[BX]的概念 [BX]和[0]類似,[0]表示記憶體單元的偏移地址是0。要完整描述一個記憶體單元,需要兩種信息:記憶體單元的地址,記憶體單元的長度(類型)。[BX]同 ...
  • 不久前參與開發了一個基於dubbo分散式框架的底層賬單系統,並實現了其中的一部分業務介面,目前需對這些介面進行壓測,以評估生產環境所能承受的最大吞吐量。筆者以其中一個查詢介面為例來回顧此次壓測的整體流程。 壓測準備: 1.調用查詢介面的測試jar包,作為dubbo-consumer,依賴了查詢服務的 ...
  • 架構好文學習,攢~~ 京東咚咚架構演進 -- By 【瞬息之間】 名詞解釋: Apache MINA: 百度百科 HAProxy: 百度百科 1.0 架構筆記: 優點:模型結構簡單 理解起來簡單;開發起來簡單;部署起來也簡單。 缺點:效率和擴展 這個模型實際上是一個高功耗低效能的模型,不活躍的連接在 ...
  • 簡單工廠模式,就是有一個工廠類,負責生成所需要的實體類。 這讓我想起了紅警,在紅警里,如果需要造一輛坦克,需要先有一個戰車工廠,在坦克工廠里選擇坦克類型,然後坦克工廠才會生成所需要的坦克。 不多說直接上代碼。 首先有個坦克類Panzer(哈哈我的俠盜獵車似乎就記得這麼一個作弊秘籍了) 有兩個類型的坦 ...
  • 在我前面有很多篇隨筆介紹了Web API 介面層的架構設計,以及對微信公眾號、企業號、小程式等模塊的分類劃分。例如在《C#開發微信門戶及應用(43)--微信各個項目模塊的定義和相互關係》介紹了相關模塊的劃分,在《基於微信小程式的系統開發準備工作》介紹了Web API的架構設計思路。本篇隨筆對之前介紹... ...
  • 參考連接:http://nqdeng.github.io/7-days-nodejs/#1.1 NodeJS基礎 什麼是NodeJS JS是腳本語言,腳本語言都需要一個解析器才能運行。對於寫在HTML頁面里的JS,瀏覽器充當瞭解析器的角色。而對於需要獨立運行的JS,NodeJS就是一個解析器。 每一 ...
  • 原文出處:https://segmentfault.com/a/1190000010371988 看了一下這篇文章,自己也手敲了一遍 ...
  • Location 對象屬性 屬性 | 描述 | hash| 設置或返回從井號 ( ) 開始的 URL(錨)。 host | 設置或返回主機名和當前 URL 的埠號。 hostname | 設置或返回當前 URL 的主機名。 href | 設置或返回完整的 URL。 pathname | 設置或返回 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...