問題描述
有多種方式,< input type = “text”> 可以改變,包括:
-
按鍵
-
複製粘貼
-
用 JavaScript 修改
-
auto-completed 通過瀏覽器或工具欄
我希望我的 JavaScript 函數隨時更改 (當前的輸入值) 被調用。而且我希望它立即被打來,而不僅僅是當輸入失去焦點時。
我正在尋找在所有瀏覽器 (最好使用 jQuery) 這樣做最乾淨,最強大的方法。
示例用例:在 Twitter Signup 頁面上,用户名字段的值顯示在其下面的 URL「http://twitter/ username」 中。
最佳解決方案
這個 jQuery 代碼可以立即對任何元素進行更改,並且可以在所有瀏覽器中運行:
$('.myElements').each(function() {
var elem = $(this);
// Save current value of element
elem.data('oldVal', elem.val());
// Look for changes in the value
elem.bind("propertychange change click keyup input paste", function(event){
// If value has changed...
if (elem.data('oldVal') != elem.val()) {
// Updated stored value
elem.data('oldVal', elem.val());
// Do action
....
}
});
});
次佳解決方案
不幸的是,我認為 setInterval 獲獎:
<input type=text id=input_id />
<script>
setInterval(function() { ObserveInputValue($('#input_id').val()); }, 100);
</script>
這是最簡單的解決方案,只有一行代碼。它也是最強大的,因為您不必擔心 input 可以獲得的所有不同的事件/方式。
在這種情況下,使用’setInterval’ 的缺點似乎不適用:
-
100ms 延遲?對於許多應用,100ms 足夠快。
-
在瀏覽器上增加了負載?一般來説,在您的頁面上添加大量 heavy-weight setIntervals 是不好的。但在這種特殊情況下,增加的頁面加載是不可檢測的。
-
它不會擴展到許多輸入?大多數頁面沒有多少輸入,您可以在同一個 setInterval 中嗅探所有的輸入。
第三種解決方案
jquery 的即時花式解決方案> = 1.9
$("#input-id").on("change keyup paste", function(){
dosomething();
})
如果還要檢測”click” 事件,只需:
$("#input-id").on("change keyup paste click", function(){
dosomething();
})
如果你的 jquery <= 1.4,只需使用"live" 而不是"on" 。
第四種方案
綁定到 oninput 事件似乎在大多數健全的瀏覽器中正常工作。 IE9 也支持它,但實現是錯誤的 (刪除字符時事件不會觸發) 。
使用 jQuery 版本 1.7+,on 方法可以像這樣綁定到事件:
$(".inputElement").on("input", null, null, callbackFunction);
第五種方案
不幸的是,沒有符合您的標準的事件或事件集。鍵盤和複製/粘貼都可以使用 keyup 事件來處理。通過 JS 進行的更改變得非常棘手。如果您可以控制設置文本框的代碼,最好的方法是修改它直接調用函數或觸發文本框上的用户事件:
// Compare the textbox's current and last value. Report a change to the console.
function watchTextbox() {
var txtInput = $('#txtInput');
var lastValue = txtInput.data('lastValue');
var currentValue = txtInput.val();
if (lastValue != currentValue) {
console.log('Value changed from ' + lastValue + ' to ' + currentValue);
txtInput.data('lastValue', currentValue);
}
}
// Record the initial value of the textbox.
$('#txtInput').data('lastValue', $('#txtInput').val());
// Bind to the keypress and user-defined set event.
$('#txtInput').bind('keypress set', null, watchTextbox);
// Example of JS code triggering the user event
$('#btnSetText').click(function (ev) {
$('#txtInput').val('abc def').trigger('set');
});
如果您無法控制該代碼,可以使用 setInterval()將’watch’ 更改為文本框:
// Check the textbox every 100 milliseconds. This seems to be pretty responsive.
setInterval(watchTextbox, 100);
這種主動監控不會捕獲更新’immediately’,但似乎足夠快,沒有明顯的滯後。正如 DrLouie 在評論中指出的那樣,如果您需要觀看大量投入,此解決方案可能不會很好地擴展。您可以隨時調整第二個參數到 setInterval()以檢查更多或更少的頻率。
第六種方案
如果您不喜歡任何其他答案,這是一個稍微不同的解決方案:
var field_selectors = ["#a", "#b"];
setInterval(function() {
$.each(field_selectors, function() {
var input = $(this);
var old = input.attr("data-old-value");
var current = input.val();
if (old !== current) {
if (typeof old != 'undefined') {
... your code ...
}
input.attr("data-old-value", current);
}
}
}, 500);
考慮到您無法依靠點擊和密鑰來捕獲上下文菜單粘貼。
第七種方案
我創建了一個示例。可能會為你工作
var typingTimer;
var doneTypingInterval = 10;
var finaldoneTypingInterval = 500;
var oldData = $("p.content").html();
$('#tyingBox').keydown(function () {
clearTimeout(typingTimer);
if ($('#tyingBox').val) {
typingTimer = setTimeout(function () {
$("p.content").html('Typing...');
}, doneTypingInterval);
}
});
$('#tyingBox').keyup(function () {
clearTimeout(typingTimer);
typingTimer = setTimeout(function () {
$("p.content").html(oldData);
}, finaldoneTypingInterval);
});
<script src="https://googleajax.admincdn.com/ajax/libs/jquery/1.9.1/jquery.min.js"></script>
<textarea id="tyingBox" tabindex="1" placeholder="Enter Message"></textarea>
<p class="content">Text will be replace here and after Stop typing it will get back</p>
第八種方案
2017 答案:input event 完全符合 IE8 的要求。
$(el).on('input', callback)
參考文獻
注:本文內容整合自 Google/Baidu/Bing 輔助翻譯的英文資料結果。如果您對結果不滿意,可以加入我們改善翻譯效果:薇曉朵技術論壇。