MutliQRCode

PC端和移动端都可用的二维码识别

目前为 2023-06-06 提交的版本。查看 最新版本

// ==UserScript==
// @name         MutliQRCode
// @namespace    https://greasyfork.org/zh-CN/users/1073349
// @version      0.4.4
// @description  PC端和移动端都可用的二维码识别
// @author       4ehex
// @match        *://*/*
// @icon         
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/jsQR.min.js
// @grant        unsafeWindow
// @license      GPL
// ==/UserScript==

(function() {
    'use strict';

    //全局变量 被选中的图片src
    var g_img_src = null;
    var g_is_moblie_env = isPhone();

    //适配移动端 很多移动端浏览器并没有实现油猴接口
    //添加'识别二维码'按钮样式
    let btn_style = document.createElement('style');
    btn_style.type = 'text/css';
    if (!g_is_moblie_env){
        btn_style.innerText = `.idtfy_div{width:5.5vw;height:2.1vh;font-size:1vh;color:#000000;background:#F8F8FF;border-radius:1.5vh;border:0.12vh solid #f2cac9;line-height:2vh;text-align:center;vertical-align:middle;z-index:99999999;display:none;position:absolute;top:20;left:20;cursor:pointer;box-shadow:0.13vh 0.13vh 0.1vh #888888;}`;
        //GM_addStyle(`.idtfy_div{width:5.5vw;height:2.1vh;font-size:1vh;color:#000000;background:#F8F8FF;border-radius:1.5vh;border:0.12vh solid #f2cac9;line-height:2vh;text-align:center;vertical-align:middle;z-index:99999999;display:none;position:absolute;top:20;left:20;cursor:pointer;box-shadow:0.13vh 0.13vh 0.1vh #888888;}`);
    }
    else{
        btn_style.innerText = `.idtfy_div{width:25vw;height:3vh;font-size:1.6vh;color:#000000;background:#F8F8FF;border-radius:1.5vh;border:0.14vw solid #f2cac9;line-height:3vh;text-align:center;vertical-align:middle;z-index:99999999;display:none;position:absolute;top:20;left:20;cursor:pointer;box-shadow:0.15vw 0.15vw 0.13vw #888888;}`;
    }
    document.head.appendChild(btn_style);

    //添加'识别二维码'按钮
    var identify_div = document.createElement('div');
    identify_div.id = 'identify_div';
    identify_div.className = 'idtfy_div';//👇添加一个图标
    identify_div.innerHTML = `<svg class="icon" style="width:1.3vh;height:1.3vh;vertical-align:middle;fill:currentColor;overflow:hidden" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3325"><path d="M613.888 715.264h-41.984C448.512 715.264 348.16 614.912 348.16 491.52s99.84-223.232 223.232-223.232h41.984c123.392 0 223.232 99.84 223.232 223.232 0.512 123.392-99.328 223.744-222.72 223.744z" fill="#CAD3FF" opacity=".2" p-id="3326"></path><path d="M873.984 940.032h-152.576c-14.336 0-25.6-11.264-25.6-25.6s11.264-25.6 25.6-25.6h147.456v-147.456c0-14.336 11.264-25.6 25.6-25.6s25.6 11.264 25.6 25.6v152.576c0 25.088-20.48 46.08-46.08 46.08zM894.464 311.296c-14.336 0-25.6-11.264-25.6-25.6v-148.48h-147.456c-14.336 0-25.6-11.264-25.6-25.6s11.264-25.6 25.6-25.6h152.576c25.6 0 46.08 20.48 46.08 46.08v153.6c0 13.824-11.264 25.6-25.6 25.6z m-20.48-174.08zM131.072 311.296c-14.336 0-25.6-11.264-25.6-25.6v-153.6c0-25.6 20.48-46.08 46.08-46.08h152.576c14.336 0 25.6 11.264 25.6 25.6s-11.264 25.6-25.6 25.6H156.672v148.48c0 13.824-11.776 25.6-25.6 25.6zM304.128 940.032H151.552c-25.6 0-46.08-20.48-46.08-46.08v-152.576c0-14.336 11.264-25.6 25.6-25.6s25.6 11.264 25.6 25.6v147.456h147.456c14.336 0 25.6 11.264 25.6 25.6s-11.264 25.6-25.6 25.6zM726.528 537.6H297.472c-14.336 0-25.6-11.264-25.6-25.6s11.264-25.6 25.6-25.6h429.056c14.336 0 25.6 11.264 25.6 25.6s-11.776 25.6-25.6 25.6z" fill="#4E63DD" p-id="3327"></path></svg> 识别二维码`;
    identify_div.onclick = OnClickedIdentifyBtn;
    document.body.appendChild(identify_div);

    //添加精简后的'notie.js' 一款纯js实现的消息弹窗
    var notie=function(){function E(a,b,c){document.activeElement.blur(),D++,setTimeout(function(){D--},1e3*e+10),1==D&&(A?(clearTimeout(B),clearTimeout(C),G(function(){F(a,b,c)})):F(a,b,c))}function F(b,c,d){A=!0;var f=0;switch(f="undefined"==typeof d?3e3:1>d?1e3:1e3*d,b){case 1:v.style.backgroundColor=g,v.onclick=function(){};break;case 2:v.style.backgroundColor=h,v.onclick=w;break;case 3:v.style.backgroundColor=i,v.onclick=w;break;case 4:v.style.backgroundColor=j,v.onclick=w}y.innerHTML=c,v.style.top="-10000px",v.style.display="table",v.style.top="-"+v.offsetHeight-5+"px",B=setTimeout(function(){a&&(v.style.boxShadow="0px 0px 10px 0px rgba(0,0,0,0.5)"),v.style.MozTransition="all "+e+"s ease",v.style.WebkitTransition="all "+e+"s ease",v.style.transition="all "+e+"s ease",v.style.top=0,C=setTimeout(function(){G(function(){})},f)},20)}function G(b){v.style.top="-"+v.offsetHeight-5+"px",setTimeout(function(){a&&(v.style.boxShadow=""),v.style.MozTransition="",v.style.WebkitTransition="",v.style.transition="",v.style.top="-10000px",A=!1,b&&b()},1e3*e+10)}var v,w,x,y,A,B,C,D,a=!0,b="18px",c="24px",d=600,e=.3,g="#57BF57",h="#E3B771",i="#E1715B",j="#4D82D6",k="#FFF",l="notie-alert-outer",m="notie-alert-inner",n="notie-alert-text",o=function(a){a.style.fontSize=window.innerWidth<=d?b:c},p=500,q=function(a,b,c){var d;return function(){var e=this,f=arguments,g=function(){d=null,c||a.apply(e,f)},h=c&&!d;clearTimeout(d),d=setTimeout(g,b),h&&a.apply(e,f)}};return window.addEventListener("keydown",function(a){var b=13==a.which||13==a.keyCode,c=27==a.which||27==a.keyCode;A&&(b||c)&&(clearTimeout(B),clearTimeout(C),G())}),"undefined"==typeof Element.prototype.addEventListener&&(Element.prototype.addEventListener=Window.prototype.addEventListener=function(a,b){return a="on"+a,this.attachEvent(a,b)}),v=document.createElement("div"),v.id=l,v.style.position="fixed",v.style.top="0",v.style.left="0",v.style.zIndex="999999999",v.style.height="auto",v.style.width="100%",v.style.display="none",v.style.textAlign="center",v.style.cursor="default",v.style.MozTransition="",v.style.WebkitTransition="",v.style.transition="",v.style.cursor="pointer",w=function(){clearTimeout(B),clearTimeout(C),G()},x=document.createElement("div"),x.id=m,x.style.padding="20px",x.style.display="table-cell",x.style.verticalAlign="middle",v.appendChild(x),y=document.createElement("span"),y.id=n,y.style.color=k,y.style.fontSize=window.innerWidth<=d?b:c,window.addEventListener("resize",q(o.bind(null,y),p),!0),x.appendChild(y),document.body.appendChild(v),A=!1,D=0,{alert:E,alert_hide:G}}();
    //👆以上混淆纯粹为了压缩体积 原作者代码: https://github.com/jaredreich/notie

    if (!g_is_moblie_env){
        //电脑端 右键弹出 双击隐藏
        window.onmousedown = function(e) {
            if (e.button == 2) {//右键
                var clickedElement = e.target;
                if ((g_img_src = GetQRSrc(clickedElement)) != null){
                    document.getElementById("identify_div").style.top = (e.pageY + 10) + "px";
                    document.getElementById("identify_div").style.left = (e.pageX) + "px";
                    document.getElementById("identify_div").style.display = 'block';
                }
            }
        };

        window.ondblclick = function(e) {//双击隐藏按钮
            document.getElementById("identify_div").style.display = 'none';
        };
    }
    else{
        //移动端 监听长按事件
        let time_out = 0, touch_time = 800;
        window.addEventListener("touchstart", function(e){
            time_out = setTimeout(function(){
                var touchedElement = e.target;
                if ((g_img_src = GetQRSrc(touchedElement)) != null) {
                    document.getElementById("identify_div").style.top = (e.touches[0].pageY - 50) + "px";
                    document.getElementById("identify_div").style.left = (e.touches[0].pageX + 20) + "px";
                    document.getElementById("identify_div").style.display = 'block';
                }
            }, touch_time);//touch_time毫秒后弹出识别按钮

            //触碰其他地方则隐藏按钮
            if (e.target.id != 'identify_div'){
                g_img_src = null;
                document.getElementById("identify_div").style.display = 'none';
            }
        });
        window.addEventListener("touchmove", function(e){
            // 如果触摸未达到 touch_time ms且开始移动,则清除计时器
            clearTimeout(time_out);
            time_out = 0;
        });
        window.addEventListener("touchend", function(e){
            // 如果触摸未达到 touch_time ms且离开屏幕,则清除计时器
            clearTimeout(time_out);
            time_out = 0;
        });
    }

    //通过UA判断是否是移动端
    function isPhone() {
        var info = navigator.userAgent;
        var isPhone = /mobile/i.test(info);
        return isPhone;
    }

    //从传入的元素中获取IMG元素的src 若没有则返回null
    function GetQRSrc(ele){
        let ret_src = null;

        if (ele.tagName == 'IMG'){
            if (ele.src != null) ret_src = ele.src;
        }
        else if(ele.tagName == 'svg'){
            ret_src = svg2b64(ele);
        }else{
            //遍历子元素看看是否有图片
            var childs = ele.childNodes;
            if (childs.length >= 6) return ret_src;
            for(var i = childs.length - 1; i >= 0; i--) {
                if (childs[i].tagName == 'IMG' && childs[i].src != null){
                    ret_src = childs[i].src;
                    break;
                }
            }
        }

        return ret_src;
    }

    //将图片URL转为jsqr需要的数据
    function Img2CanvasData(img_url){
        return new Promise((resolve, reject) => {
            const canvas = document.createElement("canvas");
            const ctx = canvas.getContext("2d");
            canvas.width = 500;
            canvas.height = 500;
            const img = new Image();
            //img.crossOrigin = '';
            img.onerror = (err) => {
                notie.alert(3, 'Image Err:' + err, 4);
            };
            img.onload = () => {
                try{
                    ctx.drawImage(img, 0, 0,300,300);
                    let data = ctx.getImageData(0, 0, 300, 300).data;
                    resolve(data);
                }
                catch(e){
                    reject(e);
                }
            };
            img.src = img_url;
        });
    }

    //cb_ok\cb_failed识别成功\失败时的回调函数
    function IdentifyQRCode(data, cb_ok, cb_failed){
        Img2CanvasData(data).then( res=>{
            const code = jsQR(res, 300, 300, {
                inversionAttempts: "dontInvert",
            });
            if (code) cb_ok(code.data, data); else cb_failed('IdentifyError', data);
        }).catch(err=>{
            cb_failed(err.name, data);
        });
    }

    //'识别二维码'按钮事件
    function OnClickedIdentifyBtn(){
        if ((typeof jsQR != 'function')){
            notie.alert(2, '外部JS脚本未加载成功!', 3);
            return;
        }

        if (g_img_src == null){
            notie.alert(2, '未选中任何图片', 2);
        }
        else {
            IdentifyQRCode(g_img_src, callback_ok, function(msg, url_){
                if (msg == 'IdentifyError'){
                    notie.alert(3, '识别失败!', 3);
                }
                else if (msg == 'SecurityError'){
                    notie.alert(4, '等待跨域代理服务器返回图片数据<div>⋘ 𝑃𝑙𝑒𝑎𝑠𝑒 𝑤𝑎𝑖𝑡... ⋙</div>', 20);
                    let cros_proxy = 'https://api.allorigins.win/get?url=' + encodeURIComponent(url_);
                    fetch(cros_proxy).then(response => {
                        if (response.ok){
                            return response.json();
                        }else{
                            return notie.alert(3, '代理服务器返回Not ok!', 3);
                        }
                    }).then(data => {
                        if (data.contents.length == 0) {
                            notie.alert(3, '图片数据为空!', 3);
                        }else{
                            IdentifyQRCode(data.contents, callback_ok, function(){
                                setTimeout(()=>{
                                    notie.alert(3, '识别失败!', 3);
                                }, 500);//不等待500ms这个提示框会弹不出来,不知道为啥
                            });
                        }
                    });
                }
                else{
                    notie.alert(3, '未知错误!', 3);
                }
            });
        }
        document.getElementById("identify_div").style.display = 'none';
    }

    //'复制'按钮单击事件
    function OnClickedCopyBtn(){
        //GM_setClipboard(document.getElementById("notie_all_text").innerText);
        navigator.clipboard.writeText(document.getElementById("notie_all_text").innerText);
        notie.alert(1, '复制成功!', 2);
    }

    //判字符串是否可能为URL
    function isMaybeURL(str) {
        let is_maybe = false;
        if ((str.indexOf('://') != -1) ||
            (str.substr(0, 3) == 'www') ||
            (str.indexOf('.com') != -1 || str.indexOf('.cn') != -1 || str.indexOf('.org') != -1 || str.indexOf('.net') != -1) ||
            (str.indexOf('.') != -1 && str.indexOf('/') != -1)) is_maybe = true;
        return is_maybe;
    }

    function callback_ok(text, url_){
        let display_text = text, ope_html = '<div><a id="notie_copy_btn" href="javascript:void(0);">复制</a>[placeholder_]<a id="notie_close_btn" style="margin-left: 20px" href="javascript:void(0);">关闭</a></div>';
        if (g_is_moblie_env && (text.length >= 40)){//如果是移动端 且识别内容过长 则隐藏一部分内容
            display_text = text.substr(0, 35) + '...';
        }
        if (isMaybeURL(text)){
            let jump_url = text;
            if (jump_url.indexOf('://') == -1) jump_url = 'https://' + jump_url;
            ope_html = ope_html.replace("[placeholder_]", '<a id="goto_btn" href="' + jump_url + '" target="_blank" style="margin-left: 20px">转到</a>');
        }
        else{
            ope_html = ope_html.replace("[placeholder_]", '<a id="notie_search_btn" href="https://www.baidu.com/s?wd=' + encodeURIComponent(text) + '" target="_blank" style="margin-left: 20px">搜索</a>');
        }
        setTimeout(()=>{
            notie.alert(1, '<a id="notie_all_text" style="display: none;">'+ text +'</a>识别到以下文本:<br><div style="word-wrap:break-word;">' + display_text + '</div>' + ope_html, 20);
            setTimeout(()=>{
                document.getElementById("notie_copy_btn").onclick = OnClickedCopyBtn;
                document.getElementById("notie_close_btn").onclick = function(){notie.alert_hide();};
            }, 500);
        }, 300);

    }

    //svg转base64
    function svg2b64(svg_ele) {
        const s = new XMLSerializer().serializeToString(svg_ele);
        const ImgBase64 = `data:image/svg+xml;base64,${window.btoa(s)}`;
        return ImgBase64;
    }
})();