SOURCE

console 命令行工具 X clear

                    
>
console
/* 补丁测试 1 :那么仿造的事件能用来选择吗? */
(function(){
    var shiftDown = false;
    var needHook = true;
    function cloneAndShift(e){
        var obj = {};
        for(var i in e){
            if(i[0] != i[0].toLowerCase()) continue;
            obj[i] = e[i]
        }
        obj.target = e.target;
        obj.shiftKey = true;
        debugger;
        return obj;
    }
    function keyHook(e){
        if(!(e instanceof KeyboardEvent)) return;
        if(!(e instanceof KeyboardEvent)) throw "Fuck!";
        if(!e.isTrusted){return;}
        if(e.target.matches(".nofix, .nofix *")) return;
        if(e.key.includes("Shift")){
            if(e.type == "keydown"){
                shiftDown = true;
            } else if(e.type == "keyup"){
                shiftDown = false;
            }
        } else {
            if(e.shiftKey == true && shiftDown == true){
                needHook = false;
                has_bug.innerHTML = "No!"
                document.removeEventListener("keydown", keyHook, true);
                document.removeEventListener("keyup", keyHook, true);
            }

            if(e.shiftKey == false && shiftDown == true){
                /* Hook! */
                var fakedEvent = new KeyboardEvent(e.type,cloneAndShift(e));
                e.stopImmediatePropagation();
                e.preventDefault();
                var res = e.target.dispatchEvent(fakedEvent);
                has_bug.innerHTML = "Yes!"
                /* 模仿默认行为 */
                if(!res) return;
                if(e.type != "keydown") return;
                var textarea = e.target;
                if(!(textarea instanceof HTMLTextAreaElement)) return;
                if(!(textarea instanceof HTMLTextAreaElement)) throw "";
                if(e.key == "ArrowLeft"){
                    if(textarea.selectionEnd == textarea.selectionStart){
                        textarea.setSelectionRange(Math.max(textarea.selectionStart-1,0),textarea.selectionEnd,"backward")
                    } else if(textarea.selectionDirection == "forward"){
                        textarea.setSelectionRange(textarea.selectionStart,textarea.selectionEnd-1,"forward")
                    } else {
                        textarea.setSelectionRange(Math.max(textarea.selectionStart-1,0),textarea.selectionEnd,"backward")
                    }
                } else if(e.key == "ArrowRight"){
                    if(textarea.selectionEnd == textarea.selectionStart){
                        textarea.setSelectionRange(textarea.selectionStart,textarea.selectionEnd+1,"forward")
                    } else if(textarea.selectionDirection == "forward"){
                        textarea.setSelectionRange(textarea.selectionStart,textarea.selectionEnd+1,"forward")
                    } else {
                        textarea.setSelectionRange(textarea.selectionStart+1,textarea.selectionEnd,"backward")
                    }
                }
            }
        }
    }
    document.addEventListener("keydown", keyHook, true);
    document.addEventListener("keyup", keyHook, true);
})();



/* 键盘事件测试程序 */
(function(){
    /**
     * htmlEncode 把一段文字用 HTML 实体编码。
     * @param {string} str 被编码的字符串
     * @return {string} 返回一个你可以放心放到 HTML 里面的字符串。
     */
    function htmlEncode(str){
        if(!htmlEncode.textarea){
            htmlEncode.textarea = document.createElement("div");
        }
        htmlEncode.textarea.textContent = str;
        return htmlEncode.textarea.innerHTML;
    }

    var select = document.querySelector("#key_test_col_select");
    var table  = document.querySelector("#key_test_result");
    var target = document.querySelector("#key_test_target");
    var cols   = new Set();
    var events = [];
    var init = function(){
        if(!(select instanceof HTMLSelectElement)) throw "Fuck 1";
        if(!(table instanceof HTMLTableElement)) throw "Fuck 2";
        cols.clear();
        cols.add("type");
        events = [];
        refreshSelect();
        refreshTable();
    }

    function refreshSelect(){
        if(!(select instanceof HTMLSelectElement)) throw "Fuck 1";
        var set2 = new Set(Array.from(select.options).map(function(option){
            return option.textContent;
        }));
        for(var i of cols){
            if(!set2.has(i)){
                var option2 = document.createElement("option");
                option2.textContent = i;
                select.appendChild(option2);
                set2.add(i);
            }
        }
    }
    var refreshTable = function(){
        if(!(select instanceof HTMLSelectElement)) throw "Fuck 1";
        var html = "";
        var selectedHeaders = Array.from(select.selectedOptions).map(function(option){
            return option.textContent;
        });
        if(selectedHeaders.length == 0){
            table.innerHTML = "";
            return;
        }
        html += "<tr>" + selectedHeaders.map(function(str){
            return "<th>" + htmlEncode(str) + "</th>";
        }).join("") + "</tr>";

        for(var event of events){
            html += "<tr>";
            for(var i of selectedHeaders){
                html += "<td>" + htmlRepresent(event[i]) + "</td>";
            }
            html += "</tr>"
        }
        table.innerHTML = html;
        if(window.hljs){
            document.querySelectorAll('code[class*=lang]').forEach((el) => {
                hljs.highlightElement(el);
            });
        }
    }
    function htmlRepresent(val){
        if(val === undefined) return "<i class='h'>(未定义)</i>";
        if(val === null)      return "<i class='h'>(null)</i>"
        if(typeof val == "function") return "<i class='h'>function(){}</i>";
        try{
        
            if(typeof val == "object" && Object.prototype.toString.call(val) != "[object Object]"){
                return "<i class='h'>(复杂对象)</i>"
            }
            return "<code class='language-javascript'>" + htmlEncode(JSON.stringify(val)) + "</code>";
        
        }catch(e){
            return "<i class='h'>出错</i>"
        };
        
    }

    function listener(e){
        if(!(e instanceof Event)) return;
        for(var i in e){
            if(e[i] !== undefined && i[0].toLowerCase() == i[0])
            cols.add(i);
        }
        events.push(e);
        refreshSelect();
        refreshTable();
    }

    select.onchange = refreshTable;
    target.addEventListener("keydown", listener);
    target.addEventListener("keyup", listener);
    init();
})();
<html lang="zh">

