2013-03-18 19:06:47 -04:00

470 lines
20 KiB
JavaScript

/*
* The MIT License
*
* Copyright (c) 2012 James Allardice
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*
* -----------------------------------------------------------------------------------------------------------------------------------------------------
*
* Placeholders is a simple polyfill for the HTML5 "placeholder" attribute. The placeholder attribute can be used on input elements of certain types
* and provides a short hint (such as a sample value or a brief description) intended to aid the user with data entry. This polyfill has been tested
* and functions correctly in Internet Explorer 6 and above, Firefox 3.6 and above, Safari 3.2 and above, Opera 9 and above and Chrome 16 and above.
* The script will be tested in further browsers in due course and the above list edited accordingly.
*
* User agents should display the value of the placeholder attribute when the element's value is the empty string and the element does not
* have focus. The user agents that have implemented support for this attribute all display the placeholder inside the element, as if it were
* the element's value, in a light grey colour to differentiate between placeholder text and value text.
*
* The Placeholders polyfill attempts to replicate the functionality of compliant user agents so that non-compliant user agents will still function
* as expected when faced with a "placeholder" attribute.
*
* The script is unobtrusive and will only apply if the placeholder attribute is not supported by the user agent in which it is running. To use a placeholder
* simply add the "placeholder" attribute to a supporting input element:
*
* <input type="text" placeholder="Placeholder text">
*
* To get this placeholder to function in non-supporting user agents simply call the init method when appropriate (the DOM must be ready for manipulation,
* unless the `live` option is true):
*
* Placeholders.init();
*
* The init method accepts one argument, `options`. It's an object that contains settings to control the behaviour of the polyfill. Currently, only 2 options
* are available:
*
* `live` - If truthy, the polyfill will apply to all supported input elements now and in the future, and dynamic
* changes to the placeholder attribute value will be reflected. If falsy, the polyfill will only apply to those elements with a placeholder attribute
* value in the DOM at the time the method is executed. If the live option is not used, the placeholders can be refreshed manually by calling `Placeholders.refresh()`.
*
* `hideOnFocus` - If truthy, the placeholder text will not disappear when the field receives focus. This option is relatively new to the spec, but is
* starting to be implemented in browsers (e.g. Safari, and now Chrome). This option is set to `true` by default, since that's where browsers seem to be heading.
*/
/*jslint browser: true */
var Placeholders = (function () {
"use strict";
/* List of input types that support the placeholder attribute. We only want to modify input elements with one of these types.
* WARNING: If an input type is not supported by a browser, the browser will choose the default type (text) and the placeholder shim will
* apply */
var validTypes = [
"text",
"search",
"url",
"tel",
"email",
"password",
"number",
"textarea"
],
//Default options, can be overridden by passing object to `init`
settings = {
live: false,
hideOnFocus: false,
className: 'placeholderspolyfill', // placeholder class name to apply to form fields
textColor: '#999', // default placeholder text color
styleImportant: true // add !important flag to placeholder style
},
//Keycodes that are not allowed when the placeholder is visible and `hideOnFocus` is `false`
badKeys = [37, 38, 39, 40],
//Used if `live` options is `true`
interval,
//Stores the input value on keydown (used when `hideOnFocus` option is `false`)
valueKeyDown,
// polyfill class name regexp
classNameRegExp = new RegExp('\\b' + settings.className + '\\b');
// The cursorToStart function attempts to jump the cursor to before the first character of input
function cursorToStart(elem) {
var range;
if (elem.createTextRange) {
range = elem.createTextRange();
range.move("character", 0);
range.select();
} else if (elem.selectionStart) {
elem.focus();
elem.setSelectionRange(0, 0);
}
}
/* The focusHandler function is executed when input elements with placeholder attributes receive a focus event. If necessary, the placeholder
* and its associated styles are removed from the element. Must be bound to an element. */
function focusHandler() {
var type;
//If the placeholder is currently visible, remove it and its associated styles
if (this.value === this.getAttribute("placeholder")) {
if (!settings.hideOnFocus) {
cursorToStart(this);
} else {
/* Remove the placeholder class name. Use a regular expression to ensure the string being searched for is a complete word, and not part of a longer
* string, on the off-chance a class name including that string also exists on the element */
this.className = this.className.replace(classNameRegExp, "");
this.value = "";
// Check if we need to switch the input type (this is the case if it's a password input)
type = this.getAttribute("data-placeholdertype");
if (type) {
this.type = type;
}
}
}
}
/* The blurHandler function is executed when input elements with placeholder attributes receive a blur event. If necessary, the placeholder
* and its associated styles are applied to the element. Must be bound to an element. */
function blurHandler() {
var type;
//If the input value is the empty string, apply the placeholder and its associated styles
if (this.value === "") {
this.className = this.className + " " + settings.className;
this.value = this.getAttribute("placeholder");
// Check if we need to switch the input type (this is the case if it's a password input)
type = this.getAttribute("data-placeholdertype");
if (type) {
this.type = "text";
}
}
}
/* The submitHandler function is executed when the containing form, if any, of a given input element is submitted. If necessary, placeholders on any
* input element descendants of the form are removed so that the placeholder value is not submitted as the element value. */
function submitHandler() {
var inputs = this.getElementsByTagName("input"),
textareas = this.getElementsByTagName("textarea"),
numInputs = inputs.length,
num = numInputs + textareas.length,
element,
placeholder,
i;
//Iterate over all descendant input elements and remove placeholder if necessary
for (i = 0; i < num; i += 1) {
element = (i < numInputs) ? inputs[i] : textareas[i - numInputs];
placeholder = element.getAttribute("placeholder");
//If the value of the input is equal to the value of the placeholder attribute we need to clear the value
if (element.value === placeholder) {
element.value = "";
}
}
}
/* The keydownHandler function is executed when the input elements with placeholder attributes receive a keydown event. It simply stores the current
* value of the input (so we can kind-of simulate the poorly-supported `input` event). Used when `hideOnFocus` option is `false`. Must be bound to an element. */
function keydownHandler(event) {
valueKeyDown = this.value;
//Prevent the use of the arrow keys (try to keep the cursor before the placeholder)
return !(valueKeyDown === this.getAttribute("placeholder") && badKeys.indexOf(event.keyCode) > -1);
}
/* The keyupHandler function is executed when the input elements with placeholder attributes receive a keyup event. It kind-of simulates the native but
* poorly-supported `input` event by checking if the key press has changed the value of the element. Used when `hideOnFocus` option is `false`. Must be bound to an element. */
function keyupHandler() {
var type;
if (this.value !== valueKeyDown) {
// Remove the placeholder
this.className = this.className.replace(classNameRegExp, "");
this.value = this.value.replace(this.getAttribute("placeholder"), "");
// Check if we need to switch the input type (this is the case if it's a password input)
type = this.getAttribute("data-placeholdertype");
if (type) {
this.type = type;
}
}
if (this.value === "") {
blurHandler.call(this);
cursorToStart(this);
}
}
//The addEventListener function binds an event handler with the context of an element to a specific event on that element. Handles old-IE and modern browsers.
function addEventListener(element, event, fn) {
if (element.addEventListener) {
return element.addEventListener(event, fn.bind(element), false);
}
if (element.attachEvent) {
return element.attachEvent("on" + event, fn.bind(element));
}
}
//The addEventListeners function binds the appropriate (depending on options) event listeners to the specified input or textarea element.
function addEventListeners(element) {
if (!settings.hideOnFocus) {
addEventListener(element, "keydown", keydownHandler);
addEventListener(element, "keyup", keyupHandler);
}
addEventListener(element, "focus", focusHandler);
addEventListener(element, "blur", blurHandler);
}
/* The updatePlaceholders function checks all input and textarea elements and updates the placeholder if necessary. Elements that have been added to the DOM since the call to
* createPlaceholders will not function correctly until this function is executed. The same goes for any existing elements whose placeholder property has been changed (via
* element.setAttribute("placeholder", "new") for example) */
function updatePlaceholders() {
//Declare variables, get references to all input and textarea elements
var inputs = document.getElementsByTagName("input"),
textareas = document.getElementsByTagName("textarea"),
numInputs = inputs.length,
num = numInputs + textareas.length,
i,
form,
element,
oldPlaceholder,
newPlaceholder;
//Iterate over all input and textarea elements and apply/update the placeholder polyfill if necessary
for (i = 0; i < num; i += 1) {
//Get the next element from either the input NodeList or the textarea NodeList, depending on how many elements we've already looped through
element = (i < numInputs) ? inputs[i] : textareas[i - numInputs];
//Get the value of the placeholder attribute
newPlaceholder = element.getAttribute("placeholder");
//Check whether the current input element is of a type that supports the placeholder attribute
if (validTypes.indexOf(element.type) > -1) {
//The input type does support the placeholder attribute. Check whether the placeholder attribute has a value
if (newPlaceholder) {
//The placeholder attribute has a value. Get the value of the current placeholder data-* attribute
oldPlaceholder = element.getAttribute("data-currentplaceholder");
//Check whether the placeholder attribute value has changed
if (newPlaceholder !== oldPlaceholder) {
//The placeholder attribute value has changed so we need to update. Check whether the placeholder should currently be visible.
if (element.value === oldPlaceholder || element.value === newPlaceholder || !element.value) {
//The placeholder should be visible so change the element value to that of the placeholder attribute and set placeholder styles
element.value = newPlaceholder;
element.className = element.className + " " + settings.className;
}
//If the current placeholder data-* attribute has no value the element wasn't present in the DOM when event handlers were bound, so bind them now
if (!oldPlaceholder) {
//If the element has a containing form bind to the submit event so we can prevent placeholder values being submitted as actual values
if (element.form) {
//Get a reference to the containing form element (if present)
form = element.form;
//The placeholdersubmit data-* attribute is set if this form has already been dealt with
if (!form.getAttribute("data-placeholdersubmit")) {
//The placeholdersubmit attribute wasn't set, so attach a submit event handler
addEventListener(form, "submit", submitHandler);
//Set the placeholdersubmit attribute so we don't repeatedly bind event handlers to this form element
form.setAttribute("data-placeholdersubmit", "true");
}
}
addEventListeners(element);
}
//Update the value of the current placeholder data-* attribute to reflect the new placeholder value
element.setAttribute("data-currentplaceholder", newPlaceholder);
}
}
}
}
}
/* The createPlaceholders function checks all input and textarea elements currently in the DOM for the placeholder attribute. If the attribute
* is present, and the element is of a type (e.g. text) that allows the placeholder attribute, it attaches the appropriate event listeners
* to the element and if necessary sets its value to that of the placeholder attribute */
function createPlaceholders() {
//Declare variables and get references to all input and textarea elements
var inputs = document.getElementsByTagName("input"),
textareas = document.getElementsByTagName("textarea"),
numInputs = inputs.length,
num = numInputs + textareas.length,
i,
element,
form,
placeholder;
//Iterate over all input elements and apply placeholder polyfill if necessary
for (i = 0; i < num; i += 1) {
//Get the next element from either the input NodeList or the textarea NodeList, depending on how many elements we've already looped through
element = (i < numInputs) ? inputs[i] : textareas[i - numInputs];
//Get the value of the placeholder attribute
placeholder = element.getAttribute("placeholder");
//Check whether or not the current element is of a type that allows the placeholder attribute
if (validTypes.indexOf(element.type) > -1) {
//The input type does support placeholders. Check that the placeholder attribute has been given a value
if (placeholder) {
// If the element type is "password", attempt to change it to "text" so we can display the placeholder value in plain text
if (element.type === "password") {
// The `type` property is read-only in IE < 9, so in those cases we just move on. The placeholder will be displayed masked
try {
element.type = "text";
element.setAttribute("data-placeholdertype", "password");
} catch (e) {}
}
//The placeholder attribute has a value. Keep track of the current placeholder value in an HTML5 data-* attribute
element.setAttribute("data-currentplaceholder", placeholder);
//If the value of the element is the empty string set the value to that of the placeholder attribute and apply the placeholder styles
if (element.value === "" || element.value === placeholder) {
element.className = element.className + " " + settings.className;
element.value = placeholder;
}
//If the element has a containing form bind to the submit event so we can prevent placeholder values being submitted as actual values
if (element.form) {
//Get a reference to the containing form element (if present)
form = element.form;
//The placeholdersubmit data-* attribute is set if this form has already been dealt with
if (!form.getAttribute("data-placeholdersubmit")) {
//The placeholdersubmit attribute wasn't set, so attach a submit event handler
addEventListener(form, "submit", submitHandler);
//Set the placeholdersubmit attribute so we don't repeatedly bind event handlers to this form element
form.setAttribute("data-placeholdersubmit", "true");
}
}
//Attach event listeners to this element
addEventListeners(element);
}
}
}
}
/* The init function checks whether or not we need to polyfill the placeholder functionality. If we do, it sets up various things
* needed throughout the script and then calls createPlaceholders to setup initial placeholders */
function init(opts) {
//Create an input element to test for the presence of the placeholder property. If the placeholder property exists, stop.
var test = document.createElement("input"),
opt,
styleElem,
styleRules,
i,
j;
//Test input element for presence of placeholder property. If it doesn't exist, the browser does not support HTML5 placeholders
if (typeof test.placeholder === "undefined") {
//HTML5 placeholder attribute not supported.
//Set the options (or use defaults)
for (opt in opts) {
if (opts.hasOwnProperty(opt)) {
settings[opt] = opts[opt];
}
}
//Create style element for placeholder styles
styleElem = document.createElement("style");
styleElem.type = "text/css";
//Create style rules as text node
var importantValue = (settings.styleImportant) ? "!important" : "";
styleRules = document.createTextNode("." + settings.className + " { color:" + settings.textColor + importantValue + "; }");
//Append style rules to newly created stylesheet
if (styleElem.styleSheet) {
styleElem.styleSheet.cssText = styleRules.nodeValue;
} else {
styleElem.appendChild(styleRules);
}
//Append new style element to the head
document.getElementsByTagName("head")[0].appendChild(styleElem);
//We use Array.prototype.indexOf later, so make sure it exists
if (!Array.prototype.indexOf) {
Array.prototype.indexOf = function (obj, start) {
for (i = (start || 0), j = this.length; i < j; i += 1) {
if (this[i] === obj) { return i; }
}
return -1;
};
}
/* We use Function.prototype.bind later, so make sure it exists. This is the MDN implementation, slightly modified to pass JSLint. See
* https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Function/bind */
if (!Function.prototype.bind) {
Function.prototype.bind = function (oThis) {
if (typeof this !== "function") {
throw new TypeError("Function.prototype.bind - what is trying to be bound is not callable");
}
var aArgs = Array.prototype.slice.call(arguments, 1),
fToBind = this,
FNop = function () {},
fBound = function () {
return fToBind.apply(this instanceof FNop
? this
: oThis,
aArgs.concat(Array.prototype.slice.call(arguments)));
};
FNop.prototype = this.prototype;
fBound.prototype = new FNop();
return fBound;
};
}
//Create placeholders for input elements currently part of the DOM
createPlaceholders();
/* If the `live` option is truthy, call updatePlaceholders repeatedly to keep up to date with any DOM changes.
* We use an interval over events such as DOMAttrModified (which are used in some other implementations of the placeholder attribute)
* since the DOM level 2 mutation events are deprecated in the level 3 spec. */
if (settings.live) {
interval = setInterval(updatePlaceholders, 100);
}
//Placeholder attribute was successfully polyfilled :)
return true;
}
//Placeholder attribute already supported by browser :)
return false;
}
//Expose public methods
return {
init: init,
refresh: updatePlaceholders
};
}());