<span id="mktg5"></span>

<i id="mktg5"><meter id="mktg5"></meter></i>

        <label id="mktg5"><meter id="mktg5"></meter></label>
        最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
        問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
        當前位置: 首頁 - 科技 - 知識百科 - 正文

        JavaScript截屏功能的實現代碼

        來源:懂視網 責編:小采 時間:2020-11-27 22:33:56
        文檔

        JavaScript截屏功能的實現代碼

        JavaScript截屏功能的實現代碼:最近參與了網易爐石盒子的相關頁面開發,在做卡組分享頁(地址:爐石盒子卡組分享),有個需求:用戶可以把這個卡組以圖片的形式分享給好友。最初的的做法是使用服務器把該頁面轉換成圖片,然后把圖片地址返回給前端。嗯,這樣也挺好的啊,而且服務器還可以對
        推薦度:
        導讀JavaScript截屏功能的實現代碼:最近參與了網易爐石盒子的相關頁面開發,在做卡組分享頁(地址:爐石盒子卡組分享),有個需求:用戶可以把這個卡組以圖片的形式分享給好友。最初的的做法是使用服務器把該頁面轉換成圖片,然后把圖片地址返回給前端。嗯,這樣也挺好的啊,而且服務器還可以對

        最近參與了網易爐石盒子的相關頁面開發,在做卡組分享頁(地址:爐石盒子卡組分享),有個需求:用戶可以把這個卡組以圖片的形式分享給好友。最初的的做法是使用服務器把該頁面轉換成圖片,然后把圖片地址返回給前端。嗯,這樣也挺好的啊,而且服務器還可以對轉換出來的圖片進行緩存,下次請求可以直接返回圖片地址了。原理上是毫無毛病的。然而,問題來了,后臺轉換的圖片和頁面內容偶爾不一致,有時候會少了一一些內容,PM姐姐就很不爽了,說這個問題一定要解決。反正頁面轉成圖片的接口是后臺做的,關我luan事啊!就在暗暗自喜的時候,悲催的事情發生的,后臺的同事說,因為頁面里面有些內容是異步加載出來的(比如底部的二維碼是通過canvas生成的),服務器轉換不穩定,有時候對異步渲染的內容無法截取。說白了,就是這問題他沒有辦法解決,前端去改吧,誰叫前端用了異步渲染呢?最后Leader讓我嘗試能不能直接用JS進行截圖了,這樣既可以減輕服務器的壓力,又可以解決上面bug。

          一開始,我覺得使用JS截圖的想法是非常荒謬的(怪我無知咯,前端這幾年發展的實在太快了):首先JS沒有權限調用操作系統的截圖功能,其次,瀏覽器(BOM)也沒有提供相關的截圖接口。我該怎么辦呢、怎么辦呢?有事找Google啊。然后搜索了一下: JS html to png ,然后來到就找到了這里:render-html-to-an-image。開始有思路了,回答中有人提到可以把dom轉成canvas,嗯!又是Canvas!我不由得興奮起來,真的是山重水復疑無路,柳暗花明又一村啊!然后再搜索一下 dom to canvas,來到了大家熟知的mdn的文檔Drawing_DOM_objects_into_a_canvas。然后就開始認(zhuang)真(bi)的看文檔。文檔開頭就說到,不可以把dom轉成canvas,但是可以把dom轉成svg,然后再把svg畫到canvas里面去。也許有人會問,為什么要先把dom轉成svg呢?這可能是因為svg使用xml表示、結構和dom一致吧。
        下面就是官方文檔的step by step的教程:

        1.Blob的媒體類型必須是"image/svg+xml"

        2.需要一個 svg 元素

        3.在 svg 元素里面插入一個 foreignObject 元素

        4.在 foreignObject 元素里面放入符合規范的 html

        把dom轉成canvas就這么簡單,就上面幾個步驟。下面是文檔給出的一上簡單的demo:

        <!doctype html>
        <html lang="en">
        <head>
         <meta charset="UTF-8">
         <title>Document</title>
        </head>
        <body>
        <canvas id="canvas" style="border:2px solid black;" width="200" height="200">
        </canvas>
        <script>
         var canvas = document.getElementById('canvas');
         var ctx = canvas.getContext('2d');
         var data = '<svg xmlns="http://www.w3.org/2000/svg" width="200" height="200">' +
         '<foreignObject width="100%" height="100%">' +
         '<div xmlns="http://www.w3.org/1999/xhtml" style="font-size:40px">' +
         '<em>I</em> like ' +
         '<span style="color:white; text-shadow:0 0 2px blue;">' +
         'cheese</span>' +
         '</div>' +
         '</foreignObject>' +
         '</svg>';
         var DOMURL = window.URL || window.webkitURL || window;
         var img = new Image();
         var svg = new Blob([data], {type: 'image/svg+xml'});
         var url = DOMURL.createObjectURL(svg);
         img.onload = function() {
         ctx.drawImage(img, 0, 0);
         DOMURL.revokeObjectURL(url);
         }
         img.src = url;
        </script>
        </body>
        </html>

        ,運行一下,哇,帥呆了,瀏覽器上出現了超酷的兩行藝術字呢!

        嗯,原來dom轉成canvas這么簡單啊?那我通過 document.body.innerHTML 把body里面的所有dom取出來,然后放到 foreignObject 元素里面,不就OK了、把整個頁面都截取下來了嗎?

        demo僅僅是個Hello World,但是實際項目中的Dom結構比這個復雜多了,比如,引入了外部樣式表、圖片、而且還可能某些標簽不符合xml規范(如缺少閉合標簽等)。下面的舉個簡單的例子,.container不是使用行內樣式的,而是在style標簽里面定義,字體紅色,轉成圖片后,樣式不生效。

        <!doctype html>
        <html lang="en">
        <head>
         <meta charset="UTF-8">
         <title>Document</title>
         <style>
         .container {
         color: red;
         }
         </style>
        </head>
        <body>
        <div class="container" >
         Hello World!
        </div>
        <canvas id="canvas" style="border:2px solid black;" width=200" height="200">
        </canvas>
        <script>
         var canvas = document.getElementById('canvas');
         var ctx = canvas.getContext('2d');
         var data = '<svg xmlns="http://www.w3.org/2000/svg" width="200" height="200">' +
         '<foreignObject width="100%" height="100%">' +
         '<div xmlns="http://www.w3.org/1999/xhtml" style="font-size:40px">' +
         document.querySelector('.container').innerHTML +
         '</div>' +
         '</foreignObject>' +
         '</svg>';
         var DOMURL = window.URL || window.webkitURL || window;
         var img = new Image();
         var svg = new Blob([data], {type: 'image/svg+xml'});
         var url = DOMURL.createObjectURL(svg);
         img.onload = function() {
         ctx.drawImage(img, 0, 0);
         DOMURL.revokeObjectURL(url);
         }
         img.src = url;
        </script>
        </body>
        </html>

        既然外部樣式不生效,那我們可以通過JS遍歷所有的dom元素,把全部的樣式通過element.style對象添加到行內樣式啊。這個思路聽起來不錯,但是,實現這個把外部樣式轉成行內樣式的函數我還真寫不出來啊。需求比較緊,也沒有那 多時間去瞎折騰了,所以,就想找找有沒有現成的庫。于是又去google一下。很幸運, 一下子就搜到了一個叫做html2canvas的庫,非常棒的一個庫,很強大、但用法非常簡單.就這么簡單的方法,就可以把我的整個頁面截圖下來了:

        function convertHtml2Canvas() {
         html2canvas(document.body, {
         allowTaint: false,
         taintTest: true
         }).then(function(canvas) {
         document.body.appendChild(canvas);
         }).catch(function(e) {
         console.error('error', e);
         });
         }

        目前還有一個問題,就是這種方法默認是把整個頁面截取下來的(就是說,會以你的innerHeight和innerWidth為邊界,會存在大量的空白),可是,我的卡組只是占了頁面的一小部分,我只想要卡組的部分啊。其實已經有了canvas就好辦了,我們可以對它進行處理啊。大概思路是:1.把上面得到的canvas對象轉成Blob并放到一個img元素。然后再把img.src繪制到canvas里面。這時候調用canvas.drawImage方法就可以截取我們想要的內容了。下面的兩個函數分別是把canvas轉成image以及反過來把image轉成canvas。

        // Converts canvas to an image
         function convertCanvasToImage(canvas) {
         var image = new Image();
         image.src = canvas.toDataURL("image/png", 0.1);
         return image;
         }
         // Converts image to canvas; returns new canvas element
         function convertImageToCanvas(image, startX, startY, width, height) {
         var canvas = document.createElement("canvas");
         canvas.width = width;
         canvas.height = height;
         canvas.getContext("2d").drawImage(image, startX, startY, width, height, 0, 0, width, height);
         return canvas;
         }

        然后,再把我們上面的寫的 convertHtml2Canvas 改成下面的:

        function convertHtml2Canvas() {
         html2canvas(document.body, {
         allowTaint: false,
         taintTest: true
         }).then(function(canvas) {
         var img = convertCanvasToImage(canvas);
         document.body.appendChild(img);
         img.onload = function() {
         img.onload = null;
         canvas = convertImageToCanvas(img, 0, 0, 384, 696);
         img.src = convertCanvasToImage(canvas).src;
         $(img).css({
         display: 'block',
         position: 'absolute',
         top: 0,
         left: 400 + 'px'
         });
         }
         }).catch(function(e) {
         console.error('error', e);
         });
         }

        這時候就可以把它的頁面的某部分內容進行截取下來了。效果如卡組分享測試頁面。頁面左邊部分是DOM結構的,右邊部分是則是使用html2canvas轉換出來的圖片。長得一模一樣,毫無毛病哈。

        關于JS頁面截圖的就寫到這里啦,因為也只剛剛接觸,很多東西也理解得不到位,歡迎各大神指點。后面會深入學習一下html2canvas的源碼,進一步理解dom to canvas的原理。

        總結

        以上所述是小編給大家介紹的JavaScript截屏功能的實現代碼,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網站的支持!

        聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

        文檔

        JavaScript截屏功能的實現代碼

        JavaScript截屏功能的實現代碼:最近參與了網易爐石盒子的相關頁面開發,在做卡組分享頁(地址:爐石盒子卡組分享),有個需求:用戶可以把這個卡組以圖片的形式分享給好友。最初的的做法是使用服務器把該頁面轉換成圖片,然后把圖片地址返回給前端。嗯,這樣也挺好的啊,而且服務器還可以對
        推薦度:
        標簽: 截屏 截圖 實現
        • 熱門焦點

        最新推薦

        猜你喜歡

        熱門推薦

        專題
        Top
        主站蜘蛛池模板: 久久久久久久免费视频| 久久er国产精品免费观看2| 波多野结衣中文字幕免费视频| 亚洲AV无码久久精品色欲| 三级网站免费观看| 久久91亚洲精品中文字幕| 久久精品视频免费播放| 亚洲精品在线网站| 日本XXX黄区免费看| 自拍偷区亚洲国内自拍| 免费被黄网站在观看| 深夜a级毛片免费无码| 区久久AAA片69亚洲| 日韩免费高清大片在线| 亚洲一区二区三区无码国产| 成人免费无码视频在线网站| 亚洲av第一网站久章草| 国产亚洲精aa成人网站| 亚欧日韩毛片在线看免费网站| 亚洲国产精品无码久久久| 免费无码不卡视频在线观看| 免费又黄又爽又猛大片午夜| 亚洲精品无码久久久久| 成人福利免费视频| 污网站在线观看免费| 久久亚洲成a人片| 成熟女人特级毛片www免费| 日韩免费高清一级毛片| 亚洲第一视频网站| 日韩激情无码免费毛片| 女人隐私秘视频黄www免费| 亚洲av永久无码嘿嘿嘿| 亚洲国产专区一区| 182tv免费观看在线视频| 精品亚洲国产成人av| 亚洲AV日韩AV高潮无码专区| 精品剧情v国产在免费线观看| 国产一级黄片儿免费看| 久久综合久久综合亚洲| 亚洲精品亚洲人成在线观看| 好吊妞在线成人免费|