Greasy Fork is available in English.

Baidu:多引擎同屏

在百度搜索页面同时显示多个搜索引擎的结果

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==UserScript==
// @name         Baidu:多引擎同屏
// @namespace    https://greasyfork.org/zh-CN/users/1073-hzhbest
// @version      1.20
// @description  在百度搜索页面同时显示多个搜索引擎的结果
// @author       hzhbest
// @match        https://www.baidu.com/*
// @icon         https://www.baidu.com/favicon.ico
// @grant        GM_xmlhttpRequest
// @run-at       document-end
// @connect      www.google.com
// @connect      www.so.com
// @connect      www.sogou.com
// @connect      cn.bing.com
// @connect      www.google.com.hk
// @connect      s.weibo.com
// @connect      www.izito.com
// @license      MIT
// ==/UserScript==


(function () {

    if (window.top.document.location.href != window.self.document.location.href) return; // 在iframe中则不运行

    //针对百度不刷新页面更新搜索结果
    const targetselector = '#wrapper_wrapper';    // 选择需要观察变动的节点
    const config = { attributes: false, childList: true, subtree: false };    // 观察器的配置(需要观察什么变动)
    const callback = function (mutationsList, observer) {
        // log('观察到变动');    // 当观察到变动时执行的回调函数
        setTimeout(init, 1000);
    };
    const observer = new MutationObserver(callback);    // 创建一个观察器实例并传入回调函数

    function goObserve(target, config) {
        observer.observe(qElem(target), config);    //以配置对目标节点进行观察
    }

    var old_kw = document.location.href;            //记录原有网址,以检查网址是否变更
    setTimeout(init, 500);                          //首次运行
    setTimeout(goObserve, 600, targetselector, config); //随后开启观察器

    //判断并推进主进程,重新观察
    function init() {
        var thisurl = document.location.href;
        var kw = getUriParam(thisurl, 'wd') || getUriParam(thisurl, 'word');
        if (!kw || kw == old_kw) return;    // 仅当搜索关键词变更时继续
        observer.disconnect();  //停止观察
        // https://www.baidu.com/s?wd=AAAA&gpc=stf%3D1577808000%2C1610035200%7Cstftype%3D2
        const bd_timepara = /(?<=gpc\=stf\%3D)[\d\%C]+(?=\%7Cstftype\%3D2)/i;
        let arri = getds(thisurl, bd_timepara, '%2C');  // 提取网址中的时间数组
        var dateset;
        if (!!arri) {
            dateset = arri.map(ds => bd_numTOdate(ds)); // 有时间范围则引入到多引擎加载中
            go(kw, dateset);
        } else {
            go(kw);
        }
        old_kw = kw;
        setTimeout(() => {
            goObserve(targetselector, config);
        }, 5000);
    }

    //主进程
    function go(kw, dateset) {

        //  ===设置开始=========================================================================

        // > 是否只在第一页显示(true / false)
        var onlyPageOne = true;

        //  **以下一行并非设置项,请勿修改! //如果只在第一页显示则在非第一页的页面直接退出
        if (onlyPageOne && getUriParam(document.location.href, 'pn') > 0) return;

        // > 直接展示的搜索结果数量(个);一般小于等于10,因为一般搜索结果一页就10项
        var resultNumber = 3;

        // > 搜索结果的显示高度(像素);小于此高度的将按实际高度显示,大于此高度的在鼠标移上时展开;
        //     要想不收缩显示,将该值设得较大(例如1000)即可
        var disHeight = 120;

        // > 与主搜索结果的水平间距(像素)
        var lgap = 50;

        // > 获取搜索结果最长尝试(超时)时间(秒)。
        var resultTimeout = 30;

        var xEngs = [

            // > 外部搜索引擎(提取结果信息线索)
            // 说明: xEngs[x].name - 搜索引擎的名字(用作标识)
            //		  xEngs[x].enable - 启用开关,1-启用,0-禁用
            //		  xEngs[x].boxname - 同屏结果框元素的 ID
            //		  xEngs[x].url - 搜索引擎的搜索 Url,其中“--keyword--”用于替换实际搜索关键词
            //		  xEngs[x].resultcss - 指向每条搜索结果的 css selector
            //		  xEngs[x].em - 搜索引擎对关键词强调的 css selector
            //        xEngs[x].timepara - 搜索引擎自由指定始末时间的URL参数模版
            //        xEngs[x].timeformat - 用于将时间对象转换为始末时间格式的函数
            //		  x - 显示顺序。放在前面的搜索引擎靠上显示结果

            {
                name: 'Google',
                enable: 1,
                boxname: "gResult",
                url: 'http://www.google.com/search?q=--keyword--',
                resultcss: 'div#rso div.g',
                em: 'em',
                timepara: '&tbs=cdr%3A1%2Ccd_min%3A--sdate--%2Ccd_max%3A--edate--',
                timeformat: function (dateobj) {
                    let d = dateobj.getDate();
                    let m = dateobj.getMonth();
                    let y = dateobj.getFullYear();
                    return (m + 1) + '%2F' + d + '%2F' + y;
                }
            },
            {
                name: '360',
                enable: 1,
                boxname: "360Result",
                url: 'http://www.so.com/s?ie=utf-8&q=--keyword--',
                resultcss: '#main li.res-list',
                em: 'em'
            },
            {
                name: 'Bing',
                enable: 1,
                boxname: "bingResult",
                url: 'http://cn.bing.com/search?q=--keyword--',
                resultcss: 'li.b_algo',
                em: 'strong',
                timepara: '&filters=ex1%3a%22ez5_--sdate--_--edate--%22',
                timeformat: function (dateobj) {
                    const d0 = Date.parse('1970-1-1');
                    const msxd = 24 * 60 * 60 * 1000;
                    return (dateobj - d0) / msxd;
                }
            },
            {
                name: 'Sogou',
                enable: 1,
                boxname: "sogouResult",
                url: 'http://www.sogou.com/web?query=--keyword--',
                resultcss: 'div.results>div>div.struct201102',
                em: 'em'
            },
            {
                name: 'GoogleCN',
                enable: 0,
                boxname: "gcnResult",
                url: 'http://www.google.com.hk/search?q=--keyword--',
                resultcss: 'div#rso div.g',
                em: 'em',
                timepara: '&tbs=cdr%3A1%2Ccd_min%3A--sdate--%2Ccd_max%3A--edate--',
                timeformat: function (dateobj) {
                    let d = dateobj.getDate();
                    let m = dateobj.getMonth();
                    let y = dateobj.getFullYear();
                    return d + '%2F' + m + '%2F' + y;
                }
            },
            {
                name: 'Weibo',
                enable: 0,
                boxname: "weiboResult",
                url: 'https://s.weibo.com/weibo?q=--keyword--',
                resultcss: 'div#pl_feedlist_index>div>div.card-wrap'
            },
            {
                name: 'izito',
                enable: 0,
                boxname: "izitoResult",
                url: 'https://www.izito.com/search?q=--keyword--',
                resultcss: 'li.organic-results__item.organic-results-item',
                em: 'strong'
            }
        ]
        //  ===Config END | 设置结束=========================================================================


        var _ID = 'resultPlus';     //同屏结果主框架ID
        var _xID = '#' + _ID;

        var b;  //同屏结果主框架

        // 页面框架提取
        var bctn = qElem('div#container');          //主搜索结果容器;该容器未载入时继续等待
        if (!bctn) { setTimeout(go, 300); return; }
        var lcol = qElem('div#content_left');       //搜索结果条目容器;以此容器位置确定插入同屏结果位置
        var lmar = (!!lcol) ? (getX(lcol) + lcol.offsetWidth + lgap) : (window.innerWidth * 0.56);    //插入同屏结果的左边缘
        var b_width = Math.min(504, window.innerWidth - lmar - 30); //同屏结果的宽度

        // 样式表
        //同屏结果主框架位置大小和底色
        var bxstyle = _xID + '{position:absolute; top:100px; left:' + lmar + 'px; background:white; z-index:10; width:' + b_width + 'px;'
            + ' line-height: 130%; border-bottom:1px solid #AACCFF;}';
        //分引擎框架样式
        var xestyle = _xID + ' .xe{border-top:1px solid #7799cc; padding: 1px 3px 3px 4px; '
            + 'background:linear-gradient(5deg, rgba(170, 204, 255,0.1) 0%, rgba(255, 255, 255,0.7) 100%),linear-gradient(0deg, rgba(255,255,255,0.65) 0%, rgba(127, 127, 203,0.3) 10%, rgb(170, 204, 255) 100%);}';
        //分引擎名称样式
        var exstyle = _xID + ' ._external {color: #6868b0 !important;}'
            + _xID + ' ._external:after {content:url("");}';
        //关闭按钮样式
        var clstyle = _xID + ' .close{position: absolute; top: 0; right: 0; padding: 0 10px;}'
            + _xID + ' .close:hover{outline: 1px solid #731616; outline-offset: -1px; background-color: #F28E8E!important; color: #731616!important;}';
        //刷新链接样式
        var rfstyle = _xID + ' ._refresh{display: none; margin: 0 10px;}'
            + _xID + ' .xe:hover ._refresh{display: inline-block;}'
        //同屏搜索结果通用样式
        var glo_style = 'body{overflow: auto !important; min-height: 1200px;}'  /*不加这个在百度自己搜索结果太少时会使同屏结果显示不全 */
            + _xID + ' p, ' + _xID + ' ul {margin: 0; padding:0;}'
            + _xID + ' li{list-style: none outside none;}'
            + _xID + ' a{color: #2626A8;}'
            + _xID + ' a div>div{display: inline;}' /*链接中的图片之类*/
            + _xID + ' svg{height: 14px !important; width: 14px !important;}' /*svg图片限制大小;通常是图标*/
            + _xID + '>div>*{height: auto!important;}' /*收缩搜索结果自带高度*/
            /*同屏结果框架样式*/
            + _xID + ' div._result{max-height:' + disHeight + 'px; margin-bottom:5px; background:white; overflow:hidden; transition: height, max-height 2.5s ease-out 1s;}'
            + _xID + ' div._result:hover{max-height: 100000px; margin-bottom:0px; padding-bottom:5px; background: #F0F7F9; transition: height, max-height 0.2s ease-out; outline: 1px solid #82BDCE;}'
            + _xID + ' div._result h3, ' + _xID + ' div._result h2{font-size:13pt!important; font-weight: 800; border-bottom: 1px solid white; margin-bottom:2px;}'
            + _xID + ' div._result:hover h3, ' + _xID + ' div._result:hover h2{border-bottom: 1px solid #a7cDd6;}'
            /*更多结果框架样式*/
            + _xID + ' ._resultMore{max-height: none!important;}'
            + _xID + ' ._re_hide>div,' + _xID + ' ._no_result{display:none;}'
            + _xID + ' ._re_more{display: block; height: 20px; width: 100%; text-align: center; background: #ddd; cursor: pointer;}'
            + _xID + ' ._re_more:hover{background: #F0F7F9!important;}'
            + _xID + ' ._re_hide ._re_more{background: white;}'
            + _xID + ' div._result+div._result,' + _xID + ' ._resultMore{border-top: 1px solid #aaccff;}';
        //应付搜索结果特殊元素变样的样式
        var tablestyle = _xID + ' td{padding: 5px 0 5px 13px !important; color: #000!important; width: fit-content !important;}'
            + _xID + ' tr+tr>td{padding: 0px 0 8px 13px !important;}' + _xID + ' td>h3{margin-left: -8px; line-height: 1.3em;}';
        var li_style = _xID + ' div._result>li {padding: 5px 0 8px 13px !important; background-image: none;}'
            + _xID + ' li h3>a:first-child,' + _xID + ' li h3>em {font-size: 13pt !important; margin-left: -8px;}'
            + _xID + ' li>div, ' + _xID + ' li>p {font-size: small;}';
        //特殊结果:与主搜索结果相同
        var mat_style = _xID + ' div._match {background: #eee; background:linear-gradient(to bottom, #eee, white); max-height:1.1em;}' + _xID + ' div._match:hover{max-height: 1000px;}'
            + '._hilire{background: #ffd!important; background: linear-gradient(to bottom, #ffd, white)!important;}'
        //同屏搜索结果-引擎专属样式
        var _w;
        var gg_style = /*Google style*/'div.yuRUbf>div>a>br,div.yuRUbf .B6fmyf,img.wA1Bge,*[role="button"]{display: none;} '
            + 'span.VuuXrf{margin: 0 3px; display: inline; background: #e9e9ffed; width: fit-content; outline: 1px solid #bac4e0; border-radius: 1em; padding: 1px 0.5em;} '
            + 'div.g a>h3+div>*,span.H9lube>div,div.byrV5b,div.P58Kof{display: inline-block;} '
            + 'div.g{margin-bottom: 0 !important; width:' + b_width + 'px !important;} .VwiC3b,.VNLkW {padding: 0 1em;}'
            + 'h3.DKV0Md{margin-top: 5px !important; padding-top: 0 !important;} '
            + 'div[class="wrBvFf OSrXXb"]{font-weight: 800;}'
            + 'div.zVvuGd{display: grid; grid-template-columns: auto auto auto auto auto auto auto auto auto;}';
        _w = 'div[id^="bingResult_"] ';
        var bg_style = /*Bing style*/ _w + '.crch,' + _w + '.sb_tsuf{display:none!important;}'
            + _w + 'h2{margin:5px 0;}' + _w + '.inner,' + _w + '.HeroTab{display: none;}'
            + _w + 'div.tptt{background: #e9e9ffed; width: fit-content; outline: 1px solid #bac4e0; border-radius: 1em; padding: 1px 0.5em;}';
        _w = 'div[id^="sogouResult_"] ';
        var sg_style = /*Sogou style*/ _w + '>div {padding: 7px 0 8px 13px !important;background-image:none;} '
            + _w + '>div>h3 {margin-left:-8px;} '
            + _w + '.img-height{float: right;} '
            + _w + '.tit-ico {background-position: left 1px;background-repeat: no-repeat;padding-left: 20px;}'
            + _w + '.item-list{display: grid; grid-template-columns: auto auto auto auto auto auto auto auto auto;}'
            + _w + '.citeurl{position: sticky; bottom: 3px; background: #e9e9ffed; width: fit-content; outline: 1px solid #bac4e0; border-radius: 1em; padding: 1px 0.5em;}';
        _w = 'div[id^="360Result_"] ';
        var sz_style = /*360 style*/ _w + 'h3>a>img {width:16px !important; height:16px !important; float: none !important;}'
            + _w + 'img {float: right;}'
            + _w + 'li:hover div>a>img {float: none;}'
            + _w + '.g-linkinfo{position: sticky; bottom: 3px; background: #e9e9ffed; width: fit-content; outline: 1px solid #bac4e0; border-radius: 1em; padding: 1px 6px !important;}';
        _w = 'div[id^="weiboResult_"] ';
        var wb_style = /*Weibo style*/ _w + 'div.menu,' + _w + '.WB_video_h5 {display: none;} '
            + _w + 'div.media{max-height: 200px; overflow: hidden;}' + _w + 'div.media:hover{overflow: auto; scrollbar-width: thin;} '
            + _w + 'div.avator{width: fit-content; float: left;} '
            + _w + 'div.avator img{float: left; margin: 3px; height:32px; width:32px;} '
            + _w + 'img.face{height:18px !important; width:18px !important;} '
            + _w + '.card-act{position: sticky; bottom: 3px; left: 300px; background: #e9e9ffed; width: fit-content; border-top: 1px solid #bac4e0;}'
            + _w + '.woo-font--retweet{margin-left: 1em; display: inline-block; height: 16px !important; content:url("");} '
            + _w + '.woo-font--comment{margin-left: 1em; display: inline-block; height: 16px !important; content:url("");} '
            + _w + 'button{border: 0; background: transparent; margin-left: 1em;}'
            + _w + '.card-top{float: right; margin-right: 6px;}' + _w + '.card-top h4{margin: 0;}'
            + _w + 'ul{display: grid; grid-template-columns: auto auto auto auto auto auto auto auto auto;}' + _w + 'ul>li{display: inline;} ' + _w + 'ul>li img{top:0 !important;}'
            + _w + '.thumbnail{min-height:0 !important;} ' + _w + '.card-wrap{padding: 10px;} ' + _w + '.info{float: left;}' + _w + '.info span{vertical-align: middle;}'
            + _w + '.from{width: 150px; float: right; height: 1.3em; overflow: hidden;}' + _w + '.from:hover{overflow: visible; background: #e3f5f1;}'
            + _w + '.txt{clear: both;}';


        var hl_styles = new Array();    //高亮样式数组
        var xEng_boxes = new Array();   //同屏搜索结果分引擎框架

        for (const i in xEngs) {
            if (xEngs[i].enable == 1) {
                xEngs[i].url = xEngs[i].url.replace('--keyword--', kw);        //替换搜索关键词到引擎网址
                console.log('xEngs[i].url: ', xEngs[i].url);
                if (!!xEngs[i].timepara && !!dateset) {                        // 若多引擎支持时间范围且主引擎使用了时间范围
                    let sdate = xEngs[i].timeformat(dateset[0]);
                    let edate = xEngs[i].timeformat(dateset[1]);
                    let timepara = xEngs[i].timepara.replace('--sdate--', sdate).replace('--edate--', edate);
                    xEngs[i].url += timepara;
                    xEngs[i].dateset = "[ " + dateTOymd(dateset[0]) + " - " + dateTOymd(dateset[1]) + " ] ";
                }
            }
            if (!!xEngs[i].em) {
                hl_styles[i] = _xID + ' div[id^="' + xEngs[i].resultcss + '_"] ' + xEngs[i].em;
            }   //形成高亮样式
        }
        var hili_style = hl_styles.join(',') + '{color: #CC0033 !important; background: #fffcec;}';


        // 插入 CSS
        var headID = document.getElementsByTagName("head")[0];
        var cssNode = creaElemIn('style', headID);
        cssNode.type = 'text/css';
        cssNode.id = '__xEngs';
        cssNode.innerHTML = hili_style + bxstyle + xestyle + clstyle + rfstyle + glo_style + exstyle + tablestyle + li_style + mat_style + gg_style + bg_style + sg_style + sz_style + wb_style;

        // 提取百度搜索结果
        var resls = qaElem('#content_left>div[class*="result"][id]:not([tpl="recommend_list"])');        //百度各条搜索结果
        var bdRes = [], bdResUrls = [], bdResNo = 0;
        // var speIDs = /imagebox_bigimages|imagebox|newsbox|videobox|blogbox/;
        for (let h = 0; h < resls.length; h++) {
            // if (speIDs.test(resls[h]).id) continue;
            var bdlnk = qElem('h3>a', resls[h]) || qElem('section a', resls[h]);
            if (!bdlnk) continue;
            bdResUrls.push(bdlnk.href.toLowerCase());               //提取主链接网址推入结果网址数组
            bdResNo++;
            resls[h].title = '第 ' + bdResNo + ' 结果';
            bdRes.push(resls[h]);                                   //推入结果数组
        }

        // 准备同屏搜索结果主框架
        b = creaElemIn('div', document.body);
        b.id = _ID;
        // 构建关闭按钮
        var close = creaElemIn('a', b);
        close.href = '#';
        close.className = 'close';
        close.addEventListener('click', () => {
            headID.removeChild(cssNode);
            document.body.removeChild(b);
        }, false);
        close.innerHTML = 'X';
        close.title = '关闭';

        // 按启用的多引擎构建结果框并获取搜索结果填入
        for (const i in xEngs) {
            if (xEngs[i].enable == 1) {
                xEng_boxes[i] = new Array();
                resultbox(b, xEng_boxes[i], xEngs[i], resultNumber);    //构建同屏各搜索引擎结果框架
                addresult(xEng_boxes[i], xEngs[i], resultNumber);       //插入搜索结果节点
            }
        }


        // FUNCTIONS

        // 构建一个搜索引擎结果框架
        function resultbox(root, box, engObj, rescnt) { //root:供插入引擎结果的主框架,box:供写入引擎结果的盒框架,engArr:供写入引擎标志的信息数组,rescnt:先显的结果数
            var c, d, e, f, r;
            //引擎名称框
            c = creaElemIn('div', root);
            c.className = 'xe';
            c.id = engObj.boxname;
            //引擎跳转链
            d = creaElemIn('a', c);
            addtext(d, engObj.name + ' ' + (engObj?.dateset || ''));
            d.href = engObj.url;
            d.target = '_blank';
            d.className = '_external';
            // 引擎刷新链
            r = creaElemIn('a', c);
            r.className = '_refresh';
            addtext(r, "刷新结果");
            r.addEventListener('click', () => {
                addresult(box, engObj, rescnt);
            }, false);
            //先显结果框
            for (let k = 0; k < rescnt; k++) {
                box[k] = creaElemIn('div', root);
                box[k].id = engObj.boxname + '_' + (k + 1);
                box[k].innerHTML = (k == 0) ? '正在获取结果...' : '...';
            }
            //后显结果框
            if (rescnt == 10) return;
            e = creaElemIn('div', root);
            e.className = '_resultMore _re_hide';
            e.id = engObj.boxname + 'More';
            for (let k = rescnt; k < 10; k++) {
                box[k] = creaElemIn('div', e);
                box[k].id = engObj.boxname + '_' + (k + 1);
            }
            //后显结果展开按钮
            f = creaElemIn('a', e);
            f.className = '_re_more';
            f.innerHTML = '↓展开↓';
            f.title = '展开更多结果';
            f.href = '#' + engObj.boxname;
            f.addEventListener('click', function (ev) {
                ev.preventDefault();
                var s = (this.parentNode.className == '_resultMore _re_hide');
                this.parentNode.className = (s) ? '_resultMore ' : '_resultMore _re_hide';
                this.innerHTML = (s) ? '↑收起↑' : '↓展开↓';
                this.title = (s) ? '收起更多结果' : '展开更多结果';
            }, false);
        }

        // 发送请求并转递返回数据
        function addresult(box, engObj, rescnt) { //box:供写入引擎结果的盒框架,engArr:供写入引擎标志的信息数组,rescnt:先显的结果数
            var timeout = function () {
                for (const i in box) {
                    set(box[i], (i == 0) ? '* 获取结果超时 *' : '');
                }
            };
            var errortimer = setTimeout(timeout, resultTimeout * 1000);

            var option = {
                method: "GET",
                url: engObj.url,
                onload: function (_h) {
                    clearTimeout(errortimer);
                    var _Node = document.createElement('div');
                    _Node.innerHTML = _h.responseText;              //包含整个页面的div
                    initresult(_Node, box, engObj.name, engObj.resultcss, rescnt);
                }
            }
            GM_xmlhttpRequest(option);
        }

        // Initialize results
        function initresult(_Node, box, engName, resSelector, rescnt) { //_Node:需处理的搜索结果,box:供写入引擎结果的盒框架,engName:引擎名称,resSelector:从搜索结果中提取单条搜索结果的选择器,rescnt:先显的结果数
            var _result = [], _resultLinkHref;
            var no_result_con = "* 无返回结果,请尝试直接访问 *";
            for (let i = 0; i < 10; i++) {
                _result[i] = (i == 0) ? no_result_con : '';      //初始化搜索结果临时容器
            }
            var _ns = qaElem(resSelector, _Node);               //按选择器提取的搜索结果数组
            for (let i = 0, j = 0; i < 12; i++) {                   //遍历搜索结果数组,排除特殊内容,放入临时容器,以j为索引
                var _n = _ns[i];
                //log(engName+" i "+i);
                if (_n == null) { //log(engName+" j "+j);
                    if (j <= rescnt && rescnt != 10) box[rescnt].parentNode.className = '_no_result';   //若无结果需要放入后显结果框,则包含展开按钮的后显结果鵟直接不显示
                    break;
                } else {  //log(engName+" j "+j);

                    //GCN url fix
                    if (engName == 'GoogleCN') fixgcn(_n);

                    //Google url fix
                    if (engName == 'Google') fixgl(_n);

                    //360 img flx
                    if (engName == '360') fixso(_n);

                    //Bing img url fix
                    if (engName == 'Bing') {
                        fixbmg(_n);
                        if (!_n.firstChild || (!_n.firstChild.firstChild && !_n.firstChild.nextSibling) || (_n.firstChild.localName == 'script' && !_n.firstChild.nextSibling)) continue;
                    }

                    //链接都在新标签页打开
                    var _resultLinks = _n.getElementsByTagName('a');
                    for (const o in _resultLinks) {
                        if (!!_resultLinks[o].href && /\:\/\//.test(_resultLinks[o].href)) {
                            _resultLinks[o].target = '_blank';
                        }
                    }

                    //处理图片框架
                    var _imgs = _n.getElementsByTagName('img');
                    for (const o in _imgs) {
                        if (!!_imgs[o].src && _imgs[o].src.indexOf('data:') == 0) {
                            _imgs[o].parentNode.style = 'width: fit-content; height: fit-content;';
                        }
                    }

                    //将处理后的结果放入临时容器
                    _result[j] = getoutterHTML(_n);

                    //检查同屏引擎搜索结果的主链接和主引擎搜索结果,找出同归结果
                    if (!_resultLinks[0]) continue;     //跳过无链接的结果
                    if (!!_resultLinks[0].href) {
                        _resultLinkHref = _resultLinks[0].href.toLowerCase();
                        for (const p in bdResUrls) {
                            if (_resultLinkHref == bdResUrls[p] || _resultLinkHref + '/' == bdResUrls[p]) {  //deal with bing's result url
                                box[j].className = '_match';
                                _result[j] = '同 百度 第 <b>' + (Number(p) + 1) + '</b> 结果' + _result[j];
                                if (bdRes[p].className.indexOf('_hilire') == -1) {
                                    bdRes[p].className += ' _hilire';
                                    bdRes[p].title += ';同时为 ' + engName + ' 第 ' + (j + 1) + ' 结果';
                                } else {
                                    bdRes[p].title += '及 ' + engName + ' 第 ' + (j + 1) + ' 结果';
                                }
                            }
                        }
                    }
                    j++;
                    if (j == 10) break;
                }
            }
            for (let i = 0; i < 10; i++) {      //将临时结果转入结果框
                set(box[i], _result[i]);
                if (_result[i]) box[i].className += ' _result';
            }
        }


        // 修复360懒图片
        function fixso(_resultcontent) {
            var imgs = qaElem('.so-lazyimg', _resultcontent);
            if (!imgs.length) return;
            for (const i in imgs) {
                if (!!imgs[i].dataset && !!imgs[i].dataset.isrc) imgs[i].src = imgs[i].dataset.isrc;
            }
        }

        // "修复"GoogleCN链接
        function fixgcn(_resultcontent) {
            var Links = qaElem('a', _resultcontent);
            for (const i in Links) {
                if (Links[i].href && Links[i].href.match(/^http:\/\/www\.google\..*\/url\?q=/i)) {
                    Links[i].href = Links[i].href.replace(/^http:\/\/www\.google\..*\/url\?q=(.*?)&.*/i, "$1");
                }
            }
        }

        // "修复"Google链接
        function fixgl(_resultcontent) {
            var Links = qaElem('a', _resultcontent);
            for (const i in Links) {
                if (Links[i].href && Links[i].href.indexOf('/') == 0) {
                    Links[i].href = 'https://www.google.com/' + Links[i].href.slice(1);
                }
            }
        }

        // 修复Bing图片地址
        function fixbmg(_resultcontent) {
            var Imgs = qaElem('img', _resultcontent);
            for (const i in Imgs) {
                if (Imgs[i].src && Imgs[i].src.indexOf('http://') != 0 && Imgs[i].src.indexOf('data:') != 0) {
                    Imgs[i].src = 'http://www.bing.com' + Imgs[i].src;
                }
            }
        }
    }


    // Create and insert an element
    function creaElemIn(tagname, destin) {
        var theElem = destin.appendChild(document.createElement(tagname));
        return theElem;
    }
    
    // Set content
    function set(elem, str) {
        elem.innerHTML = str;
    }

    // Get full HTML nodes in string
    function getoutterHTML(elem) {
        var a = elem.attributes, str = "<" + elem.tagName, i = 0; for (; i < a.length; i++)
            if (a[i].specified) str += " " + a[i].name + '="' + a[i].value + '"';
        if (!canHaveChildren(elem)) return str + " />";
        return str + ">" + elem.innerHTML + "</" + elem.tagName + ">";
    }
    function canHaveChildren(elem) {
        return !/^(area|base|basefont|col|frame|hr|img|br|input|isindex|link|meta|param)$/.test(elem.tagName.toLowerCase());
    }

    // Add text node
    function addtext(obj, text) {
        var content = document.createTextNode(text);
        obj.appendChild(content);
    }

    // scroll node into view
    function scrTo(node) {
        if (!node) return;
        if (node.getBoundingClientRect) {
            var pos = node.getBoundingClientRect();
            /*var pos_h = node.offsetHeight;*/
            document.documentElement.scrollTop = document.body.scrollTop = pos.top + window.pageYOffset - window.innerHeight / 10;
        } else {
            node.scrollIntoView();
        }
    }

    //从url中提取日期数数组[qiri,vsri]
    function getds(url, pattern, separater) {
        let ds = pattern.exec(url);
        if (!!ds) {
            let outs = ds[0];	//正则匹配出的是结果的“数组”,因此第一匹配结果需要是数组index 0的元素
            return outs.split(separater);
        } else {
            return false;
        }
    }

    //日数转换为日期对象
    function bd_numTOdate(onum) {
        return new Date(onum * 1000);
    }

    // 将日期对象转成“YYYY-MM-DD”格式
    function dateTOymd(odate) {
        return odate.getFullYear() + "-" + twodg(odate.getMonth() + 1) + "-" + twodg(odate.getDate());
    }

    // 补足两位数字
    function twodg(s) {
        if (s.length == 1) {
            s = "0" + s;
        }
        return s;
    }

    // 从url中提取名为【param】的参数的值;提取不到返回空字符
    function getUriParam(surl, param) {
        var result = surl.match(new RegExp("(\\?|&)" + param + "(\\[\\])?=([^&]*)"));
        return result ? result[3] : '';
    }

    function log() {	//debug
        var args = Array.prototype.slice.call(arguments);
        console.log.apply(console, args);
    }

    function qElem(cssSelector, root) {
        let _r = root || document;
        return _r.querySelector(cssSelector);
    }

    // get all elements by css selector
    function qaElem(cssSelector, root) {
        let _r = root || document;
        return _r.querySelectorAll(cssSelector);
    }

    // get the accumulative offsetleft of target element
    function getX(oElement) {
        var iReturnValue = 0;
        while (oElement != null) {
            iReturnValue += oElement.offsetLeft;
            oElement = oElement.offsetParent;
        }
        return iReturnValue;
    }


})();