// ========================================================================== // Project: SproutCore - JavaScript Application Framework // Copyright: ©2006-2011 Strobe Inc. and contributors. // ©2008-2011 Apple Inc. All rights reserved. // License: Licensed under MIT license (see license.js) // ========================================================================== sc_require('views/template'); /** @class */ SC.TextField = SC.TemplateView.extend( /** @scope SC.TextField.prototype */ { classNames: ['sc-text-field'], // we can't use bindAttr because of a race condition: // // when `value` is set, the bindAttr observer immediately calls // `get` in order to persist it to the DOM, but because we made // the `value` property idempotent, when it gets called by // bindAttr, it fetches the not-yet-updated value from the DOM // and returns it. // // In short, because we need to be able to catch changes to the // DOM made directly, we cannot also rely on bindAttr to update // the property: a chicken-and-egg problem. template: SC.Handlebars.compile(''), didCreateLayer: function() { var self = this; var input = this.$('input'); input.val(this._value); SC.Event.add(input, 'focus', this, this.focusIn); SC.Event.add(input, 'blur', this, this.focusOut); this.$('input').bind('change', function() { self.domValueDidChange(SC.$(this)); }); }, /** The problem this property is trying to solve is twofold: 1. Make it possible to set the value of a text field that has not yet been inserted into the DOM 2. Make sure that `value` properly reflects changes made directly to the element's `value` property. In order to achieve (2), we need to make the property volatile, so that SproutCore will call the getter no matter what if get() is called. In order to achieve (1), we need to store a local cache of the value, so that SproutCore can set the proper value as soon as the underlying DOM element is created. */ value: function(key, value) { var input = this.$('input'); if (value !== undefined) { this._value = value; input.val(value); } else if (input.length) { if(window.billy) { debugger; } this._value = value = input.val(); } else { if(window.billy) { debugger; } value = this._value; } return value; }.property().idempotent(), domValueDidChange: function(jquery) { this.set('value', jquery.val()); }, focusIn: function(event) { this.becomeFirstResponder(); this.tryToPerform('focus', event); }, focusOut: function(event) { this.resignFirstResponder(); this.tryToPerform('blur', event); }, willLoseFirstResponder: function() { this.notifyPropertyChange('value'); }, keyUp: function(evt) { this.domValueDidChange(this.$('input')); if (evt.keyCode === 13) { return this.insertNewline(evt); } else if (evt.keyCode === 27) { return this.cancel(evt); } return true; } }); SC.TextFieldSupport = /** @scope SC.TextFieldSupport */{ /** @private Used internally to store value because the layer may not exist */ _value: null, /** @type String @default null */ value: function(key, value) { var input = this.$('input'); if (value !== undefined) { this._value = value; input.val(value); } else { if (input.length > 0) { value = this._value = input.val(); } else { value = this._value; } } return value; }.property().idempotent(), didCreateLayer: function() { var input = this.$('input'); input.val(this._value); SC.Event.add(input, 'focus', this, this.focusIn); SC.Event.add(input, 'blur', this, this.focusOut); }, focusIn: function(event) { this.becomeFirstResponder(); this.tryToPerform('focus', event); }, focusOut: function(event) { this.resignFirstResponder(); this.tryToPerform('blur', event); }, /** @private Make sure our input value is synced with any bindings. In some cases, such as auto-filling, a value can get changed without an event firing. We could do this on focusOut, but blur can potentially get called after other events. */ willLoseFirstResponder: function() { this.notifyPropertyChange('value'); }, keyUp: function(event) { if (event.keyCode === 13) { return this.tryToPerform('insertNewline', event); } else if (event.keyCode === 27) { return this.tryToPerform('cancel', event); } } };