extract table for hualala

Extract table form Hualala website

// ==UserScript==
// @name         extract table for hualala   
// @namespace    http://tampermonkey.net/
// @version      0.1.5
// @description  Extract table form Hualala website
// @author       eqd
// @match        https://report.hualala.com/*
// @icon         
// @grant        GM_setClipboard
// @license MIT
// ==/UserScript==
const CopyBtnRowId = "CopyBtnRowId"
var TbData = null;
var TbHead = null;
var ExecInput = null;
var ExecBtn = null;
var onmouseover_btn = function()
{
    this.style.backgroundColor="rgb(0 114 213)";
}
var onmouseleave_btn = function()
{
    this.style.backgroundColor="rgb(0 98 183)";
}
var add_float_window = function () {
    var body = document.body;
    let div=document.createElement("div");
    div.setAttribute("id","plug_main");
    div.setAttribute("style","max-height: 540px;visibility: visible;z-index: 1000;float: left;display: block;position: fixed;top: 400px;margin: 10px;");
    div.innerHTML = `
    <div id='pg_extract' class='pg_btn' style='margin: 12px 8px 8px 8px;
    background-color: rgb(0 98 183);
    color: white;
    font-size: 13px;
    font-weight: 400;
    line-height: 20px;
    text-align: center;
    border-radius: 6px;
    text-overflow: ellipsis;
    overflow: hidden;
    white-space: nowrap;
    display: block;
    padding: 5px 20px;
    user-select: none;'>
        提取
    </div>`;
    
    var reverseBtn = div.querySelector("#pg_extract").cloneNode(true);
    reverseBtn.setAttribute("id","pg_extract_r");
    reverseBtn.innerHTML = reverseBtn.innerHTML + "反序";

    var genBtn = reverseBtn.cloneNode(true);
    genBtn.setAttribute("id","pg_gen");
    genBtn.innerHTML = "生成按钮";

    div.append(reverseBtn);
    div.append(genBtn);
    
    ExecInput = document.createElement("input");
    ExecInput.setAttribute("id","ExecIn");
    div.append(ExecInput);

    ExecBtn = reverseBtn.cloneNode(true);
    ExecBtn.setAttribute("id","pg_exec");
    ExecBtn.innerHTML = "执行";
    div.append(ExecBtn);

    body.append(div);
    
    let btns = div.querySelectorAll('.pg_btn');
    for(var b of btns)
    {
        b.onmouseover=onmouseover_btn;
        b.onmouseleave=onmouseleave_btn;
    }

    return div;
};
function isVisible(el) {
    var loopable = true,
        visible = getComputedStyle(el).display != 'none' && getComputedStyle(el).visibility != 'hidden';
        
    while(loopable && visible) {
        el = el.parentNode;
        
        if(el && el != document.body) {
            visible = getComputedStyle(el).display != 'none' && getComputedStyle(el).visibility != 'hidden';
        }else {
            loopable = false;
        }
    }
    return visible;
}
var extract_table = function(el)
{
    var trs = el.getElementsByTagName("tr");
    var cols = [];
    for(var tr of trs)
    {
        if(tr.getAttribute("aria-hidden") == "true" || tr.getAttribute("id") == CopyBtnRowId)
        {
            continue;
        }
        var rows = [];
        var tds = tr.getElementsByTagName("td");
        for(var td of tds)
        {
            var span = td.getElementsByTagName("span");
            var real = "";
            if(span.length > 0)
            {
                var inner = span[0].innerHTML;
                if(inner.startsWith("<span>"))
                {
                    real = span[0].getElementsByTagName("span")[0].innerHTML;
                }else{
                    real = inner;
                }
            }
            rows.push(real);
        }
        cols.push(rows);
    }
    return cols;
}
var extract_head = function(el)
{
    var trs = el.getElementsByTagName("tr");
    var res = [];
    var idx = 0;
    for(var tr of trs)
    {
        var ths = tr.getElementsByTagName("th");
        for(var th of ths)
        {
            var rowspan = th.getAttribute("rowspan")
            var colspan = th.getAttribute("colspan")
            if(idx == 0 && ((rowspan != undefined && parseInt(rowspan) != trs.length) || colspan != undefined) ) {continue;}
            var es = th.getElementsByTagName("span");
            for(var e of es)
            {
                var inner = e.innerHTML;
                if((e.getAttribute("class") == "title-fixed" || e.getAttribute("class") == "custom-title") && inner != "")
                {
                    if(inner.startsWith("<span>"))
                    {
                        var span = e.getElementsByTagName('span');
                        res.push(span[0].innerHTML);
                    }else{
                        res.push(inner);
                    }
                }
            }
        }
        idx+=1;
    }
    return res;
}
function checkExcelTimeStr(str)
{
    // if(/^[1-9]\d{3}-(0[1-9]|1[0-2])-(0[1-9]|[1-2][0-9]|3[0-1])\s+(20|21|22|23|[0-1]\d):[0-5]\d:[0-5]\d$/.test(str))
    // {
    //     return str.replaceAll('-','/');
    // }
    return str;
}
var setClipboard = function(table)
{
    var str = "";
    for(var i = 0;i < table.length;++i)
    {
        for(var j = 0;j < table[i].length;++j)
        {
            str += checkExcelTimeStr(table[i][j]);
            if(j < table[i].length - 1)str += '\t';
        }
        if(i < table.length - 1)str += '\n';
    }
    GM_setClipboard(str);
}
var setClipboardReverse = function(da)
{
    var str = "";
    for(var i = da.length - 1;i >= 0;--i)
    {
        for(var j = 0;j < da[i].length;++j)
        {
            str += checkExcelTimeStr(da[i][j]);
            if(j < da[i].length - 1)str += '\t';
        }
        if(i > 0)str += '\n';
    }
    GM_setClipboard(str);
}
function find_table_comm(tag)
{
    var tables = document.getElementsByTagName(tag);
    if(tables.length == 0)
    {
        return null;
    }
    var tb = null;
    for(var t of tables)
    {
        var cls = t.getAttribute("class");
        var isDisplay = isVisible(t);
        if(cls == ("ant-table-"+tag) && isDisplay)
        {
            tb = t;
        }
    }
    return tb;
}
function find_table()
{
    return find_table_comm("tbody");
}
function find_thead()
{
    return find_table_comm("thead");
}
var on_extract = function(b)
{
    var tb = find_table();
    if(tb != null)
    {
        var data = extract_table(tb);
        if(b.getAttribute("id").endsWith("_r"))
        {
            setClipboardReverse(data);
        }else{
            setClipboard(data);
        }
    }else{
        alert("没有找到表格");
    }
};
///generate col copy buttons////////////////////
function generate_copy_btn(cls,onclick,text)
{
    let btn=document.createElement("div");
    btn.setAttribute("class",cls);
    btn.style = `margin: 4px 2px 2px 2px;background-color: rgb(0 98 183);
    color: white;
    font-size: 10px;
    font-weight: 400;
    line-height: 20px;
    text-align: center;
    border-radius: 6px;
    text-overflow: ellipsis;
    overflow: hidden;
    white-space: nowrap;
    display: block;
    padding: 5px 20px;
    user-select: none;`;
    btn.onmouseover = onmouseover_btn;
    btn.onmouseleave = onmouseleave_btn;
    btn.onclick = onclick;
    btn.innerHTML = text;
    return btn;
}
function on_copy_col()
{
    var str = "";
    if(!this.reverse)
    {
        for(var i = 0;i < this.array.length;++i)
        {
            str += this.array[i];
            if(i < this.array.length - 1)str += '\n';
        }
    }else{
        for(var i = this.array.length - 1;i >= 0;--i)
        {
            str += this.array[i];
            if(i > 0)str += '\n';
        }
    }
    GM_setClipboard(str);
}
function find_or_gen_copy_rows(tb)
{
    var btn_tr = tb.querySelector("#"+CopyBtnRowId);
    if(btn_tr != null) return btn_tr;
    var trs = tb.getElementsByTagName("tr");
    var btn_tr = null;
    var first_tr = null;
    for(var tr of trs)
    {
        if(tr.getAttribute("aria-hidden") == "true")
        {
            continue;
        }
        first_tr = tr;
        btn_tr = tr.cloneNode(true);
        btn_tr.setAttribute("id",CopyBtnRowId);
        break;
    }
    if(btn_tr == null){
        alert("没有找到表格第一行");
        return;
    }
    tb.insertBefore(btn_tr,first_tr);
    return btn_tr;
}
function generate_col_btns()
{
    var tb = find_table();
    
    if(tb == null)
    {
        alert("没有找到表格");
        return;
    }
    var data = extract_table(tb);
    data.pop();
    TbHead = extract_head(find_thead());
    TbData = data;
    var btn_tr = find_or_gen_copy_rows(tb);
    var tds = btn_tr.getElementsByTagName("td");
    for(var j = 0;j < tds.length;++j)
    {
        var arr = [];
        for(var i = 0;i < data.length;++i)
        {
            arr.push(data[i][j]);
        }
        var copy = generate_copy_btn("copy_col_btn",on_copy_col,"复制");
        var copy_r = generate_copy_btn("copy_col_r_btn",on_copy_col,"反向复制");
        copy.array = arr;
        copy_r.array = arr;
        copy_r.reverse = true;
        tds[j].innerHTML = "";
        tds[j].append(copy);
        tds[j].append(copy_r);
    }
}
///////////////////////////////////////
///exec part////////////////////////////////////
var read_col = function(i,d,r)
{
    var res = [];
    if(r)
    {
        for(var x = d.length - 1;x >= 0;--x)
        {
            res.push(d[x][i]);
        }
    }else{
        for(var x = 0;x < d.length;++x)
        {
            res.push(d[x][i]);
        }
    }
    return res;
}
var find_col = function(str,reverse = true)
{
    for(var i = 0;i < TbHead.length;++i)
    {
        if(TbHead[i] == str)
        {
            return read_col(i,TbData,reverse);
        }
    }
}
//operator
var binary_operation = null;
var binary_operation_arr = function(a,op,b)
{
    if(typeof a === 'number' && typeof b === 'number')
    {
        return binary_operation(a,op,b);
    }
    var res = [];
    for(var i = 0;i < Math.min(a.length,b.length); ++i)
    {
        res.push(binary_operation(parseFloat(a[i]),op,parseFloat(b[i])));
    }
    return res;
}
var Op={};
Op.add = function(a,b){ return binary_operation_arr(a,'+',b); }
Op.sub = function(a,b){ return binary_operation_arr(a,'-',b); }
Op.mul = function(a,b){ return binary_operation_arr(a,'*',b); }
Op.div = function(a,b){ return binary_operation_arr(a,'/',b); }
Op.__supported_op = { '+':Op.add,'-':Op.sub,'*':Op.mul,'/':Op.div }
Op._is_supported_op = function(o){ return o in this.__supported_op; }
Op._get_op = function(o){ return this.__supported_op[o]; }

