千山助手

为站点千山导航添加一些实用功能,比如搜索框、更换背景和导入备份等

// ==UserScript==
// @name          千山助手
// @namespace
// @description   为站点千山导航添加一些实用功能,比如搜索框、更换背景和导入备份等
// @include       http://qianshan.co/*
// @include       http://*.qianshan.co/*
// @version       3.0
// @copyright     2018,11,01; By duke
// @github        https://github.com/DukeLuo/qianshan-helper.user.js
// @namespace https://github.com/DukeLuo/QianShan
// ==/UserScript==


;(function () {

// add html
var $styleBody = $("style[type='text/css']"),
    $body = $("body");

var cssStr = "";

cssStr += '.my-mb {  margin-bottom: 10px;}';
cssStr += '.my-btn-circle {  position: absolute;  top: 1%;  right: 1%;  width: 70px;  height: 70px;  padding: 16px;  border-radius: 35px;  text-align: center;  font-size: 12px;  line-height: 1.4;  z-index: 5;  outline: none !important;  box-shadow: 0 0 2px #555;}';
cssStr += '#search-bar {float: left; width:480px; padding: 4px 0 5px 10px; }';
cssStr += '#my-btn-search1,#my-btn-search2,#my-btn-search3 {  height: 46px;  background-color: #d3d7d4;}';
cssStr += '#my-btn-search1:focus, #my-btn-search1:active,#my-btn-search2:focus, #my-btn-search2:active,#my-btn-search3:focus, #my-btn-search3:active,#search-bar input[type="text"] {  outline: none !important;  box-shadow: none;}';
cssStr += '#my-btn-search1 {  width: 48px;  padding: 5px 6px;  border-top-left-radius: 23px;  border-bottom-left-radius: 23px;  cursor: default;}';
cssStr += '#my-btn-search3 {  width: 60px;}';
cssStr += '#search-bar .dropdown-menu {  height: 50px;  padding-left: 10px;  padding-right: 10px;  background-color: #f4f4f4;}';
cssStr += '#search-bar .dropdown-menu i {  display: inline-block;  width: 34px;  height: 34px;}';
cssStr += '#search-bar .dropdown-menu i:hover {  cursor: pointer;}';
cssStr += '#search-bar .dropdown-menu i + i {    margin-left: 8px;}';
cssStr += '.icon {  display: inline-block;  width: 100%;  height: 100%;  background-color: #ddd;  background-position: center;  background-repeat: no-repeat;}';
cssStr += '.menu {  background-color: transparent;  background-image: url("");}';
cssStr += '.google {  background-color: transparent;  background-image: url("");}';
cssStr += '.bing {  background-color: transparent;  background-image: url("");}';
cssStr += '.duckduckgo {  background-color: transparent;  background-image: url("");}';
cssStr += '.baidu {  background-color: transparent;  background-image: url("");}';
cssStr += '.search {  background-color: transparent;  background-image: url("");}';
cssStr = '<style type="text/css">' + cssStr + '</style>';

$(cssStr).insertAfter($styleBody);

var mainHtmlStr, $main,
    searchBarHtmlStr, $searchBar,
    $menuModal;

mainHtmlStr = '<div class="container"><button type="button" class="btn btn-default my-btn-circle" data-toggle="modal" data-target="#menu-context"><i class="icon menu"></i></button><div class="modal fade" id="menu-context"><div class="modal-dialog"><div class="modal-content"><div class="modal-header"><button type="button" class="close" data-dismiss="modal">&times;</button><h4 class="modal-title">设置</h4></div><div class="modal-body" style="padding: 15px;"><div class="my-mb" id="o1"><div class="row"><div class="col-sm-8"><h4 style="margin: 0; height:100%; line-height:200%;">1. 搜索框</h4></div><div class="col-sm-4"><div class="btn-group dropright"><button type="button" class="btn btn-primary" id="search-bar-btn">关闭</button><button type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown"><span class="caret"></span></button><ul class="dropdown-menu" role="menu"><li><a href="#">关闭</a></li><li><a href="#">打开</a></li></ul></div></div></div></div><div class="my-mb" id="o2"><div class="row"><div class="col-sm-8"><h4 style="margin: 0; height:100%; line-height:200%;">2. 更换背景</h4></div><div class="col-sm-4"><div class="btn-group dropright"><button type="button" class="btn btn-primary" id="bg-btn">关闭</button><button type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown"><span class="caret"></span></button><ul class="dropdown-menu" role="menu"><li><a href="#">关闭</a></li><li><a href="#">打开</a></li></ul></div></div></div></div><div class="my-mb" id="o3"><div class="row"><div class="col-sm-12"><h4>3. 备份与恢复</h4></div></div><div class="row"><div class="col-sm-7"><div class="input-group my-input-file" name="filename"><input type="text" class="form-control" placeholder="选择文件..." disabled><span class="input-group-btn"><button type="button" class="btn btn-danger" disabled>取消</button></span></div></div><div class="col-sm-1"></div><div class="col-sm-4"><button type="button" class="btn btn-primary" id="file-btn" data-sign="0">导出</button></div></div></div></div></div></div></div></div>';
searchBarHtmlStr = '<form class="input-group" id="search-bar" method="get" target="_blank" action="https://www.google.com/search"><div class="input-group-btn"><button type="button" class="btn btn-lg" id="my-btn-search1"><i class="icon google"></i></button><button type="button" class="btn dropdown-toggle" id="my-btn-search2" data-toggle="dropdown"><span class="caret"></span></button><div class="dropdown-menu dropdown-menu-right"><i class="drop-item icon duckduckgo" id="duckduckgo"></i><i class="drop-item icon baidu" id="baidu"></i><i class="drop-item icon bing" id="bing"></i><i class="drop-item icon google" id="google"></i></div></div><input type="text" class="form-control input-lg" autocomplete="off" placeholder="搜索" name="q"><div class="input-group-btn"><button type="button" class="btn btn-lg" id="my-btn-search3"><i class="icon search"></i></button></div></form>';
$main = $(mainHtmlStr);
$searchBar = $(searchBarHtmlStr);
$menuModal = $main.find("#menu-context");
$body.append($main);


// search bar
var engineConfig = {
  google: {
      name: "q",
      action: "https://www.google.com/search"
  },
  bing: {
      name: "q",
      action: "https://cn.bing.com/search"
  },
  duckduckgo: {
      name: "q",
      action: "https://duckduckgo.com/"
  },
  baidu: {
      name: "wd",
      action: "https://www.baidu.com/s"
  }
};

var $engineWrapper = $searchBar.find(".dropdown-menu"),
    $engineIcon = $searchBar.find("#my-btn-search1 i"),
    $inputField = $searchBar.find("input[type='text']"),
    $submitBtn = $searchBar.find("#my-btn-search3");

$engineWrapper.on("click", function (evt) {
    var clickedItemName = $(evt.target).attr("id"),
        config = JSON.parse(localStorage.getItem("GLOBAL")) || globalConfig;

    if (!clickedItemName && !config["isSearchBarOpen"]) {
        return ;
    }
    $searchBar.attr("action", engineConfig[clickedItemName]["action"]);
    $inputField.attr("name", engineConfig[clickedItemName]["name"]);
    $engineIcon.attr("class", "icon "+clickedItemName);
    config["searchEngine"] = clickedItemName;
    localStorage.setItem("GLOBAL", JSON.stringify(config));
});

$inputField.on("focus", function (evt) {
    $(this).on("keydown", function (evt) {
        evt.stopPropagation();
    });
});

$submitBtn.on("click", function  (evt) {
    if (!$inputField.val()) {
        return ;
    }
    $searchBar.trigger("submit");
});

var $searchBarBtn = $("#o1 #search-bar-btn");

$("#o1 .dropdown-menu").on("click", function (evt) {
    var sign = $(evt.target).text() === "打开" ? 1 : 0,
        config = JSON.parse(localStorage.getItem("GLOBAL")) || globalConfig;

    $searchBarBtn.text($(evt.target).text());
    config["isSearchBarOpen"] = sign;
    if (sign) {
        $searchBar.insertAfter($body.find(".content .navbar .navbar-header"));
    }  else {
        $searchBar.detach();
        config["searchEngine"] = "google";
    }
    localStorage.setItem("GLOBAL", JSON.stringify(config));
});


// change background
var $bgBtn = $("#o2 #bg-btn");

$("#o2 .dropdown-menu").on("click", function (evt) {
    var sign = $(evt.target).text() === "打开" ? 1 : 0,
        config = JSON.parse(localStorage.getItem("GLOBAL")) || globalConfig;;

    $bgBtn.text($(evt.target).text());
    config["isBgSet"] = sign;
    if (sign) {
        $(".input-ghost", $menuModal).one("change", function (evt) {
            var fr = new FileReader(),
                file = $(this).prop("files")[0];

            $textInput.val("");
            if (!/image*/.test(file.type)) {
                console.error("ERROR! Please select an image!");
                config["isBgSet"] = 0;
                $bgBtn.text("关闭");
                localStorage.setItem("GLOBAL", JSON.stringify(config));
                return ;
            }
            fr.onload = function (evt) {
                $body.css("backgroundImage", 'url("'+fr.result+'")');
                config["bgData"] = fr.result;
                localStorage.setItem("GLOBAL", JSON.stringify(config));
            };

            fr.readAsDataURL(file);
        });
        $(".input-ghost", $menuModal).trigger("click");
    }  else {
        $body.css("background-image", 'url("https://qianshan.sfo2.digitaloceanspaces.com/mountain.jpg")');
        config["bgData"] = "";
        localStorage.setItem("GLOBAL", JSON.stringify(config));
    }
});


// backup and restore
var $myInputFileDiv = $("#o3 .my-input-file"),
    $textInput = $myInputFileDiv.find("input[type='text']"),
    $clearBtn = $myInputFileDiv.find("button.btn-danger"),
    $fileBtn = $("#o3 #file-btn");

function createInputFileGhost() {
    var $newIF = $("<input type='file' class='input-ghost' style='display: none;'>");

    if ($menuModal.find(".input-ghost").length) {
        return ;
    }
    $newIF.attr("name", $myInputFileDiv.attr("name"));
    $newIF.on("change", function (evt) {
        $textInput.val($newIF.val().split("\\").pop());
    });
    $textInput.css("cursor", "pointer");
    $textInput.on("click", function (evt) {
        $newIF.click();
    });
    $clearBtn.on("click", function (evt) {
        $newIF.val("");
        $textInput.val("");
    });
    return $newIF;
}

$myInputFileDiv.on("mouseover", function (evt) {
    $textInput.removeAttr('disabled');
    $clearBtn.removeAttr('disabled');
    $fileBtn.text("导入");
    $fileBtn.attr("data-sign", "1");
});

$myInputFileDiv.on("mouseout", function (evt) {
    if ($textInput.val()) {
        return ;
    }
    $(".input-ghost", $menuModal).val("");
    $textInput.attr('disabled','disabled');
    $clearBtn.attr('disabled','disabled');
    $fileBtn.text("导出");
    $fileBtn.attr("data-sign", "0");
});

function getConfig(block) {
    var $block = $(block),
        $grids = $("div.website", $block),
        ret = {},
        configOfAllGrids = [],
        i;

    ret["title"] = $("div.block-name", $block).text().trim();
    for (i = 0; i < $grids.length; i++) {
        var siteObj = {};

        siteObj["name"] = $("a span", $grids[i]).text();
        siteObj["href"] = $("a", $grids[i]).attr("href");
        configOfAllGrids.push(siteObj);
    }
    ret["websites"] = configOfAllGrids;
    return ret;
}

function setConfig(block, configObj) {
    var $block = $(block),
        $grids = $("div.website", $block),
        webs = configObj["websites"],
        blockId = $block.parent().index() + 1 + "",
        categoryInStorage = JSON.parse(localStorage.getItem(blockId)) || {},
        $website, websiteInStorage,
        i;

    $("div.block-name a", $block).text(configObj["title"]);
    categoryInStorage.category_id = blockId;
    categoryInStorage.category_name = configObj["title"];
    localStorage.setItem(blockId, JSON.stringify(categoryInStorage));
    for (i = 0; i < webs.length; i++) {
        $website = $($grids[i]);
        $("a span", $website).text(webs[i]["name"]);
        $("a", $website).attr("href", webs[i]["href"]);
        websiteInStorage = JSON.parse(localStorage.getItem(blockId+(i+1))) || {};
        websiteInStorage.website_id = blockId + (i + 1);
        websiteInStorage.website_name = webs[i]["name"];
        websiteInStorage.website_link = webs[i]["href"];
        websiteInStorage.website_hotkey = [];
        websiteInStorage.website_sibling_name = $website.parents('div.building').find('a.website').text();
        websiteInStorage.website_sibling_id = $website.parents('div.building').find('a.website').attr('id');
        localStorage.setItem(blockId+(i+1), JSON.stringify(websiteInStorage));
    }
}

function configJSONFileDownload(content, filename) {
    var a = "<a style='display: none;' download='" + filename +"'></a>",
        blob = new Blob([content], {type : 'application/json'}),
        $a = $(a);

    $a.attr("href", URL.createObjectURL(blob));
    $a.appendTo($menuModal);
    $a[0].click();
    $a.remove();
}

function localJSONFileHandler(file) {
    var tempFR = new FileReader();

    return new Promise(function (resolve, reject) {
        tempFR.onload = function () {
            resolve(this.result);
        };
        tempFR.onerror = function () {
            tempFR.abort();
            reject(new Error("Problem parsing input file!"));
        };
        tempFR.readAsText(file);
    });
}

$fileBtn.on("click", function (evt) {
    var sign = +$(this).attr("data-sign"),
        $blocks = $("div.building"),
        configJSON, configOfAllBlocks, now,
        fileContent, filename,
        localFile;

    if (!$blocks.length) {
        console.error("ERROR! Please use this script at http://qianshan.co!");
        return ;
    }
    if (sign) {
        localFile = $(".input-ghost", $menuModal).prop("files")[0];
        localJSONFileHandler(localFile).then(function (json) {
            json = JSON.parse(json);
            if (!json["data"] || !json["config"]) {
                console.error("ERROR! JSON file content is incorrect!");
            }
            json = json["config"];
            $blocks.each(function (index, element) {
                setConfig(element, json[index]);
            });
            console.log("the import operation succeeded.");
        }, function (error) {
            console.error("ERROR!", error);
        });
    } else {
        configOfAllBlocks = [];
        configJSON = {};
        now = new Date();
        $blocks.each(function (index, element) {
            configOfAllBlocks.push(getConfig(element));
        });
        configJSON["data"] = now.toLocaleString();
        configJSON["config"] = configOfAllBlocks;
        fileContent = JSON.stringify(configJSON, null, 4);
        filename = "qianshan-config-" + now.getUTCFullYear() + "-"
                   + (now.getUTCMonth()+1) + "-"
                   + now.getUTCDate() + ".json";
        configJSONFileDownload(fileContent, filename);
        console.log("the export operation succeeded.");
    }
});

$menuModal.on("hidden.bs.modal", function (evt) {
    $(".input-ghost", $menuModal).val("");
    $textInput.val("");
    $textInput.attr('disabled','disabled');
    $clearBtn.attr('disabled','disabled');
    $fileBtn.text("导出");
    $fileBtn.attr("data-sign", "0");
});


//global config
var globalConfig = {
    "isSearchBarOpen": 0,        // 0 means off, 1 means on
    "searchEngine": "google",    // google is the default search engine
    "isBgSet": 0,                // 0 means no setting, 1 means setting
    "bgData": ""
};

$(function () {
    var config = localStorage.getItem("GLOBAL");

    $menuModal.append(function (index, html) {
        return createInputFileGhost();
    });
    $(".input-ghost", $menuModal).val("");
    $textInput.val("");
    if (!config) {
        return ;
    }
    config = JSON.parse(config);
    if (config["isSearchBarOpen"]) {
        $searchBarBtn.text("打开");
        $searchBar.attr("action", engineConfig[config["searchEngine"]]["action"]);
        $inputField.attr("name", engineConfig[config["searchEngine"]]["name"]);
        $engineIcon.attr("class", "icon "+config["searchEngine"]);
        $searchBar.insertAfter($body.find(".content .navbar .navbar-header"));
    }
    if (config["isBgSet"]) {
        $bgBtn.text("打开");
        $body.css("backgroundImage", "url("+config["bgData"]+")");
    }
});

})();