欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

基于casperjs和resemble.js實(shí)現(xiàn)一個(gè)像素對(duì)比服務(wù)詳解

 更新時(shí)間:2018年01月10日 09:12:41   作者:Aaaaaaaty  
這篇文章主要給大家介紹了關(guān)于基于casperjs和resemble.js實(shí)現(xiàn)一個(gè)像素對(duì)比服務(wù)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。

前言

本次分享一個(gè)提供設(shè)計(jì)稿與前端頁(yè)面進(jìn)行像素對(duì)比的node服務(wù),旨在為測(cè)試或者前端人員自己完成一個(gè)輔助性測(cè)試。相信我,在像素級(jí)別的對(duì)比下,網(wǎng)頁(yè)對(duì)設(shè)計(jì)稿的還原程度一下子就會(huì)凸顯出來(lái)。下面話不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹吧。

效果預(yù)覽


前置知識(shí)

本次用到了以下兩個(gè)庫(kù)作為輔助工具:

  • casperjs:基于PhantomJS的編寫。其內(nèi)部提供了一個(gè)無(wú)界面瀏覽器,簡(jiǎn)單來(lái)說(shuō)用它你可以以代碼的形式來(lái)完成模擬人來(lái)操作瀏覽器的操作,其中涉及鼠標(biāo)各種事件,等等非常多的功能,本次主要使用其附帶的截圖功能。
  • resemble.js:圖片像素對(duì)比工具。調(diào)用方法簡(jiǎn)單理解為,傳入兩張圖,返回一張合成圖并附帶對(duì)比參數(shù)如差別度等等?;緦?shí)現(xiàn)思路可以理解為通過(guò)將圖片轉(zhuǎn)為canvas后,獲取其圖像像素點(diǎn),之后對(duì)每個(gè)像素點(diǎn)進(jìn)行一次比對(duì)。

所以整個(gè)服務(wù)我們應(yīng)該已經(jīng)有了大題的思路即通過(guò)casperjs來(lái)進(jìn)入某個(gè)網(wǎng)站截取某個(gè)頁(yè)面,再將其與設(shè)計(jì)圖進(jìn)行比對(duì)得出結(jié)果。

整體思路

通過(guò)上圖我們應(yīng)該能整理出一個(gè)大概的流程:

  • 從前端頁(yè)面接收設(shè)計(jì)稿圖片及需要截取的網(wǎng)站地址與節(jié)點(diǎn)信息
  • 將設(shè)計(jì)稿保存到images文件夾
  • 開(kāi)啟子進(jìn)程,啟動(dòng)casperjs,完成對(duì)目標(biāo)網(wǎng)站的截取
  • 截取后請(qǐng)求form.html將圖片地址信息填入并重新傳回服務(wù)器
  • 服務(wù)端獲取圖片信息通過(guò)resemblejs將截取圖與設(shè)計(jì)稿進(jìn)行比對(duì)
  • 結(jié)果傳回前端頁(yè)面

這其中有一個(gè)問(wèn)題可能會(huì)有人注意到就是:為什么在casperjs中對(duì)目標(biāo)網(wǎng)站截圖了不能直接把信息傳回服務(wù)器中,而是選擇了再去打開(kāi)一個(gè)表單頁(yè)面通過(guò)表單的形式來(lái)提交信息?

答:首先我對(duì)casperjs和node了解都不那么深入,我理解的是首先casperjs不是一個(gè)node模塊,它是跑在操作系統(tǒng)中的,我尚且沒(méi)有發(fā)現(xiàn)怎么在casperjs中建立與node服務(wù)的通信,如果有方法一定要告訴我,因?yàn)槲艺娴牟惶私鈉asper!其次由于無(wú)法建立通信,我只能退而求其次,通過(guò)casper快速打開(kāi)一個(gè)我寫好的表單頁(yè)面并且填寫好圖片信息傳回服務(wù)器,這么做是可以完成最初的訴求。所以就有了上面from.html那段的操作。

實(shí)現(xiàn)細(xì)節(jié)

實(shí)現(xiàn)一個(gè)簡(jiǎn)易靜態(tài)服務(wù)器

因?yàn)樯婕暗絠ndex.html與form.html頁(yè)面的返回,故需要實(shí)現(xiàn)一個(gè)超級(jí)簡(jiǎn)易的靜態(tài)服務(wù)器。代碼如下:

const MIME_TYPE = {
 "css": "text/css",
 "gif": "image/gif",
 "html": "text/html",
 "ico": "image/x-icon",
 "jpeg": "image/jpeg",
 "jpg": "image/jpg",
 "js": "text/javascript",
 "json": "application/json",
 "pdf": "application/pdf",
 "png": "image/png",
 "svg": "image/svg+xml",
 "swf": "application/x-shockwave-flash",
 "tiff": "image/tiff",
 "txt": "text/plain",
 "wav": "audio/x-wav",
 "wma": "audio/x-ms-wma",
 "wmv": "video/x-ms-wmv",
 "xml": "text/xml"
}
function sendFile(filePath, res) {
 fs.open(filePath, 'r+', function(err){ //根據(jù)路徑打開(kāi)文件
  if(err){
   send404(res)
  }else{
   let ext = path.extname(filePath)
   ext = ext ? ext.slice(1) : 'unknown'
   let contentType = MIME_TYPE[ext] || "text/plain" //匹配文件類型
   fs.readFile(filePath,function(err,data){
    if(err){
     send500(res)
    }else{
     res.writeHead(200,{'content-type':contentType})
     res.end(data)
    }
   })
  }
 })
}