binary_operation = function(a,op,b)
{
    switch(op)
    {
        case '+': return a + b;
        case '-': return a - b;
        case '*': return a * b;
        case '/': return a / b;
    }
}

///parse cmd str
class Stack{
    constructor()
    {
        this.arr = [];
    }
    push(v)
    {
        this.arr.push(v);
    }
    pop()
    {
        return this.arr.pop();
    }
    get top()
    {
        if(this.arr.length == 0) {return undefined;}
        return this.arr[this.arr.length - 1];
    }
    get size() {return this.arr.length;}
    get empty() {return this.arr.length == 0;}
}
class SynTree{
    constructor()
    {
        this.a = null;
        this.b = null;
        this.op = null;
    }
    get val()
    {
        return this.op((this.a instanceof SynTree ? this.a.val : this.a),
        (this.b instanceof SynTree ? this.b.val : this.b));
    }
    get vaild()
    {
        return this.a != null && this.b != null && this.op != null;
    }
}
function check_shuck(str)
{
    if(str.length >= 2 && str[0] == '(' && str[str.length - 1] == ')')
        return str.substr(1,str.length - 2);
    return str;
}
function parse_token(i,str)
{
    var tk = "";
    var stack = new Stack();
    var next_normal = false;
    for(var x = i;x < str.length;++x)
    {
        if(str[x] == '\\')
        {
            next_normal = true;
            continue;
        }
        if(!next_normal && str[x] == '(')
        {    
            stack.push(str[x]); 
        }
        if(!next_normal && str[x] == ')')
        {
            if(stack.empty || stack.top != '(')
                throw new Error("Parse token failed!");
            stack.pop();
        }
        if(!next_normal && str[x] == ' ') {continue;}
        tk += str[x];
        if(stack.empty && x + 1 < str.length && Op._is_supported_op(str[x + 1])){return [check_shuck(tk),x];}
        if(!next_normal && stack.empty && Op._is_supported_op(tk)){return [check_shuck(tk),x];}
        next_normal = false;
    }
    if(tk.length > 0){ return [check_shuck(tk),x];}
    return [null,x]; 
}
function parse_val(str)
{
    if(str.length > 0 && str[0] == '$')
        return find_col(str.substr(1));
    if(str.length > 0 && str[0] == '#')
        return find_col(str.substr(1),false);
    return parseFloat(str);
}
function parse(str)
{
    var tree = new SynTree();
    var st = 0;
    for(var i = 0;i < str.length;++i)
    {
        let tk;
        [tk,i] = parse_token(i,str);
        if(tk == null) throw Error("Unexpected expression end");
        if(st == 0 || st == 2)
        {
            var val = parse_val(tk);
            if(val == null) {
                var sub_tree = parse(tk);
                if(!sub_tree.vaild)
                {
                    throw Error("Parse expression failed");
                }
                val = sub_tree;
            }
            if(st == 0) {
                tree.a = val;
                ++st;
            }else{
                tree.b = val;
                --st;
            }
        }else{
            if(tree.op != null)
            {
                var left_tree = tree;
                tree = new SynTree();
                tree.a = left_tree;
                left_tree = null;
            }
            tree.op = Op._get_op(tk);
            ++st;
        }
    }
    return tree;
}
function on_exec()
{
    var cmd = ExecInput.value;
    var table = TbData;
    var head = TbHead;
    var fc = find_col;
    var op = binary_operation_arr;
    var arr = parse(cmd).val; //eval(cmd);
    var cal = {};
    cal.array = arr;
    cal.on_copy_col = on_copy_col;
    cal.on_copy_col();
}
//////////////////////////////////////////////////
(function () {
    'use strict';

    var main = add_float_window();
    var extract = main.querySelector("#pg_extract");
    extract.onclick = function(){on_extract(extract);};
    var extract_r = main.querySelector("#pg_extract_r");
    extract_r.onclick = function(){on_extract(extract_r);};
    var gen = main.querySelector("#pg_gen");
    gen.onclick = generate_col_btns;

    ExecBtn.onclick = on_exec;
})();