console
(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){
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(){
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"><textarea></code>
并且很容易迁移到
<code class="language-html"><input/></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>
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);
}