解析表單并將圖片存儲(chǔ)到images文件夾

const multiparty = require('multiparty') //解析表單
let form = new multiparty.Form()
 form.parse(req, function (err, fields, files) {
  let filename = files['file'][0].originalFilename,
   targetPath = __dirname + '/images/' + filename,
  if(filename){
   fs.createReadStream(files['file'][0].path).pipe(fs.createWriteStream(targetPath))
   ...
  } 
 })

通過(guò)創(chuàng)建可讀流讀出文件內(nèi)容,再通過(guò)pipe寫入到制定路徑下即可保存上傳來(lái)的圖片。

運(yùn)行casperjs

const { spawn } = require('child_process')
spawn('casperjs', ['casper.js', filename, captureUrl, selector, id])
casperjs.stdout.on('data', (data) => {
 ...
}) 

通過(guò)spawn可以創(chuàng)建子進(jìn)程來(lái)啟動(dòng)casperjs,同樣也可以使用exec等。

截圖并提交數(shù)據(jù)到form.html

const system = require('system')
const host = 'http://10.2.45.110:3033'
const casper = require('casper').create({
 // 瀏覽器窗口大小
 viewportSize: {
  width: 1920,
  height: 4080
 }
})
const fileName = decodeURIComponent(system.args[4])
const url = decodeURIComponent(system.args[5])
const selector = decodeURIComponent(system.args[6])
const id = decodeURIComponent(system.args[7])
const time = new Date().getTime()
casper.start(url)
casper.then(function() {
  console.log('正在截圖請(qǐng)稍后')
  this.captureSelector('./images/casper'+ id + time +'.png', selector)
})
casper.then(function() {
 casper.start(host + '/form.html', function() {
  this.fill('form#contact-form', {
   'diff': './images/casper'+ id + time +'.png',
   'point': './images/' + fileName,
   'id': id
  }, true)
 })
})
casper.run()

代碼還是比較簡(jiǎn)單的,主要過(guò)程就是打開(kāi)一個(gè)頁(yè)面,然后在then中傳入你的操作,最后執(zhí)行run。在這個(gè)過(guò)程里我不太知道如何與node服務(wù)通信,故選擇了再開(kāi)一個(gè)頁(yè)面。。想深入研究的可以去看casperjs的官網(wǎng)非常詳盡!

通過(guò)resemble.js進(jìn)行像素比對(duì)并返回?cái)?shù)據(jù)

function complete(data) {
  let imgName = 'diff'+ new Date().getTime() +'.png',
   imgUrl,
   analysisTime = data.analysisTime,
   misMatchPercentage = data.misMatchPercentage,
   resultUrl = './images/' + imgName
  fs.writeFileSync(resultUrl, data.getBuffer())
  imgObj = {
   ...
  }
  let resEnd = resObj[id] // 找回最開(kāi)始的res返回給頁(yè)面數(shù)據(jù)
  resEnd.writeHead(200, {'Content-type':'application/json'})
  resEnd.end(JSON.stringify(imgObj))
 }
let result = resemble(diff).compareTo(point).ignoreColors().onComplete(complete)

這其中涉及到了一個(gè)點(diǎn),即我現(xiàn)在所得到的結(jié)果要返回給最初的請(qǐng)求里,而從一開(kāi)始的請(qǐng)求到現(xiàn)在我已經(jīng)中轉(zhuǎn)了多次,導(dǎo)致我現(xiàn)在找不到我最初的返回體res了。想了很久只能暫時(shí)采用了設(shè)定全局對(duì)象,在接收最初的請(qǐng)求后將請(qǐng)求者的ip和時(shí)間戳設(shè)定為唯一id存為該對(duì)象的key,value為當(dāng)前的res。同時(shí)整個(gè)中轉(zhuǎn)流程中時(shí)刻傳遞id,最后通過(guò)調(diào)用resObj[id]來(lái)得到一開(kāi)始的返回體,返回?cái)?shù)據(jù)。這個(gè)方法我不認(rèn)為是最優(yōu)解,但是鑒于我現(xiàn)在想不出來(lái)好方法為了跑通整個(gè)服務(wù)不得已。。如果有新的思路請(qǐng)務(wù)必告知!!

部署

安裝PhantomJS(osx)

官網(wǎng)下載: phantomjs-2.1.1-macosx.zip

解壓路徑:/User/xxx/phantomjs-2.1.1-macosx

添加環(huán)境變量:~/.bash_profile 文件中添加

