/* jQuery placeholder plugin by Andrey Kuzmin, @unsoundscapes Based on existing plugin http://mths.be/placeholder by @mathias and this demo http://robertnyman.com/2011/05/02/ by @robertnyman Adopted to toggle placeholder on user input instead of focus Released under the MIT license */ (function (factory) { 'use strict'; if (typeof define === 'function' && define.amd) { // AMD. Register as anonymous module. define(['jquery'], factory) } else { // Browser globals. factory(jQuery) } }(function ($) { 'use strict'; var isInputSupported = 'placeholder' in document.createElement('input') , isTextareaSupported = 'placeholder' in document.createElement('textarea') , $placeholders = $() function getAttributes (element) { // Return an object of element attributes var newAttrs = {} , rinlinejQuery = /^jQuery\d+$/ $.each(element.attributes, function () { if (this.specified && !rinlinejQuery.test(this.name)) { newAttrs[this.name] = this.value } }) return newAttrs } function setCaretTo (element, index) { // Set caret to specified @index if (element.createTextRange) { var range = element.createTextRange() range.move('character', index) range.select() } else if (element.selectionStart !== null) { element.focus() element.setSelectionRange(index, index) } } function Placeholder (element, options) { this.options = options || {} this.$replacement = this.$element = $(element) this.initialize.apply(this, arguments) // Cache all elements with placeholders $placeholders = $placeholders.add(element) } Placeholder.prototype = { initialize: function () { this.isHidden = true this.placeholderAttr = this.$element.attr('placeholder') // do not mess with default behavior this.$element.removeAttr('placeholder') this.isPassword = this.$element.is('[type=password]') if (this.isPassword) this.makeReplacement() this.$replacement.on({ 'keydown.placeholder': $.proxy(this.hide, this) , 'focus.placeholder drop.placeholder click.placeholder': $.proxy(this.setCaret, this) }) this.$element.on({ 'blur.placeholder keyup.placeholder': $.proxy(this.show, this) }) this.show() } // Set or get input value // Setting value toggles placeholder , val: function (value) { if (value === undefined) { return this.isHidden ? this.$element[0].value : ''; } if (value === '') { if (this.isHidden) { this.$element[0].value = value this.show() } } else { if (!this.isHidden) this.hide() this.$element[0].value = value } return this } // Hide placeholder at user input , hide: function (e) { var isActiveElement = this.$replacement.is(':focus') if (this.isHidden) return; if (!e || !(e.shiftKey && e.keyCode === 16) && e.keyCode !== 9) { this.isHidden = true if (this.isPassword) { this.$replacement.before(this.$element.show()).hide() if (isActiveElement) this.$element.focus() } else { this.$element[0].value = '' this.$element.removeClass(this.options.className) } } } // Show placeholder on blur and keyup , show: function (e) { var isActiveElement = this.$element.is(':focus') if (!this.isHidden) return; if (this.$element[0].value === '') { this.isHidden = false if (this.isPassword) { this.$element.before(this.$replacement.show()).hide() if (isActiveElement) this.$replacement.focus() } else { this.$element[0].value = this.placeholderAttr this.$element.addClass(this.options.className) if (isActiveElement) this.setCaret(e) } } } // Set caret at the beginning of the input , setCaret: function (e) { if (e && !this.isHidden) { setCaretTo(this.$replacement[0], 0) e.preventDefault() } } // Make and return replacement element , makeReplacement: function () { // we can't use $.fn.clone because ie <= 8 doesn't allow type change var replacementAttributes = $.extend( getAttributes(this.$element[0]) , { 'type': 'text' , 'value': this.placeholderAttr } ) // replacement should not have input name delete replacementAttributes.name this.$replacement = $('<input>', replacementAttributes) .data('placeholder', this) .addClass(this.options.className) return this.$replacement; } } // Override jQuery val and prop hooks $.valHooks.input = $.valHooks.textarea = $.propHooks.value = { get: function (element) { var placeholder = $(element).data('placeholder') return placeholder ? placeholder.val() : element.value; } , set: function (element, value) { var placeholder = $(element).data('placeholder') return placeholder ? placeholder.val(value) : element.value = value; } } // Plugin definition $.fn.placeholder = function (option) { return this.each(function () { var $this = $(this) , data = $this.data('placeholder') , options = $.extend({}, $.fn.placeholder.defaults, typeof option === 'object' && option) if (!data && $this.is('[placeholder]') && (options.force || !isInputSupported && $this.is('input') || !isTextareaSupported && $this.is('textarea'))) { $this.data('placeholder', data = new Placeholder(this, options)) } if (data && typeof option === 'string') data[option]() }) } $.fn.placeholder.defaults = { force: false , className: 'placeholder' } $.fn.placeholder.Constructor = Placeholder // Events $(document).on('submit.placeholder', 'form', function () { // Clear the placeholder values so they don't get submitted $placeholders.placeholder('hide') // And then restore them back setTimeout(function () { $placeholders.placeholder('show') }, 10) }) $(window).on('beforeunload.placeholder', function () { // Clear placeholders upon page reload $placeholders.placeholder('hide') }) return Placeholder }));