12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816 |
- // Copyright 2006 The Closure Library Authors. All Rights Reserved.
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS-IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- // All Rights Reserved.
- /**
- * @fileoverview Class to encapsulate an editable field. Always uses an
- * iframe to contain the editable area, never inherits the style of the
- * surrounding page, and is always a fixed height.
- *
- * @author nicksantos@google.com (Nick Santos)
- * @see ../demos/editor/editor.html
- * @see ../demos/editor/field_basic.html
- */
- goog.provide('goog.editor.Field');
- goog.provide('goog.editor.Field.EventType');
- goog.require('goog.a11y.aria');
- goog.require('goog.a11y.aria.Role');
- goog.require('goog.array');
- goog.require('goog.asserts');
- goog.require('goog.async.Delay');
- goog.require('goog.dom');
- goog.require('goog.dom.Range');
- goog.require('goog.dom.TagName');
- goog.require('goog.dom.classlist');
- goog.require('goog.dom.safe');
- goog.require('goog.editor.BrowserFeature');
- goog.require('goog.editor.Command');
- goog.require('goog.editor.Plugin');
- goog.require('goog.editor.icontent');
- goog.require('goog.editor.icontent.FieldFormatInfo');
- goog.require('goog.editor.icontent.FieldStyleInfo');
- goog.require('goog.editor.node');
- goog.require('goog.editor.range');
- goog.require('goog.events');
- goog.require('goog.events.EventHandler');
- goog.require('goog.events.EventTarget');
- goog.require('goog.events.EventType');
- goog.require('goog.events.KeyCodes');
- goog.require('goog.functions');
- goog.require('goog.html.SafeHtml');
- goog.require('goog.html.legacyconversions');
- goog.require('goog.log');
- goog.require('goog.log.Level');
- goog.require('goog.string');
- goog.require('goog.string.Unicode');
- goog.require('goog.style');
- goog.require('goog.userAgent');
- goog.require('goog.userAgent.product');
- /**
- * This class encapsulates an editable field.
- *
- * event: load Fires when the field is loaded
- * event: unload Fires when the field is unloaded (made not editable)
- *
- * event: beforechange Fires before the content of the field might change
- *
- * event: delayedchange Fires a short time after field has changed. If multiple
- * change events happen really close to each other only
- * the last one will trigger the delayedchange event.
- *
- * event: beforefocus Fires before the field becomes active
- * event: focus Fires when the field becomes active. Fires after the blur event
- * event: blur Fires when the field becomes inactive
- *
- * TODO: figure out if blur or beforefocus fires first in IE and make FF match
- *
- * @param {string} id An identifer for the field. This is used to find the
- * field and the element associated with this field.
- * @param {Document=} opt_doc The document that the element with the given
- * id can be found in. If not provided, the default document is used.
- * @constructor
- * @extends {goog.events.EventTarget}
- */
- goog.editor.Field = function(id, opt_doc) {
- goog.events.EventTarget.call(this);
- /**
- * The id for this editable field, which must match the id of the element
- * associated with this field.
- * @type {string}
- */
- this.id = id;
- /**
- * The hash code for this field. Should be equal to the id.
- * @type {string}
- * @private
- */
- this.hashCode_ = id;
- /**
- * Dom helper for the editable node.
- * @type {goog.dom.DomHelper}
- * @protected
- */
- this.editableDomHelper = null;
- /**
- * Map of class id to registered plugin.
- * @type {Object}
- * @private
- */
- this.plugins_ = {};
- /**
- * Plugins registered on this field, indexed by the goog.editor.Plugin.Op
- * that they support.
- * @type {Object<Array<goog.editor.Plugin>>}
- * @private
- */
- this.indexedPlugins_ = {};
- for (var op in goog.editor.Plugin.OPCODE) {
- this.indexedPlugins_[op] = [];
- }
- /**
- * Additional styles to install for the editable field.
- * @type {string}
- * @protected
- */
- this.cssStyles = '';
- // The field will not listen to change events until it has finished loading
- /** @private */
- this.stoppedEvents_ = {};
- this.stopEvent(goog.editor.Field.EventType.CHANGE);
- this.stopEvent(goog.editor.Field.EventType.DELAYEDCHANGE);
- /** @private */
- this.isModified_ = false;
- /** @private */
- this.isEverModified_ = false;
- /** @private */
- this.delayedChangeTimer_ = new goog.async.Delay(
- this.dispatchDelayedChange_, goog.editor.Field.DELAYED_CHANGE_FREQUENCY,
- this);
- /** @private */
- this.debouncedEvents_ = {};
- for (var key in goog.editor.Field.EventType) {
- this.debouncedEvents_[goog.editor.Field.EventType[key]] = 0;
- }
- if (goog.editor.BrowserFeature.USE_MUTATION_EVENTS) {
- /** @private */
- this.changeTimerGecko_ = new goog.async.Delay(
- this.handleChange, goog.editor.Field.CHANGE_FREQUENCY, this);
- }
- /**
- * @type {goog.events.EventHandler<!goog.editor.Field>}
- * @protected
- */
- this.eventRegister = new goog.events.EventHandler(this);
- // Wrappers around this field, to be disposed when the field is disposed.
- /** @private */
- this.wrappers_ = [];
- /** @private */
- this.loadState_ = goog.editor.Field.LoadState_.UNEDITABLE;
- var doc = opt_doc || document;
- /**
- * The dom helper for the node to be made editable.
- * @type {goog.dom.DomHelper}
- * @protected
- */
- this.originalDomHelper = goog.dom.getDomHelper(doc);
- /**
- * The original node that is being made editable, or null if it has
- * not yet been found.
- * @type {Element}
- * @protected
- */
- this.originalElement = this.originalDomHelper.getElement(this.id);
- /**
- * @private {boolean}
- */
- this.followLinkInNewWindow_ =
- goog.editor.BrowserFeature.FOLLOWS_EDITABLE_LINKS;
- // Default to the same window as the field is in.
- /** @private */
- this.appWindow_ = this.originalDomHelper.getWindow();
- };
- goog.inherits(goog.editor.Field, goog.events.EventTarget);
- /**
- * The editable dom node.
- * @type {Element}
- * TODO(user): Make this private!
- */
- goog.editor.Field.prototype.field = null;
- /**
- * Logging object.
- * @type {goog.log.Logger}
- * @protected
- */
- goog.editor.Field.prototype.logger = goog.log.getLogger('goog.editor.Field');
- /**
- * Event types that can be stopped/started.
- * @enum {string}
- */
- goog.editor.Field.EventType = {
- /**
- * Dispatched when the command state of the selection may have changed. This
- * event should be listened to for updating toolbar state.
- */
- COMMAND_VALUE_CHANGE: 'cvc',
- /**
- * Dispatched when the field is loaded and ready to use.
- */
- LOAD: 'load',
- /**
- * Dispatched when the field is fully unloaded and uneditable.
- */
- UNLOAD: 'unload',
- /**
- * Dispatched before the field contents are changed.
- */
- BEFORECHANGE: 'beforechange',
- /**
- * Dispatched when the field contents change, in FF only.
- * Used for internal resizing, please do not use.
- */
- CHANGE: 'change',
- /**
- * Dispatched on a slight delay after changes are made.
- * Use for autosave, or other times your app needs to know
- * that the field contents changed.
- */
- DELAYEDCHANGE: 'delayedchange',
- /**
- * Dispatched before focus in moved into the field.
- */
- BEFOREFOCUS: 'beforefocus',
- /**
- * Dispatched when focus is moved into the field.
- */
- FOCUS: 'focus',
- /**
- * Dispatched when the field is blurred.
- */
- BLUR: 'blur',
- /**
- * Dispatched before tab is handled by the field. This is a legacy way
- * of controlling tab behavior. Use trog.plugins.AbstractTabHandler now.
- */
- BEFORETAB: 'beforetab',
- /**
- * Dispatched after the iframe containing the field is resized, so that UI
- * components which contain it can respond.
- */
- IFRAME_RESIZED: 'ifrsz',
- /**
- * Dispatched after a user action that will eventually fire a SELECTIONCHANGE
- * event. For mouseups, this is fired immediately before SELECTIONCHANGE,
- * since {@link #handleMouseUp_} fires SELECTIONCHANGE immediately. May be
- * fired up to {@link #SELECTION_CHANGE_FREQUENCY_} ms before SELECTIONCHANGE
- * is fired in the case of keyup events, since they use
- * {@link #selectionChangeTimer_}.
- */
- BEFORESELECTIONCHANGE: 'beforeselectionchange',
- /**
- * Dispatched when the selection changes.
- * Use handleSelectionChange from plugin API instead of listening
- * directly to this event.
- */
- SELECTIONCHANGE: 'selectionchange'
- };
- /**
- * The load state of the field.
- * @enum {number}
- * @private
- */
- goog.editor.Field.LoadState_ = {
- UNEDITABLE: 0,
- LOADING: 1,
- EDITABLE: 2
- };
- /**
- * The amount of time that a debounce blocks an event.
- * TODO(nicksantos): As of 9/30/07, this is only used for blocking
- * a keyup event after a keydown. We might need to tweak this for other
- * types of events. Maybe have a per-event debounce time?
- * @type {number}
- * @private
- */
- goog.editor.Field.DEBOUNCE_TIME_MS_ = 500;
- /**
- * There is at most one "active" field at a time. By "active" field, we mean
- * a field that has focus and is being used.
- * @type {?string}
- * @private
- */
- goog.editor.Field.activeFieldId_ = null;
- /**
- * Whether this field is in "modal interaction" mode. This usually
- * means that it's being edited by a dialog.
- * @type {boolean}
- * @private
- */
- goog.editor.Field.prototype.inModalMode_ = false;
- /**
- * The window where dialogs and bubbles should be rendered.
- * @type {!Window}
- * @private
- */
- goog.editor.Field.prototype.appWindow_;
- /**
- * Target node to be used when dispatching SELECTIONCHANGE asynchronously on
- * mouseup (to avoid IE quirk). Should be set just before starting the timer and
- * nulled right after consuming.
- * @type {Node}
- * @private
- */
- goog.editor.Field.prototype.selectionChangeTarget_;
- /**
- * Flag controlling wether to capture mouse up events on the window or not.
- * @type {boolean}
- * @private
- */
- goog.editor.Field.prototype.useWindowMouseUp_ = false;
- /**
- * FLag indicating the handling of a mouse event sequence.
- * @type {boolean}
- * @private
- */
- goog.editor.Field.prototype.waitingForMouseUp_ = false;
- /**
- * Sets the active field id.
- * @param {?string} fieldId The active field id.
- */
- goog.editor.Field.setActiveFieldId = function(fieldId) {
- goog.editor.Field.activeFieldId_ = fieldId;
- };
- /**
- * @return {?string} The id of the active field.
- */
- goog.editor.Field.getActiveFieldId = function() {
- return goog.editor.Field.activeFieldId_;
- };
- /**
- * Sets flag to control whether to use window mouse up after seeing
- * a mouse down operation on the field.
- * @param {boolean} flag True to track window mouse up.
- */
- goog.editor.Field.prototype.setUseWindowMouseUp = function(flag) {
- goog.asserts.assert(
- !flag || !this.usesIframe(),
- 'procssing window mouse up should only be enabled when not using iframe');
- this.useWindowMouseUp_ = flag;
- };
- /**
- * @return {boolean} Whether we're in modal interaction mode. When this
- * returns true, another plugin is interacting with the field contents
- * in a synchronous way, and expects you not to make changes to
- * the field's DOM structure or selection.
- */
- goog.editor.Field.prototype.inModalMode = function() {
- return this.inModalMode_;
- };
- /**
- * @param {boolean} inModalMode Sets whether we're in modal interaction mode.
- */
- goog.editor.Field.prototype.setModalMode = function(inModalMode) {
- this.inModalMode_ = inModalMode;
- };
- /**
- * Returns a string usable as a hash code for this field. For field's
- * that were created with an id, the hash code is guaranteed to be the id.
- * TODO(user): I think we can get rid of this. Seems only used from editor.
- * @return {string} The hash code for this editable field.
- */
- goog.editor.Field.prototype.getHashCode = function() {
- return this.hashCode_;
- };
- /**
- * Returns the editable DOM element or null if this field
- * is not editable.
- * <p>On IE or Safari this is the element with contentEditable=true
- * (in whitebox mode, the iFrame body).
- * <p>On Gecko this is the iFrame body
- * TODO(user): How do we word this for subclass version?
- * @return {Element} The editable DOM element, defined as above.
- */
- goog.editor.Field.prototype.getElement = function() {
- return this.field;
- };
- /**
- * Returns original DOM element that is being made editable by Trogedit or
- * null if that element has not yet been found in the appropriate document.
- * @return {Element} The original element.
- */
- goog.editor.Field.prototype.getOriginalElement = function() {
- return this.originalElement;
- };
- /**
- * Registers a keyboard event listener on the field. This is necessary for
- * Gecko since the fields are contained in an iFrame and there is no way to
- * auto-propagate key events up to the main window.
- * @param {string|Array<string>} type Event type to listen for or array of
- * event types, for example goog.events.EventType.KEYDOWN.
- * @param {Function} listener Function to be used as the listener.
- * @param {boolean=} opt_capture Whether to use capture phase (optional,
- * defaults to false).
- * @param {Object=} opt_handler Object in whose scope to call the listener.
- */
- goog.editor.Field.prototype.addListener = function(
- type, listener, opt_capture, opt_handler) {
- var elem = this.getElement();
- // On Gecko, keyboard events only reliably fire on the document element when
- // using an iframe.
- if (goog.editor.BrowserFeature.USE_DOCUMENT_FOR_KEY_EVENTS && elem &&
- this.usesIframe()) {
- elem = elem.ownerDocument;
- }
- if (opt_handler) {
- this.eventRegister.listenWithScope(
- elem, type, listener, opt_capture, opt_handler);
- } else {
- this.eventRegister.listen(elem, type, listener, opt_capture);
- }
- };
- /**
- * Returns the registered plugin with the given classId.
- * @param {string} classId classId of the plugin.
- * @return {goog.editor.Plugin} Registered plugin with the given classId.
- */
- goog.editor.Field.prototype.getPluginByClassId = function(classId) {
- return this.plugins_[classId];
- };
- /**
- * Registers the plugin with the editable field.
- * @param {goog.editor.Plugin} plugin The plugin to register.
- */
- goog.editor.Field.prototype.registerPlugin = function(plugin) {
- var classId = plugin.getTrogClassId();
- if (this.plugins_[classId]) {
- goog.log.error(
- this.logger, 'Cannot register the same class of plugin twice.');
- }
- this.plugins_[classId] = plugin;
- // Only key events and execute should have these has* functions with a custom
- // handler array since they need to be very careful about performance.
- // The rest of the plugin hooks should be event-based.
- for (var op in goog.editor.Plugin.OPCODE) {
- var opcode = goog.editor.Plugin.OPCODE[op];
- if (plugin[opcode]) {
- this.indexedPlugins_[op].push(plugin);
- }
- }
- plugin.registerFieldObject(this);
- // By default we enable all plugins for fields that are currently loaded.
- if (this.isLoaded()) {
- plugin.enable(this);
- }
- };
- /**
- * Unregisters the plugin with this field.
- * @param {goog.editor.Plugin} plugin The plugin to unregister.
- */
- goog.editor.Field.prototype.unregisterPlugin = function(plugin) {
- var classId = plugin.getTrogClassId();
- if (!this.plugins_[classId]) {
- goog.log.error(
- this.logger, 'Cannot unregister a plugin that isn\'t registered.');
- }
- delete this.plugins_[classId];
- for (var op in goog.editor.Plugin.OPCODE) {
- var opcode = goog.editor.Plugin.OPCODE[op];
- if (plugin[opcode]) {
- goog.array.remove(this.indexedPlugins_[op], plugin);
- }
- }
- plugin.unregisterFieldObject(this);
- };
- /**
- * Sets the value that will replace the style attribute of this field's
- * element when the field is made non-editable. This method is called with the
- * current value of the style attribute when the field is made editable.
- * @param {string} cssText The value of the style attribute.
- */
- goog.editor.Field.prototype.setInitialStyle = function(cssText) {
- this.cssText = cssText;
- };
- /**
- * Reset the properties on the original field element to how it was before
- * it was made editable.
- */
- goog.editor.Field.prototype.resetOriginalElemProperties = function() {
- var field = this.getOriginalElement();
- field.removeAttribute('contentEditable');
- field.removeAttribute('g_editable');
- field.removeAttribute('role');
- if (!this.id) {
- field.removeAttribute('id');
- } else {
- field.id = this.id;
- }
- field.className = this.savedClassName_ || '';
- var cssText = this.cssText;
- if (!cssText) {
- field.removeAttribute('style');
- } else {
- goog.dom.setProperties(field, {'style': cssText});
- }
- if (goog.isString(this.originalFieldLineHeight_)) {
- goog.style.setStyle(field, 'lineHeight', this.originalFieldLineHeight_);
- this.originalFieldLineHeight_ = null;
- }
- };
- /**
- * Checks the modified state of the field.
- * Note: Changes that take place while the goog.editor.Field.EventType.CHANGE
- * event is stopped do not effect the modified state.
- * @param {boolean=} opt_useIsEverModified Set to true to check if the field
- * has ever been modified since it was created, otherwise checks if the field
- * has been modified since the last goog.editor.Field.EventType.DELAYEDCHANGE
- * event was dispatched.
- * @return {boolean} Whether the field has been modified.
- */
- goog.editor.Field.prototype.isModified = function(opt_useIsEverModified) {
- return opt_useIsEverModified ? this.isEverModified_ : this.isModified_;
- };
- /**
- * Number of milliseconds after a change when the change event should be fired.
- * @type {number}
- */
- goog.editor.Field.CHANGE_FREQUENCY = 15;
- /**
- * Number of milliseconds between delayed change events.
- * @type {number}
- */
- goog.editor.Field.DELAYED_CHANGE_FREQUENCY = 250;
- /**
- * @return {boolean} Whether the field is implemented as an iframe.
- */
- goog.editor.Field.prototype.usesIframe = goog.functions.TRUE;
- /**
- * @return {boolean} Whether the field should be rendered with a fixed
- * height, or should expand to fit its contents.
- */
- goog.editor.Field.prototype.isFixedHeight = goog.functions.TRUE;
- /**
- * @return {boolean} Whether the field should be refocused on input.
- * This is a workaround for the iOS bug that text input doesn't work
- * when the main window listens touch events.
- */
- goog.editor.Field.prototype.shouldRefocusOnInputMobileSafari =
- goog.functions.FALSE;
- /**
- * Map of keyCodes (not charCodes) that cause changes in the field contents.
- * @type {Object}
- * @private
- */
- goog.editor.Field.KEYS_CAUSING_CHANGES_ = {
- 46: true, // DEL
- 8: true // BACKSPACE
- };
- if (!goog.userAgent.IE) {
- // Only IE doesn't change the field by default upon tab.
- // TODO(user): This really isn't right now that we have tab plugins.
- goog.editor.Field.KEYS_CAUSING_CHANGES_[9] = true; // TAB
- }
- /**
- * Map of keyCodes (not charCodes) that when used in conjunction with the
- * Ctrl key cause changes in the field contents. These are the keys that are
- * not handled by basic formatting trogedit plugins.
- * @type {Object}
- * @private
- */
- goog.editor.Field.CTRL_KEYS_CAUSING_CHANGES_ = {
- 86: true, // V
- 88: true // X
- };
- if (goog.userAgent.WINDOWS && !goog.userAgent.GECKO) {
- // In IE and Webkit, input from IME (Input Method Editor) does not generate a
- // keypress event so we have to rely on the keydown event. This way we have
- // false positives while the user is using keyboard to select the
- // character to input, but it is still better than the false negatives
- // that ignores user's final input at all.
- goog.editor.Field.KEYS_CAUSING_CHANGES_[229] = true; // from IME;
- }
- /**
- * Returns true if the keypress generates a change in contents.
- * @param {goog.events.BrowserEvent} e The event.
- * @param {boolean} testAllKeys True to test for all types of generating keys.
- * False to test for only the keys found in
- * goog.editor.Field.KEYS_CAUSING_CHANGES_.
- * @return {boolean} Whether the keypress generates a change in contents.
- * @private
- */
- goog.editor.Field.isGeneratingKey_ = function(e, testAllKeys) {
- if (goog.editor.Field.isSpecialGeneratingKey_(e)) {
- return true;
- }
- return !!(
- testAllKeys && !(e.ctrlKey || e.metaKey) &&
- (!goog.userAgent.GECKO || e.charCode));
- };
- /**
- * Returns true if the keypress generates a change in the contents.
- * due to a special key listed in goog.editor.Field.KEYS_CAUSING_CHANGES_
- * @param {goog.events.BrowserEvent} e The event.
- * @return {boolean} Whether the keypress generated a change in the contents.
- * @private
- */
- goog.editor.Field.isSpecialGeneratingKey_ = function(e) {
- var testCtrlKeys = (e.ctrlKey || e.metaKey) &&
- e.keyCode in goog.editor.Field.CTRL_KEYS_CAUSING_CHANGES_;
- var testRegularKeys = !(e.ctrlKey || e.metaKey) &&
- e.keyCode in goog.editor.Field.KEYS_CAUSING_CHANGES_;
- return testCtrlKeys || testRegularKeys;
- };
- /**
- * Sets the application window.
- * @param {!Window} appWindow The window where dialogs and bubbles should be
- * rendered.
- */
- goog.editor.Field.prototype.setAppWindow = function(appWindow) {
- this.appWindow_ = appWindow;
- };
- /**
- * Returns the "application" window, where dialogs and bubbles
- * should be rendered.
- * @return {!Window} The window.
- */
- goog.editor.Field.prototype.getAppWindow = function() {
- return this.appWindow_;
- };
- /**
- * Sets the zIndex that the field should be based off of.
- * TODO(user): Get rid of this completely. Here for Sites.
- * Should this be set directly on UI plugins?
- *
- * @param {number} zindex The base zIndex of the editor.
- */
- goog.editor.Field.prototype.setBaseZindex = function(zindex) {
- this.baseZindex_ = zindex;
- };
- /**
- * Returns the zindex of the base level of the field.
- *
- * @return {number} The base zindex of the editor.
- */
- goog.editor.Field.prototype.getBaseZindex = function() {
- return this.baseZindex_ || 0;
- };
- /**
- * Sets up the field object and window util of this field, and enables this
- * editable field with all registered plugins.
- * This is essential to the initialization of the field.
- * It must be called when the field becomes fully loaded and editable.
- * @param {Element} field The field property.
- * @protected
- */
- goog.editor.Field.prototype.setupFieldObject = function(field) {
- this.loadState_ = goog.editor.Field.LoadState_.EDITABLE;
- this.field = field;
- this.editableDomHelper = goog.dom.getDomHelper(field);
- this.isModified_ = false;
- this.isEverModified_ = false;
- field.setAttribute('g_editable', 'true');
- goog.a11y.aria.setRole(field, goog.a11y.aria.Role.TEXTBOX);
- };
- /**
- * Help make the field not editable by setting internal data structures to null,
- * and disabling this field with all registered plugins.
- * @private
- */
- goog.editor.Field.prototype.tearDownFieldObject_ = function() {
- this.loadState_ = goog.editor.Field.LoadState_.UNEDITABLE;
- for (var classId in this.plugins_) {
- var plugin = this.plugins_[classId];
- if (!plugin.activeOnUneditableFields()) {
- plugin.disable(this);
- }
- }
- this.field = null;
- this.editableDomHelper = null;
- };
- /**
- * Initialize listeners on the field.
- * @private
- */
- goog.editor.Field.prototype.setupChangeListeners_ = function() {
- if ((goog.userAgent.product.IPHONE || goog.userAgent.product.IPAD) &&
- this.usesIframe() && this.shouldRefocusOnInputMobileSafari()) {
- // This is a workaround for the iOS bug that text input doesn't work
- // when the main window listens touch events.
- var editWindow = this.getEditableDomHelper().getWindow();
- this.boundRefocusListenerMobileSafari_ =
- goog.bind(editWindow.focus, editWindow);
- editWindow.addEventListener(
- goog.events.EventType.KEYDOWN, this.boundRefocusListenerMobileSafari_,
- false);
- editWindow.addEventListener(
- goog.events.EventType.TOUCHEND, this.boundRefocusListenerMobileSafari_,
- false);
- }
- if (goog.userAgent.OPERA && this.usesIframe()) {
- // We can't use addListener here because we need to listen on the window,
- // and removing listeners on window objects from the event register throws
- // an exception if the window is closed.
- this.boundFocusListenerOpera_ =
- goog.bind(this.dispatchFocusAndBeforeFocus_, this);
- this.boundBlurListenerOpera_ = goog.bind(this.dispatchBlur, this);
- var editWindow = this.getEditableDomHelper().getWindow();
- editWindow.addEventListener(
- goog.events.EventType.FOCUS, this.boundFocusListenerOpera_, false);
- editWindow.addEventListener(
- goog.events.EventType.BLUR, this.boundBlurListenerOpera_, false);
- } else {
- if (goog.editor.BrowserFeature.SUPPORTS_FOCUSIN) {
- this.addListener(goog.events.EventType.FOCUS, this.dispatchFocus_);
- this.addListener(
- goog.events.EventType.FOCUSIN, this.dispatchBeforeFocus_);
- } else {
- this.addListener(
- goog.events.EventType.FOCUS, this.dispatchFocusAndBeforeFocus_);
- }
- this.addListener(
- goog.events.EventType.BLUR, this.dispatchBlur,
- goog.editor.BrowserFeature.USE_MUTATION_EVENTS);
- }
- if (goog.editor.BrowserFeature.USE_MUTATION_EVENTS) {
- // Ways to detect changes in Mozilla:
- //
- // keypress - check event.charCode (only typable characters has a
- // charCode), but also keyboard commands lile Ctrl+C will
- // return a charCode.
- // dragdrop - fires when the user drops something. This does not necessary
- // lead to a change but we cannot detect if it will or not
- //
- // Known Issues: We cannot detect cut and paste using menus
- // We cannot detect when someone moves something out of the
- // field using drag and drop.
- //
- this.setupMutationEventHandlersGecko();
- } else {
- // Ways to detect that a change is about to happen in other browsers.
- // (IE and Safari have these events. Opera appears to work, but we haven't
- // researched it.)
- //
- // onbeforepaste
- // onbeforecut
- // ondrop - happens when the user drops something on the editable text
- // field the value at this time does not contain the dropped text
- // ondragleave - when the user drags something from the current document.
- // This might not cause a change if the action was copy
- // instead of move
- // onkeypress - IE only fires keypress events if the key will generate
- // output. It will not trigger for delete and backspace
- // onkeydown - For delete and backspace
- //
- // known issues: IE triggers beforepaste just by opening the edit menu
- // delete at the end should not cause beforechange
- // backspace at the beginning should not cause beforechange
- // see above in ondragleave
- // TODO(user): Why don't we dispatchBeforeChange from the
- // handleDrop event for all browsers?
- this.addListener(
- ['beforecut', 'beforepaste', 'drop', 'dragend'],
- this.dispatchBeforeChange);
- this.addListener(
- ['cut', 'paste'], goog.functions.lock(this.dispatchChange));
- this.addListener('drop', this.handleDrop_);
- }
- // TODO(user): Figure out why we use dragend vs dragdrop and
- // document this better.
- var dropEventName = goog.userAgent.WEBKIT ? 'dragend' : 'dragdrop';
- this.addListener(dropEventName, this.handleDrop_);
- this.addListener(goog.events.EventType.KEYDOWN, this.handleKeyDown_);
- this.addListener(goog.events.EventType.KEYPRESS, this.handleKeyPress_);
- this.addListener(goog.events.EventType.KEYUP, this.handleKeyUp_);
- this.selectionChangeTimer_ = new goog.async.Delay(
- this.handleSelectionChangeTimer_,
- goog.editor.Field.SELECTION_CHANGE_FREQUENCY_, this);
- if (this.followLinkInNewWindow_) {
- this.addListener(
- goog.events.EventType.CLICK, goog.editor.Field.cancelLinkClick_);
- }
- this.addListener(goog.events.EventType.MOUSEDOWN, this.handleMouseDown_);
- if (this.useWindowMouseUp_) {
- this.eventRegister.listen(
- this.editableDomHelper.getDocument(), goog.events.EventType.MOUSEUP,
- this.handleMouseUp_);
- this.addListener(goog.events.EventType.DRAGSTART, this.handleDragStart_);
- } else {
- this.addListener(goog.events.EventType.MOUSEUP, this.handleMouseUp_);
- }
- };
- /**
- * Frequency to check for selection changes.
- * @type {number}
- * @private
- */
- goog.editor.Field.SELECTION_CHANGE_FREQUENCY_ = 250;
- /**
- * Stops all listeners and timers.
- * @protected
- */
- goog.editor.Field.prototype.clearListeners = function() {
- if (this.eventRegister) {
- this.eventRegister.removeAll();
- }
- if ((goog.userAgent.product.IPHONE || goog.userAgent.product.IPAD) &&
- this.usesIframe() && this.shouldRefocusOnInputMobileSafari()) {
- try {
- var editWindow = this.getEditableDomHelper().getWindow();
- editWindow.removeEventListener(
- goog.events.EventType.KEYDOWN, this.boundRefocusListenerMobileSafari_,
- false);
- editWindow.removeEventListener(
- goog.events.EventType.TOUCHEND,
- this.boundRefocusListenerMobileSafari_, false);
- } catch (e) {
- // The editWindow no longer exists, or has been navigated to a different-
- // origin URL. Either way, the event listeners have already been removed
- // for us.
- }
- delete this.boundRefocusListenerMobileSafari_;
- }
- if (goog.userAgent.OPERA && this.usesIframe()) {
- try {
- var editWindow = this.getEditableDomHelper().getWindow();
- editWindow.removeEventListener(
- goog.events.EventType.FOCUS, this.boundFocusListenerOpera_, false);
- editWindow.removeEventListener(
- goog.events.EventType.BLUR, this.boundBlurListenerOpera_, false);
- } catch (e) {
- // The editWindow no longer exists, or has been navigated to a different-
- // origin URL. Either way, the event listeners have already been removed
- // for us.
- }
- delete this.boundFocusListenerOpera_;
- delete this.boundBlurListenerOpera_;
- }
- if (this.changeTimerGecko_) {
- this.changeTimerGecko_.stop();
- }
- this.delayedChangeTimer_.stop();
- };
- /** @override */
- goog.editor.Field.prototype.disposeInternal = function() {
- if (this.isLoading() || this.isLoaded()) {
- goog.log.warning(this.logger, 'Disposing a field that is in use.');
- }
- if (this.getOriginalElement()) {
- this.execCommand(goog.editor.Command.CLEAR_LOREM);
- }
- this.tearDownFieldObject_();
- this.clearListeners();
- this.clearFieldLoadListener_();
- this.originalDomHelper = null;
- if (this.eventRegister) {
- this.eventRegister.dispose();
- this.eventRegister = null;
- }
- this.removeAllWrappers();
- if (goog.editor.Field.getActiveFieldId() == this.id) {
- goog.editor.Field.setActiveFieldId(null);
- }
- for (var classId in this.plugins_) {
- var plugin = this.plugins_[classId];
- if (plugin.isAutoDispose()) {
- plugin.dispose();
- }
- }
- delete (this.plugins_);
- goog.editor.Field.superClass_.disposeInternal.call(this);
- };
- /**
- * Attach an wrapper to this field, to be thrown out when the field
- * is disposed.
- * @param {goog.Disposable} wrapper The wrapper to attach.
- */
- goog.editor.Field.prototype.attachWrapper = function(wrapper) {
- this.wrappers_.push(wrapper);
- };
- /**
- * Removes all wrappers and destroys them.
- */
- goog.editor.Field.prototype.removeAllWrappers = function() {
- var wrapper;
- while (wrapper = this.wrappers_.pop()) {
- wrapper.dispose();
- }
- };
- /**
- * Sets whether activating a hyperlink in this editable field will open a new
- * window or not.
- * @param {boolean} followLinkInNewWindow
- */
- goog.editor.Field.prototype.setFollowLinkInNewWindow = function(
- followLinkInNewWindow) {
- this.followLinkInNewWindow_ = followLinkInNewWindow;
- };
- /**
- * List of mutation events in Gecko browsers.
- * @type {Array<string>}
- * @protected
- */
- goog.editor.Field.MUTATION_EVENTS_GECKO = [
- 'DOMNodeInserted', 'DOMNodeRemoved', 'DOMNodeRemovedFromDocument',
- 'DOMNodeInsertedIntoDocument', 'DOMCharacterDataModified'
- ];
- /**
- * Mutation events tell us when something has changed for mozilla.
- * @protected
- */
- goog.editor.Field.prototype.setupMutationEventHandlersGecko = function() {
- // Always use DOMSubtreeModified on Gecko when not using an iframe so that
- // DOM mutations outside the Field do not trigger handleMutationEventGecko_.
- if (goog.editor.BrowserFeature.HAS_DOM_SUBTREE_MODIFIED_EVENT ||
- !this.usesIframe()) {
- this.eventRegister.listen(
- this.getElement(), 'DOMSubtreeModified',
- this.handleMutationEventGecko_);
- } else {
- var doc = this.getEditableDomHelper().getDocument();
- this.eventRegister.listen(
- doc, goog.editor.Field.MUTATION_EVENTS_GECKO,
- this.handleMutationEventGecko_, true);
- // DOMAttrModified fires for a lot of events we want to ignore. This goes
- // through a different handler so that we can ignore many of these.
- this.eventRegister.listen(
- doc, 'DOMAttrModified',
- goog.bind(
- this.handleDomAttrChange, this, this.handleMutationEventGecko_),
- true);
- }
- };
- /**
- * Handle before change key events and fire the beforetab event if appropriate.
- * This needs to happen on keydown in IE and keypress in FF.
- * @param {goog.events.BrowserEvent} e The browser event.
- * @return {boolean} Whether to still perform the default key action. Only set
- * to true if the actual event has already been canceled.
- * @private
- */
- goog.editor.Field.prototype.handleBeforeChangeKeyEvent_ = function(e) {
- // There are two reasons to block a key:
- var block =
- // #1: to intercept a tab
- // TODO: possibly don't allow clients to intercept tabs outside of LIs and
- // maybe tables as well?
- (e.keyCode == goog.events.KeyCodes.TAB && !this.dispatchBeforeTab_(e)) ||
- // #2: to block a Firefox-specific bug where Macs try to navigate
- // back a page when you hit command+left arrow or comamnd-right arrow.
- // See https://bugzilla.mozilla.org/show_bug.cgi?id=341886
- // This was fixed in Firefox 29, but still exists in older versions.
- (goog.userAgent.GECKO && e.metaKey &&
- !goog.userAgent.isVersionOrHigher(29) &&
- (e.keyCode == goog.events.KeyCodes.LEFT ||
- e.keyCode == goog.events.KeyCodes.RIGHT));
- if (block) {
- e.preventDefault();
- return false;
- } else {
- // In Gecko we have both keyCode and charCode. charCode is for human
- // readable characters like a, b and c. However pressing ctrl+c and so on
- // also causes charCode to be set.
- // TODO(arv): Del at end of field or backspace at beginning should be
- // ignored.
- this.gotGeneratingKey_ = e.charCode ||
- goog.editor.Field.isGeneratingKey_(e, goog.userAgent.GECKO);
- if (this.gotGeneratingKey_) {
- this.dispatchBeforeChange();
- // TODO(robbyw): Should we return the value of the above?
- }
- }
- return true;
- };
- /**
- * Keycodes that result in a selectionchange event (e.g. the cursor moving).
- * @type {!Object<number, number>}
- */
- goog.editor.Field.SELECTION_CHANGE_KEYCODES = {
- 8: 1, // backspace
- 9: 1, // tab
- 13: 1, // enter
- 33: 1, // page up
- 34: 1, // page down
- 35: 1, // end
- 36: 1, // home
- 37: 1, // left
- 38: 1, // up
- 39: 1, // right
- 40: 1, // down
- 46: 1 // delete
- };
- /**
- * Map of keyCodes (not charCodes) that when used in conjunction with the
- * Ctrl key cause selection changes in the field contents. These are the keys
- * that are not handled by the basic formatting trogedit plugins. Note that
- * combinations like Ctrl-left etc are already handled in
- * SELECTION_CHANGE_KEYCODES
- * @type {Object}
- * @private
- */
- goog.editor.Field.CTRL_KEYS_CAUSING_SELECTION_CHANGES_ = {
- 65: true, // A
- 86: true, // V
- 88: true // X
- };
- /**
- * Map of keyCodes (not charCodes) that might need to be handled as a keyboard
- * shortcut (even when ctrl/meta key is not pressed) by some plugin. Currently
- * it is a small list. If it grows too big we can optimize it by using ranges
- * or extending it from SELECTION_CHANGE_KEYCODES
- * @type {Object}
- * @private
- */
- goog.editor.Field.POTENTIAL_SHORTCUT_KEYCODES_ = {
- 8: 1, // backspace
- 9: 1, // tab
- 13: 1, // enter
- 27: 1, // esc
- 33: 1, // page up
- 34: 1, // page down
- 37: 1, // left
- 38: 1, // up
- 39: 1, // right
- 40: 1 // down
- };
- /**
- * Calls all the plugins of the given operation, in sequence, with the
- * given arguments. This is short-circuiting: once one plugin cancels
- * the event, no more plugins will be invoked.
- * @param {goog.editor.Plugin.Op} op A plugin op.
- * @param {...*} var_args The arguments to the plugin.
- * @return {boolean} True if one of the plugins cancel the event, false
- * otherwise.
- * @private
- */
- goog.editor.Field.prototype.invokeShortCircuitingOp_ = function(op, var_args) {
- var plugins = this.indexedPlugins_[op];
- var argList = goog.array.slice(arguments, 1);
- for (var i = 0; i < plugins.length; ++i) {
- // If the plugin returns true, that means it handled the event and
- // we shouldn't propagate to the other plugins.
- var plugin = plugins[i];
- if ((plugin.isEnabled(this) || goog.editor.Plugin.IRREPRESSIBLE_OPS[op]) &&
- plugin[goog.editor.Plugin.OPCODE[op]].apply(plugin, argList)) {
- // Only one plugin is allowed to handle the event. If for some reason
- // a plugin wants to handle it and still allow other plugins to handle
- // it, it shouldn't return true.
- return true;
- }
- }
- return false;
- };
- /**
- * Invoke this operation on all plugins with the given arguments.
- * @param {goog.editor.Plugin.Op} op A plugin op.
- * @param {...*} var_args The arguments to the plugin.
- * @private
- */
- goog.editor.Field.prototype.invokeOp_ = function(op, var_args) {
- var plugins = this.indexedPlugins_[op];
- var argList = goog.array.slice(arguments, 1);
- for (var i = 0; i < plugins.length; ++i) {
- var plugin = plugins[i];
- if (plugin.isEnabled(this) || goog.editor.Plugin.IRREPRESSIBLE_OPS[op]) {
- plugin[goog.editor.Plugin.OPCODE[op]].apply(plugin, argList);
- }
- }
- };
- /**
- * Reduce this argument over all plugins. The result of each plugin invocation
- * will be passed to the next plugin invocation. See goog.array.reduce.
- * @param {goog.editor.Plugin.Op} op A plugin op.
- * @param {string} arg The argument to reduce. For now, we assume it's a
- * string, but we should widen this later if there are reducing
- * plugins that don't operate on strings.
- * @param {...*} var_args Any extra arguments to pass to the plugin. These args
- * will not be reduced.
- * @return {string} The reduced argument.
- * @private
- */
- goog.editor.Field.prototype.reduceOp_ = function(op, arg, var_args) {
- var plugins = this.indexedPlugins_[op];
- var argList = goog.array.slice(arguments, 1);
- for (var i = 0; i < plugins.length; ++i) {
- var plugin = plugins[i];
- if (plugin.isEnabled(this) || goog.editor.Plugin.IRREPRESSIBLE_OPS[op]) {
- argList[0] = plugin[goog.editor.Plugin.OPCODE[op]].apply(plugin, argList);
- }
- }
- return argList[0];
- };
- /**
- * Prepare the given contents, then inject them into the editable field.
- * @param {?string} contents The contents to prepare.
- * @param {Element} field The field element.
- * @protected
- */
- goog.editor.Field.prototype.injectContents = function(contents, field) {
- var styles = {};
- var newHtml = this.getInjectableContents(contents, styles);
- goog.style.setStyle(field, styles);
- goog.editor.node.replaceInnerHtml(field, newHtml);
- };
- /**
- * Returns prepared contents that can be injected into the editable field.
- * @param {?string} contents The contents to prepare.
- * @param {Object} styles A map that will be populated with styles that should
- * be applied to the field element together with the contents.
- * @return {string} The prepared contents.
- */
- goog.editor.Field.prototype.getInjectableContents = function(contents, styles) {
- return this.reduceOp_(
- goog.editor.Plugin.Op.PREPARE_CONTENTS_HTML, contents || '', styles);
- };
- /**
- * Handles keydown on the field.
- * @param {goog.events.BrowserEvent} e The browser event.
- * @private
- */
- goog.editor.Field.prototype.handleKeyDown_ = function(e) {
- // Mac only fires Cmd+A for keydown, not keyup: b/22407515.
- if (goog.userAgent.MAC && e.keyCode == goog.events.KeyCodes.A) {
- this.maybeStartSelectionChangeTimer_(e);
- }
- if (!goog.editor.BrowserFeature.USE_MUTATION_EVENTS) {
- if (!this.handleBeforeChangeKeyEvent_(e)) {
- return;
- }
- }
- if (!this.invokeShortCircuitingOp_(goog.editor.Plugin.Op.KEYDOWN, e) &&
- goog.editor.BrowserFeature.USES_KEYDOWN) {
- this.handleKeyboardShortcut_(e);
- }
- };
- /**
- * Handles keypress on the field.
- * @param {goog.events.BrowserEvent} e The browser event.
- * @private
- */
- goog.editor.Field.prototype.handleKeyPress_ = function(e) {
- if (goog.editor.BrowserFeature.USE_MUTATION_EVENTS) {
- if (!this.handleBeforeChangeKeyEvent_(e)) {
- return;
- }
- } else {
- // In IE only keys that generate output trigger keypress
- // In Mozilla charCode is set for keys generating content.
- this.gotGeneratingKey_ = true;
- this.dispatchBeforeChange();
- }
- if (!this.invokeShortCircuitingOp_(goog.editor.Plugin.Op.KEYPRESS, e) &&
- !goog.editor.BrowserFeature.USES_KEYDOWN) {
- this.handleKeyboardShortcut_(e);
- }
- };
- /**
- * Handles keyup on the field.
- * @param {!goog.events.BrowserEvent} e The browser event.
- * @private
- */
- goog.editor.Field.prototype.handleKeyUp_ = function(e) {
- if (!goog.editor.BrowserFeature.USE_MUTATION_EVENTS &&
- (this.gotGeneratingKey_ ||
- goog.editor.Field.isSpecialGeneratingKey_(e))) {
- // The special keys won't have set the gotGeneratingKey flag, so we check
- // for them explicitly
- this.handleChange();
- }
- this.invokeShortCircuitingOp_(goog.editor.Plugin.Op.KEYUP, e);
- this.maybeStartSelectionChangeTimer_(e);
- };
- /**
- * Fires {@code BEFORESELECTIONCHANGE} and starts the selection change timer
- * (which will fire {@code SELECTIONCHANGE}) if the given event is a key event
- * that causes a selection change.
- * @param {!goog.events.BrowserEvent} e The browser event.
- * @private
- */
- goog.editor.Field.prototype.maybeStartSelectionChangeTimer_ = function(e) {
- if (this.isEventStopped(goog.editor.Field.EventType.SELECTIONCHANGE)) {
- return;
- }
- if (goog.editor.Field.SELECTION_CHANGE_KEYCODES[e.keyCode] ||
- ((e.ctrlKey || e.metaKey) &&
- goog.editor.Field.CTRL_KEYS_CAUSING_SELECTION_CHANGES_[e.keyCode])) {
- this.dispatchEvent(goog.editor.Field.EventType.BEFORESELECTIONCHANGE);
- this.selectionChangeTimer_.start();
- }
- };
- /**
- * Handles keyboard shortcuts on the field. Note that we bake this into our
- * handleKeyPress/handleKeyDown rather than using goog.events.KeyHandler or
- * goog.ui.KeyboardShortcutHandler for performance reasons. Since these
- * are handled on every key stroke, we do not want to be going out to the
- * event system every time.
- * @param {goog.events.BrowserEvent} e The browser event.
- * @private
- */
- goog.editor.Field.prototype.handleKeyboardShortcut_ = function(e) {
- // Alt key is used for i18n languages to enter certain characters. like
- // control + alt + z (used for IMEs) and control + alt + s for Polish.
- // So we don't invoke handleKeyboardShortcut at all for alt keys.
- if (e.altKey) {
- return;
- }
- var isModifierPressed = goog.userAgent.MAC ? e.metaKey : e.ctrlKey;
- if (isModifierPressed ||
- goog.editor.Field.POTENTIAL_SHORTCUT_KEYCODES_[e.keyCode]) {
- // TODO(user): goog.events.KeyHandler uses much more complicated logic
- // to determine key. Consider changing to what they do.
- var key = e.charCode || e.keyCode;
- if (key == 17) { // Ctrl key
- // In IE and Webkit pressing Ctrl key itself results in this event.
- return;
- }
- var stringKey = String.fromCharCode(key).toLowerCase();
- // Ctrl+Cmd+Space generates a charCode for a backtick on Mac Firefox, but
- // has the correct string key in the browser event.
- if (goog.userAgent.MAC && goog.userAgent.GECKO && stringKey == '`' &&
- e.getBrowserEvent().key == ' ') {
- stringKey = ' ';
- }
- // Converting the keyCode for "\" using fromCharCode creates "u", so we need
- // to look out for it specifically.
- if (e.keyCode == goog.events.KeyCodes.BACKSLASH) {
- stringKey = '\\';
- }
- if (this.invokeShortCircuitingOp_(
- goog.editor.Plugin.Op.SHORTCUT, e, stringKey, isModifierPressed)) {
- e.preventDefault();
- // We don't call stopPropagation as some other handler outside of
- // trogedit might need it.
- }
- }
- };
- /**
- * Executes an editing command as per the registered plugins.
- * @param {string} command The command to execute.
- * @param {...*} var_args Any additional parameters needed to execute the
- * command.
- * @return {*} False if the command wasn't handled, otherwise, the result of
- * the command.
- */
- goog.editor.Field.prototype.execCommand = function(command, var_args) {
- var args = arguments;
- var result;
- var plugins = this.indexedPlugins_[goog.editor.Plugin.Op.EXEC_COMMAND];
- for (var i = 0; i < plugins.length; ++i) {
- // If the plugin supports the command, that means it handled the
- // event and we shouldn't propagate to the other plugins.
- var plugin = plugins[i];
- if (plugin.isEnabled(this) && plugin.isSupportedCommand(command)) {
- result = plugin.execCommand.apply(plugin, args);
- break;
- }
- }
- return result;
- };
- /**
- * Gets the value of command(s).
- * @param {string|Array<string>} commands String name(s) of the command.
- * @return {*} Value of each command. Returns false (or array of falses)
- * if designMode is off or the field is otherwise uneditable, and
- * there are no activeOnUneditable plugins for the command.
- */
- goog.editor.Field.prototype.queryCommandValue = function(commands) {
- var isEditable = this.isLoaded() && this.isSelectionEditable();
- if (goog.isString(commands)) {
- return this.queryCommandValueInternal_(commands, isEditable);
- } else {
- var state = {};
- for (var i = 0; i < commands.length; i++) {
- state[commands[i]] =
- this.queryCommandValueInternal_(commands[i], isEditable);
- }
- return state;
- }
- };
- /**
- * Gets the value of this command.
- * @param {string} command The command to check.
- * @param {boolean} isEditable Whether the field is currently editable.
- * @return {*} The state of this command. Null if not handled.
- * False if the field is uneditable and there are no handlers for
- * uneditable commands.
- * @private
- */
- goog.editor.Field.prototype.queryCommandValueInternal_ = function(
- command, isEditable) {
- var plugins = this.indexedPlugins_[goog.editor.Plugin.Op.QUERY_COMMAND];
- for (var i = 0; i < plugins.length; ++i) {
- var plugin = plugins[i];
- if (plugin.isEnabled(this) && plugin.isSupportedCommand(command) &&
- (isEditable || plugin.activeOnUneditableFields())) {
- return plugin.queryCommandValue(command);
- }
- }
- return isEditable ? null : false;
- };
- /**
- * Fires a change event only if the attribute change effects the editiable
- * field. We ignore events that are internal browser events (ie scrollbar
- * state change)
- * @param {Function} handler The function to call if this is not an internal
- * browser event.
- * @param {goog.events.BrowserEvent} browserEvent The browser event.
- * @protected
- */
- goog.editor.Field.prototype.handleDomAttrChange = function(
- handler, browserEvent) {
- if (this.isEventStopped(goog.editor.Field.EventType.CHANGE)) {
- return;
- }
- var e = browserEvent.getBrowserEvent();
- // For XUL elements, since we don't care what they are doing
- try {
- if (e.originalTarget.prefix ||
- /** @type {!Element} */ (e.originalTarget).nodeName == 'scrollbar') {
- return;
- }
- } catch (ex1) {
- // Some XUL nodes don't like you reading their properties. If we got
- // the exception, this implies a XUL node so we can return.
- return;
- }
- // Check if prev and new values are different, sometimes this fires when
- // nothing has really changed.
- if (e.prevValue == e.newValue) {
- return;
- }
- handler.call(this, e);
- };
- /**
- * Handle a mutation event.
- * @param {goog.events.BrowserEvent|Event} e The browser event.
- * @private
- */
- goog.editor.Field.prototype.handleMutationEventGecko_ = function(e) {
- if (this.isEventStopped(goog.editor.Field.EventType.CHANGE)) {
- return;
- }
- e = e.getBrowserEvent ? e.getBrowserEvent() : e;
- // For people with firebug, firebug sets this property on elements it is
- // inserting into the dom.
- if (e.target.firebugIgnore) {
- return;
- }
- this.isModified_ = true;
- this.isEverModified_ = true;
- this.changeTimerGecko_.start();
- };
- /**
- * Handle drop events. Deal with focus/selection issues and set the document
- * as changed.
- * @param {goog.events.BrowserEvent} e The browser event.
- * @private
- */
- goog.editor.Field.prototype.handleDrop_ = function(e) {
- if (goog.userAgent.IE) {
- // TODO(user): This should really be done in the loremipsum plugin.
- this.execCommand(goog.editor.Command.CLEAR_LOREM, true);
- }
- // TODO(user): I just moved this code to this location, but I wonder why
- // it is only done for this case. Investigate.
- if (goog.editor.BrowserFeature.USE_MUTATION_EVENTS) {
- this.dispatchFocusAndBeforeFocus_();
- }
- this.dispatchChange();
- };
- /**
- * @return {HTMLIFrameElement} The iframe that's body is editable.
- * @protected
- */
- goog.editor.Field.prototype.getEditableIframe = function() {
- var dh;
- if (this.usesIframe() && (dh = this.getEditableDomHelper())) {
- // If the iframe has been destroyed, the dh could still exist since the
- // node may not be gc'ed, but fetching the window can fail.
- var win = dh.getWindow();
- return /** @type {HTMLIFrameElement} */ (win && win.frameElement);
- }
- return null;
- };
- /**
- * @return {goog.dom.DomHelper?} The dom helper for the editable node.
- */
- goog.editor.Field.prototype.getEditableDomHelper = function() {
- return this.editableDomHelper;
- };
- /**
- * @return {goog.dom.AbstractRange?} Closure range object wrapping the selection
- * in this field or null if this field is not currently editable.
- */
- goog.editor.Field.prototype.getRange = function() {
- var win = this.editableDomHelper && this.editableDomHelper.getWindow();
- return win && goog.dom.Range.createFromWindow(win);
- };
- /**
- * Dispatch a selection change event, optionally caused by the given browser
- * event or selecting the given target.
- * @param {goog.events.BrowserEvent=} opt_e Optional browser event causing this
- * event.
- * @param {Node=} opt_target The node the selection changed to.
- */
- goog.editor.Field.prototype.dispatchSelectionChangeEvent = function(
- opt_e, opt_target) {
- if (this.isEventStopped(goog.editor.Field.EventType.SELECTIONCHANGE)) {
- return;
- }
- // The selection is editable only if the selection is inside the
- // editable field.
- var range = this.getRange();
- var rangeContainer = range && range.getContainerElement();
- this.isSelectionEditable_ =
- !!rangeContainer && goog.dom.contains(this.getElement(), rangeContainer);
- this.dispatchCommandValueChange();
- this.dispatchEvent({
- type: goog.editor.Field.EventType.SELECTIONCHANGE,
- originalType: opt_e && opt_e.type
- });
- this.invokeShortCircuitingOp_(
- goog.editor.Plugin.Op.SELECTION, opt_e, opt_target);
- };
- /**
- * Dispatch a selection change event using a browser event that was
- * asynchronously saved earlier.
- * @private
- */
- goog.editor.Field.prototype.handleSelectionChangeTimer_ = function() {
- var t = this.selectionChangeTarget_;
- this.selectionChangeTarget_ = null;
- this.dispatchSelectionChangeEvent(undefined, t);
- };
- /**
- * This dispatches the beforechange event on the editable field
- */
- goog.editor.Field.prototype.dispatchBeforeChange = function() {
- if (this.isEventStopped(goog.editor.Field.EventType.BEFORECHANGE)) {
- return;
- }
- this.dispatchEvent(goog.editor.Field.EventType.BEFORECHANGE);
- };
- /**
- * This dispatches the beforetab event on the editable field. If this event is
- * cancelled, then the default tab behavior is prevented.
- * @param {goog.events.BrowserEvent} e The tab event.
- * @private
- * @return {boolean} The result of dispatchEvent.
- */
- goog.editor.Field.prototype.dispatchBeforeTab_ = function(e) {
- return this.dispatchEvent({
- type: goog.editor.Field.EventType.BEFORETAB,
- shiftKey: e.shiftKey,
- altKey: e.altKey,
- ctrlKey: e.ctrlKey
- });
- };
- /**
- * Temporarily ignore change events. If the time has already been set, it will
- * fire immediately now. Further setting of the timer is stopped and
- * dispatching of events is stopped until startChangeEvents is called.
- * @param {boolean=} opt_stopChange Whether to ignore base change events.
- * @param {boolean=} opt_stopDelayedChange Whether to ignore delayed change
- * events.
- */
- goog.editor.Field.prototype.stopChangeEvents = function(
- opt_stopChange, opt_stopDelayedChange) {
- if (opt_stopChange) {
- if (this.changeTimerGecko_) {
- this.changeTimerGecko_.fireIfActive();
- }
- this.stopEvent(goog.editor.Field.EventType.CHANGE);
- }
- if (opt_stopDelayedChange) {
- this.clearDelayedChange();
- this.stopEvent(goog.editor.Field.EventType.DELAYEDCHANGE);
- }
- };
- /**
- * Start change events again and fire once if desired.
- * @param {boolean=} opt_fireChange Whether to fire the change event
- * immediately.
- * @param {boolean=} opt_fireDelayedChange Whether to fire the delayed change
- * event immediately.
- */
- goog.editor.Field.prototype.startChangeEvents = function(
- opt_fireChange, opt_fireDelayedChange) {
- if (!opt_fireChange && this.changeTimerGecko_) {
- // In the case where change events were stopped and we're not firing
- // them on start, the user was trying to suppress all change or delayed
- // change events. Clear the change timer now while the events are still
- // stopped so that its firing doesn't fire a stopped change event, or
- // queue up a delayed change event that we were trying to stop.
- this.changeTimerGecko_.fireIfActive();
- }
- this.startEvent(goog.editor.Field.EventType.CHANGE);
- this.startEvent(goog.editor.Field.EventType.DELAYEDCHANGE);
- if (opt_fireChange) {
- this.handleChange();
- }
- if (opt_fireDelayedChange) {
- this.dispatchDelayedChange_();
- }
- };
- /**
- * Stops the event of the given type from being dispatched.
- * @param {goog.editor.Field.EventType} eventType type of event to stop.
- */
- goog.editor.Field.prototype.stopEvent = function(eventType) {
- this.stoppedEvents_[eventType] = 1;
- };
- /**
- * Re-starts the event of the given type being dispatched, if it had
- * previously been stopped with stopEvent().
- * @param {goog.editor.Field.EventType} eventType type of event to start.
- */
- goog.editor.Field.prototype.startEvent = function(eventType) {
- // Toggling this bit on/off instead of deleting it/re-adding it
- // saves array allocations.
- this.stoppedEvents_[eventType] = 0;
- };
- /**
- * Block an event for a short amount of time. Intended
- * for the situation where an event pair fires in quick succession
- * (e.g., mousedown/mouseup, keydown/keyup, focus/blur),
- * and we want the second event in the pair to get "debounced."
- *
- * WARNING: This should never be used to solve race conditions or for
- * mission-critical actions. It should only be used for UI improvements,
- * where it's okay if the behavior is non-deterministic.
- *
- * @param {goog.editor.Field.EventType} eventType type of event to debounce.
- */
- goog.editor.Field.prototype.debounceEvent = function(eventType) {
- this.debouncedEvents_[eventType] = goog.now();
- };
- /**
- * Checks if the event of the given type has stopped being dispatched
- * @param {goog.editor.Field.EventType} eventType type of event to check.
- * @return {boolean} true if the event has been stopped with stopEvent().
- * @protected
- */
- goog.editor.Field.prototype.isEventStopped = function(eventType) {
- return !!this.stoppedEvents_[eventType] ||
- (this.debouncedEvents_[eventType] &&
- (goog.now() - this.debouncedEvents_[eventType] <=
- goog.editor.Field.DEBOUNCE_TIME_MS_));
- };
- /**
- * Calls a function to manipulate the dom of this field. This method should be
- * used whenever Trogedit clients need to modify the dom of the field, so that
- * delayed change events are handled appropriately. Extra delayed change events
- * will cause undesired states to be added to the undo-redo stack. This method
- * will always fire at most one delayed change event, depending on the value of
- * {@code opt_preventDelayedChange}.
- *
- * @param {function()} func The function to call that will manipulate the dom.
- * @param {boolean=} opt_preventDelayedChange Whether delayed change should be
- * prevented after calling {@code func}. Defaults to always firing
- * delayed change.
- * @param {Object=} opt_handler Object in whose scope to call the listener.
- */
- goog.editor.Field.prototype.manipulateDom = function(
- func, opt_preventDelayedChange, opt_handler) {
- this.stopChangeEvents(true, true);
- // We don't want any problems with the passed in function permanently
- // stopping change events. That would break Trogedit.
- try {
- func.call(opt_handler);
- } finally {
- // If the field isn't loaded then change and delayed change events will be
- // started as part of the onload behavior.
- if (this.isLoaded()) {
- // We assume that func always modified the dom and so fire a single change
- // event. Delayed change is only fired if not prevented by the user.
- if (opt_preventDelayedChange) {
- this.startEvent(goog.editor.Field.EventType.CHANGE);
- this.handleChange();
- this.startEvent(goog.editor.Field.EventType.DELAYEDCHANGE);
- } else {
- this.dispatchChange();
- }
- }
- }
- };
- /**
- * Dispatches a command value change event.
- * @param {Array<string>=} opt_commands Commands whose state has
- * changed.
- */
- goog.editor.Field.prototype.dispatchCommandValueChange = function(
- opt_commands) {
- if (opt_commands) {
- this.dispatchEvent({
- type: goog.editor.Field.EventType.COMMAND_VALUE_CHANGE,
- commands: opt_commands
- });
- } else {
- this.dispatchEvent(goog.editor.Field.EventType.COMMAND_VALUE_CHANGE);
- }
- };
- /**
- * Dispatches the appropriate set of change events. This only fires
- * synchronous change events in blended-mode, iframe-using mozilla. It just
- * starts the appropriate timer for goog.editor.Field.EventType.DELAYEDCHANGE.
- * This also starts up change events again if they were stopped.
- *
- * @param {boolean=} opt_noDelay True if
- * goog.editor.Field.EventType.DELAYEDCHANGE should be fired syncronously.
- */
- goog.editor.Field.prototype.dispatchChange = function(opt_noDelay) {
- this.startChangeEvents(true, opt_noDelay);
- };
- /**
- * Handle a change in the Editable Field. Marks the field has modified,
- * dispatches the change event on the editable field (moz only), starts the
- * timer for the delayed change event. Note that these actions only occur if
- * the proper events are not stopped.
- */
- goog.editor.Field.prototype.handleChange = function() {
- if (this.isEventStopped(goog.editor.Field.EventType.CHANGE)) {
- return;
- }
- // Clear the changeTimerGecko_ if it's active, since any manual call to
- // handle change is equiavlent to changeTimerGecko_.fire().
- if (this.changeTimerGecko_) {
- this.changeTimerGecko_.stop();
- }
- this.isModified_ = true;
- this.isEverModified_ = true;
- if (this.isEventStopped(goog.editor.Field.EventType.DELAYEDCHANGE)) {
- return;
- }
- this.delayedChangeTimer_.start();
- };
- /**
- * Dispatch a delayed change event.
- * @private
- */
- goog.editor.Field.prototype.dispatchDelayedChange_ = function() {
- if (this.isEventStopped(goog.editor.Field.EventType.DELAYEDCHANGE)) {
- return;
- }
- // Clear the delayedChangeTimer_ if it's active, since any manual call to
- // dispatchDelayedChange_ is equivalent to delayedChangeTimer_.fire().
- this.delayedChangeTimer_.stop();
- this.isModified_ = false;
- this.dispatchEvent(goog.editor.Field.EventType.DELAYEDCHANGE);
- };
- /**
- * Don't wait for the timer and just fire the delayed change event if it's
- * pending.
- */
- goog.editor.Field.prototype.clearDelayedChange = function() {
- // The changeTimerGecko_ will queue up a delayed change so to fully clear
- // delayed change we must also clear this timer.
- if (this.changeTimerGecko_) {
- this.changeTimerGecko_.fireIfActive();
- }
- this.delayedChangeTimer_.fireIfActive();
- };
- /**
- * Dispatch beforefocus and focus for FF. Note that both of these actually
- * happen in the document's "focus" event. Unfortunately, we don't actually
- * have a way of getting in before the focus event in FF (boo! hiss!).
- * In IE, we use onfocusin for before focus and onfocus for focus.
- * @private
- */
- goog.editor.Field.prototype.dispatchFocusAndBeforeFocus_ = function() {
- this.dispatchBeforeFocus_();
- this.dispatchFocus_();
- };
- /**
- * Dispatches a before focus event.
- * @private
- */
- goog.editor.Field.prototype.dispatchBeforeFocus_ = function() {
- if (this.isEventStopped(goog.editor.Field.EventType.BEFOREFOCUS)) {
- return;
- }
- this.execCommand(goog.editor.Command.CLEAR_LOREM, true);
- this.dispatchEvent(goog.editor.Field.EventType.BEFOREFOCUS);
- };
- /**
- * Dispatches a focus event.
- * @private
- */
- goog.editor.Field.prototype.dispatchFocus_ = function() {
- if (this.isEventStopped(goog.editor.Field.EventType.FOCUS)) {
- return;
- }
- goog.editor.Field.setActiveFieldId(this.id);
- this.isSelectionEditable_ = true;
- this.dispatchEvent(goog.editor.Field.EventType.FOCUS);
- if (goog.editor.BrowserFeature
- .PUTS_CURSOR_BEFORE_FIRST_BLOCK_ELEMENT_ON_FOCUS) {
- // If the cursor is at the beginning of the field, make sure that it is
- // in the first user-visible line break, e.g.,
- // no selection: <div><p>...</p></div> --> <div><p>|cursor|...</p></div>
- // <div>|cursor|<p>...</p></div> --> <div><p>|cursor|...</p></div>
- // <body>|cursor|<p>...</p></body> --> <body><p>|cursor|...</p></body>
- var field = this.getElement();
- var range = this.getRange();
- if (range) {
- var focusNode = /** @type {!Element} */ (range.getFocusNode());
- if (range.getFocusOffset() == 0 &&
- (!focusNode || focusNode == field ||
- focusNode.tagName == goog.dom.TagName.BODY)) {
- goog.editor.range.selectNodeStart(field);
- }
- }
- }
- if (!goog.editor.BrowserFeature.CLEARS_SELECTION_WHEN_FOCUS_LEAVES &&
- this.usesIframe()) {
- var parent = this.getEditableDomHelper().getWindow().parent;
- parent.getSelection().removeAllRanges();
- }
- };
- /**
- * Dispatches a blur event.
- * @protected
- */
- goog.editor.Field.prototype.dispatchBlur = function() {
- if (this.isEventStopped(goog.editor.Field.EventType.BLUR)) {
- return;
- }
- // Another field may have already been registered as active, so only
- // clear out the active field id if we still think this field is active.
- if (goog.editor.Field.getActiveFieldId() == this.id) {
- goog.editor.Field.setActiveFieldId(null);
- }
- this.isSelectionEditable_ = false;
- this.dispatchEvent(goog.editor.Field.EventType.BLUR);
- };
- /**
- * @return {boolean} Whether the selection is editable.
- */
- goog.editor.Field.prototype.isSelectionEditable = function() {
- return this.isSelectionEditable_;
- };
- /**
- * Event handler for clicks in browsers that will follow a link when the user
- * clicks, even if it's editable. We stop the click manually
- * @param {goog.events.BrowserEvent} e The event.
- * @private
- */
- goog.editor.Field.cancelLinkClick_ = function(e) {
- if (goog.dom.getAncestorByTagNameAndClass(
- /** @type {Node} */ (e.target), goog.dom.TagName.A)) {
- e.preventDefault();
- }
- };
- /**
- * Handle mouse down inside the editable field.
- * @param {goog.events.BrowserEvent} e The event.
- * @private
- */
- goog.editor.Field.prototype.handleMouseDown_ = function(e) {
- goog.editor.Field.setActiveFieldId(this.id);
- // Open links in a new window if the user control + clicks.
- if (goog.userAgent.IE) {
- var targetElement = e.target;
- if (targetElement &&
- /** @type {!Element} */ (targetElement).tagName == goog.dom.TagName.A &&
- e.ctrlKey) {
- this.originalDomHelper.getWindow().open(targetElement.href);
- }
- }
- this.waitingForMouseUp_ = true;
- };
- /**
- * Handle drag start. Needs to cancel listening for the mouse up event on the
- * window.
- * @param {goog.events.BrowserEvent} e The event.
- * @private
- */
- goog.editor.Field.prototype.handleDragStart_ = function(e) {
- this.waitingForMouseUp_ = false;
- };
- /**
- * Handle mouse up inside the editable field.
- * @param {goog.events.BrowserEvent} e The event.
- * @private
- */
- goog.editor.Field.prototype.handleMouseUp_ = function(e) {
- if (this.useWindowMouseUp_ && !this.waitingForMouseUp_) {
- return;
- }
- this.waitingForMouseUp_ = false;
- /*
- * We fire a selection change event immediately for listeners that depend on
- * the native browser event object (e). On IE, a listener that tries to
- * retrieve the selection with goog.dom.Range may see an out-of-date
- * selection range.
- */
- this.dispatchEvent(goog.editor.Field.EventType.BEFORESELECTIONCHANGE);
- this.dispatchSelectionChangeEvent(e);
- if (goog.userAgent.IE) {
- /*
- * Fire a second selection change event for listeners that need an
- * up-to-date selection range. Save the event's target to be sent with it
- * (it's safer than saving a copy of the event itself).
- */
- this.selectionChangeTarget_ = /** @type {Node} */ (e.target);
- this.selectionChangeTimer_.start();
- }
- };
- /**
- * Retrieve the HTML contents of a field.
- *
- * Do NOT just get the innerHTML of a field directly--there's a lot of
- * processing that needs to happen.
- * @return {string} The scrubbed contents of the field.
- */
- goog.editor.Field.prototype.getCleanContents = function() {
- if (this.queryCommandValue(goog.editor.Command.USING_LOREM)) {
- return goog.string.Unicode.NBSP;
- }
- if (!this.isLoaded()) {
- // The field is uneditable, so it's ok to read contents directly.
- var elem = this.getOriginalElement();
- if (!elem) {
- goog.log.log(
- this.logger, goog.log.Level.SHOUT,
- "Couldn't get the field element to read the contents");
- }
- return elem.innerHTML;
- }
- var fieldCopy = this.getFieldCopy();
- // Allow the plugins to handle their cleanup.
- this.invokeOp_(goog.editor.Plugin.Op.CLEAN_CONTENTS_DOM, fieldCopy);
- return this.reduceOp_(
- goog.editor.Plugin.Op.CLEAN_CONTENTS_HTML, fieldCopy.innerHTML);
- };
- /**
- * Get the copy of the editable field element, which has the innerHTML set
- * correctly.
- * @return {!Element} The copy of the editable field.
- * @protected
- */
- goog.editor.Field.prototype.getFieldCopy = function() {
- var field = this.getElement();
- // Deep cloneNode strips some script tag contents in IE, so we do this.
- var fieldCopy = /** @type {Element} */ (field.cloneNode(false));
- // For some reason, when IE sets innerHtml of the cloned node, it strips
- // script tags that fall at the beginning of an element. Appending a
- // non-breaking space prevents this.
- var html = field.innerHTML;
- if (goog.userAgent.IE && html.match(/^\s*<script/i)) {
- html = goog.string.Unicode.NBSP + html;
- }
- fieldCopy.innerHTML = html;
- return fieldCopy;
- };
- /**
- * Sets the contents of the field.
- * @param {boolean} addParas Boolean to specify whether to add paragraphs
- * to long fields.
- * @param {?string} html html to insert. If html=null, then this defaults
- * to a nbsp for mozilla and an empty string for IE.
- * @param {boolean=} opt_dontFireDelayedChange True to make this content change
- * not fire a delayed change event.
- * @param {boolean=} opt_applyLorem Whether to apply lorem ipsum styles.
- * @deprecated Use setSafeHtml instead.
- */
- goog.editor.Field.prototype.setHtml = function(
- addParas, html, opt_dontFireDelayedChange, opt_applyLorem) {
- var safeHtml =
- html ? goog.html.legacyconversions.safeHtmlFromString(html) : null;
- this.setSafeHtml(
- addParas, safeHtml, opt_dontFireDelayedChange, opt_applyLorem);
- };
- /**
- * Sets the contents of the field.
- * @param {boolean} addParas Boolean to specify whether to add paragraphs
- * to long fields.
- * @param {?goog.html.SafeHtml} html html to insert. If html=null, then this
- * defaults to a nsbp for mozilla and an empty string for IE.
- * @param {boolean=} opt_dontFireDelayedChange True to make this content change
- * not fire a delayed change event.
- * @param {boolean=} opt_applyLorem Whether to apply lorem ipsum styles.
- */
- goog.editor.Field.prototype.setSafeHtml = function(
- addParas, html, opt_dontFireDelayedChange, opt_applyLorem) {
- if (this.isLoading()) {
- goog.log.error(this.logger, "Can't set html while loading Trogedit");
- return;
- }
- // Clear the lorem ipsum style, always.
- if (opt_applyLorem) {
- this.execCommand(goog.editor.Command.CLEAR_LOREM);
- }
- if (html && addParas) {
- html = goog.html.SafeHtml.create('p', {}, html);
- }
- // If we don't want change events to fire, we have to turn off change events
- // before setting the field contents, since that causes mutation events.
- if (opt_dontFireDelayedChange) {
- this.stopChangeEvents(false, true);
- }
- this.setInnerHtml_(html);
- // Set the lorem ipsum style, if the element is empty.
- if (opt_applyLorem) {
- this.execCommand(goog.editor.Command.UPDATE_LOREM);
- }
- // TODO(user): This check should probably be moved to isEventStopped and
- // startEvent.
- if (this.isLoaded()) {
- if (opt_dontFireDelayedChange) { // Turn back on change events
- // We must fire change timer if necessary before restarting change events!
- // Otherwise, the change timer firing after we restart events will cause
- // the delayed change we were trying to stop. Flow:
- // Stop delayed change
- // setInnerHtml_, this starts the change timer
- // start delayed change
- // change timer fires
- // starts delayed change timer since event was not stopped
- // delayed change fires for the delayed change we tried to stop.
- if (goog.editor.BrowserFeature.USE_MUTATION_EVENTS) {
- this.changeTimerGecko_.fireIfActive();
- }
- this.startChangeEvents();
- } else { // Mark the document as changed and fire change events.
- this.dispatchChange();
- }
- }
- };
- /**
- * Sets the inner HTML of the field. Works on both editable and
- * uneditable fields.
- * @param {?goog.html.SafeHtml} html The new inner HTML of the field.
- * @private
- */
- goog.editor.Field.prototype.setInnerHtml_ = function(html) {
- var field = this.getElement();
- if (field) {
- // Safari will put <style> tags into *new* <head> elements. When setting
- // HTML, we need to remove these spare <head>s to make sure there's a
- // clean slate, but keep the first <head>.
- // Note: We punt on this issue for the non iframe case since
- // we don't want to screw with the main document.
- if (this.usesIframe() && goog.editor.BrowserFeature.MOVES_STYLE_TO_HEAD) {
- var heads = goog.dom.getElementsByTagName(
- goog.dom.TagName.HEAD, goog.asserts.assert(field.ownerDocument));
- for (var i = heads.length - 1; i >= 1; --i) {
- heads[i].parentNode.removeChild(heads[i]);
- }
- }
- } else {
- field = this.getOriginalElement();
- }
- if (field) {
- this.injectContents(html && goog.html.SafeHtml.unwrap(html), field);
- }
- };
- /**
- * Attemps to turn on designMode for a document. This function can fail under
- * certain circumstances related to the load event, and will throw an exception.
- * @protected
- */
- goog.editor.Field.prototype.turnOnDesignModeGecko = function() {
- var doc = this.getEditableDomHelper().getDocument();
- // NOTE(nicksantos): This will fail under certain conditions, like
- // when the node has display: none. It's up to clients to ensure that
- // their fields are valid when they try to make them editable.
- doc.designMode = 'on';
- if (goog.editor.BrowserFeature.HAS_STYLE_WITH_CSS) {
- doc.execCommand('styleWithCSS', false, false);
- }
- };
- /**
- * Installs styles if needed. Only writes styles when they can't be written
- * inline directly into the field.
- * @protected
- */
- goog.editor.Field.prototype.installStyles = function() {
- if (this.cssStyles && this.shouldLoadAsynchronously()) {
- goog.style.installSafeStyleSheet(
- goog.html.legacyconversions.safeStyleSheetFromString(this.cssStyles),
- this.getElement());
- }
- };
- /**
- * Signal that the field is loaded and ready to use. Change events now are
- * in effect.
- * @private
- */
- goog.editor.Field.prototype.dispatchLoadEvent_ = function() {
- this.getElement();
- this.installStyles();
- this.startChangeEvents();
- goog.log.info(this.logger, 'Dispatching load ' + this.id);
- this.dispatchEvent(goog.editor.Field.EventType.LOAD);
- };
- /**
- * @return {boolean} Whether the field is uneditable.
- */
- goog.editor.Field.prototype.isUneditable = function() {
- return this.loadState_ == goog.editor.Field.LoadState_.UNEDITABLE;
- };
- /**
- * @return {boolean} Whether the field has finished loading.
- */
- goog.editor.Field.prototype.isLoaded = function() {
- return this.loadState_ == goog.editor.Field.LoadState_.EDITABLE;
- };
- /**
- * @return {boolean} Whether the field is in the process of loading.
- */
- goog.editor.Field.prototype.isLoading = function() {
- return this.loadState_ == goog.editor.Field.LoadState_.LOADING;
- };
- /**
- * Gives the field focus.
- */
- goog.editor.Field.prototype.focus = function() {
- if (!goog.editor.BrowserFeature.HAS_CONTENT_EDITABLE && this.usesIframe()) {
- // In designMode, only the window itself can be focused; not the element.
- this.getEditableDomHelper().getWindow().focus();
- } else {
- if (goog.userAgent.OPERA) {
- // Opera will scroll to the bottom of the focused document, even
- // if it is contained in an iframe that is scrolled to the top and
- // the bottom flows past the end of it. To prevent this,
- // save the scroll position of the document containing the editor
- // iframe, then restore it after the focus.
- var scrollX = this.appWindow_.pageXOffset;
- var scrollY = this.appWindow_.pageYOffset;
- }
- this.getElement().focus();
- if (goog.userAgent.OPERA) {
- this.appWindow_.scrollTo(
- /** @type {number} */ (scrollX), /** @type {number} */ (scrollY));
- }
- }
- };
- /**
- * Gives the field focus and places the cursor at the start of the field.
- */
- goog.editor.Field.prototype.focusAndPlaceCursorAtStart = function() {
- // NOTE(user): Excluding Gecko to maintain existing behavior post refactoring
- // placeCursorAtStart into its own method. In Gecko browsers that currently
- // have a selection the existing selection will be restored, otherwise it
- // will go to the start.
- // TODO(user): Refactor the code using this and related methods. We should
- // only mess with the selection in the case where there is not an existing
- // selection in the field.
- if (goog.editor.BrowserFeature.HAS_IE_RANGES || !goog.userAgent.GECKO) {
- this.placeCursorAtStart();
- }
- this.focus();
- };
- /**
- * Place the cursor at the start of this field. It's recommended that you only
- * use this method (and manipulate the selection in general) when there is not
- * an existing selection in the field.
- */
- goog.editor.Field.prototype.placeCursorAtStart = function() {
- this.placeCursorAtStartOrEnd_(true);
- };
- /**
- * Place the cursor at the start of this field. It's recommended that you only
- * use this method (and manipulate the selection in general) when there is not
- * an existing selection in the field.
- */
- goog.editor.Field.prototype.placeCursorAtEnd = function() {
- this.placeCursorAtStartOrEnd_(false);
- };
- /**
- * Helper method to place the cursor at the start or end of this field.
- * @param {boolean} isStart True for start, false for end.
- * @private
- */
- goog.editor.Field.prototype.placeCursorAtStartOrEnd_ = function(isStart) {
- var field = this.getElement();
- if (field) {
- var cursorPosition = isStart ? goog.editor.node.getLeftMostLeaf(field) :
- goog.editor.node.getRightMostLeaf(field);
- if (field == cursorPosition) {
- // The rightmost leaf we found was the field element itself (which likely
- // means the field element is empty). We can't place the cursor next to
- // the field element, so just place it at the beginning.
- goog.dom.Range.createCaret(field, 0).select();
- } else {
- goog.editor.range.placeCursorNextTo(cursorPosition, isStart);
- }
- this.dispatchSelectionChangeEvent();
- }
- };
- /**
- * Restore a saved range, and set the focus on the field.
- * If no range is specified, we simply set the focus.
- * @param {goog.dom.SavedRange=} opt_range A previously saved selected range.
- */
- goog.editor.Field.prototype.restoreSavedRange = function(opt_range) {
- if (opt_range) {
- opt_range.restore();
- }
- this.focus();
- };
- /**
- * Makes a field editable.
- *
- * @param {!goog.html.TrustedResourceUrl|string=} opt_iframeSrc URL to set the
- * iframe src to if necessary.
- */
- goog.editor.Field.prototype.makeEditable = function(opt_iframeSrc) {
- this.loadState_ = goog.editor.Field.LoadState_.LOADING;
- var field = this.getOriginalElement();
- // TODO: In the fieldObj, save the field's id, className, cssText
- // in order to reset it on closeField. That way, we can muck with the field's
- // css, id, class and restore to how it was at the end.
- this.nodeName = field.nodeName;
- this.savedClassName_ = field.className;
- this.setInitialStyle(field.style.cssText);
- goog.dom.classlist.add(field, 'editable');
- var iframeSrc;
- if (goog.isString(opt_iframeSrc)) {
- iframeSrc =
- goog.html.legacyconversions.trustedResourceUrlFromString(opt_iframeSrc);
- } else {
- iframeSrc = opt_iframeSrc;
- }
- this.makeEditableInternal(iframeSrc);
- };
- /**
- * Handles actually making something editable - creating necessary nodes,
- * injecting content, etc.
- * @param {!goog.html.TrustedResourceUrl=} opt_iframeSrc URL to set the iframe
- * src to if necessary.
- * @protected
- */
- goog.editor.Field.prototype.makeEditableInternal = function(opt_iframeSrc) {
- this.makeIframeField_(opt_iframeSrc);
- };
- /**
- * Handle the loading of the field (e.g. once the field is ready to setup).
- * TODO(user): this should probably just be moved into dispatchLoadEvent_.
- * @protected
- */
- goog.editor.Field.prototype.handleFieldLoad = function() {
- if (goog.userAgent.IE) {
- // This sometimes fails if the selection is invalid. This can happen, for
- // example, if you attach a CLICK handler to the field that causes the
- // field to be removed from the DOM and replaced with an editor
- // -- however, listening to another event like MOUSEDOWN does not have this
- // issue since no mouse selection has happened at that time.
- goog.dom.Range.clearSelection(this.editableDomHelper.getWindow());
- }
- if (goog.editor.Field.getActiveFieldId() != this.id) {
- this.execCommand(goog.editor.Command.UPDATE_LOREM);
- }
- this.setupChangeListeners_();
- this.dispatchLoadEvent_();
- // Enabling plugins after we fire the load event so that clients have a
- // chance to set initial field contents before we start mucking with
- // everything.
- for (var classId in this.plugins_) {
- this.plugins_[classId].enable(this);
- }
- };
- /**
- * Closes the field and cancels all pending change timers. Note that this
- * means that if a change event has not fired yet, it will not fire. Clients
- * should check fieldOj.isModified() if they depend on the final change event.
- * Throws an error if the field is already uneditable.
- *
- * @param {boolean=} opt_skipRestore True to prevent copying of editable field
- * contents back into the original node.
- */
- goog.editor.Field.prototype.makeUneditable = function(opt_skipRestore) {
- if (this.isUneditable()) {
- throw Error('makeUneditable: Field is already uneditable');
- }
- // Fire any events waiting on a timeout.
- // Clearing delayed change also clears changeTimerGecko_.
- this.clearDelayedChange();
- this.selectionChangeTimer_.fireIfActive();
- this.execCommand(goog.editor.Command.CLEAR_LOREM);
- var html = null;
- if (!opt_skipRestore && this.getElement()) {
- // Rest of cleanup is simpler if field was never initialized.
- html = this.getCleanContents();
- }
- // First clean up anything that happens in makeFieldEditable
- // (i.e. anything that needs cleanup even if field has not loaded).
- this.clearFieldLoadListener_();
- var field = this.getOriginalElement();
- if (goog.editor.Field.getActiveFieldId() == field.id) {
- goog.editor.Field.setActiveFieldId(null);
- }
- // Clear all listeners before removing the nodes from the dom - if
- // there are listeners on the iframe window, Firefox throws errors trying
- // to unlisten once the iframe is no longer in the dom.
- this.clearListeners();
- // For fields that have loaded, clean up anything that happened in
- // handleFieldOpen or later.
- // If html is provided, copy it back and reset the properties on the field
- // so that the original node will have the same properties as it did before
- // it was made editable.
- if (goog.isString(html)) {
- goog.editor.node.replaceInnerHtml(field, html);
- this.resetOriginalElemProperties();
- }
- this.restoreDom();
- this.tearDownFieldObject_();
- // On Safari, make sure to un-focus the field so that the
- // native "current field" highlight style gets removed.
- if (goog.userAgent.WEBKIT) {
- field.blur();
- }
- this.execCommand(goog.editor.Command.UPDATE_LOREM);
- this.dispatchEvent(goog.editor.Field.EventType.UNLOAD);
- };
- /**
- * Restores the dom to how it was before being made editable.
- * @protected
- */
- goog.editor.Field.prototype.restoreDom = function() {
- // TODO(user): Consider only removing the iframe if we are
- // restoring the original node, aka, if opt_html.
- var field = this.getOriginalElement();
- // TODO(robbyw): Consider throwing an error if !field.
- if (field) {
- // If the field is in the process of loading when it starts getting torn
- // up, the iframe will not exist.
- var iframe = this.getEditableIframe();
- if (iframe) {
- goog.dom.replaceNode(field, iframe);
- }
- }
- };
- /**
- * Returns true if the field needs to be loaded asynchrnously.
- * @return {boolean} True if loads are async.
- * @protected
- */
- goog.editor.Field.prototype.shouldLoadAsynchronously = function() {
- if (!goog.isDef(this.isHttps_)) {
- this.isHttps_ = false;
- if (goog.userAgent.IE && this.usesIframe()) {
- // IE iframes need to load asynchronously if they are in https as we need
- // to set an actual src on the iframe and wait for it to load.
- // Find the top-most window we have access to and see if it's https.
- // Technically this could fail if we have an http frame in an https frame
- // on the same domain (or vice versa), but walking up the window hierarchy
- // to find the first window that has an http* protocol seems like
- // overkill.
- var win = this.originalDomHelper.getWindow();
- while (win != win.parent) {
- try {
- win = win.parent;
- } catch (e) {
- break;
- }
- }
- var loc = win.location;
- this.isHttps_ =
- loc.protocol == 'https:' && loc.search.indexOf('nocheckhttps') == -1;
- }
- }
- return this.isHttps_;
- };
- /**
- * Start the editable iframe creation process for Mozilla or IE whitebox.
- * The iframes load asynchronously.
- *
- * @param {!goog.html.TrustedResourceUrl=} opt_iframeSrc URL to set the iframe
- * src to if necessary.
- * @private
- */
- goog.editor.Field.prototype.makeIframeField_ = function(opt_iframeSrc) {
- var field = this.getOriginalElement();
- // TODO(robbyw): Consider throwing an error if !field.
- if (field) {
- var html = field.innerHTML;
- // Invoke prepareContentsHtml on all plugins to prepare html for editing.
- // Make sure this is done before calling this.attachFrame which removes the
- // original element from DOM tree. Plugins may assume that the original
- // element is still in its original position in DOM.
- var styles = {};
- html = this.reduceOp_(
- goog.editor.Plugin.Op.PREPARE_CONTENTS_HTML, html, styles);
- var iframe = this.originalDomHelper.createDom(
- goog.dom.TagName.IFRAME, this.getIframeAttributes());
- // TODO(nicksantos): Figure out if this is ever needed in SAFARI?
- // In IE over HTTPS we need to wait for a load event before we set up the
- // iframe, this is to prevent a security prompt or access is denied
- // errors.
- // NOTE(user): This hasn't been confirmed. isHttps_ allows a query
- // param, nocheckhttps, which we can use to ascertain if this is actually
- // needed. It was originally thought to be needed for IE6 SP1, but
- // errors have been seen in IE7 as well.
- if (this.shouldLoadAsynchronously()) {
- // onLoad is the function to call once the iframe is ready to continue
- // loading.
- var onLoad =
- goog.bind(this.iframeFieldLoadHandler, this, iframe, html, styles);
- this.fieldLoadListenerKey_ =
- goog.events.listen(iframe, goog.events.EventType.LOAD, onLoad, true);
- if (opt_iframeSrc) {
- goog.dom.safe.setIframeSrc(iframe, opt_iframeSrc);
- }
- }
- this.attachIframe(iframe);
- // Only continue if its not IE HTTPS in which case we're waiting for load.
- if (!this.shouldLoadAsynchronously()) {
- this.iframeFieldLoadHandler(iframe, html, styles);
- }
- }
- };
- /**
- * Given the original field element, and the iframe that is destined to
- * become the editable field, styles them appropriately and add the iframe
- * to the dom.
- *
- * @param {HTMLIFrameElement} iframe The iframe element.
- * @protected
- */
- goog.editor.Field.prototype.attachIframe = function(iframe) {
- var field = this.getOriginalElement();
- // TODO(user): Why do we do these two lines .. and why whitebox only?
- iframe.className = field.className;
- iframe.id = field.id;
- goog.dom.replaceNode(iframe, field);
- };
- /**
- * @param {Object} extraStyles A map of extra styles.
- * @return {!goog.editor.icontent.FieldFormatInfo} The FieldFormatInfo
- * object for this field's configuration.
- * @protected
- */
- goog.editor.Field.prototype.getFieldFormatInfo = function(extraStyles) {
- var originalElement = this.getOriginalElement();
- var isStandardsMode = goog.editor.node.isStandardsMode(originalElement);
- return new goog.editor.icontent.FieldFormatInfo(
- this.id, isStandardsMode, false, false, extraStyles);
- };
- /**
- * Writes the html content into the iframe. Handles writing any aditional
- * styling as well.
- * @param {HTMLIFrameElement} iframe Iframe to write contents into.
- * @param {string} innerHtml The html content to write into the iframe.
- * @param {Object} extraStyles A map of extra style attributes.
- * @protected
- */
- goog.editor.Field.prototype.writeIframeContent = function(
- iframe, innerHtml, extraStyles) {
- var formatInfo = this.getFieldFormatInfo(extraStyles);
- if (this.shouldLoadAsynchronously()) {
- var doc = goog.dom.getFrameContentDocument(iframe);
- goog.editor.icontent.writeHttpsInitialIframe(formatInfo, doc, innerHtml);
- } else {
- var styleInfo = new goog.editor.icontent.FieldStyleInfo(
- this.getElement(), this.cssStyles);
- goog.editor.icontent.writeNormalInitialIframe(
- formatInfo, innerHtml, styleInfo, iframe);
- }
- };
- /**
- * The function to call when the editable iframe loads.
- *
- * @param {HTMLIFrameElement} iframe Iframe that just loaded.
- * @param {string} innerHtml Html to put inside the body of the iframe.
- * @param {Object} styles Property-value map of CSS styles to install on
- * editable field.
- * @protected
- */
- goog.editor.Field.prototype.iframeFieldLoadHandler = function(
- iframe, innerHtml, styles) {
- this.clearFieldLoadListener_();
- iframe.allowTransparency = 'true';
- this.writeIframeContent(iframe, innerHtml, styles);
- var doc = goog.dom.getFrameContentDocument(iframe);
- // Make sure to get this pointer after the doc.write as the doc.write
- // clobbers all the document contents.
- var body = doc.body;
- this.setupFieldObject(body);
- if (!goog.editor.BrowserFeature.HAS_CONTENT_EDITABLE && this.usesIframe()) {
- this.turnOnDesignModeGecko();
- }
- this.handleFieldLoad();
- };
- /**
- * Clears fieldLoadListener for a field. Must be called even (especially?) if
- * the field is not yet loaded and therefore not in this.fieldMap_
- * @private
- */
- goog.editor.Field.prototype.clearFieldLoadListener_ = function() {
- if (this.fieldLoadListenerKey_) {
- goog.events.unlistenByKey(this.fieldLoadListenerKey_);
- this.fieldLoadListenerKey_ = null;
- }
- };
- /**
- * @return {!Object} Get the HTML attributes for this field's iframe.
- * @protected
- */
- goog.editor.Field.prototype.getIframeAttributes = function() {
- var iframeStyle = 'padding:0;' + this.getOriginalElement().style.cssText;
- if (!goog.string.endsWith(iframeStyle, ';')) {
- iframeStyle += ';';
- }
- iframeStyle += 'background-color:white;';
- // Ensure that the iframe has default overflow styling. If overflow is
- // set to auto, an IE rendering bug can occur when it tries to render a
- // table at the very bottom of the field, such that the table would cause
- // a scrollbar, that makes the entire field go blank.
- if (goog.userAgent.IE) {
- iframeStyle += 'overflow:visible;';
- }
- return {'frameBorder': 0, 'style': iframeStyle};
- };
|