export PATH="$PATH:/Users/xxx/phantomjs-2.1.1-macosx/bin"

terminal輸入:phantomjs --version

能看到版本號(hào)即安裝成功

安裝casperjs

brew update && brew install casperjs

安裝resemble.js

cnpm i resemblejs //已寫進(jìn)packjson可不用安裝
brew install pkg-config cairo libpng jpeg giflib
cnpm i canvas //node內(nèi)運(yùn)行canvas

node服務(wù)

git clone https://github.com/Aaaaaaaty/gui-auto-test.git
cd gui-auto-test
cnpm i
cd pxdiff
nodemon server.js

打開(kāi)http://localhost:3033/index.html

慣例po作者的博客,不定時(shí)更新中——

參考文獻(xiàn)

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。

相關(guān)文章

  • 實(shí)例分析js和C#中使用正則表達(dá)式匹配a標(biāo)簽

    實(shí)例分析js和C#中使用正則表達(dá)式匹配a標(biāo)簽

    本文通過(guò)2個(gè)實(shí)例,對(duì)比分析了在js和c#中使用正則表達(dá)式匹配a標(biāo)簽的異同,小伙伴們自己參考下吧,有利于深刻理解正則表達(dá)式的使用。
    2014-11-11
  • javascript在IE下trim函數(shù)無(wú)法使用的解決方法

    javascript在IE下trim函數(shù)無(wú)法使用的解決方法

    這篇文章主要介紹了javascript在IE下trim函數(shù)無(wú)法使用的解決方法,分別敘述了javascript以及jQuery下的解決方案,對(duì)于WEB前端javascript設(shè)計(jì)人員進(jìn)行瀏覽器兼容性調(diào)試有不錯(cuò)的借鑒價(jià)值,需要的朋友可以參考下
    2014-09-09
  • 基于JavaScript實(shí)現(xiàn)前端文件的斷點(diǎn)續(xù)傳

    基于JavaScript實(shí)現(xiàn)前端文件的斷點(diǎn)續(xù)傳

    這篇文章主要介紹了基于JavaScript實(shí)現(xiàn)前端文件的斷點(diǎn)續(xù)傳的相關(guān)資料,需要的朋友可以參考下
    2016-10-10
  • Jquery對(duì)數(shù)組的操作技巧整理

    Jquery對(duì)數(shù)組的操作技巧整理

    這篇文章主要介紹了Jquery對(duì)數(shù)組的操作技巧,需要的朋友可以參考下
    2014-03-03
  • 一文帶你理解JS中的原型和原型鏈

    一文帶你理解JS中的原型和原型鏈

    在學(xué)習(xí)JavaScript中的繼承機(jī)制時(shí),我們常常會(huì)遇到原型和原型鏈這兩個(gè)概念,在初學(xué)階段,不理解這些概念很容易陷入迷茫,甚至?xí)?dǎo)致學(xué)習(xí) JS 的路程變得曲折,本文將介紹JavaScript原型和原型鏈的概念、設(shè)計(jì)思想以及相關(guān)的使用方法,需要的朋友可以參考下
    2023-07-07
  • 用js替換除數(shù)字與逗號(hào)以外的所有字符的代碼

    用js替換除數(shù)字與逗號(hào)以外的所有字符的代碼

    用js實(shí)現(xiàn)的替換除數(shù)字與逗號(hào)以外的所有字符的代碼,因?yàn)樾枰コ貜?fù),這樣的話就會(huì)有很多的逗號(hào)需要替換
    2014-06-06
  • layui.js實(shí)現(xiàn)的表單驗(yàn)證功能示例

    layui.js實(shí)現(xiàn)的表單驗(yàn)證功能示例

    這篇文章主要介紹了layui.js實(shí)現(xiàn)的表單驗(yàn)證功能,結(jié)合實(shí)例形式分析了基于layui.js的事件監(jiān)聽(tīng)、驗(yàn)證、判定等相關(guān)操作技巧,需要的朋友可以參考下
    2017-11-11
  • JavaScript實(shí)現(xiàn)五種不同煙花特效

    JavaScript實(shí)現(xiàn)五種不同煙花特效

    這篇文章主要給大家?guī)?lái)五個(gè)好看的基于 HTML+CSS+JS 的煙花特效,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)JavaScript有一定的幫助,需要的可以參考一下
    2022-01-01
  • 用js實(shí)現(xiàn)簡(jiǎn)單的tab選項(xiàng)卡

    用js實(shí)現(xiàn)簡(jiǎn)單的tab選項(xiàng)卡

    這篇文章主要為大家詳細(xì)介紹了用js實(shí)現(xiàn)簡(jiǎn)單的tab選項(xiàng)卡,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • 文件上傳,iframe跨域數(shù)據(jù)提交的實(shí)現(xiàn)

    文件上傳,iframe跨域數(shù)據(jù)提交的實(shí)現(xiàn)

    下面小編就為大家?guī)?lái)一篇文件上傳,iframe跨域數(shù)據(jù)提交的實(shí)現(xiàn)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2016-11-11

最新評(píng)論