Is there a way using JavaScript to disable the ability to paste text into a text field on an HTML form?
E.g. I have a simple registration form where the user is required to input their email twice. The second email entry is to verify there are no typos in the first email entry. However if the user copy/pastes their email then that defeats the purpose and I’ve been experiencing users having problems because they’ve input the wrong email and copy/pasted it.
Maybe I wasn’t clear on my question but I am not trying to prevent people from copying (or drag selecting) text on their browser. I just want to stop them from pasting input into a text field to minimize user error.
Perhaps instead of using this “hack” you can suggest another solution to the core problem of what I’m trying to solve here? I’ve done less than half a dozen user tests and this has already happened twice. My audience does not have a high level of computer proficiency.
Advertisement
Answer
I recently had to begrudgingly disable pasting in a form element. To do so, I wrote a cross-browser* implementation of Internet Explorer’s (and others’) onpaste event handler. My solution had to be independent of any third-party JavaScript libraries.
Here’s what I came up with. It doesn’t completely disable pasting (the user can paste a single character at a time, for example), but it meets my needs and avoids having to deal with keyCodes, etc.
// Register onpaste on inputs and textareas in browsers that don't // natively support it. (function () { var onload = window.onload; window.onload = function () { if (typeof onload == "function") { onload.apply(this, arguments); } var fields = []; var inputs = document.getElementsByTagName("input"); var textareas = document.getElementsByTagName("textarea"); for (var i = 0; i < inputs.length; i++) { fields.push(inputs[i]); } for (var i = 0; i < textareas.length; i++) { fields.push(textareas[i]); } for (var i = 0; i < fields.length; i++) { var field = fields[i]; if (typeof field.onpaste != "function" && !!field.getAttribute("onpaste")) { field.onpaste = eval("(function () { " + field.getAttribute("onpaste") + " })"); } if (typeof field.onpaste == "function") { var oninput = field.oninput; field.oninput = function () { if (typeof oninput == "function") { oninput.apply(this, arguments); } if (typeof this.previousValue == "undefined") { this.previousValue = this.value; } var pasted = (Math.abs(this.previousValue.length - this.value.length) > 1 && this.value != ""); if (pasted && !this.onpaste.apply(this, arguments)) { this.value = this.previousValue; } this.previousValue = this.value; }; if (field.addEventListener) { field.addEventListener("input", field.oninput, false); } else if (field.attachEvent) { field.attachEvent("oninput", field.oninput); } } } } })();
To make use of this in order to disable pasting:
<input type="text" onpaste="return false;" />
* I know oninput isn’t part of the W3C DOM spec, but all of the browsers I’ve tested this code with—Chrome 2, Safari 4, Firefox 3, Opera 10, IE6, IE7—support either oninput or onpaste. Out of all these browsers, only Opera doesn’t support onpaste, but it does support oninput.
Note: This won’t work on a console or other system that uses an on-screen keyboard (assuming the on-screen keyboard doesn’t send keys to the browser when each key is selected). If it’s possible your page/app could be used by someone with an on-screen keyboard and Opera (e.g.: Nintendo Wii, some mobile phones), don’t use this script unless you’ve tested to make sure the on-screen keyboard sends keys to the browser after each key selection.