<head>
	<meta chatset="utf-8" />
	<title>输入法选择 Bug</title>
	<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/water.css@2/out/water.css">
</head>

<body>
	
	<h1>部分 Android 输入法与 Chrominum 浏览器存在不兼容问题</h1>
	<p>众所周知,这不是什么奇怪问题。手机市场,群妖乱舞。</p>
	<p>Bug 修复前:</p>
	<textarea class="nofix">没人拦你选择这些文字,但是,如果输入法上面的选择按钮失灵,那么神仙也救不了你。</textarea>
    <p>Bug 修复后:</p>
    <textarea>这个键盘的选择效果是由脚本模仿的,但是只模仿了左和右……</textarea>
    <p>检测到 Bug?(<span id="has_bug">?</span>)</p>
    <p>
        这个“修复”只限于
        <code class="language-html">&lt;textarea&gt;</code>
        并且很容易迁移到
        <code class="language-html">&lt;input/&gt;</code>
        。其他的与文本选择相关的大多数是由 DOM Selection API 控制,逻辑
        过于复杂,暂不考虑。
    </p>
    
    <hr>

    <p>万物元凶(除了他还有 Chrominum ,这一 Bug 不在其他 App 中出现):</p>
    <figure style="text-align: center">
        <img alt="百度输入法定制版选择面板" src="" />
        <figcaption>百度输入法定制版选择面板</figcaption>
    </figure>
    <p>它发送不正常的键盘事件。以向左选择为例:</p>
    <table>
        <tr><th>type</th><th>key</th><th>shiftKey</th></tr>
        <tr><td>keydown</td><td>Shift</td><td><mark>false</mark></td></td>
        <tr><td>keydown</td><td>ArrowLeft</td><td><mark>false</mark></td></td>
        <tr><td>keyup</td><td>ArrowLeft</td><td><mark>false</mark></td></td>
        <tr><td>keyup</td><td>Shift</td><td>false</td></td>
    </table>
    <p>
        这是一个典型的异常键盘事件序列,shiftKey标识错误。
    </p>
    <p>
        不同输入法的处理不同。在某些浏览器与某些输入法的组合中,<code>shiftKey</code>被正确设置,
        而没有 Shift 键的按下事件。
    </p>
    <p>
        这些差异导致千奇百怪的行为,让许多代码编辑器脚本以及部分新式可视化编辑器失效
        (它们一般自己处理键盘事件),而同时,
        这一现象可能揭示出来,即便是 Chrominum 也不能正确应对安卓千奇百怪的生态。
    </p>
    <hr>

    <h2>附:键盘事件测试器</h2>
    <button onclick="key_test_target.classList.toggle('nofix')">切换修复标记</button>
    <textarea id="key_test_target"  class="nofix">
        字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。
        字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。字数补丁。
    </textarea>
    <p></p>
    <label for="key_test_col_select">选择列:</label>
    <select id="key_test_col_select" multiple></select>
    <table id="key_test_result"></table>
    <!-- 样式补丁 -->
    <link href="https://cdn.bootcdn.net/ajax/libs/highlight.js/11.7.0/styles/github.min.css" rel="stylesheet">
    <script delay>
        function highlight(){
            
            document.querySelectorAll('code[class*=lang]').forEach((el) => {
                hljs.highlightElement(el);
            });
        }
        
    </script>
    <script async onload="highlight()" src="https://cdn.bootcdn.net/ajax/libs/highlight.js/11.7.0/highlight.min.js"></script>
    
</body>
</html>
/* Not all styles are here */
p{text-align: justify;}
select{width: 100%;}
.h:not(:hover){opacity: 0.5}
.nofix{
    background: rgb(239,228,176)!important;
}
tr:first-child{
    position: sticky;
    top: 1em;
    height: 1.4em;
    background: rgb(153,217,234);
    
}