From 299d0dfeb92360215c9a06057c92d41c07e9e9bc Mon Sep 17 00:00:00 2001 From: Mihai Sucan Date: Sat, 10 Dec 2011 19:03:57 +0200 Subject: [PATCH] Bug 702331 - Update Orion from upstream; r=rcampbell --- .../browser/browser_644409-scratchpads.js | 18 +- browser/devtools/jar.mn | 2 +- .../scratchpad/scratchpad-manager.jsm | 2 +- browser/devtools/scratchpad/scratchpad.js | 18 +- ...ratchpad_bug_646070_chrome_context_pref.js | 15 +- ...ser_scratchpad_bug_653427_confirm_close.js | 57 +- .../test/browser_scratchpad_bug_660560_tab.js | 23 +- .../browser_scratchpad_bug_669612_unsaved.js | 122 +- .../browser_scratchpad_bug_679467_falsy.js | 15 +- ...r_scratchpad_bug_699130_edit_ui_updates.js | 6 +- .../test/browser_scratchpad_contexts.js | 17 +- .../test/browser_scratchpad_execute_print.js | 10 +- .../test/browser_scratchpad_files.js | 13 +- .../test/browser_scratchpad_initialization.js | 14 +- .../test/browser_scratchpad_inspect.js | 17 +- .../test/browser_scratchpad_open.js | 23 +- .../test/browser_scratchpad_restore.js | 15 +- .../test/browser_scratchpad_tab_switch.js | 17 +- .../scratchpad/test/browser_scratchpad_ui.js | 13 +- browser/devtools/scratchpad/test/head.js | 55 + .../sourceeditor/orion/Makefile.dryice.js | 6 + browser/devtools/sourceeditor/orion/README | 24 +- .../devtools/sourceeditor/orion/mozilla.css | 123 +- browser/devtools/sourceeditor/orion/orion.css | 173 +- browser/devtools/sourceeditor/orion/orion.js | 5282 +++++++++++------ .../sourceeditor/source-editor-orion.jsm | 213 +- .../sourceeditor/source-editor-textarea.jsm | 21 +- .../devtools/sourceeditor/source-editor.jsm | 4 +- .../devtools/sourceeditor/test/Makefile.in | 2 + .../test/browser_bug687580_drag_and_drop.js | 7 +- .../browser_bug695035_middle_click_paste.js | 84 + .../browser_sourceeditor_initialization.js | 8 +- browser/devtools/sourceeditor/test/head.js | 107 + 33 files changed, 4199 insertions(+), 2327 deletions(-) create mode 100644 browser/devtools/sourceeditor/test/browser_bug695035_middle_click_paste.js create mode 100644 browser/devtools/sourceeditor/test/head.js diff --git a/browser/components/sessionstore/test/browser/browser_644409-scratchpads.js b/browser/components/sessionstore/test/browser/browser_644409-scratchpads.js index a81746e93572..0058c44aa7b7 100644 --- a/browser/components/sessionstore/test/browser/browser_644409-scratchpads.js +++ b/browser/components/sessionstore/test/browser/browser_644409-scratchpads.js @@ -38,11 +38,19 @@ function test() { function windowObserver(aSubject, aTopic, aData) { if (aTopic == "domwindowopened") { let win = aSubject.QueryInterface(Ci.nsIDOMWindow); - win.addEventListener("load", function() { + win.addEventListener("load", function onLoad() { + win.removeEventListener("load", onLoad, false); + if (win.Scratchpad) { - let state = win.Scratchpad.getState(); - win.close(); - addState(state); + win.Scratchpad.addObserver({ + onReady: function() { + win.Scratchpad.removeObserver(this); + + let state = win.Scratchpad.getState(); + win.close(); + addState(state); + }, + }); } }, false); } @@ -56,4 +64,4 @@ function statesMatch(restored, states) { state.executionContext == restoredState.executionContext; }) }); -} \ No newline at end of file +} diff --git a/browser/devtools/jar.mn b/browser/devtools/jar.mn index 9a35144980b2..5c3679068734 100644 --- a/browser/devtools/jar.mn +++ b/browser/devtools/jar.mn @@ -2,7 +2,7 @@ browser.jar: * content/browser/inspector.html (highlighter/inspector.html) content/browser/NetworkPanel.xhtml (webconsole/NetworkPanel.xhtml) * content/browser/scratchpad.xul (scratchpad/scratchpad.xul) -* content/browser/scratchpad.js (scratchpad/scratchpad.js) + content/browser/scratchpad.js (scratchpad/scratchpad.js) * content/browser/styleeditor.xul (styleeditor/styleeditor.xul) content/browser/splitview.css (styleeditor/splitview.css) content/browser/styleeditor.css (styleeditor/styleeditor.css) diff --git a/browser/devtools/scratchpad/scratchpad-manager.jsm b/browser/devtools/scratchpad/scratchpad-manager.jsm index 42a88d863630..ac4446d4cf2d 100644 --- a/browser/devtools/scratchpad/scratchpad-manager.jsm +++ b/browser/devtools/scratchpad/scratchpad-manager.jsm @@ -104,7 +104,7 @@ var ScratchpadManager = { let enumerator = Services.wm.getEnumerator("devtools:scratchpad"); while (enumerator.hasMoreElements()) { let win = enumerator.getNext(); - if (!win.closed) { + if (!win.closed && win.Scratchpad.initialized) { this._scratchpads.push(win.Scratchpad.getState()); } } diff --git a/browser/devtools/scratchpad/scratchpad.js b/browser/devtools/scratchpad/scratchpad.js index 7ed3b496ad9e..19525596ebe1 100644 --- a/browser/devtools/scratchpad/scratchpad.js +++ b/browser/devtools/scratchpad/scratchpad.js @@ -86,6 +86,13 @@ var Scratchpad = { */ executionContext: SCRATCHPAD_CONTEXT_CONTENT, + /** + * Tells if this Scratchpad is initialized and ready for use. + * @boolean + * @see addObserver + */ + initialized: false, + /** * Retrieve the xul:notificationbox DOM element. It notifies the user when * the current code execution context is SCRATCHPAD_CONTEXT_BROWSER. @@ -733,7 +740,7 @@ var Scratchpad = { }, /** - * The Scratchpad window DOMContentLoaded event handler. This method + * The Scratchpad window load event handler. This method * initializes the Scratchpad window and source editor. * * @param nsIDOMEvent aEvent @@ -784,7 +791,9 @@ var Scratchpad = { this.onContextMenu); this.editor.focus(); this.editor.setCaretOffset(this.editor.getCharCount()); - + + this.initialized = true; + if (this.filename && !this.saved) { this.onTextChanged(); } @@ -866,7 +875,7 @@ var Scratchpad = { if (aStatus && !Components.isSuccessCode(aStatus)) { return; } - if (!document) { + if (!document || !this.initialized) { return; // file saved to disk after window has closed } document.title = document.title.replace(/^\*/, ""); @@ -904,6 +913,7 @@ var Scratchpad = { this.onContextMenu); this.editor.destroy(); this.editor = null; + this.initialized = false; }, /** @@ -1033,6 +1043,6 @@ XPCOMUtils.defineLazyGetter(Scratchpad, "strings", function () { return Services.strings.createBundle(SCRATCHPAD_L10N); }); -addEventListener("DOMContentLoaded", Scratchpad.onLoad.bind(Scratchpad), false); +addEventListener("load", Scratchpad.onLoad.bind(Scratchpad), false); addEventListener("unload", Scratchpad.onUnload.bind(Scratchpad), false); addEventListener("close", Scratchpad.onClose.bind(Scratchpad), false); diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_bug_646070_chrome_context_pref.js b/browser/devtools/scratchpad/test/browser_scratchpad_bug_646070_chrome_context_pref.js index b2c1bdb1a043..28f6b08fec3d 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_bug_646070_chrome_context_pref.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_bug_646070_chrome_context_pref.js @@ -2,24 +2,21 @@ /* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ -let gOldPref; let DEVTOOLS_CHROME_ENABLED = "devtools.chrome.enabled"; function test() { waitForExplicitFinish(); - gOldPref = Services.prefs.getBoolPref(DEVTOOLS_CHROME_ENABLED); Services.prefs.setBoolPref(DEVTOOLS_CHROME_ENABLED, true); gBrowser.selectedTab = gBrowser.addTab(); - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); + gBrowser.selectedBrowser.addEventListener("load", function onLoad() { + gBrowser.selectedBrowser.removeEventListener("load", onLoad, true); - ok(Scratchpad, "Scratchpad variable exists"); + ok(window.Scratchpad, "Scratchpad variable exists"); - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", runTests, false); + openScratchpad(runTests); }, true); content.location = "data:text/html,Scratchpad test for bug 646070 - chrome context preference"; @@ -27,8 +24,6 @@ function test() function runTests() { - gScratchpadWindow.removeEventListener("load", arguments.callee, false); - let sp = gScratchpadWindow.Scratchpad; ok(sp, "Scratchpad object exists in new window"); @@ -50,7 +45,7 @@ function runTests() ok(!chromeContextCommand.hasAttribute("disabled"), "Chrome context command is disabled"); - Services.prefs.setBoolPref(DEVTOOLS_CHROME_ENABLED, gOldPref); + Services.prefs.clearUserPref(DEVTOOLS_CHROME_ENABLED); finish(); } diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_bug_653427_confirm_close.js b/browser/devtools/scratchpad/test/browser_scratchpad_bug_653427_confirm_close.js index 9704d1b98cb8..c85ec8e5f684 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_bug_653427_confirm_close.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_bug_653427_confirm_close.js @@ -17,10 +17,10 @@ function done() } } -var ScratchpadManager = Scratchpad.ScratchpadManager; var gFile; var oldPrompt = Services.prompt; +var promptButton = -1; function test() { @@ -29,6 +29,12 @@ function test() gFile = createTempFile("fileForBug653427.tmp"); writeFile(gFile, "text", testUnsaved.call(this)); + Services.prompt = { + confirmEx: function() { + return promptButton; + } + }; + testNew(); testSavedFile(); @@ -37,28 +43,23 @@ function test() function testNew() { - let win = ScratchpadManager.openScratchpad(); - - win.addEventListener("load", function() { + openScratchpad(function(win) { win.Scratchpad.close(); - ok(win.closed, "new scratchpad window should close without prompting") done(); - }); + }, {noFocus: true}); } function testSavedFile() { - let win = ScratchpadManager.openScratchpad(); - - win.addEventListener("load", function() { + openScratchpad(function(win) { win.Scratchpad.filename = "test.js"; win.Scratchpad.saved = true; win.Scratchpad.close(); ok(win.closed, "scratchpad from file with no changes should close") done(); - }); + }, {noFocus: true}); } function testUnsaved() @@ -70,17 +71,11 @@ function testUnsaved() function testUnsavedFileCancel() { - let win = ScratchpadManager.openScratchpad(); - - win.addEventListener("load", function() { + openScratchpad(function(win) { win.Scratchpad.filename = "test.js"; win.Scratchpad.saved = false; - Services.prompt = { - confirmEx: function() { - return win.BUTTON_POSITION_CANCEL; - } - } + promptButton = win.BUTTON_POSITION_CANCEL; win.Scratchpad.close(); @@ -88,14 +83,12 @@ function testUnsavedFileCancel() win.close(); done(); - }); + }, {noFocus: true}); } function testUnsavedFileSave() { - let win = ScratchpadManager.openScratchpad(); - - win.addEventListener("load", function() { + openScratchpad(function(win) { win.Scratchpad.importFromFile(gFile, true, function(status, content) { win.Scratchpad.filename = gFile.path; win.Scratchpad.onTextSaved(); @@ -103,11 +96,7 @@ function testUnsavedFileSave() let text = "new text"; win.Scratchpad.setText(text); - Services.prompt = { - confirmEx: function() { - return win.BUTTON_POSITION_SAVE; - } - } + promptButton = win.BUTTON_POSITION_SAVE; win.Scratchpad.close(function() { readFile(gFile, function(savedContent) { @@ -118,28 +107,22 @@ function testUnsavedFileSave() ok(win.closed, 'pressing "Save" in dialog should close scratchpad'); }); - }); + }, {noFocus: true}); } function testUnsavedFileDontSave() { - let win = ScratchpadManager.openScratchpad(); - - win.addEventListener("load", function() { + openScratchpad(function(win) { win.Scratchpad.filename = gFile.path; win.Scratchpad.saved = false; - Services.prompt = { - confirmEx: function() { - return win.BUTTON_POSITION_DONT_SAVE; - } - } + promptButton = win.BUTTON_POSITION_DONT_SAVE; win.Scratchpad.close(); ok(win.closed, 'pressing "Don\'t Save" in dialog should close scratchpad'); done(); - }); + }, {noFocus: true}); } function cleanup() diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_bug_660560_tab.js b/browser/devtools/scratchpad/test/browser_scratchpad_bug_660560_tab.js index 2c505f91b655..3687c8173a25 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_bug_660560_tab.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_bug_660560_tab.js @@ -2,8 +2,6 @@ /* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ -var ScratchpadManager = Scratchpad.ScratchpadManager; - function test() { waitForExplicitFinish(); @@ -16,14 +14,7 @@ function test() Services.prefs.setIntPref("devtools.editor.tabsize", 5); - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", function onScratchpadLoad() { - gScratchpadWindow.removeEventListener("load", onScratchpadLoad, false); - - gScratchpadWindow.Scratchpad.addObserver({ - onReady: runTests - }); - }, false); + openScratchpad(runTests); }, true); content.location = "data:text/html,Scratchpad test for the Tab key, bug 660560"; @@ -34,9 +25,6 @@ function runTests() let sp = gScratchpadWindow.Scratchpad; ok(sp, "Scratchpad object exists in new window"); - is(this.onReady, runTests, "the handler runs in the context of the observer"); - sp.removeObserver(this); - ok(sp.editor.hasFocus(), "the editor has focus"); sp.setText("window.foo;"); @@ -70,19 +58,12 @@ function runTests() Services.prefs.setIntPref("devtools.editor.tabsize", 6); Services.prefs.setBoolPref("devtools.editor.expandtab", false); - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", function onScratchpadLoad() { - gScratchpadWindow.removeEventListener("load", onScratchpadLoad, false); - gScratchpadWindow.Scratchpad.addObserver({ - onReady: runTests2 - }); - }, false); + openScratchpad(runTests2); } function runTests2() { let sp = gScratchpadWindow.Scratchpad; - sp.removeObserver(this); sp.setText("window.foo;"); sp.editor.setCaretOffset(0); diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_bug_669612_unsaved.js b/browser/devtools/scratchpad/test/browser_scratchpad_bug_669612_unsaved.js index 10823e5e2fd9..71c8d6d24d68 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_bug_669612_unsaved.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_bug_669612_unsaved.js @@ -30,57 +30,37 @@ function test() function testListeners() { - let win = ScratchpadManager.openScratchpad(); + openScratchpad(function(aWin, aScratchpad) { + aScratchpad.setText("new text"); + ok(!isStar(aWin), "no star if scratchpad isn't from a file"); - win.addEventListener("load", function onScratchpadLoad() { - win.removeEventListener("load", onScratchpadLoad, false); + aScratchpad.onTextSaved(); + ok(!isStar(aWin), "no star before changing text"); - win.Scratchpad.addObserver({ - onReady: function (aScratchpad) { - aScratchpad.removeObserver(this); + aScratchpad.setText("new text2"); + ok(isStar(aWin), "shows star if scratchpad text changes"); - aScratchpad.setText("new text"); - ok(!isStar(win), "no star if scratchpad isn't from a file"); + aScratchpad.onTextSaved(); + ok(!isStar(aWin), "no star if scratchpad was just saved"); - aScratchpad.onTextSaved(); - ok(!isStar(win), "no star before changing text"); + aScratchpad.undo(); + ok(isStar(aWin), "star if scratchpad undo"); - aScratchpad.setText("new text2"); - ok(isStar(win), "shows star if scratchpad text changes"); - - aScratchpad.onTextSaved(); - ok(!isStar(win), "no star if scratchpad was just saved"); - - aScratchpad.undo(); - ok(isStar(win), "star if scratchpad undo"); - - win.close(); - done(); - } - }); - }, false); + aWin.close(); + done(); + }, {noFocus: true}); } function testErrorStatus() { - let win = ScratchpadManager.openScratchpad(); + openScratchpad(function(aWin, aScratchpad) { + aScratchpad.onTextSaved(Components.results.NS_ERROR_FAILURE); + aScratchpad.setText("new text"); + ok(!isStar(aWin), "no star if file save failed"); - win.addEventListener("load", function onScratchpadLoad() { - win.removeEventListener("load", onScratchpadLoad, false); - - win.Scratchpad.addObserver({ - onReady: function (aScratchpad) { - aScratchpad.removeObserver(this); - - aScratchpad.onTextSaved(Components.results.NS_ERROR_FAILURE); - aScratchpad.setText("new text"); - ok(!isStar(win), "no star if file save failed"); - - win.close(); - done(); - } - }); - }, false); + aWin.close(); + done(); + }, {noFocus: true}); } @@ -92,21 +72,13 @@ function testRestoreNotFromFile() }]; let [win] = ScratchpadManager.restoreSession(session); - win.addEventListener("load", function onScratchpadLoad() { - win.removeEventListener("load", onScratchpadLoad, false); + openScratchpad(function(aWin, aScratchpad) { + aScratchpad.setText("new text"); + ok(!isStar(win), "no star if restored scratchpad isn't from a file"); - win.Scratchpad.addObserver({ - onReady: function (aScratchpad) { - aScratchpad.removeObserver(this); - - aScratchpad.setText("new text"); - ok(!isStar(win), "no star if restored scratchpad isn't from a file"); - - win.close(); - done(); - } - }); - }, false); + win.close(); + done(); + }, {window: win, noFocus: true}); } function testRestoreFromFileSaved() @@ -119,23 +91,15 @@ function testRestoreFromFileSaved() }]; let [win] = ScratchpadManager.restoreSession(session); - win.addEventListener("load", function onScratchpadLoad() { - win.removeEventListener("load", onScratchpadLoad, false); + openScratchpad(function(aWin, aScratchpad) { + ok(!isStar(win), "no star before changing text in scratchpad restored from file"); - win.Scratchpad.addObserver({ - onReady: function (aScratchpad) { - aScratchpad.removeObserver(this); + aScratchpad.setText("new text"); + ok(isStar(win), "star when text changed from scratchpad restored from file"); - ok(!isStar(win), "no star before changing text in scratchpad restored from file"); - - aScratchpad.setText("new text"); - ok(isStar(win), "star when text changed from scratchpad restored from file"); - - win.close(); - done(); - } - }); - }, false); + win.close(); + done(); + }, {window: win, noFocus: true}); } function testRestoreFromFileUnsaved() @@ -148,20 +112,12 @@ function testRestoreFromFileUnsaved() }]; let [win] = ScratchpadManager.restoreSession(session); - win.addEventListener("load", function onScratchpadLoad() { - win.removeEventListener("load", onScratchpadLoad, false); + openScratchpad(function() { + ok(isStar(win), "star with scratchpad restored with unsaved text"); - win.Scratchpad.addObserver({ - onReady: function (aScratchpad) { - aScratchpad.removeObserver(this); - - ok(isStar(win), "star with scratchpad restored with unsaved text"); - - win.close(); - done(); - } - }); - }, false); + win.close(); + done(); + }, {window: win, noFocus: true}); } function isStar(win) diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_bug_679467_falsy.js b/browser/devtools/scratchpad/test/browser_scratchpad_bug_679467_falsy.js index e4ab67349876..e19754fb4f22 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_bug_679467_falsy.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_bug_679467_falsy.js @@ -2,28 +2,21 @@ /* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ -// Reference to the Scratchpad chrome window object. -let gScratchpadWindow; - function test() { waitForExplicitFinish(); gBrowser.selectedTab = gBrowser.addTab(); - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); - - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", testFalsy, false); + gBrowser.selectedBrowser.addEventListener("load", function onLoad() { + gBrowser.selectedBrowser.removeEventListener("load", onLoad, true); + openScratchpad(testFalsy); }, true); content.location = "data:text/html,

test falsy display() values in Scratchpad"; } -function testFalsy(sp) +function testFalsy() { - gScratchpadWindow.removeEventListener("load", testFalsy, false); - let sp = gScratchpadWindow.Scratchpad; verifyFalsies(sp); diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_bug_699130_edit_ui_updates.js b/browser/devtools/scratchpad/test/browser_scratchpad_bug_699130_edit_ui_updates.js index d9d4b001e101..06f95ef3b60c 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_bug_699130_edit_ui_updates.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_bug_699130_edit_ui_updates.js @@ -13,9 +13,7 @@ function test() gBrowser.selectedTab = gBrowser.addTab(); gBrowser.selectedBrowser.addEventListener("load", function onLoad() { gBrowser.selectedBrowser.removeEventListener("load", onLoad, true); - - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", runTests, false); + openScratchpad(runTests); }, true); content.location = "data:text/html,test Edit menu updates Scratchpad - bug 699130"; @@ -23,8 +21,6 @@ function test() function runTests() { - gScratchpadWindow.removeEventListener("load", runTests, false); - let sp = gScratchpadWindow.Scratchpad; let doc = gScratchpadWindow.document; let winUtils = gScratchpadWindow.QueryInterface(Ci.nsIInterfaceRequestor). diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_contexts.js b/browser/devtools/scratchpad/test/browser_scratchpad_contexts.js index 4d9f8ebee2d0..51919b65d129 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_contexts.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_contexts.js @@ -2,19 +2,14 @@ /* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ -// Reference to the Scratchpad chrome window object. -let gScratchpadWindow; - function test() { waitForExplicitFinish(); gBrowser.selectedTab = gBrowser.addTab(); - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); - - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", runTests, false); + gBrowser.selectedBrowser.addEventListener("load", function onLoad() { + gBrowser.selectedBrowser.removeEventListener("load", onLoad, true); + openScratchpad(runTests); }, true); content.location = "data:text/html,test context switch in Scratchpad"; @@ -22,8 +17,6 @@ function test() function runTests() { - gScratchpadWindow.removeEventListener("load", arguments.callee, false); - let sp = gScratchpadWindow.Scratchpad; let contentMenu = gScratchpadWindow.document.getElementById("sp-menu-content"); @@ -42,7 +35,7 @@ function runTests() is(contentMenu.getAttribute("checked"), "true", "content menuitem is checked"); - ok(!chromeMenu.hasAttribute("checked"), + isnot(chromeMenu.getAttribute("checked"), "true", "chrome menuitem is not checked"); ok(!notificationBox.currentNotification, @@ -66,7 +59,7 @@ function runTests() is(chromeMenu.getAttribute("checked"), "true", "chrome menuitem is checked"); - ok(!contentMenu.hasAttribute("checked"), + isnot(contentMenu.getAttribute("checked"), "true", "content menuitem is not checked"); ok(notificationBox.currentNotification, diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_execute_print.js b/browser/devtools/scratchpad/test/browser_scratchpad_execute_print.js index 0acf948e95c9..5278fcffc326 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_execute_print.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_execute_print.js @@ -7,11 +7,9 @@ function test() waitForExplicitFinish(); gBrowser.selectedTab = gBrowser.addTab(); - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); - - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", runTests, false); + gBrowser.selectedBrowser.addEventListener("load", function onTabLoad() { + gBrowser.selectedBrowser.removeEventListener("load", onTabLoad, true); + openScratchpad(runTests); }, true); content.location = "data:text/html,

test run() and display() in Scratchpad"; @@ -19,8 +17,6 @@ function test() function runTests() { - gScratchpadWindow.removeEventListener("load", arguments.callee, false); - let sp = gScratchpadWindow.Scratchpad; content.wrappedJSObject.foobarBug636725 = 1; diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_files.js b/browser/devtools/scratchpad/test/browser_scratchpad_files.js index e5ffddaed70e..5e50c24bc4e3 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_files.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_files.js @@ -5,9 +5,6 @@ Cu.import("resource://gre/modules/NetUtil.jsm"); Cu.import("resource://gre/modules/FileUtils.jsm"); -// Reference to the Scratchpad chrome window object. -let gScratchpadWindow; - // Reference to the Scratchpad object. let gScratchpad; @@ -22,11 +19,9 @@ function test() waitForExplicitFinish(); gBrowser.selectedTab = gBrowser.addTab(); - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); - - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", runTests, false); + gBrowser.selectedBrowser.addEventListener("load", function onLoad() { + gBrowser.selectedBrowser.removeEventListener("load", onLoad, true); + openScratchpad(runTests); }, true); content.location = "data:text/html,

test file open and save in Scratchpad"; @@ -34,8 +29,6 @@ function test() function runTests() { - gScratchpadWindow.removeEventListener("load", arguments.callee, false); - gScratchpad = gScratchpadWindow.Scratchpad; // Create a temporary file. diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_initialization.js b/browser/devtools/scratchpad/test/browser_scratchpad_initialization.js index b926598f1ceb..67bca826a208 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_initialization.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_initialization.js @@ -2,21 +2,17 @@ /* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ -// Reference to the Scratchpad chrome window object. -let gScratchpadWindow; - function test() { waitForExplicitFinish(); gBrowser.selectedTab = gBrowser.addTab(); - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); + gBrowser.selectedBrowser.addEventListener("load", function onLoad() { + gBrowser.selectedBrowser.removeEventListener("load", onLoad, true); - ok(Scratchpad, "Scratchpad variable exists"); + ok(window.Scratchpad, "Scratchpad variable exists"); - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", runTests, false); + openScratchpad(runTests); }, true); content.location = "data:text/html,initialization test for Scratchpad"; @@ -24,8 +20,6 @@ function test() function runTests() { - gScratchpadWindow.removeEventListener("load", arguments.callee, false); - let sp = gScratchpadWindow.Scratchpad; ok(sp, "Scratchpad object exists in new window"); is(typeof sp.run, "function", "Scratchpad.run() exists"); diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_inspect.js b/browser/devtools/scratchpad/test/browser_scratchpad_inspect.js index cd52d7a0349c..ed4c19f0d084 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_inspect.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_inspect.js @@ -2,19 +2,14 @@ /* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ -// Reference to the Scratchpad chrome window object. -let gScratchpadWindow; - function test() { waitForExplicitFinish(); gBrowser.selectedTab = gBrowser.addTab(); - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); - - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", runTests, false); + gBrowser.selectedBrowser.addEventListener("load", function onLoad() { + gBrowser.selectedBrowser.removeEventListener("load", onLoad, true); + openScratchpad(runTests); }, true); content.location = "data:text/html,foobarBug636725" + @@ -23,8 +18,6 @@ function test() function runTests() { - gScratchpadWindow.removeEventListener("load", arguments.callee, false); - let sp = gScratchpadWindow.Scratchpad; sp.setText("document"); @@ -34,8 +27,8 @@ function runTests() let propPanel = document.querySelector(".scratchpad_propertyPanel"); ok(propPanel, "property panel is open"); - propPanel.addEventListener("popupshown", function() { - propPanel.removeEventListener("popupshown", arguments.callee, false); + propPanel.addEventListener("popupshown", function onPopupShown() { + propPanel.removeEventListener("popupshown", onPopupShown, false); let tree = propPanel.querySelector("tree"); ok(tree, "property panel tree found"); diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_open.js b/browser/devtools/scratchpad/test/browser_scratchpad_open.js index 2847c3ecb9d0..5af2ffaf2c54 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_open.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_open.js @@ -2,8 +2,6 @@ /* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ -var ScratchpadManager = Scratchpad.ScratchpadManager; - // only finish() when correct number of tests are done const expected = 3; var count = 0; @@ -15,7 +13,6 @@ function done() } } - function test() { waitForExplicitFinish(); @@ -26,11 +23,7 @@ function test() function testOpen() { - let win = ScratchpadManager.openScratchpad(); - - win.addEventListener("load", function onScratchpadLoad() { - win.removeEventListener("load", onScratchpadLoad, false); - + openScratchpad(function(win) { is(win.Scratchpad.filename, undefined, "Default filename is undefined"); is(win.Scratchpad.getText(), win.Scratchpad.strings.GetStringFromName("scratchpadIntro"), @@ -40,7 +33,7 @@ function testOpen() win.close(); done(); - }, false); + }, {noFocus: true}); } function testOpenWithState() @@ -51,25 +44,19 @@ function testOpenWithState() text: "test text" }; - let win = ScratchpadManager.openScratchpad(state); - - win.addEventListener("load", function onScratchpadLoad() { - win.removeEventListener("load", onScratchpadLoad, false); - + openScratchpad(function(win) { is(win.Scratchpad.filename, state.filename, "Filename loaded from state"); is(win.Scratchpad.executionContext, state.executionContext, "Execution context loaded from state"); is(win.Scratchpad.getText(), state.text, "Content loaded from state"); win.close(); done(); - }, false); + }, {state: state, noFocus: true}); } function testOpenInvalidState() { - let state = 7; - - let win = ScratchpadManager.openScratchpad(state); + let win = openScratchpad(null, {state: 7}); ok(!win, "no scratchpad opened if state is not an object"); done(); } diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_restore.js b/browser/devtools/scratchpad/test/browser_scratchpad_restore.js index c1e1b0b921d2..a83c4213ce7e 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_restore.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_restore.js @@ -48,11 +48,7 @@ function testRestore() asyncMap(states, function(state, done) { // Open some scratchpad windows - let win = ScratchpadManager.openScratchpad(state); - win.addEventListener("load", function onScratchpadLoad() { - removeEventListener("load", onScratchpadLoad, false); - done(win); - }, false) + openScratchpad(done, {state: state, noFocus: true}); }, function(wins) { // Then save the windows to session store ScratchpadManager.saveOpenWindows(); @@ -74,12 +70,11 @@ function testRestore() is(restoredWins.length, 3, "Three scratchad windows restored"); asyncMap(restoredWins, function(restoredWin, done) { - restoredWin.addEventListener("load", function onScratchpadLoad() { - restoredWin.removeEventListener("load", onScratchpadLoad, false); - let state = restoredWin.Scratchpad.getState(); - restoredWin.close(); + openScratchpad(function(aWin) { + let state = aWin.Scratchpad.getState(); + aWin.close(); done(state); - }, false); + }, {window: restoredWin, noFocus: true}); }, function(restoredStates) { // Then make sure they were restored with the right states ok(statesMatch(restoredStates, states), diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_tab_switch.js b/browser/devtools/scratchpad/test/browser_scratchpad_tab_switch.js index 754adf99d15c..2db5a3c5a54e 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_tab_switch.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_tab_switch.js @@ -2,8 +2,6 @@ /* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ -// Reference to the Scratchpad chrome window object. -let gScratchpadWindow; let tab1; let tab2; let sp; @@ -14,15 +12,14 @@ function test() tab1 = gBrowser.addTab(); gBrowser.selectedTab = tab1; - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); + gBrowser.selectedBrowser.addEventListener("load", function onLoad1() { + gBrowser.selectedBrowser.removeEventListener("load", onLoad1, true); tab2 = gBrowser.addTab(); gBrowser.selectedTab = tab2; - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", runTests, false); + gBrowser.selectedBrowser.addEventListener("load", function onLoad2() { + gBrowser.selectedBrowser.removeEventListener("load", onLoad2, true); + openScratchpad(runTests); }, true); content.location = "data:text/html,test context switch in Scratchpad tab 2"; }, true); @@ -32,8 +29,6 @@ function test() function runTests() { - gScratchpadWindow.removeEventListener("load", runTests, true); - sp = gScratchpadWindow.Scratchpad; let contentMenu = gScratchpadWindow.document.getElementById("sp-menu-content"); @@ -52,7 +47,7 @@ function runTests() is(contentMenu.getAttribute("checked"), "true", "content menuitem is checked"); - ok(!browserMenu.hasAttribute("checked"), + isnot(browserMenu.getAttribute("checked"), "true", "chrome menuitem is not checked"); is(notificationBox.currentNotification, null, diff --git a/browser/devtools/scratchpad/test/browser_scratchpad_ui.js b/browser/devtools/scratchpad/test/browser_scratchpad_ui.js index 14058607076f..e388e0490ce5 100644 --- a/browser/devtools/scratchpad/test/browser_scratchpad_ui.js +++ b/browser/devtools/scratchpad/test/browser_scratchpad_ui.js @@ -2,19 +2,14 @@ /* Any copyright is dedicated to the Public Domain. http://creativecommons.org/publicdomain/zero/1.0/ */ -// Reference to the Scratchpad chrome window object. -let gScratchpadWindow; - function test() { waitForExplicitFinish(); gBrowser.selectedTab = gBrowser.addTab(); - gBrowser.selectedBrowser.addEventListener("load", function() { - gBrowser.selectedBrowser.removeEventListener("load", arguments.callee, true); - - gScratchpadWindow = Scratchpad.openScratchpad(); - gScratchpadWindow.addEventListener("load", runTests, false); + gBrowser.selectedBrowser.addEventListener("load", function onLoad() { + gBrowser.selectedBrowser.removeEventListener("load", onLoad, true); + openScratchpad(runTests); }, true); content.location = "data:text/html,foobarBug636725" + @@ -23,8 +18,6 @@ function test() function runTests() { - gScratchpadWindow.removeEventListener("load", arguments.callee, false); - let sp = gScratchpadWindow.Scratchpad; let doc = gScratchpadWindow.document; diff --git a/browser/devtools/scratchpad/test/head.js b/browser/devtools/scratchpad/test/head.js index e8244441c753..210dc57e094a 100644 --- a/browser/devtools/scratchpad/test/head.js +++ b/browser/devtools/scratchpad/test/head.js @@ -6,6 +6,61 @@ let gScratchpadWindow; // Reference to the Scratchpad chrome window object +/** + * Open a Scratchpad window. + * + * @param function aReadyCallback + * Optional. The function you want invoked when the Scratchpad instance + * is ready. + * @param object aOptions + * Optional. Options for opening the scratchpad: + * - window + * Provide this if there's already a Scratchpad window you want to wait + * loading for. + * - state + * Scratchpad state object. This is used when Scratchpad is open. + * - noFocus + * Boolean that tells you do not want the opened window to receive + * focus. + * @return nsIDOMWindow + * The new window object that holds Scratchpad. Note that the + * gScratchpadWindow global is also updated to reference the new window + * object. + */ +function openScratchpad(aReadyCallback, aOptions) +{ + aOptions = aOptions || {}; + + let win = aOptions.window || + Scratchpad.ScratchpadManager.openScratchpad(aOptions.state); + if (!win) { + return; + } + + let onLoad = function() { + win.removeEventListener("load", onLoad, false); + + win.Scratchpad.addObserver({ + onReady: function(aScratchpad) { + aScratchpad.removeObserver(this); + + if (aOptions.noFocus) { + aReadyCallback(win, aScratchpad); + } else { + waitForFocus(aReadyCallback.bind(null, win, aScratchpad), win); + } + } + }); + }; + + if (aReadyCallback) { + win.addEventListener("load", onLoad, false); + } + + gScratchpadWindow = win; + return gScratchpadWindow; +} + function cleanup() { if (gScratchpadWindow) { diff --git a/browser/devtools/sourceeditor/orion/Makefile.dryice.js b/browser/devtools/sourceeditor/orion/Makefile.dryice.js index cafc3e87bb6c..efcb73f640c3 100755 --- a/browser/devtools/sourceeditor/orion/Makefile.dryice.js +++ b/browser/devtools/sourceeditor/orion/Makefile.dryice.js @@ -45,12 +45,17 @@ var js_src = copy.createDataObject(); copy({ source: [ + ORION_EDITOR + "/orion/textview/global.js", + ORION_EDITOR + "/orion/textview/eventTarget.js", + ORION_EDITOR + "/orion/editor/regex.js", ORION_EDITOR + "/orion/textview/keyBinding.js", ORION_EDITOR + "/orion/textview/rulers.js", ORION_EDITOR + "/orion/textview/undoStack.js", ORION_EDITOR + "/orion/textview/textModel.js", + ORION_EDITOR + "/orion/textview/annotations.js", ORION_EDITOR + "/orion/textview/tooltip.js", ORION_EDITOR + "/orion/textview/textView.js", + ORION_EDITOR + "/orion/textview/textDND.js", ORION_EDITOR + "/orion/editor/htmlGrammar.js", ORION_EDITOR + "/orion/editor/textMateStyler.js", ORION_EDITOR + "/examples/textview/textStyler.js", @@ -69,6 +74,7 @@ copy({ source: [ ORION_EDITOR + "/orion/textview/textview.css", ORION_EDITOR + "/orion/textview/rulers.css", + ORION_EDITOR + "/orion/textview/annotations.css", ORION_EDITOR + "/examples/textview/textstyler.css", ORION_EDITOR + "/examples/editor/htmlStyles.css", ], diff --git a/browser/devtools/sourceeditor/orion/README b/browser/devtools/sourceeditor/orion/README index c8703c947f46..2465f87bdb21 100644 --- a/browser/devtools/sourceeditor/orion/README +++ b/browser/devtools/sourceeditor/orion/README @@ -8,25 +8,11 @@ The Orion editor web site: http://www.eclipse.org/orion To upgrade Orion to a newer version see the UPGRADE file. -Orion version: git clone from 2011-10-26 - commit hash 0ab295660e1f7d33ca2bfb8558b3b7492d2c5aa5 - + patch for Eclipse Bug 358623 - Drag and Drop support: - https://github.com/mihaisucan/orion.client/tree/bug-358623 - see https://bugs.eclipse.org/bugs/show_bug.cgi?id=358623 - + patch for Eclipse Bug 362286 - Monaco font line height: - https://github.com/mihaisucan/orion.client/tree/bug-362286 - see https://bugs.eclipse.org/bugs/show_bug.cgi?id=362286 - + patch for Eclipse Bug 362107 - Ctrl-Up/Down failure on Linux: - https://github.com/mihaisucan/orion.client/tree/bug-362107 - see https://bugs.eclipse.org/bugs/show_bug.cgi?id=362107 - + patch for Eclipse Bug 362428 - _getXToOffset() throws: - https://github.com/mihaisucan/orion.client/tree/bug-362428 - see https://bugs.eclipse.org/bugs/show_bug.cgi?id=362428 - + patch for Eclipse Bug 362835 - Pasted HTML shows twice: - https://github.com/mihaisucan/orion.client/tree/bug-362835 - see https://bugs.eclipse.org/bugs/show_bug.cgi?id=362835 - + patch for Eclipse Bug 363508 - Selection is broken after TextView hide/unhide - see https://bugs.eclipse.org/bugs/show_bug.cgi?id=363508 +Orion version: git clone from 2011-12-09 + commit hash d8a6dc01d9c561d6eb99f03b64c8c78ce785c59d + + patch for Eclipse Bug 366312 - right-clicking outside of the selection causes the caret to move + https://github.com/mihaisucan/orion.client/tree/bug-366312 + see https://bugs.eclipse.org/bugs/show_bug.cgi?id=366312 # License diff --git a/browser/devtools/sourceeditor/orion/mozilla.css b/browser/devtools/sourceeditor/orion/mozilla.css index 8d476a8f96e2..214a8477014d 100644 --- a/browser/devtools/sourceeditor/orion/mozilla.css +++ b/browser/devtools/sourceeditor/orion/mozilla.css @@ -2,15 +2,124 @@ http://creativecommons.org/publicdomain/zero/1.0/ */ .viewContainer { - font-size: inherit; /* inherit browser's default monospace font size */ + background: -moz-Dialog; + font-family: monospace; + font-size: inherit; /* inherit browser's default monospace font size */ } +.view { + background: #fff; +} + +.readonly > .view { + background: #f6f6f6; +} + +/* Styles for rulers */ +.ruler { + background-color: white; +} +.ruler.lines { + border-right: 1px solid lightgray; + text-align: right; +} + +/* Styles for the line number ruler */ .rulerLines { - background: -moz-Dialog; - color: -moz-DialogText; - min-width: 1.4em; - padding-left: 4px; - padding-right: 4px; - text-align: end; + background: -moz-Dialog; + color: -moz-DialogText; + min-width: 1.4em; + padding-left: 4px; + padding-right: 4px; + text-align: end; +} + +.token_singleline_comment { + color: green; +} + +.token_multiline_comment { + color: green; +} + +.token_doc_comment { + color: #00008F; +} + +.token_doc_html_markup { + color: #7F7F9F; +} + +.token_doc_tag { + color: #7F9FBF; +} + +.token_task_tag { + color: #7F9FBF; +} + +.token_string { + color: blue; +} + +.token_keyword { + color: darkred; + font-weight: bold; +} + +.token_space { + /* images/white_space.png */ + background-image: url(""); + background-repeat: no-repeat; + background-position: center center; +} + +.token_tab { + /* images/white_tab.png */ + background-image: url(""); + background-repeat: no-repeat; + background-position: left center; +} + +.line_caret { + background: #EAF2FE; +} + +.readonly .line_caret { + background: #fcfcfc; +} + +/* Styling for html syntax highlighting */ +.entity-name-tag { + color: #3f7f7f; +} + +.entity-other-attribute-name { + color: #7f007f; +} + +.punctuation-definition-comment { + color: #3f5fbf; +} + +.comment { + color: #3f5fbf +} + +.string-quoted { + color: #2a00ff; + font-style: italic; +} + +.invalid { + color: red; + font-weight: bold; +} + +.annotationRange.currentBracket { +} + +.annotationRange.matchingBracket { + outline: 1px solid red; } diff --git a/browser/devtools/sourceeditor/orion/orion.css b/browser/devtools/sourceeditor/orion/orion.css index 48207cf0a185..855159755281 100644 --- a/browser/devtools/sourceeditor/orion/orion.css +++ b/browser/devtools/sourceeditor/orion/orion.css @@ -3,9 +3,13 @@ } .viewContainer { + background-color: #eeeeee; font-family: monospace; font-size: 10pt; } +::-webkit-scrollbar-corner { + background-color: #eeeeee; +} .viewContent { }/* Styles for rulers */ @@ -25,16 +29,172 @@ text-align: right; } .ruler.overview { + border-left: 1px solid lightgray; width: 14px; } /* Styles for the line number ruler */ .rulerLines { - background-color: white; } .rulerLines.even .rulerLines.odd { -}.token_singleline_comment { +}/* Styles for the annotation ruler (all lines) */ +.annotation { +} +.annotation.error, +.annotation.warning +.annotation.task, +.annotation.bookmark, +.annotation.breakpoint, +.annotation.collapsed +.annotation.expanded { +} + +/* Styles for the annotation ruler (first line) */ +.annotationHTML { + cursor: pointer; + width: 16px; + height: 16px; + display: inline-block; + vertical-align: middle; + background-position: center; + background-repeat: no-repeat; +} +.annotationHTML.error { + /* images/error.gif */ + background-image: url(""); +} +.annotationHTML.warning { + /* images/warning.gif */ + background-image: url(""); +} +.annotationHTML.task { + /* images/task.gif */ + background-image: url(""); +} +.annotationHTML.bookmark { + /* images/bookmark.gif */ + background-image: url(""); +} +.annotationHTML.breakpoint { + /* images/breakpoint.gif */ + background-image: url(""); +} +.annotationHTML.collapsed { + /* images/collapsed.png */ + width: 14px; + height: 14px; + background-image: url(""); +} +.annotationHTML.expanded { + /* images/expanded.png */ + width: 14px; + height: 14px; + background-image: url(""); +} +.annotationHTML.multiple { + /* images/multiple.gif */ + background-image: url(""); +} +.annotationHTML.overlay { + /* images/plus.png */ + background-image: url(""); + background-position: right bottom; + position: relative; + top: -16px; +} +.annotationHTML.currentBracket { + /* images/currentBracket.png */ + background-image: url(""); +} +.annotationHTML.matchingBracket { + /* images/matchingBracket.png */ + background-image: url(""); +} +.annotationHTML.currentLine { + /* images/currentLine.gif */ + background-image: url(""); +} + +/* Styles for the overview ruler */ +.annotationOverview { + cursor: pointer; + border-radius: 2px; + left: 2px; + width: 8px; +} +.annotationOverview.task { + background-color: lightgreen; + border: 1px solid green; +} +.annotationOverview.breakpoint { + background-color: lightblue; + border: 1px solid blue; +} +.annotationOverview.bookmark { + background-color: yellow; + border: 1px solid orange; +} +.annotationOverview.error { + background-color: lightcoral; + border: 1px solid darkred; +} +.annotationOverview.warning { + background-color: Gold; + border: 1px solid black; +} +.annotationOverview.currentBracket { + background-color: lightgray; + border: 1px solid red; +} +.annotationOverview.matchingBracket { + background-color: lightgray; + border: 1px solid red; +} +.annotationOverview.currentLine { + background-color: #EAF2FE; + border: 1px solid black; +} + +/* Styles for text range */ +.annotationRange { + background-repeat: repeat-x; + background-position: left bottom; +} +.annotationRange.task { + /* images/squiggly_task.png */ + background-image: url(""); +} +.annotationRange.breakpoint { + /* images/squiggly_breakpoint.png */ + background-image: url(""); +} +.annotationRange.bookmark { + /* images/squiggly_bookmark.png */ + background-image: url(""); +} +.annotationRange.error { + /* images/squiggly_error.png */ + background-image: url(""); +} +.annotationRange.warning { + /* images/squiggly_warning.png */ + background-image: url(""); +} +.annotationRange.currentBracket { +} +.annotationRange.matchingBracket { + outline: 1px solid red; +} + +/* Styles for lines of text */ +.annotationLine { +} +.annotationLine.currentLine { + background-color: #EAF2FE; +} + +.token_singleline_comment { color: green; } @@ -67,15 +227,6 @@ font-weight: bold; } -.token_bracket_outline { - outline: 1px solid red; -} - -.token_bracket { - color: white; - background-color: grey; -} - .token_space { /* images/white_space.png */ background-image: url(""); diff --git a/browser/devtools/sourceeditor/orion/orion.js b/browser/devtools/sourceeditor/orion/orion.js index 7b73fb1f02f4..93d2681bfc6f 100644 --- a/browser/devtools/sourceeditor/orion/orion.js +++ b/browser/devtools/sourceeditor/orion/orion.js @@ -1,4 +1,289 @@ /******************************************************************************* + * @license + * Copyright (c) 2010, 2011 IBM Corporation and others. + * All rights reserved. This program and the accompanying materials are made + * available under the terms of the Eclipse Public License v1.0 + * (http://www.eclipse.org/legal/epl-v10.html), and the Eclipse Distribution + * License v1.0 (http://www.eclipse.org/org/documents/edl-v10.html). + * + * Contributors: + * Felipe Heidrich (IBM Corporation) - initial API and implementation + * Silenio Quarti (IBM Corporation) - initial API and implementation + * Mihai Sucan (Mozilla Foundation) - fix for Bug#364214 + */ + +/*global window */ + +/** + * Evaluates the definition function and mixes in the returned module with + * the module specified by moduleName. + *

+ * This function is intented to by used when RequireJS is not available. + *

+ * + * @param {String[]} deps The array of dependency names. + * @param {Function} callback The definition function. + * @param {String} moduleName The mixin module name. + */ +if (!window.define) { + window.define = function(deps, callback, moduleName) { + var module = this; + var split = (moduleName || "").split("/"), i, j; + for (i = 0; i < split.length; i++) { + module = module[split[i]] = (module[split[i]] || {}); + } + var depModules = [], depModule; + for (j = 0; j < deps.length; j++) { + depModule = this; + split = deps[j].split("/"); + for (i = 0; i < split.length - 1; i++) { + depModule = depModule[split[i]] = (depModule[split[i]] || {}); + } + depModules.push(depModule); + } + var newModule = callback.apply(this, depModules); + for (var p in newModule) { + if (newModule.hasOwnProperty(p)) { + module[p] = newModule[p]; + } + } + }; +} + +/** + * Require/get the defined modules. + *

+ * This function is intented to by used when RequireJS is not available. + *

+ * + * @param {String[]|String} deps The array of dependency names. This can also be + * a string, a single dependency name. + * @param {Function} [callback] Optional, the callback function to execute when + * multiple dependencies are required. The callback arguments will have + * references to each module in the same order as the deps array. + * @returns {Object|undefined} If the deps parameter is a string, then this + * function returns the required module definition, otherwise undefined is + * returned. + */ +if (!window.require) { + window.require = function(deps, callback) { + var depsArr = typeof deps === "string" ? [deps] : deps; + var depModules = [], depModule, split, i, j; + for (j = 0; j < depsArr.length; j++) { + depModule = this; + split = depsArr[j].split("/"); + for (i = 0; i < split.length - 1; i++) { + depModule = depModule[split[i]] = (depModule[split[i]] || {}); + } + depModules.push(depModule); + } + if (callback) { + callback.apply(this, depModules); + } + return typeof deps === "string" ? depModules[0] : undefined; + }; +}/******************************************************************************* + * Copyright (c) 2010, 2011 IBM Corporation and others. + * All rights reserved. This program and the accompanying materials are made + * available under the terms of the Eclipse Public License v1.0 + * (http://www.eclipse.org/legal/epl-v10.html), and the Eclipse Distribution + * License v1.0 (http://www.eclipse.org/org/documents/edl-v10.html). + * + * Contributors: + * Felipe Heidrich (IBM Corporation) - initial API and implementation + * Silenio Quarti (IBM Corporation) - initial API and implementation + ******************************************************************************/ + +/*global define */ +define([], function() { + /** + * Constructs a new EventTarget object. + * + * @class + * @name orion.textview.EventTarget + */ + function EventTarget() { + } + /** + * Adds in the event target interface into the specified object. + * + * @param {Object} object The object to add in the event target interface. + */ + EventTarget.addMixin = function(object) { + var proto = EventTarget.prototype; + for (var p in proto) { + if (proto.hasOwnProperty(p)) { + object[p] = proto[p]; + } + } + }; + EventTarget.prototype = /** @lends orion.textview.EventTarget.prototype */ { + /** + * Adds an event listener to this event target. + * + * @param {String} type The event type. + * @param {Function|EventListener} listener The function or the EventListener that will be executed when the event happens. + * @param {Boolean} [useCapture=false] true if the listener should be trigged in the capture phase. + * + * @see #removeEventListener + */ + addEventListener: function(type, listener, useCapture) { + if (!this._eventTypes) { this._eventTypes = {}; } + var state = this._eventTypes[type]; + if (!state) { + state = this._eventTypes[type] = {level: 0, listeners: []}; + } + var listeners = state.listeners; + listeners.push({listener: listener, useCapture: useCapture}); + }, + /** + * Dispatches the given event to the listeners added to this event target. + * @param {Event} evt The event to dispatch. + */ + dispatchEvent: function(evt) { + if (!this._eventTypes) { return; } + var type = evt.type; + var state = this._eventTypes[type]; + if (state) { + var listeners = state.listeners; + try { + state.level++; + if (listeners) { + for (var i=0, len=listeners.length; i < len; i++) { + if (listeners[i]) { + var l = listeners[i].listener; + if (typeof l === "function") { + l.call(this, evt); + } else if (l.handleEvent && typeof l.handleEvent === "function") { + l.handleEvent(evt); + } + } + } + } + } finally { + state.level--; + if (state.compact && state.level === 0) { + for (var j=listeners.length - 1; j >= 0; j--) { + if (!listeners[j]) { + listeners.splice(j, 1); + } + } + if (listeners.length === 0) { + delete this._eventTypes[type]; + } + state.compact = false; + } + } + } + }, + /** + * Returns whether there is a listener for the specified event type. + * + * @param {String} type The event type + * + * @see #addEventListener + * @see #removeEventListener + */ + isListening: function(type) { + if (!this._eventTypes) { return false; } + return this._eventTypes[type] !== undefined; + }, + /** + * Removes an event listener from the event target. + *

+ * All the parameters must be the same ones used to add the listener. + *

+ * + * @param {String} type The event type + * @param {Function|EventListener} listener The function or the EventListener that will be executed when the event happens. + * @param {Boolean} [useCapture=false] true if the listener should be trigged in the capture phase. + * + * @see #addEventListener + */ + removeEventListener: function(type, listener, useCapture){ + if (!this._eventTypes) { return; } + var state = this._eventTypes[type]; + if (state) { + var listeners = state.listeners; + for (var i=0, len=listeners.length; i < len; i++) { + var l = listeners[i]; + if (l && l.listener === listener && l.useCapture === useCapture) { + if (state.level !== 0) { + listeners[i] = null; + state.compact = true; + } else { + listeners.splice(i, 1); + } + break; + } + } + if (listeners.length === 0) { + delete this._eventTypes[type]; + } + } + } + }; + return {EventTarget: EventTarget}; +}, "orion/textview"); +/******************************************************************************* + * @license + * Copyright (c) 2011 IBM Corporation and others. + * All rights reserved. This program and the accompanying materials are made + * available under the terms of the Eclipse Public License v1.0 + * (http://www.eclipse.org/legal/epl-v10.html), and the Eclipse Distribution + * License v1.0 (http://www.eclipse.org/org/documents/edl-v10.html). + * + * Contributors: + * IBM Corporation - initial API and implementation + *******************************************************************************/ +/*global define */ +/*jslint browser:true regexp:false*/ +/** + * @name orion.editor.regex + * @class Utilities for dealing with regular expressions. + * @description Utilities for dealing with regular expressions. + */ +define([], function() { + /** + * @methodOf orion.editor.regex + * @static + * @description Escapes regex special characters in the input string. + * @param {String} str The string to escape. + * @returns {String} A copy of str with regex special characters escaped. + */ + function escape(str) { + return str.replace(/([\\$\^*\/+?\.\(\)|{}\[\]])/g, "\\$&"); + } + + /** + * @methodOf orion.editor.regex + * @static + * @description Parses a pattern and flags out of a regex literal string. + * @param {String} str The string to parse. Should look something like "/ab+c/" or "/ab+c/i". + * @returns {Object} If str looks like a regex literal, returns an object with properties + *
+ *
pattern
{String}
+ *
flags
{String}
+ *
otherwise returns null. + */ + function parse(str) { + var regexp = /^\s*\/(.+)\/([gim]{0,3})\s*$/.exec(str); + if (regexp) { + return { + pattern : regexp[1], + flags : regexp[2] + }; + } + return null; + } + + return { + escape: escape, + parse: parse + }; +}, "orion/editor"); +/******************************************************************************* + * @license * Copyright (c) 2010, 2011 IBM Corporation and others. * All rights reserved. This program and the accompanying materials are made * available under the terms of the Eclipse Public License v1.0 @@ -12,38 +297,29 @@ /*global window define */ -/** - * @namespace The global container for Orion APIs. - */ -var orion = orion || {}; -/** - * @namespace The container for textview APIs. - */ -orion.textview = orion.textview || {}; - -/** - * Constructs a new key binding with the given key code and modifiers. - * - * @param {String|Number} keyCode the key code. - * @param {Boolean} mod1 the primary modifier (usually Command on Mac and Control on other platforms). - * @param {Boolean} mod2 the secondary modifier (usually Shift). - * @param {Boolean} mod3 the third modifier (usually Alt). - * @param {Boolean} mod4 the fourth modifier (usually Control on the Mac). - * - * @class A KeyBinding represents of a key code and a modifier state that can be triggered by the user using the keyboard. - * @name orion.textview.KeyBinding - * - * @property {String|Number} keyCode The key code. - * @property {Boolean} mod1 The primary modifier (usually Command on Mac and Control on other platforms). - * @property {Boolean} mod2 The secondary modifier (usually Shift). - * @property {Boolean} mod3 The third modifier (usually Alt). - * @property {Boolean} mod4 The fourth modifier (usually Control on the Mac). - * - * @see orion.textview.TextView#setKeyBinding - */ -orion.textview.KeyBinding = (function() { +define([], function() { var isMac = window.navigator.platform.indexOf("Mac") !== -1; - /** @private */ + + /** + * Constructs a new key binding with the given key code and modifiers. + * + * @param {String|Number} keyCode the key code. + * @param {Boolean} mod1 the primary modifier (usually Command on Mac and Control on other platforms). + * @param {Boolean} mod2 the secondary modifier (usually Shift). + * @param {Boolean} mod3 the third modifier (usually Alt). + * @param {Boolean} mod4 the fourth modifier (usually Control on the Mac). + * + * @class A KeyBinding represents of a key code and a modifier state that can be triggered by the user using the keyboard. + * @name orion.textview.KeyBinding + * + * @property {String|Number} keyCode The key code. + * @property {Boolean} mod1 The primary modifier (usually Command on Mac and Control on other platforms). + * @property {Boolean} mod2 The secondary modifier (usually Shift). + * @property {Boolean} mod3 The third modifier (usually Alt). + * @property {Boolean} mod4 The fourth modifier (usually Control on the Mac). + * + * @see orion.textview.TextView#setKeyBinding + */ function KeyBinding (keyCode, mod1, mod2, mod3, mod4) { if (typeof(keyCode) === "string") { this.keyCode = keyCode.toUpperCase().charCodeAt(0); @@ -89,16 +365,10 @@ orion.textview.KeyBinding = (function() { return true; } }; - return KeyBinding; -}()); - -if (typeof window !== "undefined" && typeof window.define !== "undefined") { - define([], function() { - return orion.textview; - }); -} - + return {KeyBinding: KeyBinding}; +}, "orion/textview"); /******************************************************************************* + * @license * Copyright (c) 2010, 2011 IBM Corporation and others. * All rights reserved. This program and the accompanying materials are made * available under the terms of the Eclipse Public License v1.0 @@ -108,49 +378,40 @@ if (typeof window !== "undefined" && typeof window.define !== "undefined") { * Contributors: IBM Corporation - initial API and implementation ******************************************************************************/ -/*global window define setTimeout clearTimeout setInterval clearInterval Node */ +/*global define setTimeout clearTimeout setInterval clearInterval Node */ -/** - * @namespace The global container for Orion APIs. - */ -var orion = orion || {}; -/** - * @namespace The container for textview APIs. - */ -orion.textview = orion.textview || {}; +define(['orion/textview/tooltip'], function(mTooltip) { -/** - * Constructs a new ruler. - *

- * The default implementation does not implement all the methods in the interface - * and is useful only for objects implementing rulers. - *

- * - * @param {orion.textview.AnnotationModel} annotationModel the annotation model for the ruler. - * @param {String} [rulerLocation="left"] the location for the ruler. - * @param {String} [rulerOverview="page"] the overview for the ruler. - * @param {orion.textview.Style} [rulerStyle] the style for the ruler. - * - * @class This interface represents a ruler for the text view. - *

- * A Ruler is a graphical element that is placed either on the left or on the right side of - * the view. It can be used to provide the view with per line decoration such as line numbering, - * bookmarks, breakpoints, folding disclosures, etc. - *

- * There are two types of rulers: page and document. A page ruler only shows the content for the lines that are - * visible, while a document ruler always shows the whole content. - *

- * See:
- * {@link orion.textview.LineNumberRuler}
- * {@link orion.textview.AnnotationRuler}
- * {@link orion.textview.OverviewRuler}
- * {@link orion.textview.TextView}
- * {@link orion.textview.TextView#addRuler} - *

- * @name orion.textview.Ruler - */ -orion.textview.Ruler = (function() { - /** @private */ + /** + * Constructs a new ruler. + *

+ * The default implementation does not implement all the methods in the interface + * and is useful only for objects implementing rulers. + *

+ * + * @param {orion.textview.AnnotationModel} annotationModel the annotation model for the ruler. + * @param {String} [rulerLocation="left"] the location for the ruler. + * @param {String} [rulerOverview="page"] the overview for the ruler. + * @param {orion.textview.Style} [rulerStyle] the style for the ruler. + * + * @class This interface represents a ruler for the text view. + *

+ * A Ruler is a graphical element that is placed either on the left or on the right side of + * the view. It can be used to provide the view with per line decoration such as line numbering, + * bookmarks, breakpoints, folding disclosures, etc. + *

+ * There are two types of rulers: page and document. A page ruler only shows the content for the lines that are + * visible, while a document ruler always shows the whole content. + *

+ * See:
+ * {@link orion.textview.LineNumberRuler}
+ * {@link orion.textview.AnnotationRuler}
+ * {@link orion.textview.OverviewRuler}
+ * {@link orion.textview.TextView}
+ * {@link orion.textview.TextView#addRuler} + *

+ * @name orion.textview.Ruler + */ function Ruler (annotationModel, rulerLocation, rulerOverview, rulerStyle) { this._location = rulerLocation || "left"; this._overview = rulerOverview || "page"; @@ -158,8 +419,11 @@ orion.textview.Ruler = (function() { this._types = []; this._view = null; var self = this; - this._annotationModelListener = { - onChanged: function(e) { + this._listener = { + onTextModelChanged: function(e) { + self._onTextModelChanged(e); + }, + onAnnotationModelChanged: function(e) { self._onAnnotationModelChanged(e); } }; @@ -210,7 +474,7 @@ orion.textview.Ruler = (function() { var annotation = annotations.next(); if (!this.isAnnotationTypeVisible(annotation.type)) { continue; } var annotationLineStart = baseModel.getLineAtOffset(annotation.start); - var annotationLineEnd = baseModel.getLineAtOffset(annotation.end - 1); + var annotationLineEnd = baseModel.getLineAtOffset(Math.max(annotation.start, annotation.end - 1)); for (var lineIndex = annotationLineStart; lineIndex<=annotationLineEnd; lineIndex++) { var visualLineIndex = lineIndex; if (model !== baseModel) { @@ -332,11 +596,11 @@ orion.textview.Ruler = (function() { */ setAnnotationModel: function (annotationModel) { if (this._annotationModel) { - this._annotationModel.removeListener(this._annotationModelListener); + this._annotationModel.removEventListener("Changed", this._listener.onAnnotationModelChanged); } this._annotationModel = annotationModel; if (this._annotationModel) { - this._annotationModel.addListener(this._annotationModelListener); + this._annotationModel.addEventListener("Changed", this._listener.onAnnotationModelChanged); } }, /** @@ -374,11 +638,11 @@ orion.textview.Ruler = (function() { */ setView: function (view) { if (this._onTextModelChanged && this._view) { - this._view.removeEventListener("ModelChanged", this, this._onTextModelChanged); + this._view.removeEventListener("ModelChanged", this._listener.onTextModelChanged); } this._view = view; if (this._onTextModelChanged && this._view) { - this._view.addEventListener("ModelChanged", this, this._onTextModelChanged); + this._view.addEventListener("ModelChanged", this._listener.onTextModelChanged); } }, /** @@ -407,7 +671,7 @@ orion.textview.Ruler = (function() { * @param {DOMEvent} e the mouse move event. */ onMouseMove: function(lineIndex, e) { - var tooltip = orion.textview.Tooltip.getTooltip(this._view); + var tooltip = mTooltip.Tooltip.getTooltip(this._view); if (!tooltip) { return; } if (tooltip.isVisible() && this._tooltipLineIndex === lineIndex) { return; } this._tooltipLineIndex = lineIndex; @@ -426,7 +690,9 @@ orion.textview.Ruler = (function() { * @param {Number} lineIndex the line index of the annotation under the pointer. * @param {DOMEvent} e the mouse over event. */ - onMouseOver: this._onMouseMove, + onMouseOver: function(lineIndex, e) { + this.onMouseMove(lineIndex, e); + }, /** * This event is sent when the mouse pointer exits a line annotation. * @@ -435,7 +701,7 @@ orion.textview.Ruler = (function() { * @param {DOMEvent} e the mouse out event. */ onMouseOut: function(lineIndex, e) { - var tooltip = orion.textview.Tooltip.getTooltip(this._view); + var tooltip = mTooltip.Tooltip.getTooltip(this._view); if (!tooltip) { return; } tooltip.setTarget(null); }, @@ -566,37 +832,34 @@ orion.textview.Ruler = (function() { return result; } }; - return Ruler; -}()); -/** - * Constructs a new line numbering ruler. - * - * @param {String} [rulerLocation="left"] the location for the ruler. - * @param {orion.textview.Style} [rulerStyle=undefined] the style for the ruler. - * @param {orion.textview.Style} [oddStyle={style: {backgroundColor: "white"}] the style for lines with odd line index. - * @param {orion.textview.Style} [evenStyle={backgroundColor: "white"}] the style for lines with even line index. - * - * @augments orion.textview.Ruler - * @class This objects implements a line numbering ruler. - * - *

See:
- * {@link orion.textview.Ruler} - *

- * @name orion.textview.LineNumberRuler - */ -orion.textview.LineNumberRuler = (function() { - /** @private */ + /** + * Constructs a new line numbering ruler. + * + * @param {orion.textview.AnnotationModel} annotationModel the annotation model for the ruler. + * @param {String} [rulerLocation="left"] the location for the ruler. + * @param {orion.textview.Style} [rulerStyle=undefined] the style for the ruler. + * @param {orion.textview.Style} [oddStyle={style: {backgroundColor: "white"}] the style for lines with odd line index. + * @param {orion.textview.Style} [evenStyle={backgroundColor: "white"}] the style for lines with even line index. + * + * @augments orion.textview.Ruler + * @class This objects implements a line numbering ruler. + * + *

See:
+ * {@link orion.textview.Ruler} + *

+ * @name orion.textview.LineNumberRuler + */ function LineNumberRuler (annotationModel, rulerLocation, rulerStyle, oddStyle, evenStyle) { - orion.textview.Ruler.call(this, annotationModel, rulerLocation, "page", rulerStyle); + Ruler.call(this, annotationModel, rulerLocation, "page", rulerStyle); this._oddStyle = oddStyle || {style: {backgroundColor: "white"}}; this._evenStyle = evenStyle || {style: {backgroundColor: "white"}}; this._numOfDigits = 0; } - LineNumberRuler.prototype = new orion.textview.Ruler(); + LineNumberRuler.prototype = new Ruler(); /** @ignore */ LineNumberRuler.prototype.getAnnotations = function(startLine, endLine) { - var result = orion.textview.Ruler.prototype.getAnnotations.call(this, startLine, endLine); + var result = Ruler.prototype.getAnnotations.call(this, startLine, endLine); var model = this._view.getModel(); for (var lineIndex = startLine; lineIndex < endLine; lineIndex++) { var style = lineIndex & 1 ? this._oddStyle : this._evenStyle; @@ -628,74 +891,67 @@ orion.textview.LineNumberRuler = (function() { this._view.redrawLines(startLine, model.getLineCount(), this); } }; - return LineNumberRuler; -}()); -/** - * @class This is class represents an annotation for the AnnotationRuler. - *

- * See:
- * {@link orion.textview.AnnotationRuler} - *

- * - * @name orion.textview.Annotation - * - * @property {String} [html=""] The html content for the annotation, typically contains an image. - * @property {orion.textview.Style} [style] the style for the annotation. - * @property {orion.textview.Style} [overviewStyle] the style for the annotation in the overview ruler. - */ -/** - * Contructs a new annotation ruler. - * - * @param {String} [rulerLocation="left"] the location for the ruler. - * @param {orion.textview.Style} [rulerStyle=undefined] the style for the ruler. - * @param {orion.textview.Annotation} [defaultAnnotation] the default annotation. - * - * @augments orion.textview.Ruler - * @class This objects implements an annotation ruler. - * - *

See:
- * {@link orion.textview.Ruler}
- * {@link orion.textview.Annotation} - *

- * @name orion.textview.AnnotationRuler - */ -orion.textview.AnnotationRuler = (function() { - /** @private */ - function AnnotationRuler (annotationModel, rulerLocation, rulerStyle) { - orion.textview.Ruler.call(this, annotationModel, rulerLocation, "page", rulerStyle); - } - AnnotationRuler.prototype = new orion.textview.Ruler(); - return AnnotationRuler; -}()); - -/** - * Contructs an overview ruler. - *

- * The overview ruler is used in conjunction with a AnnotationRuler, for each annotation in the - * AnnotationRuler this ruler displays a mark in the overview. Clicking on the mark causes the - * view to scroll to the annotated line. - *

- * - * @param {String} [rulerLocation="left"] the location for the ruler. - * @param {orion.textview.Style} [rulerStyle=undefined] the style for the ruler. - * @param {orion.textview.AnnotationRuler} [annotationRuler] the annotation ruler for the overview. - * - * @augments orion.textview.Ruler - * @class This objects implements an overview ruler. - * - *

See:
- * {@link orion.textview.AnnotationRuler}
- * {@link orion.textview.Ruler} - *

- * @name orion.textview.OverviewRuler - */ -orion.textview.OverviewRuler = (function() { - /** @private */ - function OverviewRuler (annotationModel, rulerLocation, rulerStyle) { - orion.textview.Ruler.call(this, annotationModel, rulerLocation, "document", rulerStyle); + /** + * @class This is class represents an annotation for the AnnotationRuler. + *

+ * See:
+ * {@link orion.textview.AnnotationRuler} + *

+ * + * @name orion.textview.Annotation + * + * @property {String} [html=""] The html content for the annotation, typically contains an image. + * @property {orion.textview.Style} [style] the style for the annotation. + * @property {orion.textview.Style} [overviewStyle] the style for the annotation in the overview ruler. + */ + /** + * Constructs a new annotation ruler. + * + * @param {orion.textview.AnnotationModel} annotationModel the annotation model for the ruler. + * @param {String} [rulerLocation="left"] the location for the ruler. + * @param {orion.textview.Style} [rulerStyle=undefined] the style for the ruler. + * @param {orion.textview.Annotation} [defaultAnnotation] the default annotation. + * + * @augments orion.textview.Ruler + * @class This objects implements an annotation ruler. + * + *

See:
+ * {@link orion.textview.Ruler}
+ * {@link orion.textview.Annotation} + *

+ * @name orion.textview.AnnotationRuler + */ + function AnnotationRuler (annotationModel, rulerLocation, rulerStyle) { + Ruler.call(this, annotationModel, rulerLocation, "page", rulerStyle); } - OverviewRuler.prototype = new orion.textview.Ruler(); + AnnotationRuler.prototype = new Ruler(); + + /** + * Constructs a new overview ruler. + *

+ * The overview ruler is used in conjunction with a AnnotationRuler, for each annotation in the + * AnnotationRuler this ruler displays a mark in the overview. Clicking on the mark causes the + * view to scroll to the annotated line. + *

+ * + * @param {orion.textview.AnnotationModel} annotationModel the annotation model for the ruler. + * @param {String} [rulerLocation="left"] the location for the ruler. + * @param {orion.textview.Style} [rulerStyle=undefined] the style for the ruler. + * + * @augments orion.textview.Ruler + * @class This objects implements an overview ruler. + * + *

See:
+ * {@link orion.textview.AnnotationRuler}
+ * {@link orion.textview.Ruler} + *

+ * @name orion.textview.OverviewRuler + */ + function OverviewRuler (annotationModel, rulerLocation, rulerStyle) { + Ruler.call(this, annotationModel, rulerLocation, "document", rulerStyle); + } + OverviewRuler.prototype = new Ruler(); /** @ignore */ OverviewRuler.prototype.getRulerStyle = function() { @@ -719,7 +975,7 @@ orion.textview.OverviewRuler = (function() { } return "Line: " + (mapLine + 1); } - return orion.textview.Ruler.prototype._getTooltipContents.call(this, lineIndex, annotations); + return Ruler.prototype._getTooltipContents.call(this, lineIndex, annotations); }; /** @ignore */ OverviewRuler.prototype._mergeAnnotation = function(previousAnnotation, annotation, annotationLineIndex, annotationLineCount) { @@ -733,15 +989,27 @@ orion.textview.OverviewRuler = (function() { } return result; }; - return OverviewRuler; -}()); -orion.textview.FoldingRuler = (function() { - /** @private */ + /** + * Constructs a new folding ruler. + * + * @param {orion.textview.AnnotationModel} annotationModel the annotation model for the ruler. + * @param {String} [rulerLocation="left"] the location for the ruler. + * @param {orion.textview.Style} [rulerStyle=undefined] the style for the ruler. + * + * @augments orion.textview.Ruler + * @class This objects implements an overview ruler. + * + *

See:
+ * {@link orion.textview.AnnotationRuler}
+ * {@link orion.textview.Ruler} + *

+ * @name orion.textview.OverviewRuler + */ function FoldingRuler (annotationModel, rulerLocation, rulerStyle) { - orion.textview.AnnotationRuler.call(this, annotationModel, rulerLocation, rulerStyle); + AnnotationRuler.call(this, annotationModel, rulerLocation, rulerStyle); } - FoldingRuler.prototype = new orion.textview.AnnotationRuler(); + FoldingRuler.prototype = new AnnotationRuler(); /** @ignore */ FoldingRuler.prototype.onClick = function(lineIndex, e) { @@ -763,7 +1031,7 @@ orion.textview.FoldingRuler = (function() { annotation = a; } if (annotation) { - var tooltip = orion.textview.Tooltip.getTooltip(this._view); + var tooltip = mTooltip.Tooltip.getTooltip(this._view); if (tooltip) { tooltip.setTarget(null); } @@ -782,12 +1050,12 @@ orion.textview.FoldingRuler = (function() { return null; } } - return orion.textview.AnnotationRuler.prototype._getTooltipContents.call(this, lineIndex, annotations); + return AnnotationRuler.prototype._getTooltipContents.call(this, lineIndex, annotations); }; /** @ignore */ FoldingRuler.prototype._onAnnotationModelChanged = function(e) { if (e.textModelChangedEvent) { - orion.textview.AnnotationRuler.prototype._onAnnotationModelChanged.call(this, e); + AnnotationRuler.prototype._onAnnotationModelChanged.call(this, e); return; } var view = this._view; @@ -815,15 +1083,16 @@ orion.textview.FoldingRuler = (function() { } }; - return FoldingRuler; -}()); - -if (typeof window !== "undefined" && typeof window.define !== "undefined") { - define(['orion/textview/tooltip'], function() { - return orion.textview; - }); -} + return { + Ruler: Ruler, + AnnotationRuler: AnnotationRuler, + LineNumberRuler: LineNumberRuler, + OverviewRuler: OverviewRuler, + FoldingRuler: FoldingRuler + }; +}, "orion/textview"); /******************************************************************************* + * @license * Copyright (c) 2010, 2011 IBM Corporation and others. * All rights reserved. This program and the accompanying materials are made * available under the terms of the Eclipse Public License v1.0 @@ -833,33 +1102,10 @@ if (typeof window !== "undefined" && typeof window.define !== "undefined") { * Contributors: IBM Corporation - initial API and implementation ******************************************************************************/ -/*global window define */ +/*global define */ -/** - * @namespace The global container for Orion APIs. - */ -var orion = orion || {}; -/** - * @namespace The container for textview APIs. - */ -orion.textview = orion.textview || {}; +define([], function() { -/** - * Constructs a new UndoStack on a text view. - * - * @param {orion.textview.TextView} view the text view for the undo stack. - * @param {Number} [size=100] the size for the undo stack. - * - * @name orion.textview.UndoStack - * @class The UndoStack is used to record the history of a text model associated to an view. Every - * change to the model is added to stack, allowing the application to undo and redo these changes. - * - *

- * See:
- * {@link orion.textview.TextView}
- *

- */ -orion.textview.UndoStack = (function() { /** * Constructs a new Change object. * @@ -867,53 +1113,50 @@ orion.textview.UndoStack = (function() { * @name orion.textview.Change * @private */ - var Change = (function() { - function Change(offset, text, previousText) { - this.offset = offset; - this.text = text; - this.previousText = previousText; - } - Change.prototype = { - /** @ignore */ - undo: function (view, select) { - this._doUndoRedo(this.offset, this.previousText, this.text, view, select); - }, - /** @ignore */ - redo: function (view, select) { - this._doUndoRedo(this.offset, this.text, this.previousText, view, select); - }, - _doUndoRedo: function(offset, text, previousText, view, select) { - var model = view.getModel(); - /* - * TODO UndoStack should be changing the text in the base model. - * This is code needs to change when modifications in the base - * model are supported properly by the projection model. - */ - if (model.mapOffset && view.annotationModel) { - var mapOffset = model.mapOffset(offset, true); - if (mapOffset < 0) { - var annotationModel = view.annotationModel; - var iter = annotationModel.getAnnotations(offset, offset + 1); - while (iter.hasNext()) { - var annotation = iter.next(); - if (annotation.type === "orion.annotation.folding") { - annotation.expand(); - mapOffset = model.mapOffset(offset, true); - break; - } + function Change(offset, text, previousText) { + this.offset = offset; + this.text = text; + this.previousText = previousText; + } + Change.prototype = { + /** @ignore */ + undo: function (view, select) { + this._doUndoRedo(this.offset, this.previousText, this.text, view, select); + }, + /** @ignore */ + redo: function (view, select) { + this._doUndoRedo(this.offset, this.text, this.previousText, view, select); + }, + _doUndoRedo: function(offset, text, previousText, view, select) { + var model = view.getModel(); + /* + * TODO UndoStack should be changing the text in the base model. + * This is code needs to change when modifications in the base + * model are supported properly by the projection model. + */ + if (model.mapOffset && view.annotationModel) { + var mapOffset = model.mapOffset(offset, true); + if (mapOffset < 0) { + var annotationModel = view.annotationModel; + var iter = annotationModel.getAnnotations(offset, offset + 1); + while (iter.hasNext()) { + var annotation = iter.next(); + if (annotation.type === "orion.annotation.folding") { + annotation.expand(); + mapOffset = model.mapOffset(offset, true); + break; } } - if (mapOffset < 0) { return; } - offset = mapOffset; - } - view.setText(text, offset, offset + previousText.length); - if (select) { - view.setSelection(offset, offset + text.length); } + if (mapOffset < 0) { return; } + offset = mapOffset; } - }; - return Change; - }()); + view.setText(text, offset, offset + previousText.length); + if (select) { + view.setSelection(offset, offset + text.length); + } + } + }; /** * Constructs a new CompoundChange object. @@ -922,44 +1165,63 @@ orion.textview.UndoStack = (function() { * @name orion.textview.CompoundChange * @private */ - var CompoundChange = (function() { - function CompoundChange (selection, caret) { - this.selection = selection; - this.caret = caret; - this.changes = []; - } - CompoundChange.prototype = { - /** @ignore */ - add: function (change) { - this.changes.push(change); - }, - /** @ignore */ - undo: function (view, select) { - for (var i=this.changes.length - 1; i >= 0; i--) { - this.changes[i].undo(view, false); - } - if (select) { - var start = this.selection.start; - var end = this.selection.end; - view.setSelection(this.caret ? start : end, this.caret ? end : start); - } - }, - /** @ignore */ - redo: function (view, select) { - for (var i = 0; i < this.changes.length; i++) { - this.changes[i].redo(view, false); - } - if (select) { - var start = this.selection.start; - var end = this.selection.end; - view.setSelection(this.caret ? start : end, this.caret ? end : start); - } + function CompoundChange () { + this.changes = []; + } + CompoundChange.prototype = { + /** @ignore */ + add: function (change) { + this.changes.push(change); + }, + /** @ignore */ + end: function (view) { + this.endSelection = view.getSelection(); + this.endCaret = view.getCaretOffset(); + }, + /** @ignore */ + undo: function (view, select) { + for (var i=this.changes.length - 1; i >= 0; i--) { + this.changes[i].undo(view, false); } - }; - return CompoundChange; - }()); + if (select) { + var start = this.startSelection.start; + var end = this.startSelection.end; + view.setSelection(this.startCaret ? start : end, this.startCaret ? end : start); + } + }, + /** @ignore */ + redo: function (view, select) { + for (var i = 0; i < this.changes.length; i++) { + this.changes[i].redo(view, false); + } + if (select) { + var start = this.endSelection.start; + var end = this.endSelection.end; + view.setSelection(this.endCaret ? start : end, this.endCaret ? end : start); + } + }, + /** @ignore */ + start: function (view) { + this.startSelection = view.getSelection(); + this.startCaret = view.getCaretOffset(); + } + }; - /** @private */ + /** + * Constructs a new UndoStack on a text view. + * + * @param {orion.textview.TextView} view the text view for the undo stack. + * @param {Number} [size=100] the size for the undo stack. + * + * @name orion.textview.UndoStack + * @class The UndoStack is used to record the history of a text model associated to an view. Every + * change to the model is added to stack, allowing the application to undo and redo these changes. + * + *

+ * See:
+ * {@link orion.textview.TextView}
+ *

+ */ function UndoStack (view, size) { this.view = view; this.size = size !== undefined ? size : 100; @@ -970,14 +1232,16 @@ orion.textview.UndoStack = (function() { } this.model = model; var self = this; - this._modelListener = { + this._listener = { onChanging: function(e) { - self._onModelChanging(e); + self._onChanging(e); + }, + onDestroy: function(e) { + self._onDestroy(e); } }; - model.addListener(this._modelListener); - view._undoStack = this; - view.addEventListener("Destroy", this, this._onDestroy); + model.addEventListener("Changing", this._listener.onChanging); + view.addEventListener("Destroy", this._listener.onDestroy); } UndoStack.prototype = /** @lends orion.textview.UndoStack.prototype */ { /** @@ -1064,6 +1328,9 @@ orion.textview.UndoStack = (function() { * @see #startCompoundChange */ endCompoundChange: function() { + if (this.compoundChange) { + this.compoundChange.end(this.view); + } this.compoundChange = undefined; }, /** @@ -1144,9 +1411,10 @@ orion.textview.UndoStack = (function() { */ startCompoundChange: function() { this._commitUndo(); - var change = new CompoundChange(this.view.getSelection(), this.view.getCaretOffset()); + var change = new CompoundChange(); this.add(change); this.compoundChange = change; + this.compoundChange.start(this.view); }, _commitUndo: function () { if (this._undoStart !== undefined) { @@ -1159,12 +1427,11 @@ orion.textview.UndoStack = (function() { this._undoText = ""; } }, - _onDestroy: function() { - this.model.removeListener(this._modelListener); - this.view.removeEventListener("Destroy", this, this._onDestroy); - this.view._undoStack = null; + _onDestroy: function(evt) { + this.model.removeEventListener("Changing", this._listener.onChanging); + this.view.removeEventListener("Destroy", this._listener.onDestroy); }, - _onModelChanging: function(e) { + _onChanging: function(e) { var newText = e.text; var start = e.start; var removedCharCount = e.removedCharCount; @@ -1199,15 +1466,13 @@ orion.textview.UndoStack = (function() { this.add(new Change(start, newText, this.model.getText(start, start + removedCharCount))); } }; - return UndoStack; -}()); - -if (typeof window !== "undefined" && typeof window.define !== "undefined") { - define([], function() { - return orion.textview; - }); -} + + return { + UndoStack: UndoStack + }; +}, "orion/textview"); /******************************************************************************* + * @license * Copyright (c) 2010, 2011 IBM Corporation and others. * All rights reserved. This program and the accompanying materials are made * available under the terms of the Eclipse Public License v1.0 @@ -1219,40 +1484,32 @@ if (typeof window !== "undefined" && typeof window.define !== "undefined") { * Silenio Quarti (IBM Corporation) - initial API and implementation ******************************************************************************/ -/*global window define */ +/*global define window*/ -/** - * @namespace The global container for Orion APIs. - */ -var orion = orion || {}; -/** - * @namespace The container for textview APIs. - */ -orion.textview = orion.textview || {}; - -/** - * Constructs a new TextModel with the given text and default line delimiter. - * - * @param {String} [text=""] the text that the model will store - * @param {String} [lineDelimiter=platform delimiter] the line delimiter used when inserting new lines to the model. - * - * @name orion.textview.TextModel - * @class The TextModel is an interface that provides text for the view. Applications may - * implement the TextModel interface to provide a custom store for the view content. The - * view interacts with its text model in order to access and update the text that is being - * displayed and edited in the view. This is the default implementation. - *

- * See:
- * {@link orion.textview.TextView}
- * {@link orion.textview.TextView#setModel} - *

- */ -orion.textview.TextModel = (function() { +define(['orion/textview/eventTarget'], function(mEventTarget) { var isWindows = window.navigator.platform.indexOf("Win") !== -1; - /** @private */ + /** + * Constructs a new TextModel with the given text and default line delimiter. + * + * @param {String} [text=""] the text that the model will store + * @param {String} [lineDelimiter=platform delimiter] the line delimiter used when inserting new lines to the model. + * + * @name orion.textview.TextModel + * @class The TextModel is an interface that provides text for the view. Applications may + * implement the TextModel interface to provide a custom store for the view content. The + * view interacts with its text model in order to access and update the text that is being + * displayed and edited in the view. This is the default implementation. + *

+ * See:
+ * {@link orion.textview.TextView}
+ * {@link orion.textview.TextView#setModel} + *

+ * @borrows orion.textview.EventTarget#addEventListener as #addEventListener + * @borrows orion.textview.EventTarget#removeEventListener as #removeEventListener + * @borrows orion.textview.EventTarget#dispatchEvent as #dispatchEvent + */ function TextModel(text, lineDelimiter) { - this._listeners = []; this._lastLineIndex = -1; this._text = [""]; this._lineOffsets = [0]; @@ -1261,33 +1518,6 @@ orion.textview.TextModel = (function() { } TextModel.prototype = /** @lends orion.textview.TextModel.prototype */ { - /** - * Adds a listener to the model. - * - * @param {Object} listener the listener to add. - * @param {Function} [listener.onChanged] see {@link #onChanged}. - * @param {Function} [listener.onChanging] see {@link #onChanging}. - * - * @see removeListener - */ - addListener: function(listener) { - this._listeners.push(listener); - }, - /** - * Removes a listener from the model. - * - * @param {Object} listener the listener to remove - * - * @see #addListener - */ - removeListener: function(listener) { - for (var i = 0; i < this._listeners.length; i++) { - if (this._listeners[i] === listener) { - this._listeners.splice(i, 1); - return; - } - } - }, /** * Returns the number of characters in the model. * @@ -1512,12 +1742,7 @@ orion.textview.TextModel = (function() { * @param {orion.textview.ModelChangingEvent} modelChangingEvent the changing event */ onChanging: function(modelChangingEvent) { - for (var i = 0; i < this._listeners.length; i++) { - var l = this._listeners[i]; - if (l && l.onChanging) { - l.onChanging(modelChangingEvent); - } - } + return this.dispatchEvent(modelChangingEvent); }, /** * Notifies all listeners that the text has changed. @@ -1534,12 +1759,7 @@ orion.textview.TextModel = (function() { * @param {orion.textview.ModelChangedEvent} modelChangedEvent the changed event */ onChanged: function(modelChangedEvent) { - for (var i = 0; i < this._listeners.length; i++) { - var l = this._listeners[i]; - if (l && l.onChanged) { - l.onChanged(modelChangedEvent); - } - } + return this.dispatchEvent(modelChangedEvent); }, /** * Sets the line delimiter that is used by the view @@ -1616,6 +1836,7 @@ orion.textview.TextModel = (function() { } var modelChangingEvent = { + type: "Changing", text: text, start: eventStart, removedCharCount: removedCharCount, @@ -1679,6 +1900,7 @@ orion.textview.TextModel = (function() { if (this._text.length === 0) { this._text = [""]; } var modelChangedEvent = { + type: "Changed", start: eventStart, removedCharCount: removedCharCount, addedCharCount: addedCharCount, @@ -1688,16 +1910,631 @@ orion.textview.TextModel = (function() { this.onChanged(modelChangedEvent); } }; + mEventTarget.EventTarget.addMixin(TextModel.prototype); - return TextModel; -}()); + return {TextModel: TextModel}; +}, "orion/textview");/******************************************************************************* + * @license + * Copyright (c) 2010, 2011 IBM Corporation and others. + * All rights reserved. This program and the accompanying materials are made + * available under the terms of the Eclipse Public License v1.0 + * (http://www.eclipse.org/legal/epl-v10.html), and the Eclipse Distribution + * License v1.0 (http://www.eclipse.org/org/documents/edl-v10.html). + * + * Contributors: + * Felipe Heidrich (IBM Corporation) - initial API and implementation + * Silenio Quarti (IBM Corporation) - initial API and implementation + ******************************************************************************/ -if (typeof window !== "undefined" && typeof window.define !== "undefined") { - define([], function() { - return orion.textview; - }); -} +/*global define */ + +define(['orion/textview/eventTarget'], function(mEventTarget) { + /** + * @class This object represents a decoration attached to a range of text. Annotations are added to a + * AnnotationModel which is attached to a TextModel. + *

+ * See:
+ * {@link orion.textview.AnnotationModel}
+ * {@link orion.textview.Ruler}
+ *

+ * @name orion.textview.Annotation + * + * @property {String} type The annotation type (for example, orion.annotation.error). + * @property {Number} start The start offset of the annotation in the text model. + * @property {Number} end The end offset of the annotation in the text model. + * @property {String} html The HTML displayed for the annotation. + * @property {String} title The text description for the annotation. + * @property {orion.textview.Style} style The style information for the annotation used in the annotations ruler and tooltips. + * @property {orion.textview.Style} overviewStyle The style information for the annotation used in the overview ruler. + * @property {orion.textview.Style} rangeStyle The style information for the annotation used in the text view to decorate a range of text. + * @property {orion.textview.Style} lineStyle The style information for the annotation used in the text view to decorate a line of text. + */ + /** + * Constructs a new folding annotation. + * + * @param {orion.textview.ProjectionTextModel} projectionModel The projection text model. + * @param {String} type The annotation type. + * @param {Number} start The start offset of the annotation in the text model. + * @param {Number} end The end offset of the annotation in the text model. + * @param {String} expandedHTML The HTML displayed for this annotation when it is expanded. + * @param {orion.textview.Style} expandedStyle The style information for the annotation when it is expanded. + * @param {String} collapsedHTML The HTML displayed for this annotation when it is collapsed. + * @param {orion.textview.Style} collapsedStyle The style information for the annotation when it is collapsed. + * + * @class This object represents a folding annotation. + * @name orion.textview.FoldingAnnotation + */ + function FoldingAnnotation (projectionModel, type, start, end, expandedHTML, expandedStyle, collapsedHTML, collapsedStyle) { + this.type = type; + this.start = start; + this.end = end; + this._projectionModel = projectionModel; + this._expandedHTML = this.html = expandedHTML; + this._expandedStyle = this.style = expandedStyle; + this._collapsedHTML = collapsedHTML; + this._collapsedStyle = collapsedStyle; + this.expanded = true; + } + + FoldingAnnotation.prototype = /** @lends orion.textview.FoldingAnnotation.prototype */ { + /** + * Collapses the annotation. + */ + collapse: function () { + if (!this.expanded) { return; } + this.expanded = false; + this.html = this._collapsedHTML; + this.style = this._collapsedStyle; + var projectionModel = this._projectionModel; + var baseModel = projectionModel.getBaseModel(); + this._projection = { + start: baseModel.getLineStart(baseModel.getLineAtOffset(this.start) + 1), + end: baseModel.getLineEnd(baseModel.getLineAtOffset(this.end), true) + }; + projectionModel.addProjection(this._projection); + }, + /** + * Expands the annotation. + */ + expand: function () { + if (this.expanded) { return; } + this.expanded = true; + this.html = this._expandedHTML; + this.style = this._expandedStyle; + this._projectionModel.removeProjection(this._projection); + } + }; + + /** + * Constructs an annotation model. + * + * @param {textModel} textModel The text model. + * + * @class This object manages annotations for a TextModel. + *

+ * See:
+ * {@link orion.textview.Annotation}
+ * {@link orion.textview.TextModel}
+ *

+ * @name orion.textview.AnnotationModel + * @borrows orion.textview.EventTarget#addEventListener as #addEventListener + * @borrows orion.textview.EventTarget#removeEventListener as #removeEventListener + * @borrows orion.textview.EventTarget#dispatchEvent as #dispatchEvent + */ + function AnnotationModel(textModel) { + this._annotations = []; + var self = this; + this._listener = { + onChanged: function(modelChangedEvent) { + self._onChanged(modelChangedEvent); + } + }; + this.setTextModel(textModel); + } + + AnnotationModel.prototype = /** @lends orion.textview.AnnotationModel.prototype */ { + /** + * Adds an annotation to the annotation model. + *

The annotation model listeners are notified of this change.

+ * + * @param {orion.textview.Annotation} annotation the annotation to be added. + * + * @see #removeAnnotation + */ + addAnnotation: function(annotation) { + if (!annotation) { return; } + var annotations = this._annotations; + var index = this._binarySearch(annotations, annotation.start); + annotations.splice(index, 0, annotation); + var e = { + type: "Changed", + added: [annotation], + removed: [], + changed: [] + }; + this.onChanged(e); + }, + /** + * Returns the text model. + * + * @return {orion.textview.TextModel} The text model. + * + * @see #setTextModel + */ + getTextModel: function() { + return this._model; + }, + /** + * @class This object represents an annotation iterator. + *

+ * See:
+ * {@link orion.textview.AnnotationModel#getAnnotations}
+ *

+ * @name orion.textview.AnnotationIterator + * + * @property {Function} hasNext Determines whether there are more annotations in the iterator. + * @property {Function} next Returns the next annotation in the iterator. + */ + /** + * Returns an iterator of annotations for the given range of text. + * + * @param {Number} start the start offset of the range. + * @param {Number} end the end offset of the range. + * @return {orion.textview.AnnotationIterator} an annotation iterartor. + */ + getAnnotations: function(start, end) { + var annotations = this._annotations, current; + //TODO binary search does not work for range intersection when there are overlaping ranges, need interval search tree for this + var i = 0; + var skip = function() { + while (i < annotations.length) { + var a = annotations[i++]; + if ((start === a.start) || (start > a.start ? start < a.end : a.start < end)) { + return a; + } + if (a.start >= end) { + break; + } + } + return null; + }; + current = skip(); + return { + next: function() { + var result = current; + if (result) { current = skip(); } + return result; + }, + hasNext: function() { + return current !== null; + } + }; + }, + /** + * Notifies the annotation model that the given annotation has been modified. + *

The annotation model listeners are notified of this change.

+ * + * @param {orion.textview.Annotation} annotation the modified annotation. + * + * @see #addAnnotation + */ + modifyAnnotation: function(annotation) { + if (!annotation) { return; } + var index = this._getAnnotationIndex(annotation); + if (index < 0) { return; } + var e = { + type: "Changed", + added: [], + removed: [], + changed: [annotation] + }; + this.onChanged(e); + }, + /** + * Notifies all listeners that the annotation model has changed. + * + * @param {orion.textview.Annotation[]} added The list of annotation being added to the model. + * @param {orion.textview.Annotation[]} changed The list of annotation modified in the model. + * @param {orion.textview.Annotation[]} removed The list of annotation being removed from the model. + * @param {ModelChangedEvent} textModelChangedEvent the text model changed event that trigger this change, can be null if the change was trigger by a method call (for example, {@link #addAnnotation}). + */ + onChanged: function(e) { + return this.dispatchEvent(e); + }, + /** + * Removes all annotations of the given type. All annotations + * are removed if the type is not specified. + *

The annotation model listeners are notified of this change. Only one changed event is generated.

+ * + * @param {Object} type the type of annotations to be removed. + * + * @see #removeAnnotation + */ + removeAnnotations: function(type) { + var annotations = this._annotations; + var removed, i; + if (type) { + removed = []; + for (i = annotations.length - 1; i >= 0; i--) { + var annotation = annotations[i]; + if (annotation.type === type) { + annotations.splice(i, 1); + } + removed.splice(0, 0, annotation); + } + } else { + removed = annotations; + annotations = []; + } + var e = { + type: "Changed", + removed: removed, + added: [], + changed: [] + }; + this.onChanged(e); + }, + /** + * Removes an annotation from the annotation model. + *

The annotation model listeners are notified of this change.

+ * + * @param {orion.textview.Annotation} annotation the annotation to be removed. + * + * @see #addAnnotation + */ + removeAnnotation: function(annotation) { + if (!annotation) { return; } + var index = this._getAnnotationIndex(annotation); + if (index < 0) { return; } + var e = { + type: "Changed", + removed: this._annotations.splice(index, 1), + added: [], + changed: [] + }; + this.onChanged(e); + }, + /** + * Removes and adds the specifed annotations to the annotation model. + *

The annotation model listeners are notified of this change. Only one changed event is generated.

+ * + * @param {orion.textview.Annotation} remove the annotations to be removed. + * @param {orion.textview.Annotation} add the annotations to be added. + * + * @see #addAnnotation + * @see #removeAnnotation + */ + replaceAnnotations: function(remove, add) { + var annotations = this._annotations, i, index, annotation, removed = []; + if (remove) { + for (i = remove.length - 1; i >= 0; i--) { + annotation = remove[i]; + index = this._getAnnotationIndex(annotation); + if (index < 0) { continue; } + annotations.splice(index, 1); + removed.splice(0, 0, annotation); + } + } + if (!add) { add = []; } + for (i = 0; i < add.length; i++) { + annotation = add[i]; + index = this._binarySearch(annotations, annotation.start); + annotations.splice(index, 0, annotation); + } + var e = { + type: "Changed", + removed: removed, + added: add, + changed: [] + }; + this.onChanged(e); + }, + /** + * Sets the text model of the annotation model. The annotation + * model listens for changes in the text model to update and remove + * annotations that are affected by the change. + * + * @param {orion.textview.TextModel} textModel the text model. + * + * @see #getTextModel + */ + setTextModel: function(textModel) { + if (this._model) { + this._model.removeEventListener("Changed", this._listener.onChanged); + } + this._model = textModel; + if (this._model) { + this._model.addEventListener("Changed", this._listener.onChanged); + } + }, + /** @ignore */ + _binarySearch: function (array, offset) { + var high = array.length, low = -1, index; + while (high - low > 1) { + index = Math.floor((high + low) / 2); + if (offset <= array[index].start) { + high = index; + } else { + low = index; + } + } + return high; + }, + /** @ignore */ + _getAnnotationIndex: function(annotation) { + var annotations = this._annotations; + var index = this._binarySearch(annotations, annotation.start); + while (index < annotations.length && annotations[index].start === annotation.start) { + if (annotations[index] === annotation) { + return index; + } + index++; + } + return -1; + }, + /** @ignore */ + _onChanged: function(modelChangedEvent) { + var start = modelChangedEvent.start; + var addedCharCount = modelChangedEvent.addedCharCount; + var removedCharCount = modelChangedEvent.removedCharCount; + var annotations = this._annotations, end = start + removedCharCount; + //TODO binary search does not work for range intersection when there are overlaping ranges, need interval search tree for this + var startIndex = 0; + if (!(0 <= startIndex && startIndex < annotations.length)) { return; } + var e = { + type: "Changed", + added: [], + removed: [], + changed: [], + textModelChangedEvent: modelChangedEvent + }; + var changeCount = addedCharCount - removedCharCount, i; + for (i = startIndex; i < annotations.length; i++) { + var annotation = annotations[i]; + if (annotation.start >= end) { + annotation.start += changeCount; + annotation.end += changeCount; + e.changed.push(annotation); + } else if (annotation.end <= start) { + //nothing + } else if (annotation.start < start && end < annotation.end) { + annotation.end += changeCount; + e.changed.push(annotation); + } else { + annotations.splice(i, 1); + e.removed.push(annotation); + i--; + } + } + if (e.added.length > 0 || e.removed.length > 0 || e.changed.length > 0) { + this.onChanged(e); + } + } + }; + mEventTarget.EventTarget.addMixin(AnnotationModel.prototype); + + /** + * Constructs a new styler for annotations. + * + * @param {orion.textview.TextView} view The styler view. + * @param {orion.textview.AnnotationModel} view The styler annotation model. + * + * @class This object represents a styler for annotation attached to a text view. + * @name orion.textview.AnnotationStyler + */ + function AnnotationStyler (view, annotationModel) { + this._view = view; + this._annotationModel = annotationModel; + this._types = []; + var self = this; + this._listener = { + onDestroy: function(e) { + self._onDestroy(e); + }, + onLineStyle: function(e) { + self._onLineStyle(e); + }, + onChanged: function(e) { + self._onAnnotationModelChanged(e); + } + }; + view.addEventListener("Destroy", this._listener.onDestroy); + view.addEventListener("LineStyle", this._listener.onLineStyle); + annotationModel.addEventListener("Changed", this._listener.onChanged); + } + AnnotationStyler.prototype = /** @lends orion.textview.AnnotationStyler.prototype */ { + /** + * Adds an annotation type to the receiver. + *

+ * Only annotations of the specified types will be shown by + * this receiver. + *

+ * + * @param type {Object} the annotation type to be shown + * + * @see #removeAnnotationType + * @see #isAnnotationTypeVisible + */ + addAnnotationType: function(type) { + this._types.push(type); + }, + /** + * Destroys the styler. + *

+ * Removes all listeners added by this styler. + *

+ */ + destroy: function() { + var view = this._view; + if (view) { + view.removeEventListener("Destroy", this._listener.onDestroy); + view.removeEventListener("LineStyle", this._listener.onLineStyle); + this.view = null; + } + var annotationModel = this._annotationModel; + if (annotationModel) { + annotationModel.removeEventListener("Changed", this._listener.onChanged); + annotationModel = null; + } + }, + /** + * Returns whether the receiver shows annotations of the specified type. + * + * @param {Object} type the annotation type + * @returns {Boolean} whether the specified annotation type is shown + * + * @see #addAnnotationType + * @see #removeAnnotationType + */ + isAnnotationTypeVisible: function(type) { + for (var i = 0; i < this._types.length; i++) { + if (this._types[i] === type) { + return true; + } + } + return false; + }, + /** + * Removes an annotation type from the receiver. + * + * @param {Object} type the annotation type to be removed + * + * @see #addAnnotationType + * @see #isAnnotationTypeVisible + */ + removeAnnotationType: function(type) { + for (var i = 0; i < this._types.length; i++) { + if (this._types[i] === type) { + this._types.splice(i, 1); + break; + } + } + }, + _mergeStyle: function(result, style) { + if (style) { + if (!result) { result = {}; } + if (result.styleClass && style.styleClass && result.styleClass !== style.styleClass) { + result.styleClass += " " + style.styleClass; + } else { + result.styleClass = style.styleClass; + } + var prop; + if (style.style) { + if (!result.style) { result.style = {}; } + for (prop in style.style) { + if (!result.style[prop]) { + result.style[prop] = style.style[prop]; + } + } + } + if (style.attributes) { + if (!result.attributes) { result.attributes = {}; } + for (prop in style.attributes) { + if (!result.attributes[prop]) { + result.attributes[prop] = style.attributes[prop]; + } + } + } + } + return result; + }, + _mergeStyleRanges: function(ranges, styleRange) { + if (!ranges) { return; } + for (var i=0; i= range.end) { continue; } + var mergedStyle = this._mergeStyle({}, range.style); + mergedStyle = this._mergeStyle(mergedStyle, styleRange.style); + if (styleRange.start <= range.start && styleRange.end >= range.end) { + ranges[i] = {start: range.start, end: range.end, style: mergedStyle}; + } else if (styleRange.start > range.start && styleRange.end < range.end) { + ranges.splice(i, 1, + {start: range.start, end: styleRange.start, style: range.style}, + {start: styleRange.start, end: styleRange.end, style: mergedStyle}, + {start: styleRange.end, end: range.end, style: range.style}); + i += 2; + } else if (styleRange.start > range.start) { + ranges.splice(i, 1, + {start: range.start, end: styleRange.start, style: range.style}, + {start: styleRange.start, end: range.end, style: mergedStyle}); + i += 1; + } else if (styleRange.end < range.end) { + ranges.splice(i, 1, + {start: range.start, end: styleRange.end, style: mergedStyle}, + {start: styleRange.end, end: range.end, style: range.style}); + i += 1; + } + } + }, + _onAnnotationModelChanged: function(e) { + if (e.textModelChangedEvent) { + return; + } + var view = this._view; + if (!view) { return; } + var self = this; + var model = view.getModel(); + function redraw(changes) { + for (var i = 0; i < changes.length; i++) { + if (!self.isAnnotationTypeVisible(changes[i].type)) { continue; } + var start = changes[i].start; + var end = changes[i].end; + if (model.getBaseModel) { + start = model.mapOffset(start, true); + end = model.mapOffset(end, true); + } + if (start !== -1 && end !== -1) { + view.redrawRange(start, end); + } + } + } + redraw(e.added); + redraw(e.removed); + redraw(e.changed); + }, + _onDestroy: function(e) { + this.destroy(); + }, + _onLineStyle: function (e) { + var annotationModel = this._annotationModel; + var viewModel = this._view.getModel(); + var baseModel = annotationModel.getTextModel(); + var start = e.lineStart; + var end = e.lineStart + e.lineText.length; + if (baseModel !== viewModel) { + start = viewModel.mapOffset(start); + end = viewModel.mapOffset(end); + } + var annotations = annotationModel.getAnnotations(start, end); + while (annotations.hasNext()) { + var annotation = annotations.next(); + if (!this.isAnnotationTypeVisible(annotation.type)) { continue; } + if (annotation.rangeStyle) { + var annotationStart = annotation.start; + var annotationEnd = annotation.end; + if (baseModel !== viewModel) { + annotationStart = viewModel.mapOffset(annotationStart, true); + annotationEnd = viewModel.mapOffset(annotationEnd, true); + } + this._mergeStyleRanges(e.ranges, {start: annotationStart, end: annotationEnd, style: annotation.rangeStyle}); + } + if (annotation.lineStyle) { + e.style = this._mergeStyle({}, e.style); + e.style = this._mergeStyle(e.style, annotation.lineStyle); + } + } + } + }; + + return { + FoldingAnnotation: FoldingAnnotation, + AnnotationModel: AnnotationModel, + AnnotationStyler: AnnotationStyler + }; +}, "orion/textview"); /******************************************************************************* + * @license * Copyright (c) 2010, 2011 IBM Corporation and others. * All rights reserved. This program and the accompanying materials are made * available under the terms of the Eclipse Public License v1.0 @@ -1707,19 +2544,10 @@ if (typeof window !== "undefined" && typeof window.define !== "undefined") { * Contributors: IBM Corporation - initial API and implementation ******************************************************************************/ -/*global window define setTimeout clearTimeout setInterval clearInterval Node */ +/*global define setTimeout clearTimeout setInterval clearInterval Node */ -/** - * @namespace The global container for Orion APIs. - */ -var orion = orion || {}; -/** - * @namespace The container for textview APIs. - */ -orion.textview = orion.textview || {}; +define(['orion/textview/textView', 'orion/textview/textModel', 'orion/textview/projectionTextModel'], function(mTextView, mTextModel, mProjectionTextModel) { -/** @ignore */ -orion.textview.Tooltip = (function() { /** @private */ function Tooltip (view) { this._view = view; @@ -1816,14 +2644,15 @@ orion.textview.Tooltip = (function() { (contentsDiv = this._htmlParent).innerHTML = contents; } else if (contents instanceof Node) { (contentsDiv = this._htmlParent).appendChild(contents); - } else if (contents instanceof orion.textview.ProjectionTextModel) { + } else if (contents instanceof mProjectionTextModel.ProjectionTextModel) { if (!this._contentsView) { - this._emptyModel = new orion.textview.TextModel(""); + this._emptyModel = new mTextModel.TextModel(""); //TODO need hook into setup.js (or editor.js) to create a text view (and styler) - var newView = this._contentsView = new orion.textview.TextView({ + var newView = this._contentsView = new mTextView.TextView({ model: this._emptyModel, parent: this._viewParent, tabSize: 4, + sync: true, stylesheet: ["/orion/textview/tooltip.css", "/orion/textview/rulers.css", "/examples/textview/textstyler.css", "/css/default-theme.css"] }); @@ -1831,7 +2660,12 @@ orion.textview.Tooltip = (function() { newView._clientDiv.contentEditable = false; //TODO need to find a better way of sharing the styler for multiple views var view = this._view; - newView.addEventListener("LineStyle", view, view.onLineStyle); + var listener = { + onLineStyle: function(e) { + view.onLineStyle(e); + } + }; + newView.addEventListener("LineStyle", listener.onLineStyle); } var contentsView = this._contentsView; contentsView.setModel(contents); @@ -1893,16 +2727,16 @@ orion.textview.Tooltip = (function() { annotation = annotations[0]; if (annotation.title) { title = annotation.title.replace(//g, ">"); - return annotation.html + " " + title; + return "
" + annotation.html + " " + title + "
"; } else { - var newModel = new orion.textview.ProjectionTextModel(baseModel); + var newModel = new mProjectionTextModel.ProjectionTextModel(baseModel); var lineStart = baseModel.getLineStart(baseModel.getLineAtOffset(annotation.start)); newModel.addProjection({start: annotation.end, end: newModel.getCharCount()}); newModel.addProjection({start: 0, end: lineStart}); return newModel; } } else { - var tooltipHTML = "Multiple annotations:
"; + var tooltipHTML = "
Multiple annotations:
"; for (var i = 0; i < annotations.length; i++) { annotation = annotations[i]; title = annotation.title; @@ -1910,7 +2744,7 @@ orion.textview.Tooltip = (function() { title = getText(annotation.start, annotation.end); } title = title.replace(//g, ">"); - tooltipHTML += annotation.html + " " + title + "
"; + tooltipHTML += "
" + annotation.html + " " + title + "
"; } return tooltipHTML; } @@ -1935,14 +2769,10 @@ orion.textview.Tooltip = (function() { return value || defaultValue; } }; - return Tooltip; -}()); - -if (typeof window !== "undefined" && typeof window.define !== "undefined") { - define([], function() { - return orion.textview; - }); -}/******************************************************************************* + return {Tooltip: Tooltip}; +}, "orion/textview"); +/******************************************************************************* + * @license * Copyright (c) 2010, 2011 IBM Corporation and others. * All rights reserved. This program and the accompanying materials are made * available under the terms of the Eclipse Public License v1.0 @@ -1952,35 +2782,12 @@ if (typeof window !== "undefined" && typeof window.define !== "undefined") { * Contributors: * Felipe Heidrich (IBM Corporation) - initial API and implementation * Silenio Quarti (IBM Corporation) - initial API and implementation - * Mihai Sucan (Mozilla Foundation) - fix for Bug#334583 Bug#348471 Bug#349485 Bug#350595 Bug#360726 Bug#361180 Bug#358623 Bug#362286 Bug#362107 Bug#362428 Bug#362835 Bug#363508 + * Mihai Sucan (Mozilla Foundation) - fix for Bug#334583 Bug#348471 Bug#349485 Bug#350595 Bug#360726 Bug#361180 Bug#362835 Bug#362428 Bug#362286 Bug#354270 Bug#361474 Bug#363945 Bug#366312 ******************************************************************************/ /*global window document navigator setTimeout clearTimeout XMLHttpRequest define */ -/** - * @namespace The global container for Orion APIs. - */ -var orion = orion || {}; -/** - * @namespace The container for textview APIs. - */ -orion.textview = orion.textview || {}; - -/** - * Constructs a new text view. - * - * @param options the view options. - * @param {String|DOMElement} options.parent the parent element for the view, it can be either a DOM element or an ID for a DOM element. - * @param {orion.textview.TextModel} [options.model] the text model for the view. If this options is not set the view creates an empty {@link orion.textview.TextModel}. - * @param {Boolean} [options.readonly=false] whether or not the view is read-only. - * @param {Boolean} [options.fullSelection=true] whether or not the view is in full selection mode. - * @param {String|String[]} [options.stylesheet] one or more stylesheet URIs for the view. - * @param {Number} [options.tabSize] The number of spaces in a tab. - * - * @class A TextView is a user interface for editing text. - * @name orion.textview.TextView - */ -orion.textview.TextView = (function() { +define(['orion/textview/textModel', 'orion/textview/keyBinding', 'orion/textview/eventTarget'], function(mTextModel, mKeyBinding, mEventTarget) { /** @private */ function addHandler(node, type, handler, capture) { @@ -1998,19 +2805,22 @@ orion.textview.TextView = (function() { node.detachEvent("on" + type, handler); } } - var isIE = document.selection && window.ActiveXObject && /MSIE/.test(navigator.userAgent) ? document.documentMode : undefined; - var isFirefox = parseFloat(navigator.userAgent.split("Firefox/")[1] || navigator.userAgent.split("Minefield/")[1]) || undefined; - var isOpera = navigator.userAgent.indexOf("Opera") !== -1; - var isChrome = navigator.userAgent.indexOf("Chrome") !== -1; - var isSafari = navigator.userAgent.indexOf("Safari") !== -1; - var isWebkit = navigator.userAgent.indexOf("WebKit") !== -1; - var isPad = navigator.userAgent.indexOf("iPad") !== -1; + var userAgent = navigator.userAgent; + var isIE; + if (document.selection && window.ActiveXObject && /MSIE/.test(userAgent)) { + isIE = document.documentMode ? document.documentMode : 7; + } + var isFirefox = parseFloat(userAgent.split("Firefox/")[1] || userAgent.split("Minefield/")[1]) || undefined; + var isOpera = userAgent.indexOf("Opera") !== -1; + var isChrome = userAgent.indexOf("Chrome") !== -1; + var isSafari = userAgent.indexOf("Safari") !== -1 && !isChrome; + var isWebkit = userAgent.indexOf("WebKit") !== -1; + var isPad = userAgent.indexOf("iPad") !== -1; var isMac = navigator.platform.indexOf("Mac") !== -1; var isWindows = navigator.platform.indexOf("Win") !== -1; var isLinux = navigator.platform.indexOf("Linux") !== -1; var isW3CEvents = typeof window.document.documentElement.addEventListener === "function"; var isRangeRects = (!isIE || isIE >= 9) && typeof window.document.createRange().getBoundingClientRect === "function"; - var isDnD = isFirefox || isWebkit; // drag and drop support var platformDelimiter = isWindows ? "\r\n" : "\n"; /** @@ -2019,185 +2829,108 @@ orion.textview.TextView = (function() { * @class A Selection represents a range of selected text in the view. * @name orion.textview.Selection */ - var Selection = (function() { + function Selection (start, end, caret) { + /** + * The selection start offset. + * + * @name orion.textview.Selection#start + */ + this.start = start; + /** + * The selection end offset. + * + * @name orion.textview.Selection#end + */ + this.end = end; /** @private */ - function Selection (start, end, caret) { - /** - * The selection start offset. - * - * @name orion.textview.Selection#start - */ - this.start = start; - /** - * The selection end offset. - * - * @name orion.textview.Selection#end - */ - this.end = end; - /** @private */ - this.caret = caret; //true if the start, false if the caret is at end - } - Selection.prototype = /** @lends orion.textview.Selection.prototype */ { - /** @private */ - clone: function() { - return new Selection(this.start, this.end, this.caret); - }, - /** @private */ - collapse: function() { - if (this.caret) { - this.end = this.start; - } else { - this.start = this.end; - } - }, - /** @private */ - extend: function (offset) { - if (this.caret) { - this.start = offset; - } else { - this.end = offset; - } - if (this.start > this.end) { - var tmp = this.start; - this.start = this.end; - this.end = tmp; - this.caret = !this.caret; - } - }, - /** @private */ - setCaret: function(offset) { + this.caret = caret; //true if the start, false if the caret is at end + } + Selection.prototype = /** @lends orion.textview.Selection.prototype */ { + /** @private */ + clone: function() { + return new Selection(this.start, this.end, this.caret); + }, + /** @private */ + collapse: function() { + if (this.caret) { + this.end = this.start; + } else { + this.start = this.end; + } + }, + /** @private */ + extend: function (offset) { + if (this.caret) { this.start = offset; + } else { this.end = offset; - this.caret = false; - }, - /** @private */ - getCaret: function() { - return this.caret ? this.start : this.end; - }, - /** @private */ - toString: function() { - return "start=" + this.start + " end=" + this.end + (this.caret ? " caret is at start" : " caret is at end"); - }, - /** @private */ - isEmpty: function() { - return this.start === this.end; - }, - /** @private */ - equals: function(object) { - return this.caret === object.caret && this.start === object.start && this.end === object.end; } - }; - return Selection; - }()); - - /** - * Constructs a new EventTable object. - * - * @class - * @name orion.textview.EventTable - * @private - */ - var EventTable = (function() { + if (this.start > this.end) { + var tmp = this.start; + this.start = this.end; + this.end = tmp; + this.caret = !this.caret; + } + }, /** @private */ - function EventTable(){ - this._types = {}; + setCaret: function(offset) { + this.start = offset; + this.end = offset; + this.caret = false; + }, + /** @private */ + getCaret: function() { + return this.caret ? this.start : this.end; + }, + /** @private */ + toString: function() { + return "start=" + this.start + " end=" + this.end + (this.caret ? " caret is at start" : " caret is at end"); + }, + /** @private */ + isEmpty: function() { + return this.start === this.end; + }, + /** @private */ + equals: function(object) { + return this.caret === object.caret && this.start === object.start && this.end === object.end; } - EventTable.prototype = /** @lends EventTable.prototype */ { - /** @private */ - addEventListener: function(type, context, func, data) { - var state = this._types[type]; - if (!state) { - state = this._types[type] = {level: 0, listeners: []}; - } - var listener = { - context: context, - func: func, - data: data - }; - var listeners = state.listeners; - listeners.push(listener); - }, - /** @private */ - sendEvent: function(type, event) { - var state = this._types[type]; - if (state) { - var listeners = state.listeners; - try { - state.level++; - if (listeners) { - for (var i=0, len=listeners.length; i < len; i++) { - var l = listeners[i]; - if (l && l.context && l.func) { - l.func.call(l.context, event, l.data); - } - } - } - } finally { - state.level--; - if (state.compact && state.level === 0) { - for (var j=listeners.length - 1; j >= 0; j--) { - if (!listeners[j]) { - listeners.splice(j, 1); - } - } - state.compact = false; - } - } - } - }, - /** @private */ - removeEventListener: function(type, context, func, data){ - var state = this._types[type]; - if (state) { - var listeners = state.listeners; - for (var i=0, len=listeners.length; i < len; i++) { - var l = listeners[i]; - if (l && l.context === context && l.func === func && l.data === data) { - if (state.level !== 0) { - listeners[i] = null; - state.compact = true; - } else { - listeners.splice(i, 1); - } - break; - } - } - } - } - }; - return EventTable; - }()); - - /** @private */ + }; + /** + * @class This object describes the options for the text view. + *

+ * See:
+ * {@link orion.textview.TextView}
+ * {@link orion.textview.TextView#setOptions} + * {@link orion.textview.TextView#getOptions} + *

+ * @name orion.textview.TextViewOptions + * + * @property {String|DOMElement} parent the parent element for the view, it can be either a DOM element or an ID for a DOM element. + * @property {orion.textview.TextModel} [model] the text model for the view. If it is not set the view creates an empty {@link orion.textview.TextModel}. + * @property {Boolean} [readonly=false] whether or not the view is read-only. + * @property {Boolean} [fullSelection=true] whether or not the view is in full selection mode. + * @property {Boolean} [sync=false] whether or not the view creation should be synchronous (if possible). + * @property {Boolean} [expandTab=false] whether or not the tab key inserts white spaces + * @property {String|String[]} [stylesheet] one or more stylesheet URIs for the view. + * @property {String} [themeClass] the CSS class for the view theming. + * @property {Number} [tabSize] The number of spaces in a tab. + */ + /** + * Constructs a new text view. + * + * @param {orion.textview.TextViewOptions} options the view options. + * + * @class A TextView is a user interface for editing text. + * @name orion.textview.TextView + * @borrows orion.textview.EventTarget#addEventListener as #addEventListener + * @borrows orion.textview.EventTarget#removeEventListener as #removeEventListener + * @borrows orion.textview.EventTarget#dispatchEvent as #dispatchEvent + */ function TextView (options) { this._init(options); } TextView.prototype = /** @lends orion.textview.TextView.prototype */ { - /** - * Adds an event listener to the text view. - * - * @param {String} type the event type. The supported events are: - *
    - *
  • "Modify" See {@link #onModify}
  • - *
  • "Selection" See {@link #onSelection}
  • - *
  • "Scroll" See {@link #onScroll}
  • - *
  • "Verify" See {@link #onVerify}
  • - *
  • "Destroy" See {@link #onDestroy}
  • - *
  • "LineStyle" See {@link #onLineStyle}
  • - *
  • "ModelChanging" See {@link #onModelChanging}
  • - *
  • "ModelChanged" See {@link #onModelChanged}
  • - *
- * @param {Object} context the context of the function. - * @param {Function} func the function that will be executed when the event happens. - * The function should take an event as the first parameter and optional data as the second parameter. - * @param {Object} [data] optional data passed to the function. - * - * @see #removeEventListener - */ - addEventListener: function(type, context, func, data) { - this._eventTable.addEventListener(type, context, func, data); - }, /** * Adds a ruler to the text view. * @@ -2212,6 +2945,7 @@ orion.textview.TextView = (function() { computeSize: function() { var w = 0, h = 0; var model = this._model, clientDiv = this._clientDiv; + if (!clientDiv) { return {width: w, height: h}; } var clientWidth = clientDiv.style.width; /* * Feature in WekKit. Webkit limits the width of the lines @@ -2228,7 +2962,7 @@ orion.textview.TextView = (function() { var document = this._frameDocument; for (var lineIndex=0; lineIndextrue if the text view has focus, otherwise false. + */ + hasFocus: function() { + return this._hasFocus; + }, /** * Returns all action names defined in the text view. *

@@ -2466,6 +3206,7 @@ orion.textview.TextView = (function() { * @see #setTopIndex */ getBottomIndex: function(fullyVisible) { + if (!this._clientDiv) { return 0; } return this._getBottomIndex(fullyVisible); }, /** @@ -2483,6 +3224,7 @@ orion.textview.TextView = (function() { * @see #convert */ getBottomPixel: function() { + if (!this._clientDiv) { return 0; } return this._getScroll().y + this._getClientHeight(); }, /** @@ -2513,6 +3255,7 @@ orion.textview.TextView = (function() { * @see #convert */ getClientArea: function() { + if (!this._clientDiv) { return {x: 0, y: 0, width: 0, height: 0}; } var scroll = this._getScroll(); return {x: scroll.x, y: scroll.y, width: this._getClientWidth(), height: this._getClientHeight()}; }, @@ -2530,6 +3273,7 @@ orion.textview.TextView = (function() { * @see #convert */ getHorizontalPixel: function() { + if (!this._clientDiv) { return 0; } return this._getScroll().x; }, /** @@ -2561,6 +3305,7 @@ orion.textview.TextView = (function() { * @see #getLinePixel */ getLineHeight: function(lineIndex) { + if (!this._clientDiv) { return 0; } return this._getLineHeight(); }, /** @@ -2577,6 +3322,7 @@ orion.textview.TextView = (function() { * @see #convert */ getLinePixel: function(lineIndex) { + if (!this._clientDiv) { return 0; } lineIndex = Math.min(Math.max(0, lineIndex), this._model.getLineCount()); var lineHeight = this._getLineHeight(); return lineHeight * lineIndex; @@ -2596,6 +3342,7 @@ orion.textview.TextView = (function() { * @see #convert */ getLocationAtOffset: function(offset) { + if (!this._clientDiv) { return {x: 0, y: 0}; } var model = this._model; offset = Math.min(Math.max(0, offset), model.getCharCount()); var lineIndex = model.getLineAtOffset(offset); @@ -2606,6 +3353,44 @@ orion.textview.TextView = (function() { var y = this.getLinePixel(lineIndex); return {x: x, y: y}; }, + /** + * Returns the specified view options. + *

+ * The returned value is either a orion.textview.TextViewOptions or an option value. An option value is returned when only one string paremeter + * is specified. A orion.textview.TextViewOptions is returned when there are no paremeters, or the parameters are a list of options names or a + * orion.textview.TextViewOptions. All view options are returned when there no paremeters. + *

+ * + * @param {String|orion.textview.TextViewOptions} [options] The options to return. + * @return {Object|orion.textview.TextViewOptions} The requested options or an option value. + * + * @see #setOptions + */ + getOptions: function() { + var options; + if (arguments.length === 0) { + options = this._defaultOptions(); + } else if (arguments.length === 1) { + var arg = arguments[0]; + if (typeof arg === "string") { + return this._clone(this["_" + arg]); + } + options = arg; + } else { + options = {}; + for (var index in arguments) { + if (arguments.hasOwnProperty(index)) { + options[arguments[index]] = undefined; + } + } + } + for (var option in options) { + if (options.hasOwnProperty(option)) { + options[option] = this._clone(this["_" + option]); + } + } + return options; + }, /** * Returns the text model of the text view. * @@ -2625,6 +3410,7 @@ orion.textview.TextView = (function() { * @see #getLocationAtOffset */ getOffsetAtLocation: function(x, y) { + if (!this._clientDiv) { return 0; } var scroll = this._getScroll(); var viewRect = this._viewDiv.getBoundingClientRect(); var viewPad = this._getViewPadding(); @@ -2689,6 +3475,7 @@ orion.textview.TextView = (function() { * @see #setTopIndex */ getTopIndex: function(fullyVisible) { + if (!this._clientDiv) { return 0; } return this._getTopIndex(fullyVisible); }, /** @@ -2706,6 +3493,7 @@ orion.textview.TextView = (function() { * @see #convert */ getTopPixel: function() { + if (!this._clientDiv) { return 0; } return this._getScroll().y; }, /** @@ -2727,6 +3515,7 @@ orion.textview.TextView = (function() { * @see #getActions */ invokeAction: function (name, defaultAction) { + if (!this._clientDiv) { return; } var actions = this._actions; for (var i = 0; i < actions.length; i++) { var a = actions[i]; @@ -2761,9 +3550,30 @@ orion.textview.TextView = (function() { * @event * @param {orion.textview.ContextMenuEvent} contextMenuEvent the event */ - onContextMenu: function(contextMenuEvent) { - this._eventTable.sendEvent("ContextMenu", contextMenuEvent); + onContextMenu: function(contextMenuEvent) { + return this.dispatchEvent(contextMenuEvent); }, + onDragStart: function(dragEvent) { + return this.dispatchEvent(dragEvent); + }, + onDrag: function(dragEvent) { + return this.dispatchEvent(dragEvent); + }, + onDragEnd: function(dragEvent) { + return this.dispatchEvent(dragEvent); + }, + onDragEnter: function(dragEvent) { + return this.dispatchEvent(dragEvent); + }, + onDragOver: function(dragEvent) { + return this.dispatchEvent(dragEvent); + }, + onDragLeave: function(dragEvent) { + return this.dispatchEvent(dragEvent); + }, + onDrop: function(dragEvent) { + return this.dispatchEvent(dragEvent); + }, /** * @class This is the event sent when the text view is destroyed. *

@@ -2782,7 +3592,7 @@ orion.textview.TextView = (function() { * @see #destroy */ onDestroy: function(destroyEvent) { - this._eventTable.sendEvent("Destroy", destroyEvent); + return this.dispatchEvent(destroyEvent); }, /** * @class This object is used to define style information for the text view. @@ -2834,7 +3644,25 @@ orion.textview.TextView = (function() { * @param {orion.textview.LineStyleEvent} lineStyleEvent the event */ onLineStyle: function(lineStyleEvent) { - this._eventTable.sendEvent("LineStyle", lineStyleEvent); + return this.dispatchEvent(lineStyleEvent); + }, + /** + * @class This is the event sent when the text view has loaded its contents. + *

+ * See:
+ * {@link orion.textview.TextView}
+ * {@link orion.textview.TextView#event:onLoad} + *

+ * @name orion.textview.LoadEvent + */ + /** + * This event is sent when the text view has loaded its contents. + * + * @event + * @param {orion.textview.LoadEvent} loadEvent the event + */ + onLoad: function(loadEvent) { + return this.dispatchEvent(loadEvent); }, /** * @class This is the event sent when the text in the model has changed. @@ -2859,7 +3687,7 @@ orion.textview.TextView = (function() { * @param {orion.textview.ModelChangedEvent} modelChangedEvent the event */ onModelChanged: function(modelChangedEvent) { - this._eventTable.sendEvent("ModelChanged", modelChangedEvent); + return this.dispatchEvent(modelChangedEvent); }, /** * @class This is the event sent when the text in the model is about to change. @@ -2885,7 +3713,7 @@ orion.textview.TextView = (function() { * @param {orion.textview.ModelChangingEvent} modelChangingEvent the event */ onModelChanging: function(modelChangingEvent) { - this._eventTable.sendEvent("ModelChanging", modelChangingEvent); + return this.dispatchEvent(modelChangingEvent); }, /** * @class This is the event sent when the text is modified by the text view. @@ -2907,7 +3735,22 @@ orion.textview.TextView = (function() { * @param {orion.textview.ModifyEvent} modifyEvent the event */ onModify: function(modifyEvent) { - this._eventTable.sendEvent("Modify", modifyEvent); + return this.dispatchEvent(modifyEvent); + }, + onMouseDown: function(mouseEvent) { + return this.dispatchEvent(mouseEvent); + }, + onMouseUp: function(mouseEvent) { + return this.dispatchEvent(mouseEvent); + }, + onMouseMove: function(mouseEvent) { + return this.dispatchEvent(mouseEvent); + }, + onMouseOver: function(mouseEvent) { + return this.dispatchEvent(mouseEvent); + }, + onMouseOut: function(mouseEvent) { + return this.dispatchEvent(mouseEvent); }, /** * @class This is the event sent when the selection changes in the text view. @@ -2928,7 +3771,7 @@ orion.textview.TextView = (function() { * @param {orion.textview.SelectionEvent} selectionEvent the event */ onSelection: function(selectionEvent) { - this._eventTable.sendEvent("Selection", selectionEvent); + return this.dispatchEvent(selectionEvent); }, /** * @class This is the event sent when the text view scrolls. @@ -2949,7 +3792,7 @@ orion.textview.TextView = (function() { * @param {orion.textview.ScrollEvent} scrollEvent the event */ onScroll: function(scrollEvent) { - this._eventTable.sendEvent("Scroll", scrollEvent); + return this.dispatchEvent(scrollEvent); }, /** * @class This is the event sent when the text is about to be modified by the text view. @@ -2979,7 +3822,77 @@ orion.textview.TextView = (function() { * @param {orion.textview.VerifyEvent} verifyEvent the event */ onVerify: function(verifyEvent) { - this._eventTable.sendEvent("Verify", verifyEvent); + return this.dispatchEvent(verifyEvent); + }, + /** + * @class This is the event sent when the text view has unloaded its contents. + *

+ * See:
+ * {@link orion.textview.TextView}
+ * {@link orion.textview.TextView#event:onLoad} + *

+ * @name orion.textview.UnloadEvent + */ + /** + * This event is sent when the text view has unloaded its contents. + * + * @event + * @param {orion.textview.UnloadEvent} unloadEvent the event + */ + onUnload: function(unloadEvent) { + return this.dispatchEvent(unloadEvent); + }, + /** + * @class This is the event sent when the text view is focused. + *

+ * See:
+ * {@link orion.textview.TextView}
+ * {@link orion.textview.TextView#event:onFocus}
+ *

+ * @name orion.textview.FocusEvent + */ + /** + * This event is sent when the text view is focused. + * + * @event + * @param {orion.textview.FocusEvent} focusEvent the event + */ + onFocus: function(focusEvent) { + return this.dispatchEvent(focusEvent); + }, + /** + * @class This is the event sent when the text view goes out of focus. + *

+ * See:
+ * {@link orion.textview.TextView}
+ * {@link orion.textview.TextView#event:onBlur}
+ *

+ * @name orion.textview.BlurEvent + */ + /** + * This event is sent when the text view goes out of focus. + * + * @event + * @param {orion.textview.BlurEvent} blurEvent the event + */ + onBlur: function(blurEvent) { + return this.dispatchEvent(blurEvent); + }, + /** + * Redraws the entire view, including rulers. + * + * @see #redrawLines + * @see #redrawRange + * @see #setRedraw + */ + redraw: function() { + if (this._redrawCount > 0) { return; } + var lineCount = this._model.getLineCount(); + var rulers = this.getRulers(); + for (var i = 0; i < rulers.length; i++) { + this.redrawLines(0, lineCount, rulers[i]); + } + this.redrawLines(0, lineCount); }, /** * Redraws the text in the given line range. @@ -2989,6 +3902,10 @@ orion.textview.TextView = (function() { * * @param {Number} [startLine=0] the start line * @param {Number} [endLine=line count] the end line + * + * @see #redraw + * @see #redrawRange + * @see #setRedraw */ redrawLines: function(startLine, endLine, ruler) { if (this._redrawCount > 0) { return; } @@ -3038,32 +3955,20 @@ orion.textview.TextView = (function() { * * @param {Number} [start=0] the start offset of text range * @param {Number} [end=char count] the end offset of text range + * + * @see #redraw + * @see #redrawLines + * @see #setRedraw */ redrawRange: function(start, end) { + if (this._redrawCount > 0) { return; } var model = this._model; if (start === undefined) { start = 0; } if (end === undefined) { end = model.getCharCount(); } - if (start === end) { return; } var startLine = model.getLineAtOffset(start); - var endLine = model.getLineAtOffset(Math.max(0, end - 1)) + 1; + var endLine = model.getLineAtOffset(Math.max(start, end - 1)) + 1; this.redrawLines(startLine, endLine); }, - /** - * Removes an event listener from the text view. - *

- * All the parameters must be the same ones used to add the listener. - *

- * - * @param {String} type the event type. - * @param {Object} context the context of the function. - * @param {Function} func the function that will be executed when the event happens. - * @param {Object} [data] optional data passed to the function. - * - * @see #addEventListener - */ - removeEventListener: function(type, context, func, data) { - this._eventTable.removeEventListener(type, context, func, data); - }, /** * Removes a ruler from the text view. * @@ -3187,19 +4092,28 @@ orion.textview.TextView = (function() { * @see #convert */ setHorizontalPixel: function(pixel) { + if (!this._clientDiv) { return; } pixel = Math.max(0, pixel); this._scrollView(pixel - this._getScroll().x, 0); }, + /** + * Sets whether the view should update the DOM. + *

+ * This can be used to improve the performance. + *

+ * When the flag is set to true, + * the entire view is marked as needing to be redrawn. + * Nested calls to this method are stacked. + *

+ * + * @param {Boolean} redraw the new redraw state + * + * @see #redraw + */ setRedraw: function(redraw) { if (redraw) { if (--this._redrawCount === 0) { - var lineCount = this._model.getLineCount(); - var rulers = this.getRulers(); - for (var i = 0; i < rulers.length; i++) { - this.redrawLines(0, lineCount, rulers[i]); - } - this.redrawLines(0, lineCount); - this._queueUpdatePage(); + this.redraw(); } } else { this._redrawCount++; @@ -3213,13 +4127,15 @@ orion.textview.TextView = (function() { setModel: function(model) { if (!model) { return; } if (model === this._model) { return; } - this._model.removeListener(this._modelListener); + this._model.removeEventListener("Changing", this._modelListener.onChanging); + this._model.removeEventListener("Changed", this._modelListener.onChanged); var oldLineCount = this._model.getLineCount(); var oldCharCount = this._model.getCharCount(); var newLineCount = model.getLineCount(); var newCharCount = model.getCharCount(); var newText = model.getText(); var e = { + type: "ModelChanging", text: newText, start: 0, removedCharCount: oldCharCount, @@ -3230,6 +4146,7 @@ orion.textview.TextView = (function() { this.onModelChanging(e); this._model = model; e = { + type: "ModelChanged", start: 0, removedCharCount: oldCharCount, addedCharCount: newCharCount, @@ -3237,10 +4154,55 @@ orion.textview.TextView = (function() { addedLineCount: newLineCount }; this.onModelChanged(e); - this._model.addListener(this._modelListener); + this._model.addEventListener("Changing", this._modelListener.onChanging); + this._model.addEventListener("Changed", this._modelListener.onChanged); this._reset(); this._updatePage(); }, + /** + * Sets the view options for the view. + * + * @param {orion.textview.TextViewOptions} options the view options. + * + * @see #getOptions + */ + setOptions: function (options) { + var defaultOptions = this._defaultOptions(); + var recreate = false, option, created = this._clientDiv; + if (created) { + for (option in options) { + if (options.hasOwnProperty(option)) { + if (defaultOptions[option].recreate) { + recreate = true; + break; + } + } + } + } + var changed = false; + for (option in options) { + if (options.hasOwnProperty(option)) { + var newValue = options[option], oldValue = this["_" + option]; + if (this._compare(oldValue, newValue)) { continue; } + changed = true; + if (!recreate) { + var update = defaultOptions[option].update; + if (created && update) { + update.call(this, newValue); + continue; + } + } + this["_" + option] = this._clone(newValue); + } + } + if (changed) { + if (recreate) { + var oldParent = this._frame.parentNode; + oldParent.removeChild(this._frame); + this._parent.appendChild(this._frame); + } + } + }, /** * Sets the text view selection. *

@@ -3307,14 +4269,7 @@ orion.textview.TextView = (function() { * force the clientDiv to loose and receive focus if the it is focused. */ if (isFirefox) { - var clientDiv = this._clientDiv; - if (clientDiv) { - var hasFocus = this._hasFocus; - if (hasFocus) { clientDiv.blur(); } - clientDiv.contentEditable = false; - clientDiv.contentEditable = true; - if (hasFocus) { clientDiv.focus(); } - } + this._fixCaret(); } } }, @@ -3331,6 +4286,7 @@ orion.textview.TextView = (function() { * @see #getTopIndex */ setTopIndex: function(topIndex) { + if (!this._clientDiv) { return; } var model = this._model; if (model.getCharCount() === 0) { return; @@ -3361,6 +4317,7 @@ orion.textview.TextView = (function() { * @see #convert */ setTopPixel: function(pixel) { + if (!this._clientDiv) { return; } var lineHeight = this._getLineHeight(); var clientHeight = this._getClientHeight(); var lineCount = this._model.getLineCount(); @@ -3382,6 +4339,12 @@ orion.textview.TextView = (function() { /**************************************** Event handlers *********************************/ _handleBodyMouseDown: function (e) { if (!e) { e = window.event; } + if (isFirefox) { + this._clientDiv.contentEditable = false; + (this._overlayDiv || this._clientDiv).draggable = true; + this._ignoreBlur = true; + } + /* * Prevent clicks outside of the view from taking focus * away the view. Note that in Firefox and Opera clicking on the @@ -3389,7 +4352,7 @@ orion.textview.TextView = (function() { * do not have this problem and stopping the click over the * scrollbar for them causes mouse capture problems. */ - var topNode = isOpera ? this._clientDiv : this._overlayDiv || this._viewDiv; + var topNode = isOpera || (isFirefox && !this._overlayDiv) ? this._clientDiv : this._overlayDiv || this._viewDiv; var temp = e.target ? e.target : e.srcElement; while (temp) { @@ -3409,8 +4372,26 @@ orion.textview.TextView = (function() { setTimeout(function() { topNode.releaseCapture(); }, 0); } }, + _handleBodyMouseUp: function (e) { + if (!e) { e = window.event; } + if (isFirefox) { + this._clientDiv.contentEditable = true; + (this._overlayDiv || this._clientDiv).draggable = false; + + /* + * Bug in Firefox. For some reason, Firefox stops showing the caret + * in some cases. For example when the user cancels a drag operation + * by pressing ESC. The fix is to detect that the drag operation was + * cancelled, toggle the contentEditable state and force the clientDiv + * to loose and receive focus if the it is focused. + */ + this._fixCaret(); + this._ignoreBlur = false; + } + }, _handleBlur: function (e) { if (!e) { e = window.event; } + if (this._ignoreBlur) { return; } this._hasFocus = false; /* * Bug in IE 8 and earlier. For some reason when text is deselected @@ -3434,15 +4415,27 @@ orion.textview.TextView = (function() { this._selDiv3.style.background = color; } } + if (!this._ignoreFocus) { + this.onBlur({type: "Blur"}); + } }, _handleContextMenu: function (e) { if (!e) { e = window.event; } - var scroll = this._getScroll(); - var viewRect = this._viewDiv.getBoundingClientRect(); - var viewPad = this._getViewPadding(); - var x = e.clientX + scroll.x - viewRect.left - viewPad.left; - var y = e.clientY + scroll.y - viewRect.top - viewPad.top; - this.onContextMenu({x: x, y: y, screenX: e.screenX, screenY: e.screenY}); + if (isFirefox && this._lastMouseButton === 3) { + // We need to update the DOM selection, because on + // right-click the caret moves to the mouse location. + // See bug 366312. + var timeDiff = e.timeStamp - this._lastMouseTime; + if (timeDiff <= this._clickTime) { + this._updateDOMSelection(); + } + } + if (this.isListening("ContextMenu")) { + var evt = this._createMouseEvent("ContextMenu", e); + evt.screenX = e.screenX; + evt.screenY = e.screenY; + this.onContextMenu(evt); + } if (e.preventDefault) { e.preventDefault(); } return false; }, @@ -3494,129 +4487,98 @@ orion.textview.TextView = (function() { }, _handleDragStart: function (e) { if (!e) { e = window.event; } - if (isDnD) { - var sel = this._getSelection(); - var text = !sel.isEmpty() ? this._getBaseText(sel.start, sel.end) : ""; - if (text) { - e.dataTransfer.effectAllowed = "copyMove"; - e.dataTransfer.setData("text/plain", text); - // TODO: generate a drag image to be a better visual indicatator of the drag operation. - this._dragStartSelection = {start: sel.start, end: sel.end}; - this.focus(); - return; - } + if (isFirefox) { + var self = this; + setTimeout(function() { + self._clientDiv.contentEditable = true; + self._clientDiv.draggable = false; + self._ignoreBlur = false; + }, 0); + } + if (this.isListening("DragStart") && this._dragOffset !== -1) { + this._isMouseDown = false; + this.onDragStart(this._createMouseEvent("DragStart", e)); + this._dragOffset = -1; + } else { + if (e.preventDefault) { e.preventDefault(); } + return false; + } + }, + _handleDrag: function (e) { + if (!e) { e = window.event; } + if (this.isListening("Drag")) { + this.onDrag(this._createMouseEvent("Drag", e)); } - if (e.preventDefault) { e.preventDefault(); } - return false; }, _handleDragEnd: function (e) { if (!e) { e = window.event; } - if (e.preventDefault) { e.preventDefault(); } - var startSel = this._dragStartSelection; - var drop = this._dropDestination; - if (startSel && e.dataTransfer.dropEffect === "move") { - var offset = 0; - if (drop && drop.offset < Math.min(startSel.start, startSel.end)) { - offset = drop.length; - } - var change = { - text: "", - start: startSel.start + offset, - end: startSel.end + offset - }; - this._modifyContent(change, false); + this._dropTarget = false; + this._dragOffset = -1; + if (this.isListening("DragEnd")) { + this.onDragEnd(this._createMouseEvent("DragEnd", e)); } - if (this._undoStack && drop) { - this._undoStack.endCompoundChange(); + if (isFirefox) { + this._fixCaret(); } - this._dragNode.draggable = false; - this._dragStartSelection = null; - this._dropDestination = null; - return false; }, _handleDragEnter: function (e) { if (!e) { e = window.event; } - if (e.preventDefault) { e.preventDefault(); } - var types = e.dataTransfer.types; - var allowed = false; - var types = isDnD ? e.dataTransfer.types : null; - if (types) { - // Firefox gives a .types of type StringList, while Webkit gives us an actual string. - allowed = types.contains ? types.contains("text/plain") : types.indexOf("text/plain"); + var prevent = true; + this._dropTarget = true; + if (this.isListening("DragEnter")) { + prevent = false; + this.onDragEnter(this._createMouseEvent("DragEnter", e)); } - if (allowed) { - e.dataTransfer.dropEffect = "copyMove"; - this.focus(); - return true; + /* + * Webkit will not send drop events if this event is not prevented, as spec in HTML5. + * Firefox and IE do not follow this spec for contentEditable. Note that preventing this + * event will result is loss of functionality (insertion mark, etc). + */ + if (isWebkit || prevent) { + if (e.preventDefault) { e.preventDefault(); } + return false; } - e.dataTransfer.dropEffect = "none"; - return false; }, _handleDragOver: function (e) { if (!e) { e = window.event; } - if (e.preventDefault) { e.preventDefault(); } - var allowed = false; - var types = isDnD ? e.dataTransfer.types : null; - if (types) { - allowed = types.contains ? types.contains("text/plain") : types.indexOf("text/plain"); + var prevent = true; + if (this.isListening("DragOver")) { + prevent = false; + this.onDragOver(this._createMouseEvent("DragOver", e)); } - if (!allowed) { - e.dataTransfer.dropEffect = "none"; + /* + * Webkit will not send drop events if this event is not prevented, as spec in HTML5. + * Firefox and IE do not follow this spec for contentEditable. Note that preventing this + * event will result is loss of functionality (insertion mark, etc). + */ + if (isWebkit || prevent) { + if (prevent) { e.dataTransfer.dropEffect = "none"; } + if (e.preventDefault) { e.preventDefault(); } return false; } - - var destLine = this._getYToLine(e.clientY); - var destOffset = this._getXToOffset(destLine, e.clientX); - - var startSel = this._dragStartSelection; - if (startSel && startSel.start <= destOffset && destOffset <= startSel.end) { - e.dataTransfer.dropEffect = "none"; - return false; + }, + _handleDragLeave: function (e) { + if (!e) { e = window.event; } + this._dropTarget = false; + if (this.isListening("DragLeave")) { + this.onDragLeave(this._createMouseEvent("DragLeave", e)); } - - if (!startSel) { - // Hide the selection when the user drags something coming from the outside. - // TODO: make sure the cursor is actually visible. It's not visible in Firefox during drag, only in Chrome... - this.setSelection(destOffset, destOffset, true); - } - - return true; }, _handleDrop: function (e) { if (!e) { e = window.event; } + this._dropTarget = false; + if (this.isListening("Drop")) { + this.onDrop(this._createMouseEvent("Drop", e)); + } + /* + * This event must be prevented otherwise the user agent will modify + * the DOM. Note that preventing the event on some user agents (i.e. IE) + * indicates that the operation is cancelled. This causes the dropEffect to + * be set to none in the dragend event causing the implementor to not execute + * the code responsible by the move effect. + */ if (e.preventDefault) { e.preventDefault(); } - var allowed = false; - var types = isDnD ? e.dataTransfer.types : null; - if (types) { - allowed = types.contains ? types.contains("text/plain") : types.indexOf("text/plain"); - } - if (!allowed) { - return false; - } - - var destLine = this._getYToLine(e.clientY); - var destOffset = this._getXToOffset(destLine, e.clientX); - var startSel = this._dragStartSelection; - - if (startSel && startSel.start <= destOffset && destOffset <= startSel.end) { - return false; - } - - var text = e.dataTransfer.getData("text/plain"); - this.setSelection(destOffset, destOffset, true); - - if (startSel) { - this._dropDestination = {offset: destOffset, length: text.length}; - if (this._undoStack) { - this._undoStack.startCompoundChange(); - } - } else { - this._dragNode.draggable = false; - } - - this._doContent(text); - this.focus(); - return true; + return false; }, _handleDocFocus: function (e) { if (!e) { e = window.event; } @@ -3642,6 +4604,9 @@ orion.textview.TextView = (function() { this._selDiv3.style.background = color; } } + if (!this._ignoreFocus) { + this.onFocus({type: "Focus"}); + } }, _handleKeyDown: function (e) { if (!e) { e = window.event; } @@ -3662,7 +4627,7 @@ orion.textview.TextView = (function() { this._setLinksVisible(false); } if (e.keyCode === 229) { - if (this.readonly) { + if (this._readonly) { if (e.preventDefault) { e.preventDefault(); } return false; } @@ -3775,21 +4740,26 @@ orion.textview.TextView = (function() { _handleLoad: function (e) { var state = this._getVisible(); if (state === "visible" || (state === "hidden" && isWebkit)) { - this._createView(!e); + this._createView(); } }, _handleMouse: function (e) { + var result = true; var target = this._frameWindow; - if (isIE) { target = this._clientDiv; } + if (isIE || (isFirefox && !this._overlayDiv)) { target = this._clientDiv; } if (this._overlayDiv) { + if (this._hasFocus) { + this._ignoreFocus = true; + } var self = this; setTimeout(function () { self.focus(); + self._ignoreFocus = false; }, 0); } if (this._clickCount === 1) { - this._setGrab(target); - this._setSelectionTo(e.clientX, e.clientY, e.shiftKey); + result = this._setSelectionTo(e.clientX, e.clientY, e.shiftKey, !isOpera && this.isListening("DragStart")); + if (result) { this._setGrab(target); } } else { /* * Feature in IE8 and older, the sequence of events in the IE8 event model @@ -3810,9 +4780,13 @@ orion.textview.TextView = (function() { this._setSelectionTo(e.clientX, e.clientY, e.shiftKey); this._doubleClickSelection = this._getSelection(); } + return result; }, _handleMouseDown: function (e) { if (!e) { e = window.event; } + if (this.isListening("MouseDown")) { + this.onMouseDown(this._createMouseEvent("MouseDown", e)); + } if (this._linksVisible) { var target = e.target || e.srcElement; if (target.tagName !== "A") { @@ -3821,59 +4795,82 @@ orion.textview.TextView = (function() { return; } } - var left = e.which ? e.button === 0 : e.button === 1; this._commitIME(); - if (left) { - var deltaX = Math.abs(this._lastMouseX - e.clientX); - var deltaY = Math.abs(this._lastMouseY - e.clientY); - var time = e.timeStamp ? e.timeStamp : new Date().getTime(); - if ((time - this._lastMouseTime) <= this._clickTime && deltaX <= this._clickDist && deltaY <= this._clickDist) { + + var button = e.which; // 1 - left, 2 - middle, 3 - right + if (!button) { + // if IE 8 or older + if (e.button === 4) { button = 2; } + if (e.button === 2) { button = 3; } + if (e.button === 1) { button = 1; } + } + + // For middle click we always need getTime(). See _getClipboardText(). + var time = button !== 2 && e.timeStamp ? e.timeStamp : new Date().getTime(); + var timeDiff = time - this._lastMouseTime; + var deltaX = Math.abs(this._lastMouseX - e.clientX); + var deltaY = Math.abs(this._lastMouseY - e.clientY); + var sameButton = this._lastMouseButton === button; + this._lastMouseX = e.clientX; + this._lastMouseY = e.clientY; + this._lastMouseTime = time; + this._lastMouseButton = button; + + if (button === 1) { + this._isMouseDown = true; + if (sameButton && timeDiff <= this._clickTime && deltaX <= this._clickDist && deltaY <= this._clickDist) { this._clickCount++; } else { this._clickCount = 1; } - this._lastMouseX = e.clientX; - this._lastMouseY = e.clientY; - this._lastMouseTime = time; - - // Selection drag support - if (isDnD && this._clickCount === 1) { - var inSelection = false; - var selection = this._getSelection(); - if (!selection.isEmpty()) { - var clickLine = this._getYToLine(e.clientY); - var clickOffset = this._getXToOffset(clickLine, e.clientX); - inSelection = selection.start < clickOffset && clickOffset < selection.end; - } - - // Webkit fails to allow dragging if .draggable is set to true during mousedown. - // But Firefox makes it a requirement to set .draggable to true. - this._dragNode.draggable = !isWebkit && inSelection; - - if (inSelection) { - return; // allow the dragstart event - } - } - if (this._dragNode && this._dragNode.draggable) { - this._dragNode.draggable = false; - } - - this._isMouseDown = true; - this._handleMouse(e); - if (isOpera || isChrome) { + if (this._handleMouse(e) && (isOpera || isChrome || (isFirefox && !this._overlayDiv))) { if (!this._hasFocus) { this.focus(); } - } - if (e.preventDefault) { e.preventDefault(); } } }, + _handleMouseOver: function (e) { + if (!e) { e = window.event; } + if (this.isListening("MouseOver")) { + this.onMouseOver(this._createMouseEvent("MouseOver", e)); + } + }, + _handleMouseOut: function (e) { + if (!e) { e = window.event; } + if (this.isListening("MouseOut")) { + this.onMouseOut(this._createMouseEvent("MouseOut", e)); + } + }, _handleMouseMove: function (e) { if (!e) { e = window.event; } - this._setLinksVisible(!this._isMouseDown && (isMac ? e.metaKey : e.ctrlKey)); - if (!this._isMouseDown) { + if (this.isListening("MouseMove")) { + var topNode = this._overlayDiv || this._clientDiv; + var temp = e.target ? e.target : e.srcElement; + while (temp) { + if (topNode === temp) { + this.onMouseMove(this._createMouseEvent("MouseMove", e)); + break; + } + temp = temp.parentNode; + } + } + if (this._dropTarget) { + return; + } + /* + * Bug in IE9. IE sends one mouse event when the user changes the text by + * pasting or undo. These operations usually happen with the Ctrl key + * down which causes the view to enter link mode. Link mode does not end + * because there are no further events. The fix is to only enter link + * mode when the coordinates of the mouse move event have changed. + */ + var changed = this._linksVisible || this._lastMouseMoveX !== e.clientX || this._lastMouseMoveY !== e.clientY; + this._lastMouseMoveX = e.clientX; + this._lastMouseMoveY = e.clientY; + this._setLinksVisible(changed && !this._isMouseDown && (isMac ? e.metaKey : e.ctrlKey)); + if (!this._isMouseDown || this._dragOffset !== -1) { return; } /* @@ -3946,21 +4943,36 @@ orion.textview.TextView = (function() { } } }, + _createMouseEvent: function(type, e) { + var scroll = this._getScroll(); + var viewRect = this._viewDiv.getBoundingClientRect(); + var viewPad = this._getViewPadding(); + var x = e.clientX + scroll.x - viewRect.left - viewPad.left; + var y = e.clientY + scroll.y - viewRect.top; + return { + type: type, + event: e, + x: x, + y: y + }; + }, _handleMouseUp: function (e) { if (!e) { e = window.event; } + if (this.isListening("MouseUp")) { + this.onMouseUp(this._createMouseEvent("MouseUp", e)); + } if (this._linksVisible) { return; } var left = e.which ? e.button === 0 : e.button === 1; if (left) { - if (this._dragNode && this._dragNode.draggable) { - this._dragNode.draggable = false; - if (!this._dragStartSelection) { - this._setSelectionTo(e.clientX, e.clientY, false); - } - this.focus(); + if (this._dragOffset !== -1) { + var selection = this._getSelection(); + selection.extend(this._dragOffset); + selection.collapse(); + this._setSelection(selection, true, true); + this._dragOffset = -1; } - this._isMouseDown = false; this._endAutoScroll(); @@ -3978,6 +4990,17 @@ orion.textview.TextView = (function() { * mouse down and ungrab on mouse move when the button 1 is not set. */ if (isW3CEvents) { this._setGrab(null); } + + /* + * Note that there cases when Firefox sets the DOM selection in mouse up. + * This happens for example after a cancelled drag operation. + * + * Note that on Chrome and IE, the caret stops blicking if mouse up is + * prevented. + */ + if (isFirefox) { + e.preventDefault(); + } } }, _handleMouseWheel: function (e) { @@ -4062,7 +5085,11 @@ orion.textview.TextView = (function() { * Bug in IE, */ var self = this; - setTimeout(function() {self._updateDOMSelection();}, 0); + this._ignoreFocus = true; + setTimeout(function() { + self._updateDOMSelection(); + this._ignoreFocus = false; + }, 0); } if (e.preventDefault) { e.preventDefault(); } return false; @@ -4130,8 +5157,9 @@ orion.textview.TextView = (function() { this._hScroll = scroll.x; this._vScroll = scroll.y; this._commitIME(); - this._updatePage(); + this._updatePage(oldY === scroll.y); var e = { + type: "Scroll", oldValue: {x: oldX, y: oldY}, newValue: scroll }; @@ -4320,7 +5348,7 @@ orion.textview.TextView = (function() { } }, 0); } -// e.preventDefault(); +// e.preventDefault(); }, /************************************ Actions ******************************************/ @@ -4360,12 +5388,22 @@ orion.textview.TextView = (function() { var model = this._model; var caret = selection.getCaret(); var lineIndex = model.getLineAtOffset(caret); - if (caret === model.getLineStart(lineIndex)) { + var lineStart = model.getLineStart(lineIndex); + if (caret === lineStart) { if (lineIndex > 0) { selection.extend(model.getLineEnd(lineIndex - 1)); } } else { - selection.extend(this._getOffset(caret, args.unit, -1)); + var removeTab = false; + if (this._expandTab && args.unit === "character" && (caret - lineStart) % this._tabSize === 0) { + var lineText = model.getText(lineStart, caret); + removeTab = !/[^ ]/.test(lineText); // Only spaces between line start and caret. + } + if (removeTab) { + selection.extend(caret - this._tabSize); + } else { + selection.extend(this._getOffset(caret, args.unit, -1)); + } } } this._modifyContent({text: "", start: selection.start, end: selection.end}, true); @@ -4491,7 +5529,7 @@ orion.textview.TextView = (function() { if (lineIndex + 1 < model.getLineCount()) { var scrollX = this._getScroll().x; var x = this._columnX; - if (x === -1 || args.select || args.wholeLine) { + if (x === -1 || args.wholeLine || (args.select && isIE)) { var offset = args.wholeLine ? model.getLineEnd(lineIndex + 1) : caret; x = this._getOffsetToX(offset) + scrollX; } @@ -4510,7 +5548,7 @@ orion.textview.TextView = (function() { if (lineIndex > 0) { var scrollX = this._getScroll().x; var x = this._columnX; - if (x === -1 || args.select || args.wholeLine) { + if (x === -1 || args.wholeLine || (args.select && isIE)) { var offset = args.wholeLine ? model.getLineStart(lineIndex - 1) : caret; x = this._getOffsetToX(offset) + scrollX; } @@ -4535,7 +5573,7 @@ orion.textview.TextView = (function() { var scrollLines = Math.min(lineCount - caretLine - 1, lines); scrollLines = Math.max(1, scrollLines); var x = this._columnX; - if (x === -1 || args.select) { + if (x === -1 || (args.select && isIE)) { x = this._getOffsetToX(caret) + scroll.x; } selection.extend(this._getXToOffset(caretLine + scrollLines, x - scroll.x)); @@ -4562,7 +5600,7 @@ orion.textview.TextView = (function() { var lines = Math.floor(clientHeight / lineHeight); var scrollLines = Math.max(1, Math.min(caretLine, lines)); var x = this._columnX; - if (x === -1 || args.select) { + if (x === -1 || (args.select && isIE)) { x = this._getOffsetToX(caret) + scroll.x; } selection.extend(this._getXToOffset(caretLine - scrollLines, x - scroll.x)); @@ -4574,11 +5612,19 @@ orion.textview.TextView = (function() { return true; }, _doPaste: function(e) { - var text = this._getClipboardText(e); - if (text) { - this._doContent(text); - } - return text !== null; + var self = this; + var result = this._getClipboardText(e, function(text) { + if (text) { + if (isLinux && self._lastMouseButton === 2) { + var timeDiff = new Date().getTime() - self._lastMouseTime; + if (timeDiff <= self._clickTime) { + self._setSelectionTo(self._lastMouseX, self._lastMouseY); + } + } + self._doContent(text); + } + }); + return result !== null; }, _doScroll: function (args) { var type = args.type; @@ -4616,12 +5662,27 @@ orion.textview.TextView = (function() { return true; }, _doTab: function (args) { - this._doContent("\t"); + var text = "\t"; + if (this._expandTab) { + var model = this._model; + var caret = this._getSelection().getCaret(); + var lineIndex = model.getLineAtOffset(caret); + var lineStart = model.getLineStart(lineIndex); + var spaces = this._tabSize - ((caret - lineStart) % this._tabSize); + text = (new Array(spaces + 1)).join(" "); + } + this._doContent(text); return true; }, /************************************ Internals ******************************************/ - _applyStyle: function(style, node) { + _applyStyle: function(style, node, reset) { + if (reset) { + var attrs = node.attributes; + for (var i= attrs.length; i-->0;) { + node.removeAttributeNode(attrs[i]); + } + } if (!style) { return; } @@ -4710,7 +5771,17 @@ orion.textview.TextView = (function() { if (h4 > h3) { fontStyle = 3; } - this._largestFontStyle = fontStyle; + var style; + if (fontStyle !== 0) { + style = {style: {}}; + if ((fontStyle & 1) !== 0) { + style.style.fontStyle = "italic"; + } + if ((fontStyle & 2) !== 0) { + style.style.fontWeight = "bold"; + } + } + this._largestFontStyle = style; parent.removeChild(line); return lineHeight; }, @@ -4754,6 +5825,40 @@ orion.textview.TextView = (function() { this._setSelection(selection, true); return true; }, + _clone: function (obj) { + /*Note that this code only works because of the limited types used in TextViewOptions */ + if (obj instanceof Array) { + return obj.slice(0); + } + return obj; + }, + _compare: function (s1, s2) { + if (s1 === s2) { return true; } + if (s1 && !s2 || !s1 && s2) { return false; } + if ((s1 && s1.constructor === String) || (s2 && s2.constructor === String)) { return false; } + if (s1 instanceof Array || s2 instanceof Array) { + if (!(s1 instanceof Array && s2 instanceof Array)) { return false; } + if (s1.length !== s2.length) { return false; } + for (var i = 0; i < s1.length; i++) { + if (!this._compare(s1[i], s2[i])) { + return false; + } + } + return true; + } + if (!(s1 instanceof Object) || !(s2 instanceof Object)) { return false; } + var p; + for (p in s1) { + if (s1.hasOwnProperty(p)) { + if (!s2.hasOwnProperty(p)) { return false; } + if (!this._compare(s1[p], s2[p])) {return false; } + } + } + for (p in s2) { + if (!s1.hasOwnProperty(p)) { return false; } + } + return true; + }, _commitIME: function () { if (this._imeOffset === -1) { return; } // make the state of the IME match the state the view expects it be in @@ -4805,7 +5910,7 @@ orion.textview.TextView = (function() { } }, _createActions: function () { - var KeyBinding = orion.textview.KeyBinding; + var KeyBinding = mKeyBinding.KeyBinding; //no duplicate keybindings var bindings = this._keyBindings = []; @@ -4827,20 +5932,27 @@ orion.textview.TextView = (function() { bindings.push({name: "scrollTextEnd", keyBinding: new KeyBinding(35), predefined: true}); bindings.push({name: "textStart", keyBinding: new KeyBinding(38, true), predefined: true}); bindings.push({name: "textEnd", keyBinding: new KeyBinding(40, true), predefined: true}); + bindings.push({name: "scrollPageUp", keyBinding: new KeyBinding(38, null, null, null, true), predefined: true}); + bindings.push({name: "scrollPageDown", keyBinding: new KeyBinding(40, null, null, null, true), predefined: true}); + bindings.push({name: "lineStart", keyBinding: new KeyBinding(37, null, null, null, true), predefined: true}); + bindings.push({name: "lineEnd", keyBinding: new KeyBinding(39, null, null, null, true), predefined: true}); + //TODO These two actions should be changed to paragraph start and paragraph end when word wrap is implemented + bindings.push({name: "lineStart", keyBinding: new KeyBinding(38, null, null, true), predefined: true}); + bindings.push({name: "lineEnd", keyBinding: new KeyBinding(40, null, null, true), predefined: true}); } else { bindings.push({name: "pageUp", keyBinding: new KeyBinding(33), predefined: true}); bindings.push({name: "pageDown", keyBinding: new KeyBinding(34), predefined: true}); bindings.push({name: "lineStart", keyBinding: new KeyBinding(36), predefined: true}); bindings.push({name: "lineEnd", keyBinding: new KeyBinding(35), predefined: true}); - if (isLinux) { - bindings.push({name: "lineStartUp", keyBinding: new KeyBinding(38, true), predefined: true}); - bindings.push({name: "lineEndDown", keyBinding: new KeyBinding(40, true), predefined: true}); - } bindings.push({name: "wordPrevious", keyBinding: new KeyBinding(37, true), predefined: true}); bindings.push({name: "wordNext", keyBinding: new KeyBinding(39, true), predefined: true}); bindings.push({name: "textStart", keyBinding: new KeyBinding(36, true), predefined: true}); bindings.push({name: "textEnd", keyBinding: new KeyBinding(35, true), predefined: true}); } + if (isFirefox && isLinux) { + bindings.push({name: "lineUp", keyBinding: new KeyBinding(38, true), predefined: true}); + bindings.push({name: "lineDown", keyBinding: new KeyBinding(40, true), predefined: true}); + } // Select Cursor Navigation bindings.push({name: "selectLineUp", keyBinding: new KeyBinding(38, null, true), predefined: true}); @@ -4858,6 +5970,11 @@ orion.textview.TextView = (function() { bindings.push({name: "selectTextEnd", keyBinding: new KeyBinding(35, null, true), predefined: true}); bindings.push({name: "selectTextStart", keyBinding: new KeyBinding(38, true, true), predefined: true}); bindings.push({name: "selectTextEnd", keyBinding: new KeyBinding(40, true, true), predefined: true}); + bindings.push({name: "selectLineStart", keyBinding: new KeyBinding(37, null, true, null, true), predefined: true}); + bindings.push({name: "selectLineEnd", keyBinding: new KeyBinding(39, null, true, null, true), predefined: true}); + //TODO These two actions should be changed to select paragraph start and select paragraph end when word wrap is implemented + bindings.push({name: "selectLineStart", keyBinding: new KeyBinding(38, null, true, true), predefined: true}); + bindings.push({name: "selectLineEnd", keyBinding: new KeyBinding(40, null, true, true), predefined: true}); } else { if (isLinux) { bindings.push({name: "selectWholeLineUp", keyBinding: new KeyBinding(38, true, true), predefined: true}); @@ -4936,8 +6053,6 @@ orion.textview.TextView = (function() { {name: "lineDown", defaultHandler: function() {return self._doLineDown({select: false});}}, {name: "lineStart", defaultHandler: function() {return self._doHome({select: false, ctrl:false});}}, {name: "lineEnd", defaultHandler: function() {return self._doEnd({select: false, ctrl:false});}}, - {name: "lineStartUp", defaultHandler: function() {return self._doLineUp({select: false, wholeLine:true});}}, - {name: "lineEndDown", defaultHandler: function() {return self._doLineDown({select: false, wholeLine:true});}}, {name: "charPrevious", defaultHandler: function() {return self._doCursorPrevious({select: false, unit:"character"});}}, {name: "charNext", defaultHandler: function() {return self._doCursorNext({select: false, unit:"character"});}}, {name: "pageUp", defaultHandler: function() {return self._doPageUp({select: false});}}, @@ -4982,49 +6097,20 @@ orion.textview.TextView = (function() { {name: "paste", defaultHandler: function() {return self._doPaste();}} ]; }, - _createLine: function(parent, sibling, document, lineIndex, model) { + _createLine: function(parent, div, document, lineIndex, model) { var lineText = model.getLine(lineIndex); var lineStart = model.getLineStart(lineIndex); - var e = {textView: this, lineIndex: lineIndex, lineText: lineText, lineStart: lineStart}; + var e = {type:"LineStyle", textView: this, lineIndex: lineIndex, lineText: lineText, lineStart: lineStart}; this.onLineStyle(e); - var child = document.createElement("DIV"); - child.lineIndex = lineIndex; - this._applyStyle(e.style, child); - if (lineText.length !== 0) { - var start = 0; - var tabSize = this._customTabSize; - if (tabSize && tabSize !== 8) { - var tabIndex = lineText.indexOf("\t"), ignoreChars = 0; - while (tabIndex !== -1) { - this._createRanges(child, document, e.ranges, start, tabIndex, lineText, lineStart); - var spacesCount = tabSize - ((tabIndex + ignoreChars) % tabSize); - var spaces = "\u00A0"; - for (var i = 1; i < spacesCount; i++) { - spaces += " "; - } - var tabSpan = document.createElement("SPAN"); - tabSpan.appendChild(document.createTextNode(spaces)); - tabSpan.ignoreChars = spacesCount - 1; - ignoreChars += tabSpan.ignoreChars; - if (e.ranges) { - for (var j = 0; j < e.ranges.length; j++) { - var range = e.ranges[j]; - var styleStart = range.start - lineStart; - var styleEnd = range.end - lineStart; - if (styleStart > tabIndex) { break; } - if (styleStart <= tabIndex && tabIndex < styleEnd) { - this._applyStyle(range.style, tabSpan); - break; - } - } - } - child.appendChild(tabSpan); - start = tabIndex + 1; - tabIndex = lineText.indexOf("\t", start); - } - } - this._createRanges(child, document, e.ranges, start, lineText.length, lineText, lineStart); + var lineDiv = div || document.createElement("DIV"); + if (!div || !this._compare(div.viewStyle, e.style)) { + this._applyStyle(e.style, lineDiv, div); + lineDiv.viewStyle = e.style; } + lineDiv.lineIndex = lineIndex; + var ranges = []; + var data = {tabOffset: 0, ranges: ranges}; + this._createRanges(e.ranges, lineText, 0, lineText.length, lineStart, data); /* * A trailing span with a whitespace is added for three different reasons: @@ -5034,14 +6120,6 @@ orion.textview.TextView = (function() { * selection at the end of the line when the line is fully selected. * 3. The height of a div with only an empty span is zero. */ - var span = document.createElement("SPAN"); - span.ignoreChars = 1; - if ((this._largestFontStyle & 1) !== 0) { - span.style.fontStyle = "italic"; - } - if ((this._largestFontStyle & 2) !== 0) { - span.style.fontWeight = "bold"; - } var c = " "; if (!this._fullSelection && isIE < 9) { /* @@ -5061,13 +6139,76 @@ orion.textview.TextView = (function() { */ c = "\u200C"; } - span.appendChild(document.createTextNode(c)); - child.appendChild(span); + ranges.push({text: c, style: this._largestFontStyle, ignoreChars: 1}); - parent.insertBefore(child, sibling); - return child; + var range, span, style, oldSpan, oldStyle, text, oldText, end = 0, oldEnd = 0, next; + var changeCount, changeStart; + if (div) { + var modelChangedEvent = div.modelChangedEvent; + if (modelChangedEvent) { + if (modelChangedEvent.removedLineCount === 0 && modelChangedEvent.addedLineCount === 0) { + changeStart = modelChangedEvent.start - lineStart; + changeCount = modelChangedEvent.addedCharCount - modelChangedEvent.removedCharCount; + } else { + changeStart = -1; + } + delete div.modelChangedEvent; + } + oldSpan = div.firstChild; + } + for (var i = 0; i < ranges.length; i++) { + range = ranges[i]; + text = range.text; + end += text.length; + style = range.style; + if (oldSpan) { + oldText = oldSpan.firstChild.data; + oldStyle = oldSpan.viewStyle; + if (oldText === text && this._compare(style, oldStyle)) { + oldEnd += oldText.length; + delete oldSpan._rectsCache; + span = oldSpan = oldSpan.nextSibling; + continue; + } else { + while (oldSpan) { + if (changeStart !== -1) { + var spanEnd = end; + if (spanEnd >= changeStart) { + spanEnd -= changeCount; + } + var length = oldSpan.firstChild.data.length; + if (oldEnd + length > spanEnd) { break; } + oldEnd += length; + } + next = oldSpan.nextSibling; + lineDiv.removeChild(oldSpan); + oldSpan = next; + } + } + } + span = this._createSpan(lineDiv, document, text, style, range.ignoreChars); + if (oldSpan) { + lineDiv.insertBefore(span, oldSpan); + } else { + lineDiv.appendChild(span); + } + if (div) { + div.lineWidth = undefined; + } + } + if (div) { + var tmp = span ? span.nextSibling : null; + while (tmp) { + next = tmp.nextSibling; + div.removeChild(tmp); + tmp = next; + } + } else { + parent.appendChild(lineDiv); + } + return lineDiv; }, - _createRanges: function(parent, document, ranges, start, end, text, lineStart) { + _createRanges: function(ranges, text, start, end, lineStart, data) { if (start >= end) { return; } if (ranges) { for (var i = 0; i < ranges.length; i++) { @@ -5080,18 +6221,55 @@ orion.textview.TextView = (function() { styleStart = Math.max(start, styleStart); styleEnd = Math.min(end, styleEnd); if (start < styleStart) { - parent.appendChild(this._createRange(parent, document, text.substring(start, styleStart), null)); + this._createRange(text, start, styleStart, null, data); } - parent.appendChild(this._createRange(parent, document, text.substring(styleStart, styleEnd), range.style)); + while (i + 1 < ranges.length && ranges[i + 1].start - lineStart === styleEnd && this._compare(range.style, ranges[i + 1].style)) { + range = ranges[i + 1]; + styleEnd = Math.min(lineStart + end, range.end) - lineStart; + i++; + } + this._createRange(text, styleStart, styleEnd, range.style, data); start = styleEnd; } } } if (start < end) { - parent.appendChild(this._createRange(parent, document, text.substring(start, end), null)); + this._createRange(text, start, end, null, data); } }, - _createRange: function(parent, document, text, style) { + _createRange: function(text, start, end, style, data) { + if (start >= end) { return; } + var tabSize = this._customTabSize, range; + if (tabSize && tabSize !== 8) { + var tabIndex = text.indexOf("\t", start); + while (tabIndex !== -1 && tabIndex < end) { + if (start < tabIndex) { + range = {text: text.substring(start, tabIndex), style: style}; + data.ranges.push(range); + data.tabOffset += range.text.length; + } + var spacesCount = tabSize - (data.tabOffset % tabSize); + if (spacesCount > 0) { + //TODO hack to preserve text length in getDOMText() + var spaces = "\u00A0"; + for (var i = 1; i < spacesCount; i++) { + spaces += " "; + } + range = {text: spaces, style: style, ignoreChars: spacesCount - 1}; + data.ranges.push(range); + data.tabOffset += range.text.length; + } + start = tabIndex + 1; + tabIndex = text.indexOf("\t", start); + } + } + if (start < end) { + range = {text: text.substring(start, end), style: style}; + data.ranges.push(range); + data.tabOffset += range.text.length; + } + }, + _createSpan: function(parent, document, text, style, ignoreChars) { var isLink = style && style.tagName === "A"; if (isLink) { parent.hasLink = true; } var tagName = isLink && this._linksVisible ? "A" : "SPAN"; @@ -5103,6 +6281,9 @@ orion.textview.TextView = (function() { addHandler(child, "click", function(e) { return self._handleLinkClick(e); }, false); } child.viewStyle = style; + if (ignoreChars) { + child.ignoreChars = ignoreChars; + } return child; }, _createRuler: function(ruler) { @@ -5172,7 +6353,10 @@ orion.textview.TextView = (function() { * parent is not connected to the document. Only create it when the load * event is trigged. */ - addHandler(frame, "load", this._loadHandler = function(e) { self._handleLoad(e); }); + this._loadHandler = function(e) { + self._handleLoad(e); + }; + addHandler(frame, "load", this._loadHandler, !!isFirefox); if (!isWebkit) { /* * Feature in IE and Firefox. It is not possible to get the style of an @@ -5188,9 +6372,11 @@ orion.textview.TextView = (function() { } parent.appendChild(frame); /* create synchronously if possible */ - this._handleLoad(); + if (this._sync) { + this._handleLoad(); + } }, - _getFrameHTML: function(sync) { + _getFrameHTML: function() { var html = []; html.push(""); html.push(""); @@ -5206,9 +6392,8 @@ orion.textview.TextView = (function() { if (this._stylesheet) { var stylesheet = typeof(this._stylesheet) === "string" ? [this._stylesheet] : this._stylesheet; for (var i = 0; i < stylesheet.length; i++) { - if (sync) { + if (this._sync) { try { - //Force CSS to be loaded synchronously so lineHeight can be calculated var objXml = new XMLHttpRequest(); if (objXml.overrideMimeType) { objXml.overrideMimeType("text/css"); @@ -5226,29 +6411,64 @@ orion.textview.TextView = (function() { html.push("'>"); } } + /* + * Feature in WebKit. In WebKit, window load will not wait for the style sheets + * to be loaded unless there is script element after the style sheet link elements. + */ + html.push(""); html.push(""); html.push(""); html.push(""); return html.join(""); }, - _createView: function(sync) { + _createView: function() { + if (this._frameDocument) { return; } + var frameWindow = this._frameWindow = this._frame.contentWindow; + var frameDocument = this._frameDocument = frameWindow.document; + var self = this; + function write() { + frameDocument.open(); + frameDocument.write(self._getFrameHTML()); + frameDocument.close(); + self._windowLoadHandler = function(e) { + self._createContent(); + }; + addHandler(frameWindow, "load", self._windowLoadHandler); + } + /* + * Bug in Firefox. Firefox does not send window load event if document.write + * is done inside of the frame load event handler. + */ + if (isFirefox && !this._sync) { + setTimeout(write, 0); + } else { + write(); + } + if (this._sync) { + this._createContent(); + } + }, + _createContent: function() { if (this._clientDiv) { return; } - if (this._ignoreCreate) { return; } - this._ignoreCreate = true; - - var frame = this._frame; var parent = this._parent; var parentDocument = this._parentDocument; - var frameWindow = frame.contentWindow; - this._frameWindow = frameWindow; - var frameDocument = frameWindow.document; - this._frameDocument = frameDocument; - frameDocument.open(); - frameDocument.write(this._getFrameHTML(true)); - frameDocument.close(); - + var frameDocument = this._frameDocument; + /* + * Bug in Safari. Safari sends the window load event before the + * style sheets are loaded. The fix is to defer creation of the + * contents until the document readyState changes to complete. + */ + var self = this; + if (!this._sync && frameDocument.readyState !== "complete") { + setTimeout(function() { + self._createContent(); + }, 10); + return; + } var body = frameDocument.body; - body.className = "viewContainer"; + this._setThemeClass(this._themeClass, true); body.style.margin = "0px"; body.style.borderWidth = "0px"; body.style.padding = "0px"; @@ -5316,93 +6536,47 @@ orion.textview.TextView = (function() { scrollDiv.style.borderWidth = "0px"; scrollDiv.style.padding = "0px"; viewDiv.appendChild(scrollDiv); - - if (isPad || (this._fullSelection && !isWebkit)) { - this._hightlightRGB = "Highlight"; - var selDiv1 = frameDocument.createElement("DIV"); - this._selDiv1 = selDiv1; - selDiv1.id = "selDiv1"; - selDiv1.style.position = "fixed"; - selDiv1.style.borderWidth = "0px"; - selDiv1.style.margin = "0px"; - selDiv1.style.padding = "0px"; - selDiv1.style.MozOutline = "none"; - selDiv1.style.outline = "none"; - selDiv1.style.background = this._hightlightRGB; - selDiv1.style.width="0px"; - selDiv1.style.height="0px"; - scrollDiv.appendChild(selDiv1); - var selDiv2 = frameDocument.createElement("DIV"); - this._selDiv2 = selDiv2; - selDiv2.id = "selDiv2"; - selDiv2.style.position = "fixed"; - selDiv2.style.borderWidth = "0px"; - selDiv2.style.margin = "0px"; - selDiv2.style.padding = "0px"; - selDiv2.style.MozOutline = "none"; - selDiv2.style.outline = "none"; - selDiv2.style.background = this._hightlightRGB; - selDiv2.style.width="0px"; - selDiv2.style.height="0px"; - scrollDiv.appendChild(selDiv2); - var selDiv3 = frameDocument.createElement("DIV"); - this._selDiv3 = selDiv3; - selDiv3.id = "selDiv3"; - selDiv3.style.position = "fixed"; - selDiv3.style.borderWidth = "0px"; - selDiv3.style.margin = "0px"; - selDiv3.style.padding = "0px"; - selDiv3.style.MozOutline = "none"; - selDiv3.style.outline = "none"; - selDiv3.style.background = this._hightlightRGB; - selDiv3.style.width="0px"; - selDiv3.style.height="0px"; - scrollDiv.appendChild(selDiv3); + + if (isFirefox) { + var clipDiv = frameDocument.createElement("DIV"); + this._clipDiv = clipDiv; + clipDiv.id = "clipDiv"; + clipDiv.style.position = "fixed"; + clipDiv.style.overflow = "hidden"; + clipDiv.style.margin = "0px"; + clipDiv.style.borderWidth = "0px"; + clipDiv.style.padding = "0px"; + scrollDiv.appendChild(clipDiv); - /* - * Bug in Firefox. The Highlight color is mapped to list selection - * background instead of the text selection background. The fix - * is to map known colors using a table or fallback to light blue. - */ - if (isFirefox && isMac) { - var style = this._frameWindow.getComputedStyle(selDiv3, null); - var rgb = style.getPropertyValue("background-color"); - switch (rgb) { - case "rgb(119, 141, 168)": rgb = "rgb(199, 208, 218)"; break; - case "rgb(127, 127, 127)": rgb = "rgb(198, 198, 198)"; break; - case "rgb(255, 193, 31)": rgb = "rgb(250, 236, 115)"; break; - case "rgb(243, 70, 72)": rgb = "rgb(255, 176, 139)"; break; - case "rgb(255, 138, 34)": rgb = "rgb(255, 209, 129)"; break; - case "rgb(102, 197, 71)": rgb = "rgb(194, 249, 144)"; break; - case "rgb(140, 78, 184)": rgb = "rgb(232, 184, 255)"; break; - default: rgb = "rgb(180, 213, 255)"; break; - } - this._hightlightRGB = rgb; - selDiv1.style.background = rgb; - selDiv2.style.background = rgb; - selDiv3.style.background = rgb; - var styleSheet = frameDocument.styleSheets[0]; - styleSheet.insertRule("::-moz-selection {background: " + rgb + "; }", 0); - } + var clipScrollDiv = frameDocument.createElement("DIV"); + this._clipScrollDiv = clipScrollDiv; + clipScrollDiv.id = "clipScrollDiv"; + clipScrollDiv.style.position = "absolute"; + clipScrollDiv.style.height = "1px"; + clipScrollDiv.style.top = "-1000px"; + clipDiv.appendChild(clipScrollDiv); } + + this._setFullSelection(this._fullSelection, true); var clientDiv = frameDocument.createElement("DIV"); clientDiv.className = "viewContent"; this._clientDiv = clientDiv; clientDiv.id = "clientDiv"; clientDiv.style.whiteSpace = "pre"; - clientDiv.style.position = "fixed"; + clientDiv.style.position = this._clipDiv ? "absolute" : "fixed"; clientDiv.style.borderWidth = "0px"; clientDiv.style.margin = "0px"; clientDiv.style.padding = "0px"; clientDiv.style.MozOutline = "none"; clientDiv.style.outline = "none"; + clientDiv.style.zIndex = "1"; if (isPad) { clientDiv.style.WebkitTapHighlightColor = "transparent"; } - scrollDiv.appendChild(clientDiv); + (this._clipDiv || scrollDiv).appendChild(clientDiv); - if (isFirefox) { + if (isFirefox && !clientDiv.setCapture) { var overlayDiv = frameDocument.createElement("DIV"); this._overlayDiv = overlayDiv; overlayDiv.id = "overlayDiv"; @@ -5411,54 +6585,50 @@ orion.textview.TextView = (function() { overlayDiv.style.margin = clientDiv.style.margin; overlayDiv.style.padding = clientDiv.style.padding; overlayDiv.style.cursor = "text"; - overlayDiv.style.zIndex = "1"; - scrollDiv.appendChild(overlayDiv); + overlayDiv.style.zIndex = "2"; + (this._clipDiv || scrollDiv).appendChild(overlayDiv); } if (!isPad) { clientDiv.contentEditable = "true"; } - if (isDnD) { - this._dragNode = this._overlayDiv || this._clientDiv; - } this._lineHeight = this._calculateLineHeight(); this._viewPadding = this._calculatePadding(); if (isIE) { body.style.lineHeight = this._lineHeight + "px"; } - if (this._tabSize) { - if (isOpera) { - clientDiv.style.OTabSize = this._tabSize+""; - } else if (isFirefox >= 4) { - clientDiv.style.MozTabSize = this._tabSize+""; - } else if (this._tabSize !== 8) { - this._customTabSize = this._tabSize; - } - } + this._setTabSize(this._tabSize, true); this._hookEvents(); var rulers = this._rulers; for (var i=0; i 0 || v > 0) { - var self = this; - setTimeout(function() { - self._scrollView(h, v); - }, 0); - } + this._updatePage(); + var h = this._hScroll, v = this._vScroll; + this._vScroll = this._hScroll = 0; + if (h > 0 || v > 0) { + viewDiv.scrollLeft = h; + viewDiv.scrollTop = v; } - this._ignoreCreate = false; + this.onLoad({type: "Load"}); + }, + _defaultOptions: function() { + return { + parent: {value: undefined, recreate: true, update: null}, + model: {value: undefined, recreate: false, update: this.setModel}, + readonly: {value: false, recreate: false, update: null}, + fullSelection: {value: true, recreate: false, update: this._setFullSelection}, + tabSize: {value: 8, recreate: false, update: this._setTabSize}, + expandTab: {value: false, recreate: false, update: null}, + stylesheet: {value: [], recreate: true, update: null}, + themeClass: {value: undefined, recreate: false, update: this._setThemeClass}, + sync: {value: false, recreate: false, update: null} + }; }, _destroyFrame: function() { var frame = this._frame; if (!frame) { return; } if (this._loadHandler) { - removeHandler(frame, "load", this._loadHandler); + removeHandler(frame, "load", this._loadHandler, !!isFirefox); this._loadHandler = null; } if (this._attrModifiedHandler) { @@ -5467,8 +6637,6 @@ orion.textview.TextView = (function() { } frame.parentNode.removeChild(frame); this._frame = null; - this._frameDocument = null; - this._frameWindow = null; }, _destroyRuler: function(ruler) { var side = ruler.getLocation(); @@ -5490,6 +6658,10 @@ orion.textview.TextView = (function() { if (!clientDiv) { return; } this._setGrab(null); this._unhookEvents(); + if (this._windowLoadHandler) { + removeHandler(this._frameWindow, "load", this._windowLoadHandler); + this._windowLoadHandler = null; + } /* Destroy timers */ if (this._autoScrollTimerID) { @@ -5511,15 +6683,20 @@ orion.textview.TextView = (function() { this._selDiv1 = null; this._selDiv2 = null; this._selDiv3 = null; + this._insertedSelRule = false; this._textArea = null; this._clipboardDiv = null; this._scrollDiv = null; this._viewDiv = null; + this._clipDiv = null; + this._clipScrollDiv = null; this._clientDiv = null; this._overlayDiv = null; - this._dragNode = null; this._leftDiv = null; this._rightDiv = null; + this._frameDocument = null; + this._frameWindow = null; + this.onUnload({type: "Unload"}); }, _doAutoScroll: function (direction, x, y) { this._autoScrollDir = direction; @@ -5534,6 +6711,18 @@ orion.textview.TextView = (function() { this._autoScrollDir = undefined; this._autoScrollTimerID = undefined; }, + _fixCaret: function() { + var clientDiv = this._clientDiv; + if (clientDiv) { + var hasFocus = this._hasFocus; + this._ignoreFocus = true; + if (hasFocus) { clientDiv.blur(); } + clientDiv.contentEditable = false; + clientDiv.contentEditable = true; + if (hasFocus) { clientDiv.focus(); } + this._ignoreFocus = false; + } + }, _getBaseText: function(start, end) { var model = this._model; /* This is the only case the view access the base model, alternatively the view could use a event to application to customize the text */ @@ -5646,7 +6835,7 @@ orion.textview.TextView = (function() { var viewPad = this._getViewPadding(); return Math.max(0, this._viewDiv.clientWidth - viewPad.left - viewPad.right); }, - _getClipboardText: function (event) { + _getClipboardText: function (event, handler) { var delimiter = this._model.getLineDelimiter(); var clipboadText, text; if (this._frameWindow.clipboardData) { @@ -5654,51 +6843,65 @@ orion.textview.TextView = (function() { clipboadText = []; text = this._frameWindow.clipboardData.getData("Text"); this._convertDelimiter(text, function(t) {clipboadText.push(t);}, function() {clipboadText.push(delimiter);}); - return clipboadText.join(""); + text = clipboadText.join(""); + if (handler) { handler(text); } + return text; } if (isFirefox) { + this._ignoreFocus = true; var document = this._frameDocument; var clipboardDiv = this._clipboardDiv; clipboardDiv.innerHTML = "

";
 				clipboardDiv.firstChild.focus();
 				var self = this;
 				var _getText = function() {
-					var text = self._getTextFromElement(clipboardDiv);
+					var noteText = self._getTextFromElement(clipboardDiv);
 					clipboardDiv.innerHTML = "";
 					clipboadText = [];
-					self._convertDelimiter(text, function(t) {clipboadText.push(t);}, function() {clipboadText.push(delimiter);});
+					self._convertDelimiter(noteText, function(t) {clipboadText.push(t);}, function() {clipboadText.push(delimiter);});
 					return clipboadText.join("");
 				};
 				
 				/* Try execCommand first. Works on firefox with clipboard permission. */
 				var result = false;
 				this._ignorePaste = true;
-				try {
-					result = document.execCommand("paste", false, null);
-				} catch (ex) {
-					// Firefox can throw even when execCommand() works, see bug 362835
-					result = clipboardDiv.childNodes.length > 1 || clipboardDiv.firstChild && clipboardDiv.firstChild.childNodes.length > 0;
+
+				/* Do not try execCommand if middle-click is used, because if we do, we get the clipboard text, not the primary selection text. */
+				if (!isLinux || this._lastMouseButton !== 2) {
+					try {
+						result = document.execCommand("paste", false, null);
+					} catch (ex) {
+						/* Firefox can throw even when execCommand() works, see bug 362835. */
+						result = clipboardDiv.childNodes.length > 1 || clipboardDiv.firstChild && clipboardDiv.firstChild.childNodes.length > 0;
+					}
 				}
 				this._ignorePaste = false;
 				if (!result) {
-					/*
-					* Try native paste in DOM, works for firefox during the paste event.
-					*/
+					/* Try native paste in DOM, works for firefox during the paste event. */
 					if (event) {
 						setTimeout(function() {
 							self.focus();
-							var text = _getText();
-							if (text) { self._doContent(text); }
+							text = _getText();
+							if (text && handler) {
+								handler(text);
+							}
+							self._ignoreFocus = false;
 						}, 0);
 						return null;
 					} else {
 						/* no event and no clipboard permission, paste can't be performed */
 						this.focus();
+						this._ignoreFocus = false;
 						return "";
 					}
 				}
 				this.focus();
-				return _getText();
+				this._ignoreFocus = false;
+				text = _getText();
+				if (text && handler) {
+					handler(text);
+				}
+				return text;
 			}
 			//webkit
 			if (event && event.clipboardData) {
@@ -5709,7 +6912,11 @@ orion.textview.TextView = (function() {
 				clipboadText = [];
 				text = event.clipboardData.getData("text/plain");
 				this._convertDelimiter(text, function(t) {clipboadText.push(t);}, function() {clipboadText.push(delimiter);});
-				return clipboadText.join("");
+				text = clipboadText.join("");
+				if (text && handler) {
+					handler(text);
+				}
+				return text;
 			} else {
 				//TODO try paste using extension (Chrome only)
 			}
@@ -5749,8 +6956,8 @@ orion.textview.TextView = (function() {
 			newRange.selectNode(element);
 
 			var selection = window.getSelection();
-			var oldRanges = [];
-			for (var i = 0; i < selection.rangeCount; i++) {
+			var oldRanges = [], i;
+			for (i = 0; i < selection.rangeCount; i++) {
 				oldRanges.push(selection.getRangeAt(i));
 			}
 
@@ -5761,7 +6968,7 @@ orion.textview.TextView = (function() {
 			var text = selection.toString();
 
 			selection.removeAllRanges();
-			for (var i = 0; i < oldRanges.length; i++) {
+			for (i = 0; i < oldRanges.length; i++) {
 				selection.addRange(oldRanges[i]);
 			}
 
@@ -6217,7 +7424,8 @@ orion.textview.TextView = (function() {
 					self._onModelChanged(modelChangedEvent);
 				}
 			};
-			this._model.addListener(this._modelListener);
+			this._model.addEventListener("Changing", this._modelListener.onChanging);
+			this._model.addEventListener("Changed", this._modelListener.onChanged);
 			
 			var clientDiv = this._clientDiv;
 			var viewDiv = this._viewDiv;
@@ -6242,7 +7450,6 @@ orion.textview.TextView = (function() {
 				handlers.push({target: touchDiv, type: "touchend", handler: function(e) { return self._handleTouchEnd(e); }});
 			} else {
 				var topNode = this._overlayDiv || this._clientDiv;
-				var dragNode = this._dragNode || topNode;
 				var grabNode = isIE ? clientDiv : this._frameWindow;
 				handlers.push({target: clientDiv, type: "keydown", handler: function(e) { return self._handleKeyDown(e);}});
 				handlers.push({target: clientDiv, type: "keypress", handler: function(e) { return self._handleKeyPress(e);}});
@@ -6253,16 +7460,19 @@ orion.textview.TextView = (function() {
 				handlers.push({target: clientDiv, type: "cut", handler: function(e) { return self._handleCut(e);}});
 				handlers.push({target: clientDiv, type: "paste", handler: function(e) { return self._handlePaste(e);}});
 				handlers.push({target: clientDiv, type: "mousedown", handler: function(e) { return self._handleMouseDown(e);}});
+				handlers.push({target: clientDiv, type: "mouseover", handler: function(e) { return self._handleMouseOver(e);}});
+				handlers.push({target: clientDiv, type: "mouseout", handler: function(e) { return self._handleMouseOut(e);}});
 				handlers.push({target: grabNode, type: "mouseup", handler: function(e) { return self._handleMouseUp(e);}});
 				handlers.push({target: grabNode, type: "mousemove", handler: function(e) { return self._handleMouseMove(e);}});
 				handlers.push({target: body, type: "mousedown", handler: function(e) { return self._handleBodyMouseDown(e);}});
-				handlers.push({target: dragNode, type: "dragstart", handler: function(e) { return self._handleDragStart(e);}});
-				if (isDnD) {
-					handlers.push({target: dragNode, type: "dragend", handler: function(e) { return self._handleDragEnd(e);}});
-					handlers.push({target: dragNode, type: "dragenter", handler: function(e) { return self._handleDragEnter(e);}});
-				}
-				handlers.push({target: dragNode, type: "dragover", handler: function(e) { return self._handleDragOver(e);}});
-				handlers.push({target: dragNode, type: "drop", handler: function(e) { return self._handleDrop(e);}});
+				handlers.push({target: body, type: "mouseup", handler: function(e) { return self._handleBodyMouseUp(e);}});
+				handlers.push({target: topNode, type: "dragstart", handler: function(e) { return self._handleDragStart(e);}});
+				handlers.push({target: topNode, type: "drag", handler: function(e) { return self._handleDrag(e);}});
+				handlers.push({target: topNode, type: "dragend", handler: function(e) { return self._handleDragEnd(e);}});
+				handlers.push({target: topNode, type: "dragenter", handler: function(e) { return self._handleDragEnter(e);}});
+				handlers.push({target: topNode, type: "dragover", handler: function(e) { return self._handleDragOver(e);}});
+				handlers.push({target: topNode, type: "dragleave", handler: function(e) { return self._handleDragLeave(e);}});
+				handlers.push({target: topNode, type: "drop", handler: function(e) { return self._handleDrop(e);}});
 				if (isChrome) {
 					handlers.push({target: this._parentDocument, type: "mousemove", handler: function(e) { return self._handleMouseMove(e);}});
 					handlers.push({target: this._parentDocument, type: "mouseup", handler: function(e) { return self._handleMouseUp(e);}});
@@ -6282,6 +7492,8 @@ orion.textview.TextView = (function() {
 				}
 				if (this._overlayDiv) {
 					handlers.push({target: this._overlayDiv, type: "mousedown", handler: function(e) { return self._handleMouseDown(e);}});
+					handlers.push({target: this._overlayDiv, type: "mouseover", handler: function(e) { return self._handleMouseOver(e);}});
+					handlers.push({target: this._overlayDiv, type: "mouseout", handler: function(e) { return self._handleMouseOut(e);}});
 					handlers.push({target: this._overlayDiv, type: "contextmenu", handler: function(e) { return self._handleContextMenu(e); }});
 				}
 				if (!isW3CEvents) {
@@ -6299,32 +7511,30 @@ orion.textview.TextView = (function() {
 				parent = window.document.getElementById(parent);
 			}
 			if (!parent) { throw "no parent"; }
-			this._parent = parent;
-			this._model = options.model ? options.model : new orion.textview.TextModel();
-			this.readonly = options.readonly === true;
-			this._fullSelection = options.fullSelection === undefined || options.fullSelection;
-			/* 
-			* Bug in IE 8. For some reason, during scrolling IE does not reflow the elements
-			* that are used to compute the location for the selection divs. This causes the
-			* divs to be placed at the wrong location. The fix is to disabled full selection for IE8.
-			*/
-			if (isIE < 9) {
-				this._fullSelection = false;
+			options.parent = parent;
+			options.model = options.model || new mTextModel.TextModel();
+			var defaultOptions = this._defaultOptions();
+			for (var option in defaultOptions) {
+				if (defaultOptions.hasOwnProperty(option)) {
+					var value;
+					if (options[option] !== undefined) {
+						value = options[option];
+					} else {
+						value = defaultOptions[option].value;
+					}
+					this["_" + option] = value;
+				}
 			}
-			this._stylesheet = options.stylesheet;
-			this._tabSize = options.tabSize;
 			this._rulers = [];
 			this._selection = new Selection (0, 0, false);
 			this._linksVisible = false;
-			this._eventTable = new EventTable();
 			this._redrawCount = 0;
 			this._maxLineWidth = 0;
 			this._maxLineIndex = -1;
 			this._ignoreSelect = true;
+			this._ignoreFocus = false;
 			this._columnX = -1;
-			
-			this._dragStartSelection = null;
-			this._dropDestination = null;
+			this._dragOffset = -1;
 
 			/* Auto scroll */
 			this._autoScrollX = null;
@@ -6357,10 +7567,10 @@ orion.textview.TextView = (function() {
 			this._createFrame();
 		},
 		_modifyContent: function(e, updateCaret) {
-			if (this.readonly && !e._code) {
+			if (this._readonly && !e._code) {
 				return;
 			}
-
+			e.type = "Verify";
 			this.onVerify(e);
 
 			if (e.text === null || e.text === undefined) { return; }
@@ -6378,10 +7588,12 @@ orion.textview.TextView = (function() {
 				selection.setCaret(e.start + e.text.length);
 				this._setSelection(selection, true);
 			}
-			this.onModify({});
+			this.onModify({type: "Modify"});
 		},
 		_onModelChanged: function(modelChangedEvent) {
+			modelChangedEvent.type = "ModelChanged";
 			this.onModelChanged(modelChangedEvent);
+			modelChangedEvent.type = "Changed";
 			var start = modelChangedEvent.start;
 			var addedCharCount = modelChangedEvent.addedCharCount;
 			var removedCharCount = modelChangedEvent.removedCharCount;
@@ -6406,7 +7618,14 @@ orion.textview.TextView = (function() {
 			while (child) {
 				var lineIndex = child.lineIndex;
 				if (startLine <= lineIndex && lineIndex <= startLine + removedLineCount) {
-					child.lineChanged = true;
+					if (startLine === lineIndex && !child.modelChangedEvent && !child.lineRemoved) {
+						child.modelChangedEvent = modelChangedEvent;
+						child.lineChanged = true;
+					} else {
+						child.lineRemoved = true;
+						child.lineChanged = false;
+						child.modelChangedEvent = null;
+					}
 				}
 				if (lineIndex > startLine + removedLineCount) {
 					child.lineIndex = lineIndex + addedLineCount - removedLineCount;
@@ -6421,7 +7640,9 @@ orion.textview.TextView = (function() {
 			this._updatePage();
 		},
 		_onModelChanging: function(modelChangingEvent) {
+			modelChangingEvent.type = "ModelChanging";
 			this.onModelChanging(modelChangingEvent);
+			modelChangingEvent.type = "Changing";
 		},
 		_queueUpdatePage: function() {
 			if (this._updateTimer) { return; }
@@ -6447,7 +7668,7 @@ orion.textview.TextView = (function() {
 			if (clientDiv) {
 				var child = clientDiv.firstChild;
 				while (child) {
-					child.lineChanged = true;
+					child.lineRemoved = true;
 					child = child.nextSibling;
 				}
 				/*
@@ -6456,11 +7677,13 @@ orion.textview.TextView = (function() {
 				* force the clientDiv to loose and receive focus if the it is focused.
 				*/
 				if (isFirefox) {
+					this._ignoreFocus = false;
 					var hasFocus = this._hasFocus;
 					if (hasFocus) { clientDiv.blur(); }
 					clientDiv.contentEditable = false;
 					clientDiv.contentEditable = true;
 					if (hasFocus) { clientDiv.focus(); }
+					this._ignoreFocus = false;
 				}
 			}
 		},
@@ -6505,7 +7728,7 @@ orion.textview.TextView = (function() {
 			/*
 			* Scrolling is done only by setting the scrollLeft and scrollTop fields in the
 			* view div. This causes an updatePage from the scroll event. In some browsers 
-			* this event is asynchromous and forcing update page to run synchronously
+			* this event is asynchronous and forcing update page to run synchronously
 			* leads to redraw problems. 
 			* On Chrome 11, the view redrawing at times when holding PageDown/PageUp key.
 			* On Firefox 4 for Linux, the view redraws the first page when holding 
@@ -6720,6 +7943,12 @@ orion.textview.TextView = (function() {
 					var right = clientRect.right;
 					var top = viewRect.top + viewPad.top;
 					var bottom = clientRect.bottom;
+					var hd = 0, vd = 0;
+					if (this._clipDiv) {
+						var clipRect = this._clipDiv.getBoundingClientRect();
+						hd = clipRect.left - this._clipDiv.scrollLeft;
+						vd = clipRect.top;
+					}
 					var r;
 					var endLineBounds = this._getLineBoundingClientRect(endNode);
 					if (endOffset === 0) {
@@ -6738,8 +7967,8 @@ orion.textview.TextView = (function() {
 					var sel1Top = Math.min(bottom, Math.max(top, startLineBounds.top));
 					var sel1Right = right;
 					var sel1Bottom = Math.min(bottom, Math.max(top, startLineBounds.bottom));
-					sel1Div.style.left = sel1Left + "px";
-					sel1Div.style.top = sel1Top + "px";
+					sel1Div.style.left = (sel1Left - hd) + "px";
+					sel1Div.style.top = (sel1Top - vd) + "px";
 					sel1Div.style.width = Math.max(0, sel1Right - sel1Left) + "px";
 					sel1Div.style.height = Math.max(0, sel1Bottom - sel1Top) + (isPad ? 1 : 0) + "px";
 					if (isPad) {
@@ -6758,8 +7987,8 @@ orion.textview.TextView = (function() {
 						var sel3Right = Math.min(right, Math.max(left, r));
 						var sel3Bottom = Math.min(bottom, Math.max(top, endLineBounds.bottom));
 						var sel3Div = this._selDiv3;
-						sel3Div.style.left = sel3Left + "px";
-						sel3Div.style.top = sel3Top + "px";
+						sel3Div.style.left = (sel3Left - hd) + "px";
+						sel3Div.style.top = (sel3Top - vd) + "px";
 						sel3Div.style.width = Math.max(0, sel3Right - sel3Left - handleWidth) + "px";
 						sel3Div.style.height = Math.max(0, sel3Bottom - sel3Top) + "px";
 						if (isPad) {
@@ -6767,8 +7996,8 @@ orion.textview.TextView = (function() {
 						}
 						if (sel3Top - sel1Bottom > 0) {
 							var sel2Div = this._selDiv2;
-							sel2Div.style.left = left + "px";
-							sel2Div.style.top = sel1Bottom + "px";
+							sel2Div.style.left = (left - hd)  + "px";
+							sel2Div.style.top = (sel1Bottom - vd) + "px";
 							sel2Div.style.width = Math.max(0, right - left) + "px";
 							sel2Div.style.height = Math.max(0, sel3Top - sel1Bottom) + (isPad ? 1 : 0) + "px";
 						}
@@ -6815,7 +8044,7 @@ orion.textview.TextView = (function() {
 						var next = lineChild.nextSibling;
 						var style = lineChild.viewStyle;
 						if (style && style.tagName === "A") {
-							line.replaceChild(this._createRange(line, document, lineChild.firstChild.data, style), lineChild);
+							line.replaceChild(this._createSpan(line, document, lineChild.firstChild.data, style), lineChild);
 						}
 						lineChild = next;
 					}
@@ -6831,6 +8060,7 @@ orion.textview.TextView = (function() {
 				if (!oldSelection.equals(selection)) {
 					this._selection = selection;
 					var e = {
+						type: "Selection",
 						oldValue: {start:oldSelection.start, end:oldSelection.end},
 						newValue: {start:selection.start, end:selection.end}
 					};
@@ -6853,12 +8083,18 @@ orion.textview.TextView = (function() {
 				if (update) { this._updateDOMSelection(); }
 			}
 		},
-		_setSelectionTo: function (x,y,extent) {
+		_setSelectionTo: function (x, y, extent, drag) {
 			var model = this._model, offset;
 			var selection = this._getSelection();
 			var lineIndex = this._getYToLine(y);
 			if (this._clickCount === 1) {
 				offset = this._getXToOffset(lineIndex, x);
+				if (drag && !extent) {
+					if (selection.start <= offset && offset < selection.end) {
+						this._dragOffset = offset;
+						return false;
+					}
+				}
 				selection.extend(offset);
 				if (!extent) { selection.collapse(); }
 			} else {
@@ -6897,6 +8133,155 @@ orion.textview.TextView = (function() {
 				selection.extend(end);
 			} 
 			this._setSelection(selection, true, true);
+			return true;
+		},
+		_setFullSelection: function(fullSelection, init) {
+			this._fullSelection = fullSelection;
+			
+			/* 
+			* Bug in IE 8. For some reason, during scrolling IE does not reflow the elements
+			* that are used to compute the location for the selection divs. This causes the
+			* divs to be placed at the wrong location. The fix is to disabled full selection for IE8.
+			*/
+			if (isIE < 9) {
+				this._fullSelection = false;
+			}
+			if (isWebkit) {
+				this._fullSelection = true;
+			}
+			var parent = this._clipDiv || this._scrollDiv;
+			if (!parent) {
+				return;
+			}
+			if (!isPad && !this._fullSelection) {
+				if (this._selDiv1) {
+					parent.removeChild(this._selDiv1);
+					this._selDiv1 = null;
+				}
+				if (this._selDiv2) {
+					parent.removeChild(this._selDiv2);
+					this._selDiv2 = null;
+				}
+				if (this._selDiv3) {
+					parent.removeChild(this._selDiv3);
+					this._selDiv3 = null;
+				}
+				return;
+			}
+			
+			if (!this._selDiv1 && (isPad || (this._fullSelection && !isWebkit))) {
+				var frameDocument = this._frameDocument;
+				this._hightlightRGB = "Highlight";
+				var selDiv1 = frameDocument.createElement("DIV");
+				this._selDiv1 = selDiv1;
+				selDiv1.id = "selDiv1";
+				selDiv1.style.position = this._clipDiv ? "absolute" : "fixed";
+				selDiv1.style.borderWidth = "0px";
+				selDiv1.style.margin = "0px";
+				selDiv1.style.padding = "0px";
+				selDiv1.style.MozOutline = "none";
+				selDiv1.style.outline = "none";
+				selDiv1.style.background = this._hightlightRGB;
+				selDiv1.style.width = "0px";
+				selDiv1.style.height = "0px";
+				selDiv1.style.zIndex = "0";
+				parent.appendChild(selDiv1);
+				var selDiv2 = frameDocument.createElement("DIV");
+				this._selDiv2 = selDiv2;
+				selDiv2.id = "selDiv2";
+				selDiv2.style.position = this._clipDiv ? "absolute" : "fixed";
+				selDiv2.style.borderWidth = "0px";
+				selDiv2.style.margin = "0px";
+				selDiv2.style.padding = "0px";
+				selDiv2.style.MozOutline = "none";
+				selDiv2.style.outline = "none";
+				selDiv2.style.background = this._hightlightRGB;
+				selDiv2.style.width = "0px";
+				selDiv2.style.height = "0px";
+				selDiv2.style.zIndex = "0";
+				parent.appendChild(selDiv2);
+				var selDiv3 = frameDocument.createElement("DIV");
+				this._selDiv3 = selDiv3;
+				selDiv3.id = "selDiv3";
+				selDiv3.style.position = this._clipDiv ? "absolute" : "fixed";
+				selDiv3.style.borderWidth = "0px";
+				selDiv3.style.margin = "0px";
+				selDiv3.style.padding = "0px";
+				selDiv3.style.MozOutline = "none";
+				selDiv3.style.outline = "none";
+				selDiv3.style.background = this._hightlightRGB;
+				selDiv3.style.width = "0px";
+				selDiv3.style.height = "0px";
+				selDiv3.style.zIndex = "0";
+				parent.appendChild(selDiv3);
+				
+				/*
+				* Bug in Firefox. The Highlight color is mapped to list selection
+				* background instead of the text selection background.  The fix
+				* is to map known colors using a table or fallback to light blue.
+				*/
+				if (isFirefox && isMac) {
+					var style = this._frameWindow.getComputedStyle(selDiv3, null);
+					var rgb = style.getPropertyValue("background-color");
+					switch (rgb) {
+						case "rgb(119, 141, 168)": rgb = "rgb(199, 208, 218)"; break;
+						case "rgb(127, 127, 127)": rgb = "rgb(198, 198, 198)"; break;
+						case "rgb(255, 193, 31)": rgb = "rgb(250, 236, 115)"; break;
+						case "rgb(243, 70, 72)": rgb = "rgb(255, 176, 139)"; break;
+						case "rgb(255, 138, 34)": rgb = "rgb(255, 209, 129)"; break;
+						case "rgb(102, 197, 71)": rgb = "rgb(194, 249, 144)"; break;
+						case "rgb(140, 78, 184)": rgb = "rgb(232, 184, 255)"; break;
+						default: rgb = "rgb(180, 213, 255)"; break;
+					}
+					this._hightlightRGB = rgb;
+					selDiv1.style.background = rgb;
+					selDiv2.style.background = rgb;
+					selDiv3.style.background = rgb;
+					if (!this._insertedSelRule) {
+						var styleSheet = frameDocument.styleSheets[0];
+						styleSheet.insertRule("::-moz-selection {background: " + rgb + "; }", 0);
+						this._insertedSelRule = true;
+					}
+				}
+				if (!init) {
+					this._updateDOMSelection();
+				}
+			}
+		},
+		_setTabSize: function (tabSize, init) {
+			this._tabSize = tabSize;
+			this._customTabSize = undefined;
+			var clientDiv = this._clientDiv;
+			if (isOpera) {
+				if (clientDiv) { clientDiv.style.OTabSize = this._tabSize+""; }
+			} else if (isFirefox >= 4) {
+				if (clientDiv) {  clientDiv.style.MozTabSize = this._tabSize+""; }
+			} else if (this._tabSize !== 8) {
+				this._customTabSize = this._tabSize;
+				if (!init) {
+					this.redrawLines();
+				}
+			}
+		},
+		_setThemeClass: function (themeClass, init) {
+			this._themeClass = themeClass;
+			var document = this._frameDocument;
+			if (document) {
+				var viewContainerClass = "viewContainer";
+				if (this._themeClass) { viewContainerClass += " " + this._themeClass; }
+				document.body.className = viewContainerClass;
+				if (!init) {
+					if (isIE) {
+						document.body.style.lineHeight = "normal";
+					}
+					this._lineHeight = this._calculateLineHeight();
+					this._viewPadding = this._calculatePadding();
+					if (isIE) {
+						document.body.style.lineHeight = this._lineHeight + "px";
+					}
+					this.redraw();
+				}
+			}
 		},
 		_showCaret: function (allSelection, pageScroll) {
 			if (!this._clientDiv) { return; }
@@ -6997,7 +8382,8 @@ orion.textview.TextView = (function() {
 			this._imeOffset = selection.start;
 		},
 		_unhookEvents: function() {
-			this._model.removeListener(this._modelListener);
+			this._model.removeEventListener("Changing", this._modelListener.onChanging);
+			this._model.removeEventListener("Changed", this._modelListener.onChanged);
 			this._modelListener = null;
 			for (var i=0; i 0) { return; }
-			if (this._updateTimer) { 
+			if (this._updateTimer) {
 				clearTimeout(this._updateTimer);
 				this._updateTimer = null;
+				hScrollOnly = false;
 			}
-			var document = this._frameDocument;
-			var viewDiv = this._viewDiv;
 			var clientDiv = this._clientDiv;
 			if (!clientDiv) { return; }
-			var frameWidth = this._getFrameWidth();
-			var frameHeight = this._getFrameHeight();
-			document.body.style.width = frameWidth + "px";
-			document.body.style.height = frameHeight + "px";
-			var viewPad = this._getViewPadding();
-			
-			/* Update view height in order to have client height computed */
-			viewDiv.style.height = Math.max(0, (frameHeight - viewPad.top - viewPad.bottom)) + "px";
-			
 			var model = this._model;
+			var scroll = this._getScroll();
+			var viewPad = this._getViewPadding();
+			var lineCount = model.getLineCount();
 			var lineHeight = this._getLineHeight();
-			var scrollY = this._getScroll().y;
-			var firstLine = Math.max(0, scrollY) / lineHeight;
+			var firstLine = Math.max(0, scroll.y) / lineHeight;
 			var topIndex = Math.floor(firstLine);
 			var lineStart = Math.max(0, topIndex - 1);
 			var top = Math.round((firstLine - lineStart) * lineHeight);
-			var lineCount = model.getLineCount();
-			var clientHeight = this._getClientHeight();
-			var partialY = Math.round((firstLine - topIndex) * lineHeight);
-			var linesPerPage = Math.floor((clientHeight + partialY) / lineHeight);
-			var bottomIndex = Math.min(topIndex + linesPerPage, lineCount - 1);
-			var lineEnd = Math.min(bottomIndex + 1, lineCount - 1);
-			this._partialY = partialY;
-			
-			var lineIndex, lineWidth;
-			var child = clientDiv.firstChild;
-			while (child) {
-				lineIndex = child.lineIndex;
-				var nextChild = child.nextSibling;
-				if (!(lineStart <= lineIndex && lineIndex <= lineEnd) || child.lineChanged || child.lineIndex === -1) {
-					if (this._mouseWheelLine === child) {
-						child.style.display = "none";
-						child.lineIndex = -1;
+			var partialY = this._partialY = Math.round((firstLine - topIndex) * lineHeight);
+			var scrollWidth, scrollHeight = lineCount * lineHeight;
+			var leftWidth, clientWidth, clientHeight;
+			if (hScrollOnly) {
+				clientWidth = this._getClientWidth();
+				clientHeight = this._getClientHeight();
+				leftWidth = this._leftDiv ? this._leftDiv.scrollWidth : 0;
+				scrollWidth = Math.max(this._maxLineWidth, clientWidth);
+			} else {
+				var document = this._frameDocument;
+				var frameWidth = this._getFrameWidth();
+				var frameHeight = this._getFrameHeight();
+				document.body.style.width = frameWidth + "px";
+				document.body.style.height = frameHeight + "px";
+
+				/* Update view height in order to have client height computed */
+				var viewDiv = this._viewDiv;
+				viewDiv.style.height = Math.max(0, (frameHeight - viewPad.top - viewPad.bottom)) + "px";
+				clientHeight = this._getClientHeight();
+				var linesPerPage = Math.floor((clientHeight + partialY) / lineHeight);
+				var bottomIndex = Math.min(topIndex + linesPerPage, lineCount - 1);
+				var lineEnd = Math.min(bottomIndex + 1, lineCount - 1);
+				
+				var lineIndex, lineWidth;
+				var child = clientDiv.firstChild;
+				while (child) {
+					lineIndex = child.lineIndex;
+					var nextChild = child.nextSibling;
+					if (!(lineStart <= lineIndex && lineIndex <= lineEnd) || child.lineRemoved || child.lineIndex === -1) {
+						if (this._mouseWheelLine === child) {
+							child.style.display = "none";
+							child.lineIndex = -1;
+						} else {
+							clientDiv.removeChild(child);
+						}
+					}
+					child = nextChild;
+				}
+	
+				child = this._getLineNext();
+				var frag = document.createDocumentFragment();
+				for (lineIndex=lineStart; lineIndex<=lineEnd; lineIndex++) {
+					if (!child || child.lineIndex > lineIndex) {
+						this._createLine(frag, null, document, lineIndex, model);
 					} else {
-						clientDiv.removeChild(child);
+						if (frag.firstChild) {
+							clientDiv.insertBefore(frag, child);
+							frag = document.createDocumentFragment();
+						}
+						if (child && child.lineChanged) {
+							child = this._createLine(frag, child, document, lineIndex, model);
+							child.lineChanged = false;
+						}
+						child = this._getLineNext(child);
 					}
 				}
-				child = nextChild;
-			}
-
-			child = this._getLineNext();
-			var frag = document.createDocumentFragment();
-			for (lineIndex=lineStart; lineIndex<=lineEnd; lineIndex++) {
-				if (!child || child.lineIndex > lineIndex) {
-					this._createLine(frag, null, document, lineIndex, model);
-				} else {
-					if (frag.firstChild) {
-						clientDiv.insertBefore(frag, child);
-						frag = document.createDocumentFragment();
+				if (frag.firstChild) { clientDiv.insertBefore(frag, child); }
+	
+				/*
+				* Feature in WekKit. Webkit limits the width of the lines
+				* computed below to the width of the client div.  This causes
+				* the lines to be wrapped even though "pre" is set.  The fix
+				* is to set the width of the client div to a larger number
+				* before computing the lines width.  Note that this value is
+				* reset to the appropriate value further down.
+				*/ 
+				if (isWebkit) {
+					clientDiv.style.width = (0x7FFFF).toString() + "px";
+				}
+	
+				var rect;
+				child = this._getLineNext();
+				while (child) {
+					lineWidth = child.lineWidth;
+					if (lineWidth === undefined) {
+						rect = this._getLineBoundingClientRect(child);
+						lineWidth = child.lineWidth = rect.right - rect.left;
 					}
-					child = this._getLineNext(child);
-				}
-			}
-			if (frag.firstChild) { clientDiv.insertBefore(frag, child); }
-
-			/*
-			* Feature in WekKit. Webkit limits the width of the lines
-			* computed below to the width of the client div.  This causes
-			* the lines to be wrapped even though "pre" is set.  The fix
-			* is to set the width of the client div to a larger number
-			* before computing the lines width.  Note that this value is
-			* reset to the appropriate value further down.
-			*/ 
-			if (isWebkit) {
-				clientDiv.style.width = (0x7FFFF).toString() + "px";
-			}
-
-			var rect;
-			child = this._getLineNext();
-			while (child) {
-				lineWidth = child.lineWidth;
-				if (lineWidth === undefined) {
-					rect = this._getLineBoundingClientRect(child);
-					lineWidth = child.lineWidth = rect.right - rect.left;
-				}
-				if (lineWidth >= this._maxLineWidth) {
-					this._maxLineWidth = lineWidth;
-					this._maxLineIndex = child.lineIndex;
-				}
-				if (child.lineIndex === topIndex) { this._topChild = child; }
-				if (child.lineIndex === bottomIndex) { this._bottomChild = child; }
-				if (this._checkMaxLineIndex === child.lineIndex) { this._checkMaxLineIndex = -1; }
-				child = this._getLineNext(child);
-			}
-			if (this._checkMaxLineIndex !== -1) {
-				lineIndex = this._checkMaxLineIndex;
-				this._checkMaxLineIndex = -1;
-				if (0 <= lineIndex && lineIndex < lineCount) {
-					var dummy = this._createLine(clientDiv, null, document, lineIndex, model);
-					rect = this._getLineBoundingClientRect(dummy);
-					lineWidth = rect.right - rect.left;
 					if (lineWidth >= this._maxLineWidth) {
 						this._maxLineWidth = lineWidth;
-						this._maxLineIndex = lineIndex;
+						this._maxLineIndex = child.lineIndex;
 					}
-					clientDiv.removeChild(dummy);
+					if (child.lineIndex === topIndex) { this._topChild = child; }
+					if (child.lineIndex === bottomIndex) { this._bottomChild = child; }
+					if (this._checkMaxLineIndex === child.lineIndex) { this._checkMaxLineIndex = -1; }
+					child = this._getLineNext(child);
 				}
-			}
-
-			// Update rulers
-			this._updateRuler(this._leftDiv, topIndex, bottomIndex);
-			this._updateRuler(this._rightDiv, topIndex, bottomIndex);
-			
-			var leftWidth = this._leftDiv ? this._leftDiv.scrollWidth : 0;
-			var rightWidth = this._rightDiv ? this._rightDiv.scrollWidth : 0;
-			viewDiv.style.left = leftWidth + "px";
-			viewDiv.style.width = Math.max(0, frameWidth - leftWidth - rightWidth - viewPad.left - viewPad.right) + "px";
-			if (this._rightDiv) {
-				this._rightDiv.style.left = (frameWidth - rightWidth) + "px"; 
-			}
-			
-			var scrollDiv = this._scrollDiv;
-			/* Need to set the height first in order for the width to consider the vertical scrollbar */
-			var scrollHeight = lineCount * lineHeight;
-			scrollDiv.style.height = scrollHeight + "px";
-			// TODO if frameHeightWithoutHScrollbar < scrollHeight  < frameHeightWithHScrollbar and the horizontal bar is visible, 
-			// then the clientWidth is wrong because the vertical scrollbar is showing. To correct code should hide both scrollbars 
-			// at this point.
-			var clientWidth = this._getClientWidth();
-			var width = Math.max(this._maxLineWidth, clientWidth);
-			/*
-			* Except by IE 8 and earlier, all other browsers are not allocating enough space for the right padding 
-			* in the scrollbar. It is possible this a bug since all other paddings are considered.
-			*/
-			var scrollWidth = width;
-			if (!isIE || isIE >= 9) { width += viewPad.right; }
-			scrollDiv.style.width = width + "px";
-
-			// Get the left scroll after setting the width of the scrollDiv as this can change the horizontal scroll offset.
-			var scroll = this._getScroll();
-			var left = scroll.x;
-			var clipLeft = left;
-			var clipTop = top;
-			var clipRight = left + clientWidth;
-			var clipBottom = top + clientHeight;
-			if (clipLeft === 0) { clipLeft -= viewPad.left; }
-			if (clipTop === 0) { clipTop -= viewPad.top; }
-			if (clipRight === scrollWidth) { clipRight += viewPad.right; }
-			if (scroll.y + clientHeight === scrollHeight) { clipBottom += viewPad.bottom; }
-			clientDiv.style.clip = "rect(" + clipTop + "px," + clipRight + "px," + clipBottom + "px," + clipLeft + "px)";
-			clientDiv.style.left = (-left + leftWidth + viewPad.left) + "px";
-			clientDiv.style.top = (-top + viewPad.top) + "px";
-			clientDiv.style.width = (isWebkit ? scrollWidth : clientWidth + left) + "px";
-			clientDiv.style.height = (clientHeight + top) + "px";
-			var overlayDiv = this._overlayDiv;
-			if (overlayDiv) {
-				overlayDiv.style.clip = clientDiv.style.clip;
-				overlayDiv.style.left = clientDiv.style.left;
-				overlayDiv.style.top = clientDiv.style.top;
-				overlayDiv.style.width = clientDiv.style.width;
-				overlayDiv.style.height = clientDiv.style.height;
-			}
-			function _updateRulerSize(divRuler) {
-				if (!divRuler) { return; }
+				if (this._checkMaxLineIndex !== -1) {
+					lineIndex = this._checkMaxLineIndex;
+					this._checkMaxLineIndex = -1;
+					if (0 <= lineIndex && lineIndex < lineCount) {
+						var dummy = this._createLine(clientDiv, null, document, lineIndex, model);
+						rect = this._getLineBoundingClientRect(dummy);
+						lineWidth = rect.right - rect.left;
+						if (lineWidth >= this._maxLineWidth) {
+							this._maxLineWidth = lineWidth;
+							this._maxLineIndex = lineIndex;
+						}
+						clientDiv.removeChild(dummy);
+					}
+				}
+	
+				// Update rulers
+				this._updateRuler(this._leftDiv, topIndex, bottomIndex);
+				this._updateRuler(this._rightDiv, topIndex, bottomIndex);
+				
+				leftWidth = this._leftDiv ? this._leftDiv.scrollWidth : 0;
+				var rightWidth = this._rightDiv ? this._rightDiv.scrollWidth : 0;
+				viewDiv.style.left = leftWidth + "px";
+				viewDiv.style.width = Math.max(0, frameWidth - leftWidth - rightWidth - viewPad.left - viewPad.right) + "px";
+				if (this._rightDiv) {
+					this._rightDiv.style.left = (frameWidth - rightWidth) + "px"; 
+				}
+				/* Need to set the height first in order for the width to consider the vertical scrollbar */
+				var scrollDiv = this._scrollDiv;
+				scrollDiv.style.height = scrollHeight + "px";
+				/*
+				* TODO if frameHeightWithoutHScrollbar < scrollHeight  < frameHeightWithHScrollbar and the horizontal bar is visible, 
+				* then the clientWidth is wrong because the vertical scrollbar is showing. To correct code should hide both scrollbars 
+				* at this point.
+				*/
+				clientWidth = this._getClientWidth();
+				var width = Math.max(this._maxLineWidth, clientWidth);
+				/*
+				* Except by IE 8 and earlier, all other browsers are not allocating enough space for the right padding 
+				* in the scrollbar. It is possible this a bug since all other paddings are considered.
+				*/
+				scrollWidth = width;
+				if (!isIE || isIE >= 9) { width += viewPad.right; }
+				scrollDiv.style.width = width + "px";
+				if (this._clipScrollDiv) {
+					this._clipScrollDiv.style.width = width + "px";
+				}
+				/* Get the left scroll after setting the width of the scrollDiv as this can change the horizontal scroll offset. */
+				scroll = this._getScroll();
 				var rulerHeight = clientHeight + viewPad.top + viewPad.bottom;
-				var cells = divRuler.firstChild.rows[0].cells;
-				for (var i = 0; i < cells.length; i++) {
-					var div = cells[i].firstChild;
-					var offset = lineHeight;
-					if (div._ruler.getOverview() === "page") { offset += partialY; }
-					div.style.top = -offset + "px";
-					div.style.height = (rulerHeight + offset) + "px";
-					div = div.nextSibling;
-				}
-				divRuler.style.height = rulerHeight + "px";
+				this._updateRulerSize(this._leftDiv, rulerHeight);
+				this._updateRulerSize(this._rightDiv, rulerHeight);
 			}
-			_updateRulerSize(this._leftDiv);
-			_updateRulerSize(this._rightDiv);
-			if (isPad) {
-				var self = this;
-				setTimeout(function() {self._resizeTouchDiv();}, 0);
+			var left = scroll.x;	
+			var clipDiv = this._clipDiv;
+			var overlayDiv = this._overlayDiv;
+			var clipLeft, clipTop;
+			if (clipDiv) {
+				clipDiv.scrollLeft = left;			
+				clipLeft = leftWidth + viewPad.left;
+				clipTop = viewPad.top;
+				var clipWidth = clientWidth;
+				var clipHeight = clientHeight;
+				var clientLeft = 0, clientTop = -top;
+				if (scroll.x === 0) {
+					clipLeft -= viewPad.left;
+					clipWidth += viewPad.left;
+					clientLeft = viewPad.left;
+				} 
+				if (scroll.x + clientWidth === scrollWidth) {
+					clipWidth += viewPad.right;
+				}
+				if (scroll.y === 0) {
+					clipTop -= viewPad.top;
+					clipHeight += viewPad.top;
+					clientTop += viewPad.top;
+				}
+				if (scroll.y + clientHeight === scrollHeight) { 
+					clipHeight += viewPad.bottom; 
+				}
+				clipDiv.style.left = clipLeft + "px";
+				clipDiv.style.top = clipTop + "px";
+				clipDiv.style.width = clipWidth + "px";
+				clipDiv.style.height = clipHeight + "px";
+				clientDiv.style.left = clientLeft + "px";
+				clientDiv.style.top = clientTop + "px";
+				clientDiv.style.width = scrollWidth + "px";
+				clientDiv.style.height = (clientHeight + top) + "px";
+				if (overlayDiv) {
+					overlayDiv.style.left = clientDiv.style.left;
+					overlayDiv.style.top = clientDiv.style.top;
+					overlayDiv.style.width = clientDiv.style.width;
+					overlayDiv.style.height = clientDiv.style.height;
+				}
+			} else {
+				clipLeft = left;
+				clipTop = top;
+				var clipRight = left + clientWidth;
+				var clipBottom = top + clientHeight;
+				if (clipLeft === 0) { clipLeft -= viewPad.left; }
+				if (clipTop === 0) { clipTop -= viewPad.top; }
+				if (clipRight === scrollWidth) { clipRight += viewPad.right; }
+				if (scroll.y + clientHeight === scrollHeight) { clipBottom += viewPad.bottom; }
+				clientDiv.style.clip = "rect(" + clipTop + "px," + clipRight + "px," + clipBottom + "px," + clipLeft + "px)";
+				clientDiv.style.left = (-left + leftWidth + viewPad.left) + "px";
+				clientDiv.style.width = (isWebkit ? scrollWidth : clientWidth + left) + "px";
+				if (!hScrollOnly) {
+					clientDiv.style.top = (-top + viewPad.top) + "px";
+					clientDiv.style.height = (clientHeight + top) + "px";
+				}
+				if (overlayDiv) {
+					overlayDiv.style.clip = clientDiv.style.clip;
+					overlayDiv.style.left = clientDiv.style.left;
+					overlayDiv.style.width = clientDiv.style.width;
+					if (!hScrollOnly) {
+						overlayDiv.style.top = clientDiv.style.top;
+						overlayDiv.style.height = clientDiv.style.height;
+					}
+				}
 			}
 			this._updateDOMSelection();
 
@@ -7243,6 +8671,25 @@ orion.textview.TextView = (function() {
 					this._showCaret();
 				}
 			}
+			if (isPad) {
+				var self = this;
+				setTimeout(function() {self._resizeTouchDiv();}, 0);
+			}
+		},
+		_updateRulerSize: function (divRuler, rulerHeight) {
+			if (!divRuler) { return; }
+			var partialY = this._partialY;
+			var lineHeight = this._getLineHeight();
+			var cells = divRuler.firstChild.rows[0].cells;
+			for (var i = 0; i < cells.length; i++) {
+				var div = cells[i].firstChild;
+				var offset = lineHeight;
+				if (div._ruler.getOverview() === "page") { offset += partialY; }
+				div.style.top = -offset + "px";
+				div.style.height = (rulerHeight + offset) + "px";
+				div = div.nextSibling;
+			}
+			divRuler.style.height = rulerHeight + "px";
 		},
 		_updateRuler: function (divRuler, topIndex, bottomIndex) {
 			if (!divRuler) { return; }
@@ -7371,16 +8818,151 @@ orion.textview.TextView = (function() {
 			}
 		}
 	};//end prototype
+	mEventTarget.EventTarget.addMixin(TextView.prototype);
 	
-	return TextView;
-}());
+	return {TextView: TextView};
+}, "orion/textview");
 
-if (typeof window !== "undefined" && typeof window.define !== "undefined") {
-	define(['orion/textview/textModel', 'orion/textview/keyBinding'], function() {
-		return orion.textview;
-	});
-}
-/******************************************************************************* 
+/*******************************************************************************
+ * @license
+ * Copyright (c) 2010, 2011 IBM Corporation and others.
+ * All rights reserved. This program and the accompanying materials are made 
+ * available under the terms of the Eclipse Public License v1.0 
+ * (http://www.eclipse.org/legal/epl-v10.html), and the Eclipse Distribution 
+ * License v1.0 (http://www.eclipse.org/org/documents/edl-v10.html). 
+ * 
+ * Contributors: 
+ *		Felipe Heidrich (IBM Corporation) - initial API and implementation
+ *		Silenio Quarti (IBM Corporation) - initial API and implementation
+ ******************************************************************************/
+ 
+/*global define */
+
+define([], function() {
+
+	function TextDND(view, undoStack) {
+		this._view = view;
+		this._undoStack = undoStack;
+		this._dragSelection = null;
+		this._dropOffset = -1;
+		this._dropText = null;
+		var self = this;
+		this._listener = {
+			onDragStart: function (evt) {
+				self._onDragStart(evt);
+			},
+			onDragEnd: function (evt) {
+				self._onDragEnd(evt);
+			},
+			onDragEnter: function (evt) {
+				self._onDragEnter(evt);
+			},
+			onDragOver: function (evt) {
+				self._onDragOver(evt);
+			},
+			onDrop: function (evt) {
+				self._onDrop(evt);
+			},
+			onDestroy: function (evt) {
+				self._onDestroy(evt);
+			}
+		};
+		view.addEventListener("DragStart", this._listener.onDragStart);
+		view.addEventListener("DragEnd", this._listener.onDragEnd);
+		view.addEventListener("DragEnter", this._listener.onDragEnter);
+		view.addEventListener("DragOver", this._listener.onDragOver);
+		view.addEventListener("Drop", this._listener.onDrop);
+		view.addEventListener("Destroy", this._listener.onDestroy);
+	}
+	TextDND.prototype = {
+		destroy: function() {
+			var view = this._view;
+			if (!view) { return; }
+			view.removeEventListener("DragStart", this._listener.onDragStart);
+			view.removeEventListener("DragEnd", this._listener.onDragEnd);
+			view.removeEventListener("DragEnter", this._listener.onDragEnter);
+			view.removeEventListener("DragOver", this._listener.onDragOver);
+			view.removeEventListener("Drop", this._listener.onDrop);
+			view.removeEventListener("Destroy", this._listener.onDestroy);
+			this._view = null;
+		},
+		_onDestroy: function(e) {
+			this.destroy();
+		},
+		_onDragStart: function(e) {
+			var view = this._view;
+			var selection = view.getSelection();
+			var model = view.getModel();
+			if (model.getBaseModel) {
+				selection.start = model.mapOffset(selection.start);
+				selection.end = model.mapOffset(selection.end);
+				model = model.getBaseModel();
+			}
+			var text = model.getText(selection.start, selection.end);
+			if (text) {
+				this._dragSelection = selection;
+				e.event.dataTransfer.effectAllowed = "copyMove";
+				e.event.dataTransfer.setData("Text", text);
+			}
+		},
+		_onDragEnd: function(e) {
+			var view = this._view;
+			if (this._dragSelection) {
+				if (this._undoStack) { this._undoStack.startCompoundChange(); }
+				var move = e.event.dataTransfer.dropEffect === "move";
+				if (move) {
+					view.setText("", this._dragSelection.start, this._dragSelection.end);
+				}
+				if (this._dropText) {
+					var text = this._dropText;
+					var offset = this._dropOffset;
+					if (move) {
+						if (offset >= this._dragSelection.end) {
+							offset -= this._dragSelection.end - this._dragSelection.start;
+						} else if (offset >= this._dragSelection.start) {
+							offset = this._dragSelection.start;
+						}
+					}
+					view.setText(text, offset, offset);
+					view.setSelection(offset, offset + text.length);
+					this._dropText = null;
+					this._dropOffset = -1;
+				}
+				if (this._undoStack) { this._undoStack.endCompoundChange(); }
+			}
+			this._dragSelection = null;
+		},
+		_onDragEnter: function(e) {
+			this._onDragOver(e);
+		},
+		_onDragOver: function(e) {
+			var types = e.event.dataTransfer.types;
+			if (types) {
+				var allowed = types.contains ? types.contains("text/plain") : types.indexOf("text/plain") !== -1;
+				if (!allowed) {
+					e.event.dataTransfer.dropEffect = "none";
+				}
+			}
+		},
+		_onDrop: function(e) {
+			var view = this._view;
+			var text = e.event.dataTransfer.getData("Text");
+			if (text) {
+				var offset = view.getOffsetAtLocation(e.x, e.y);
+				if (this._dragSelection) {
+					this._dropOffset = offset;
+					this._dropText = text;
+				} else {
+					view.setText(text, offset, offset);
+					view.setSelection(offset, offset + text.length);
+				}
+			}
+		}
+	};
+
+	return {TextDND: TextDND};
+}, "orion/textview");/******************************************************************************* 
+ * @license
  * Copyright (c) 2011 IBM Corporation and others.
  * All rights reserved. This program and the accompanying materials are made 
  * available under the terms of the Eclipse Public License v1.0 
@@ -7391,104 +8973,100 @@ if (typeof window !== "undefined" && typeof window.define !== "undefined") {
  ******************************************************************************/
 
 /*jslint */
-/*global define orion:true window */
+/*global define */
 
-var orion = orion || {};
+define([], function() {
 
-orion.editor = orion.editor || {};
-
-/**
- * Provides a grammar that can do some very rough syntax highlighting for HTML.
- * @class orion.syntax.HtmlGrammar
- */
-orion.editor.HtmlGrammar = (function() {
-	return {
-		/**
-		 * What kind of highlight provider we are.
-		 * @public
-		 * @type String
-		 */
-		type: "grammar",
-		
-		/**
-		 * The file extensions that we provide rules for.
-		 * @public
-		 * @type String[]
-		 */
-		fileTypes: [ "html", "htm" ],
-		
-		/**
-		 * Object containing the grammar rules.
-		 * @public
-		 * @type Object
-		 */
-		grammar: {
-			"name": "HTML",
-			"scopeName": "source.html",
-			"uuid": "3B5C76FB-EBB5-D930-F40C-047D082CE99B",
-			"patterns": [
-				// TODO unicode?
-				{
-					"match": "]+>",
-					"name": "entity.name.tag.doctype.html"
-				},
-				{
-					"begin": "",
-					"beginCaptures": {
-						"0": { "name": "punctuation.definition.comment.html" }
+	/**
+	 * Provides a grammar that can do some very rough syntax highlighting for HTML.
+	 * @class orion.syntax.HtmlGrammar
+	 */
+	function HtmlGrammar() {
+		return {
+			/**
+			 * What kind of highlight provider we are.
+			 * @public
+			 * @type String
+			 */
+			type: "grammar",
+			
+			/**
+			 * The file extensions that we provide rules for.
+			 * @public
+			 * @type String[]
+			 */
+			fileTypes: [ "html", "htm" ],
+			
+			/**
+			 * Object containing the grammar rules.
+			 * @public
+			 * @type Object
+			 */
+			grammar: {
+				"name": "HTML",
+				"scopeName": "source.html",
+				"uuid": "3B5C76FB-EBB5-D930-F40C-047D082CE99B",
+				"patterns": [
+					// TODO unicode?
+					{
+						"match": "]+>",
+						"name": "entity.name.tag.doctype.html"
 					},
-					"endCaptures": {
-						"0": { "name": "punctuation.definition.comment.html" }
+					{
+						"begin": "",
+						"beginCaptures": {
+							"0": { "name": "punctuation.definition.comment.html" }
+						},
+						"endCaptures": {
+							"0": { "name": "punctuation.definition.comment.html" }
+						},
+						"patterns": [
+							{
+								"match": "--",
+								"name": "invalid.illegal.badcomment.html"
+							}
+						],
+						"contentName": "comment.block.html"
 					},
-					"patterns": [
-						{
-							"match": "--",
-							"name": "invalid.illegal.badcomment.html"
-						}
-					],
-					"contentName": "comment.block.html"
-				},
-				{ // startDelimiter + tagName
-					"match": "<[A-Za-z0-9_\\-:]+(?= ?)",
-					"name": "entity.name.tag.html"
-				},
-				{ "include": "#attrName" },
-				{ "include": "#qString" },
-				{ "include": "#qqString" },
-				// TODO attrName, qString, qqString should be applied first while inside a tag
-				{ // startDelimiter + slash + tagName + endDelimiter
-					"match": "",
-					"name": "entity.name.tag.html"
-				},
-				{ // end delimiter of open tag
-					"match": ">", 
-					"name": "entity.name.tag.html"
-				} ],
-			"repository": {
-				"attrName": { // attribute name
-					"match": "[A-Za-z\\-:]+(?=\\s*=\\s*['\"])",
-					"name": "entity.other.attribute.name.html"
-				},
-				"qqString": { // double quoted string
-					"match": "(\")[^\"]+(\")",
-					"name": "string.quoted.double.html"
-				},
-				"qString": { // single quoted string
-					"match": "(')[^']+(\')",
-					"name": "string.quoted.single.html"
+					{ // startDelimiter + tagName
+						"match": "<[A-Za-z0-9_\\-:]+(?= ?)",
+						"name": "entity.name.tag.html"
+					},
+					{ "include": "#attrName" },
+					{ "include": "#qString" },
+					{ "include": "#qqString" },
+					// TODO attrName, qString, qqString should be applied first while inside a tag
+					{ // startDelimiter + slash + tagName + endDelimiter
+						"match": "",
+						"name": "entity.name.tag.html"
+					},
+					{ // end delimiter of open tag
+						"match": ">", 
+						"name": "entity.name.tag.html"
+					} ],
+				"repository": {
+					"attrName": { // attribute name
+						"match": "[A-Za-z\\-:]+(?=\\s*=\\s*['\"])",
+						"name": "entity.other.attribute.name.html"
+					},
+					"qqString": { // double quoted string
+						"match": "(\")[^\"]+(\")",
+						"name": "string.quoted.double.html"
+					},
+					"qString": { // single quoted string
+						"match": "(')[^']+(\')",
+						"name": "string.quoted.single.html"
+					}
 				}
 			}
-		}
-	};
-}());
+		};
+	}
 
-if (typeof window !== "undefined" && typeof window.define !== "undefined") {
-	define([], function() {
-		return orion.editor;
-	});
-}
+	return {HtmlGrammar: HtmlGrammar};
+}, "orion/editor");
 /******************************************************************************* 
+ * @license
  * Copyright (c) 2011 IBM Corporation and others.
  * All rights reserved. This program and the accompanying materials are made 
  * available under the terms of the Eclipse Public License v1.0 
@@ -7499,85 +9077,11 @@ if (typeof window !== "undefined" && typeof window.define !== "undefined") {
  ******************************************************************************/
 
 /*jslint regexp:false laxbreak:true*/
-/*global define window*/
+/*global define */
 
-var orion = orion || {};
-orion.editor = orion.editor || {};
+define(['orion/editor/regex'], function(mRegex) {
 
-/**
- * @class A styler that does nothing, but can be extended by concrete stylers. To extend, call 
- * {@link orion.editor.AbstractStyler.extend} and provide implementations of one or more of
- * the {@link #_onSelection}, {@link #_onModelChanged}, {@link #_onDestroy} and {@link #_onLineStyle} methods.
- * @name orion.editor.AbstractStyler
- */
-orion.editor.AbstractStyler = (function() {
-	/** @inner */
-	function AbstractStyler() {
-	}
-	AbstractStyler.prototype = /** @lends orion.editor.AbstractStyler.prototype */ {
-		/**
-		 * Initializes this styler with a TextView. If you are extending AbstractStyler,
-		 * you must call this from your subclass's constructor function.
-		 * @param {orion.textview.TextView} textView The TextView to provide styling for.
-		 */
-		initialize: function(textView) {
-			this.textView = textView;
-			
-			textView.addEventListener("Selection", this, this._onSelection);
-			textView.addEventListener("ModelChanged", this, this._onModelChanged);
-			textView.addEventListener("Destroy", this, this._onDestroy);
-			textView.addEventListener("LineStyle", this, this._onLineStyle);
-			textView.redrawLines();
-		},
-		/** Destroys this styler and removes all listeners. Called by the editor. */
-		destroy: function() {
-			if (this.textView) {
-				this.textView.removeEventListener("Selection", this, this._onSelection);
-				this.textView.removeEventListener("ModelChanged", this, this._onModelChanged);
-				this.textView.removeEventListener("Destroy", this, this._onDestroy);
-				this.textView.removeEventListener("LineStyle", this, this._onLineStyle);
-				this.textView = null;
-			}
-		},
-		/** To be overridden by subclass.
-		 * @public
-		 */
-		_onSelection: function(/**eclipse.SelectionEvent*/ selectionEvent) {},
-		/** To be overridden by subclass.
-		 * @public
-		 */
-		_onModelChanged: function(/**eclipse.ModelChangedEvent*/ modelChangedEvent) {},
-		/** To be overridden by subclass.
-		 * @public
-		 */
-		_onDestroy: function(/**eclipse.DestroyEvent*/ destroyEvent) {},
-		/** To be overridden by subclass.
-		 * @public
-		 */
-		_onLineStyle: function(/**eclipse.LineStyleEvent*/ lineStyleEvent) {}
-	};
-	
-	return AbstractStyler;
-}());
-
-/**
- * Helper for extending {@link orion.editor.AbstractStyler}.
- * @methodOf orion.editor.AbstractStyler
- * @static
- * @param {Function} subCtor The constructor function for the subclass.
- * @param {Object} [proto] Object to be mixed into the subclass's prototype. This object should 
- * contain your implementation of _onSelection, _onModelChanged, etc.
- */
-orion.editor.AbstractStyler.extend = function(subCtor, proto) {
-	if (typeof(subCtor) !== "function") { throw new Error("Function expected"); }
-	subCtor.prototype = new orion.editor.AbstractStyler();
-	subCtor.constructor = subCtor;
-	for (var p in proto) {
-		if (proto.hasOwnProperty(p)) { subCtor.prototype[p] = proto[p]; }
-	}
-};
-
-orion.editor.RegexUtil = {
+var RegexUtil = {
 	// Rules to detect some unsupported Oniguruma features
 	unsupported: [
 		{regex: /\(\?[ims\-]:/, func: function(match) { return "option on/off for subexp"; }},
@@ -7639,12 +9143,12 @@ orion.editor.RegexUtil = {
 		var i;
 		
 		// Handle global "x" flag (whitespace/comments)
-		str = orion.editor.RegexUtil.processGlobalFlag("x", str, function(subexp) {
+		str = RegexUtil.processGlobalFlag("x", str, function(subexp) {
 				return normalize(subexp);
 			});
 		
 		// Handle global "i" flag (case-insensitive)
-		str = orion.editor.RegexUtil.processGlobalFlag("i", str, function(subexp) {
+		str = RegexUtil.processGlobalFlag("i", str, function(subexp) {
 				flags += "i";
 				return subexp;
 			});
@@ -7716,7 +9220,7 @@ orion.editor.RegexUtil = {
 			var backrefMatch = /\\(\d+)/.exec(term);
 			if (backrefMatch) {
 				var text = sub[backrefMatch[1]] || "";
-				array.push(escape ? orion.editor.RegexUtil.escapeRegex(text) : text);
+				array.push(escape ? mRegex.escape(text) : text);
 			} else {
 				array.push(term);
 			}
@@ -7724,11 +9228,6 @@ orion.editor.RegexUtil = {
 		return new RegExp(array.join(""));
 	},
 	
-	/** @returns {String} The input string with regex special characters escaped. */
-	escapeRegex: function(/**String*/ str) {
-		return str.replace(/([\\$\^*\/+?\.\(\)|{}\[\]])/g, "\\$&");
-	},
-	
 	/**
 	 * Builds a version of regex with every non-capturing term converted into a capturing group. This is a workaround
 	 * for JavaScript's lack of API to get the index at which a matched group begins in the input string.

@@ -7904,73 +9403,71 @@ orion.editor.RegexUtil = { } }; -/** - * @name orion.editor.TextMateStyler - * @class A styler that knows how to apply a subset of the TextMate grammar format to style a line. - * - *

Styling from a grammar:

- *

Each scope name given in the grammar is converted to an array of CSS class names. For example - * a region of text with scope keyword.control.php will be assigned the CSS classes
- * keyword, keyword-control, keyword-control-php

- * - *

A CSS file can give rules matching any of these class names to provide generic or more specific styling. - * For example,

- *

.keyword { font-color: blue; }

- *

colors all keywords blue, while

- *

.keyword-control-php { font-weight: bold; }

- *

bolds only PHP control keywords.

- * - *

This is useful when using grammars that adhere to TextMate's - * scope name conventions, - * as a single CSS rule can provide consistent styling to similar constructs across different languages.

- * - *

Top-level grammar constructs:

- *
  • patterns, repository (with limitations, see "Other Features") are supported.
  • - *
  • scopeName, firstLineMatch, foldingStartMarker, foldingStopMarker are not supported.
  • - *
  • fileTypes is not supported. When using the Orion service registry, the "orion.edit.highlighter" - * service serves a similar purpose.
  • - *
- * - *

Regular expression constructs:

- *
    - *
  • match patterns are supported.
  • - *
  • begin .. end patterns are supported.
  • - *
  • The "extended" regex forms (?x) and (?x:...) are supported, but only when they - * apply to the entire regex pattern.
  • - *
  • Matching is done using native JavaScript RegExps. As a result, many features of the Oniguruma regex - * engine used by TextMate are not supported. - * Unsupported features include: - *
    • Named captures
    • - *
    • Setting flags inside subgroups (eg. (?i:a)b)
    • - *
    • Lookbehind and negative lookbehind
    • - *
    • Subexpression call
    • - *
    • etc.
    • - *
    - *
  • - *
- * - *

Scope-assignment constructs:

- *
    - *
  • captures, beginCaptures, endCaptures are supported.
  • - *
  • name and contentName are supported.
  • - *
- * - *

Other features:

- *
    - *
  • applyEndPatternLast is supported.
  • - *
  • include is supported, but only when it references a rule in the current grammar's repository. - * Including $self, $base, or rule.from.another.grammar is not supported.
  • - *
- * - * @description Creates a new TextMateStyler. - * @extends orion.editor.AbstractStyler - * @param {orion.textview.TextView} textView The TextView to provide styling for. - * @param {Object} grammar The TextMate grammar to use for styling the TextView, as a JavaScript object. You can - * produce this object by running a PList-to-JavaScript conversion tool on a TextMate .tmLanguage file. - * @param {Object[]} [externalGrammars] Additional grammar objects that will be used to resolve named rule references. - */ -orion.editor.TextMateStyler = (function() { - /** @inner */ + /** + * @name orion.editor.TextMateStyler + * @class A styler that knows how to apply a subset of the TextMate grammar format to style a line. + * + *

Styling from a grammar:

+ *

Each scope name given in the grammar is converted to an array of CSS class names. For example + * a region of text with scope keyword.control.php will be assigned the CSS classes
+ * keyword, keyword-control, keyword-control-php

+ * + *

A CSS file can give rules matching any of these class names to provide generic or more specific styling. + * For example,

+ *

.keyword { font-color: blue; }

+ *

colors all keywords blue, while

+ *

.keyword-control-php { font-weight: bold; }

+ *

bolds only PHP control keywords.

+ * + *

This is useful when using grammars that adhere to TextMate's + * scope name conventions, + * as a single CSS rule can provide consistent styling to similar constructs across different languages.

+ * + *

Top-level grammar constructs:

+ *
  • patterns, repository (with limitations, see "Other Features") are supported.
  • + *
  • scopeName, firstLineMatch, foldingStartMarker, foldingStopMarker are not supported.
  • + *
  • fileTypes is not supported. When using the Orion service registry, the "orion.edit.highlighter" + * service serves a similar purpose.
  • + *
+ * + *

Regular expression constructs:

+ *
    + *
  • match patterns are supported.
  • + *
  • begin .. end patterns are supported.
  • + *
  • The "extended" regex forms (?x) and (?x:...) are supported, but only when they + * apply to the entire regex pattern.
  • + *
  • Matching is done using native JavaScript RegExps. As a result, many features of the Oniguruma regex + * engine used by TextMate are not supported. + * Unsupported features include: + *
    • Named captures
    • + *
    • Setting flags inside subgroups (eg. (?i:a)b)
    • + *
    • Lookbehind and negative lookbehind
    • + *
    • Subexpression call
    • + *
    • etc.
    • + *
    + *
  • + *
+ * + *

Scope-assignment constructs:

+ *
    + *
  • captures, beginCaptures, endCaptures are supported.
  • + *
  • name and contentName are supported.
  • + *
+ * + *

Other features:

+ *
    + *
  • applyEndPatternLast is supported.
  • + *
  • include is supported, but only when it references a rule in the current grammar's repository. + * Including $self, $base, or rule.from.another.grammar is not supported.
  • + *
+ * + * @description Creates a new TextMateStyler. + * @extends orion.editor.AbstractStyler + * @param {orion.textview.TextView} textView The TextView to provide styling for. + * @param {Object} grammar The TextMate grammar to use for styling the TextView, as a JavaScript object. You can + * produce this object by running a PList-to-JavaScript conversion tool on a TextMate .tmLanguage file. + * @param {Object[]} [externalGrammars] Additional grammar objects that will be used to resolve named rule references. + */ function TextMateStyler(textView, grammar, externalGrammars) { this.initialize(textView); // Copy grammar object(s) since we will mutate them @@ -7982,7 +9479,41 @@ orion.editor.TextMateStyler = (function() { this._allGrammars = {}; /* key: {String} scopeName of grammar, value: {Object} grammar */ this.preprocess(this.grammar); } - orion.editor.AbstractStyler.extend(TextMateStyler, /** @lends orion.editor.TextMateStyler.prototype */ { + TextMateStyler.prototype = /** @lends orion.editor.TextMateStyler.prototype */ { + initialize: function(textView) { + this.textView = textView; + var self = this; + this._listener = { + onModelChanged: function(e) { + self.onModelChanged(e); + }, + onDestroy: function(e) { + self.onDestroy(e); + }, + onLineStyle: function(e) { + self.onLineStyle(e); + } + }; + textView.addEventListener("ModelChanged", this._listener.onModelChanged); + textView.addEventListener("Destroy", this._listener.onDestroy); + textView.addEventListener("LineStyle", this._listener.onLineStyle); + textView.redrawLines(); + }, + onDestroy: function(/**eclipse.DestroyEvent*/ e) { + this.destroy(); + }, + destroy: function() { + if (this.textView) { + this.textView.removeEventListener("ModelChanged", this._listener.onModelChanged); + this.textView.removeEventListener("Destroy", this._listener.onDestroy); + this.textView.removeEventListener("LineStyle", this._listener.onLineStyle); + this.textView = null; + } + this.grammar = null; + this._styles = null; + this._tree = null; + this._listener = null; + }, /** @private */ copy: function(obj) { return JSON.parse(JSON.stringify(obj)); @@ -7995,7 +9526,7 @@ orion.editor.TextMateStyler = (function() { if (rule._resolvedRule && rule._typedRule) { continue; } -// console.debug("Process " + (rule.include || rule.name)); +// console.debug("Process " + (rule.include || rule.name)); // Look up include'd rule, create typed *Rule instance rule._resolvedRule = this._resolve(rule); @@ -8066,23 +9597,23 @@ orion.editor.TextMateStyler = (function() { function BeginEndRule(/**Object*/ rule) { this.rule = rule; // TODO: the TextMate blog claims that "end" is optional. - this.beginRegex = orion.editor.RegexUtil.toRegExp(rule.begin); - this.endRegex = orion.editor.RegexUtil.toRegExp(rule.end); + this.beginRegex = RegexUtil.toRegExp(rule.begin); + this.endRegex = RegexUtil.toRegExp(rule.end); this.subrules = rule.patterns || []; - this.endRegexHasBackRef = orion.editor.RegexUtil.hasBackReference(this.endRegex); + this.endRegexHasBackRef = RegexUtil.hasBackReference(this.endRegex); // Deal with non-0 captures - var complexCaptures = orion.editor.RegexUtil.complexCaptures(rule.captures); - var complexBeginEnd = orion.editor.RegexUtil.complexCaptures(rule.beginCaptures) || orion.editor.RegexUtil.complexCaptures(rule.endCaptures); + var complexCaptures = RegexUtil.complexCaptures(rule.captures); + var complexBeginEnd = RegexUtil.complexCaptures(rule.beginCaptures) || RegexUtil.complexCaptures(rule.endCaptures); this.isComplex = complexCaptures || complexBeginEnd; if (this.isComplex) { - var bg = orion.editor.RegexUtil.groupify(this.beginRegex); + var bg = RegexUtil.groupify(this.beginRegex); this.beginRegex = bg[0]; this.beginOld2New = bg[1]; this.beginConsuming = bg[2]; - var eg = orion.editor.RegexUtil.groupify(this.endRegex, this.beginOld2New /*Update end's backrefs to begin's new group #s*/); + var eg = RegexUtil.groupify(this.endRegex, this.beginOld2New /*Update end's backrefs to begin's new group #s*/); this.endRegex = eg[0]; this.endOld2New = eg[1]; this.endConsuming = eg[2]; @@ -8098,10 +9629,10 @@ orion.editor.TextMateStyler = (function() { MatchRule: (function() { function MatchRule(/**Object*/ rule) { this.rule = rule; - this.matchRegex = orion.editor.RegexUtil.toRegExp(rule.match); - this.isComplex = orion.editor.RegexUtil.complexCaptures(rule.captures); + this.matchRegex = RegexUtil.toRegExp(rule.match); + this.isComplex = RegexUtil.complexCaptures(rule.captures); if (this.isComplex) { - var mg = orion.editor.RegexUtil.groupify(this.matchRegex); + var mg = RegexUtil.groupify(this.matchRegex); this.matchRegex = mg[0]; this.matchOld2New = mg[1]; this.matchConsuming = mg[2]; @@ -8192,7 +9723,7 @@ orion.editor.TextMateStyler = (function() { // Build a new regex if the "end" regex has backrefs since they refer to matched groups of beginMatch if (rule.endRegexHasBackRef) { - this.endRegexSubstituted = orion.editor.RegexUtil.getSubstitutedRegex(rule.endRegex, beginMatch); + this.endRegexSubstituted = RegexUtil.getSubstitutedRegex(rule.endRegex, beginMatch); } else { this.endRegexSubstituted = null; } @@ -8284,7 +9815,7 @@ orion.editor.TextMateStyler = (function() { this._tree = root; this.parse(this._tree, false, 0); }, - _onModelChanged: function(/**eclipse.ModelChangedEvent*/ e) { + onModelChanged: function(/**eclipse.ModelChangedEvent*/ e) { var addedCharCount = e.addedCharCount, addedLineCount = e.addedLineCount, removedCharCount = e.removedCharCount, @@ -8684,7 +10215,7 @@ orion.editor.TextMateStyler = (function() { node.parent.children.length = node.getIndexInParent() + 1; } }, - _onLineStyle: function(/**eclipse.LineStyleEvent*/ e) { + onLineStyle: function(/**eclipse.LineStyleEvent*/ e) { function byStart(r1, r2) { return r1.start - r2.start; } @@ -8702,7 +10233,7 @@ orion.editor.TextMateStyler = (function() { var scopes = this.getLineScope(model, node, lineStart, lineEnd); e.ranges = this.toStyleRanges(scopes); -// // Editor requires StyleRanges must be in ascending order by 'start', or else some will be ignored + // Editor requires StyleRanges must be in ascending order by 'start', or else some will be ignored e.ranges.sort(byStart); }, /** Runs parse algorithm on [start, end] in the context of node, assigning scope as we find matches. @@ -8866,13 +10397,6 @@ orion.editor.TextMateStyler = (function() { } return result.reverse(); }, - _onSelection: function(e) { - }, - _onDestroy: function(/**eclipse.DestroyEvent*/ e) { - this.grammar = null; - this._styles = null; - this._tree = null; - }, /** * Applies the grammar to obtain the {@link eclipse.StyleRange[]} for the given line. * @returns eclipse.StyleRange[] @@ -8890,17 +10414,15 @@ orion.editor.TextMateStyler = (function() { } return styleRanges; } - }); - return TextMateStyler; -}()); - -if (typeof window !== "undefined" && typeof window.define !== "undefined") { - define([], function() { - return orion.editor; - }); -} - + }; + + return { + RegexUtil: RegexUtil, + TextMateStyler: TextMateStyler + }; +}, "orion/editor"); /******************************************************************************* + * @license * Copyright (c) 2010, 2011 IBM Corporation and others. * All rights reserved. This program and the accompanying materials are made * available under the terms of the Eclipse Public License v1.0 @@ -8910,12 +10432,9 @@ if (typeof window !== "undefined" && typeof window.define !== "undefined") { * Contributors: IBM Corporation - initial API and implementation ******************************************************************************/ -/*global document window navigator define orion */ +/*global document window navigator define */ -var examples = examples || {}; -examples.textview = examples.textview || {}; - -examples.textview.TextStyler = (function() { +define(['orion/textview/annotations'], function(mAnnotations) { var JS_KEYWORDS = ["break", @@ -8988,7 +10507,6 @@ examples.textview.TextStyler = (function() { var TASK_TAG = 12; // Styles - var isIE = document.selection && window.ActiveXObject && /MSIE/.test(navigator.userAgent) ? document.documentMode : undefined; var singleCommentStyle = {styleClass: "token_singleline_comment"}; var multiCommentStyle = {styleClass: "token_multiline_comment"}; var docCommentStyle = {styleClass: "token_doc_comment"}; @@ -8999,88 +10517,87 @@ examples.textview.TextStyler = (function() { var keywordStyle = {styleClass: "token_keyword"}; var spaceStyle = {styleClass: "token_space"}; var tabStyle = {styleClass: "token_tab"}; - var bracketStyle = {styleClass: isIE < 9 ? "token_bracket" : "token_bracket_outline"}; var caretLineStyle = {styleClass: "line_caret"}; - var Scanner = (function() { - function Scanner (keywords, whitespacesVisible) { - this.keywords = keywords; - this.whitespacesVisible = whitespacesVisible; - this.setText(""); - } - - Scanner.prototype = { - getOffset: function() { - return this.offset; - }, - getStartOffset: function() { - return this.startOffset; - }, - getData: function() { - return this.text.substring(this.startOffset, this.offset); - }, - getDataLength: function() { - return this.offset - this.startOffset; - }, - _default: function(c) { - var keywords = this.keywords; - switch (c) { - case 32: // SPACE - case 9: // TAB - if (this.whitespacesVisible) { - return c === 32 ? WHITE_SPACE : WHITE_TAB; - } + function Scanner (keywords, whitespacesVisible) { + this.keywords = keywords; + this.whitespacesVisible = whitespacesVisible; + this.setText(""); + } + + Scanner.prototype = { + getOffset: function() { + return this.offset; + }, + getStartOffset: function() { + return this.startOffset; + }, + getData: function() { + return this.text.substring(this.startOffset, this.offset); + }, + getDataLength: function() { + return this.offset - this.startOffset; + }, + _default: function(c) { + var keywords = this.keywords; + switch (c) { + case 32: // SPACE + case 9: // TAB + if (this.whitespacesVisible) { + return c === 32 ? WHITE_SPACE : WHITE_TAB; + } + do { + c = this._read(); + } while(c === 32 || c === 9); + this._unread(c); + return WHITE; + case 123: // { + case 125: // } + case 40: // ( + case 41: // ) + case 91: // [ + case 93: // ] + case 60: // < + case 62: // > + // BRACKETS + return c; + default: + var isCSS = this.isCSS; + if ((97 <= c && c <= 122) || (65 <= c && c <= 90) || c === 95 || (48 <= c && c <= 57) || (0x2d === c && isCSS)) { //LETTER OR UNDERSCORE OR NUMBER + var off = this.offset - 1; do { c = this._read(); - } while(c === 32 || c === 9); + } while((97 <= c && c <= 122) || (65 <= c && c <= 90) || c === 95 || (48 <= c && c <= 57) || (0x2d === c && isCSS)); //LETTER OR UNDERSCORE OR NUMBER this._unread(c); - return WHITE; - case 123: // { - case 125: // } - case 40: // ( - case 41: // ) - case 91: // [ - case 93: // ] - case 60: // < - case 62: // > - // BRACKETS - return c; - default: - var isCSS = this.isCSS; - if ((97 <= c && c <= 122) || (65 <= c && c <= 90) || c === 95 || (48 <= c && c <= 57) || (0x2d === c && isCSS)) { //LETTER OR UNDERSCORE OR NUMBER - var off = this.offset - 1; - do { - c = this._read(); - } while((97 <= c && c <= 122) || (65 <= c && c <= 90) || c === 95 || (48 <= c && c <= 57) || (0x2d === c && isCSS)); //LETTER OR UNDERSCORE OR NUMBER - this._unread(c); - if (keywords.length > 0) { - var word = this.text.substring(off, this.offset); - //TODO slow - for (var i=0; i 0) { + var word = this.text.substring(off, this.offset); + //TODO slow + for (var i=0; i comment - c = this._read(); + } + return UNKOWN; + } + }, + _read: function() { + if (this.offset < this.text.length) { + return this.text.charCodeAt(this.offset++); + } + return -1; + }, + _unread: function(c) { + if (c !== -1) { this.offset--; } + }, + nextToken: function() { + this.startOffset = this.offset; + while (true) { + var c = this._read(); + switch (c) { + case -1: return null; + case 47: // SLASH -> comment + c = this._read(); + if (!this.isCSS) { if (c === 47) { // SLASH -> single line while (true) { c = this._read(); @@ -9090,201 +10607,188 @@ examples.textview.TextStyler = (function() { } } } - if (c === 42) { // STAR -> multi line - c = this._read(); - var token = MULTILINE_COMMENT; - if (c === 42) { - token = DOC_COMMENT; - } - while (true) { - while (c === 42) { - c = this._read(); - if (c === 47) { - return token; - } - } - if (c === -1) { - this._unread(c); + } + if (c === 42) { // STAR -> multi line + c = this._read(); + var token = MULTILINE_COMMENT; + if (c === 42) { + token = DOC_COMMENT; + } + while (true) { + while (c === 42) { + c = this._read(); + if (c === 47) { return token; } + } + if (c === -1) { + this._unread(c); + return token; + } + c = this._read(); + } + } + this._unread(c); + return UNKOWN; + case 39: // SINGLE QUOTE -> char const + while(true) { + c = this._read(); + switch (c) { + case 39: + return STRING; + case 13: + case 10: + case -1: + this._unread(c); + return STRING; + case 92: // BACKSLASH c = this._read(); - } + break; } - this._unread(c); - return UNKOWN; - case 39: // SINGLE QUOTE -> char const - while(true) { - c = this._read(); - switch (c) { - case 39: - return STRING; - case 13: - case 10: - case -1: - this._unread(c); - return STRING; - case 92: // BACKSLASH - c = this._read(); - break; - } + } + break; + case 34: // DOUBLE QUOTE -> string + while(true) { + c = this._read(); + switch (c) { + case 34: // DOUBLE QUOTE + return STRING; + case 13: + case 10: + case -1: + this._unread(c); + return STRING; + case 92: // BACKSLASH + c = this._read(); + break; } - break; - case 34: // DOUBLE QUOTE -> string - while(true) { - c = this._read(); - switch (c) { - case 34: // DOUBLE QUOTE - return STRING; - case 13: - case 10: - case -1: - this._unread(c); - return STRING; - case 92: // BACKSLASH - c = this._read(); - break; - } - } - break; - default: - return this._default(c); - } - } - }, - setText: function(text) { - this.text = text; - this.offset = 0; - this.startOffset = 0; - } - }; - return Scanner; - }()); - - var WhitespaceScanner = (function() { - function WhitespaceScanner () { - Scanner.call(this, null, true); - } - WhitespaceScanner.prototype = new Scanner(null); - WhitespaceScanner.prototype.nextToken = function() { - this.startOffset = this.offset; - while (true) { - var c = this._read(); - switch (c) { - case -1: return null; - case 32: // SPACE - return WHITE_SPACE; - case 9: // TAB - return WHITE_TAB; + } + break; default: - do { - c = this._read(); - } while(!(c === 32 || c === 9 || c === -1)); - this._unread(c); - return UNKOWN; + return this._default(c); } } - }; - - return WhitespaceScanner; - }()); - - var CommentScanner = (function() { - function CommentScanner (whitespacesVisible) { - Scanner.call(this, null, whitespacesVisible); + }, + setText: function(text) { + this.text = text; + this.offset = 0; + this.startOffset = 0; } - CommentScanner.prototype = new Scanner(null); - CommentScanner.prototype.setType = function(type) { - this._type = type; - }; - CommentScanner.prototype.nextToken = function() { - this.startOffset = this.offset; - while (true) { - var c = this._read(); - switch (c) { - case -1: return null; - case 32: // SPACE - case 9: // TAB - if (this.whitespacesVisible) { - return c === 32 ? WHITE_SPACE : WHITE_TAB; - } + }; + + function WhitespaceScanner () { + Scanner.call(this, null, true); + } + WhitespaceScanner.prototype = new Scanner(null); + WhitespaceScanner.prototype.nextToken = function() { + this.startOffset = this.offset; + while (true) { + var c = this._read(); + switch (c) { + case -1: return null; + case 32: // SPACE + return WHITE_SPACE; + case 9: // TAB + return WHITE_TAB; + default: + do { + c = this._read(); + } while(!(c === 32 || c === 9 || c === -1)); + this._unread(c); + return UNKOWN; + } + } + }; + + function CommentScanner (whitespacesVisible) { + Scanner.call(this, null, whitespacesVisible); + } + CommentScanner.prototype = new Scanner(null); + CommentScanner.prototype.setType = function(type) { + this._type = type; + }; + CommentScanner.prototype.nextToken = function() { + this.startOffset = this.offset; + while (true) { + var c = this._read(); + switch (c) { + case -1: return null; + case 32: // SPACE + case 9: // TAB + if (this.whitespacesVisible) { + return c === 32 ? WHITE_SPACE : WHITE_TAB; + } + do { + c = this._read(); + } while(c === 32 || c === 9); + this._unread(c); + return WHITE; + case 60: // < + if (this._type === DOC_COMMENT) { do { c = this._read(); - } while(c === 32 || c === 9); + } while(!(c === 62 || c === -1)); // > + if (c === 62) { + return HTML_MARKUP; + } + } + return UNKOWN; + case 64: // @ + if (this._type === DOC_COMMENT) { + do { + c = this._read(); + } while((97 <= c && c <= 122) || (65 <= c && c <= 90) || c === 95 || (48 <= c && c <= 57)); //LETTER OR UNDERSCORE OR NUMBER this._unread(c); - return WHITE; - case 60: // < - if (this._type === DOC_COMMENT) { - do { + return DOC_TAG; + } + return UNKOWN; + case 84: // T + if ((c = this._read()) === 79) { // O + if ((c = this._read()) === 68) { // D + if ((c = this._read()) === 79) { // O c = this._read(); - } while(!(c === 62 || c === -1)); // > - if (c === 62) { - return HTML_MARKUP; - } - } - return UNKOWN; - case 64: // @ - if (this._type === DOC_COMMENT) { - do { - c = this._read(); - } while((97 <= c && c <= 122) || (65 <= c && c <= 90) || c === 95 || (48 <= c && c <= 57)); //LETTER OR UNDERSCORE OR NUMBER - this._unread(c); - return DOC_TAG; - } - return UNKOWN; - case 84: // T - if ((c = this._read()) === 79) { // O - if ((c = this._read()) === 68) { // D - if ((c = this._read()) === 79) { // O - c = this._read(); - if (!((97 <= c && c <= 122) || (65 <= c && c <= 90) || c === 95 || (48 <= c && c <= 57))) { - this._unread(c); - return TASK_TAG; - } - this._unread(c); - } else { + if (!((97 <= c && c <= 122) || (65 <= c && c <= 90) || c === 95 || (48 <= c && c <= 57))) { this._unread(c); + return TASK_TAG; } + this._unread(c); } else { this._unread(c); } } else { this._unread(c); } - //FALL THROUGH - default: - do { - c = this._read(); - } while(!(c === 32 || c === 9 || c === -1 || c === 60 || c === 64 || c === 84)); + } else { this._unread(c); - return UNKOWN; - } + } + //FALL THROUGH + default: + do { + c = this._read(); + } while(!(c === 32 || c === 9 || c === -1 || c === 60 || c === 64 || c === 84)); + this._unread(c); + return UNKOWN; } - }; - - return CommentScanner; - }()); - - var FirstScanner = (function() { - function FirstScanner () { - Scanner.call(this, null, false); } - FirstScanner.prototype = new Scanner(null); - FirstScanner.prototype._default = function(c) { - while(true) { - c = this._read(); - switch (c) { - case 47: // SLASH - case 34: // DOUBLE QUOTE - case 39: // SINGLE QUOTE - case -1: - this._unread(c); - return UNKOWN; - } + }; + + function FirstScanner () { + Scanner.call(this, null, false); + } + FirstScanner.prototype = new Scanner(null); + FirstScanner.prototype._default = function(c) { + while(true) { + c = this._read(); + switch (c) { + case 47: // SLASH + case 34: // DOUBLE QUOTE + case 39: // SINGLE QUOTE + case -1: + this._unread(c); + return UNKOWN; } - }; - - return FirstScanner; - }()); + } + }; function TextStyler (view, lang, annotationModel) { this.commentStart = "/*"; @@ -9297,38 +10801,47 @@ examples.textview.TextStyler = (function() { } this.whitespacesVisible = false; this.detectHyperlinks = true; - this.highlightCaretLine = true; + this.highlightCaretLine = false; this.foldingEnabled = true; this.detectTasks = true; this._scanner = new Scanner(keywords, this.whitespacesVisible); - //TODO this scanner is not the best/correct way to parse CSS - if (lang === "css") { - this._scanner.isCSS = true; - } this._firstScanner = new FirstScanner(); this._commentScanner = new CommentScanner(this.whitespacesVisible); this._whitespaceScanner = new WhitespaceScanner(); + //TODO these scanners are not the best/correct way to parse CSS + if (lang === "css") { + this._scanner.isCSS = true; + this._firstScanner.isCSS = true; + } this.view = view; this.annotationModel = annotationModel; - this._currentBracket = undefined; - this._matchingBracket = undefined; + this._bracketAnnotations = undefined; - view.addEventListener("Selection", this, this._onSelection); + var self = this; + this._listener = { + onChanged: function(e) { + self._onModelChanged(e); + }, + onDestroy: function(e) { + self._onDestroy(e); + }, + onLineStyle: function(e) { + self._onLineStyle(e); + }, + onSelection: function(e) { + self._onSelection(e); + } + }; var model = view.getModel(); if (model.getBaseModel) { - var self = this; - this._baseModelListener = { - onChanged: function(modelChangedEvent) { - self._onModelChanged(modelChangedEvent); - } - }; - model.getBaseModel().addListener(this._baseModelListener); + model.getBaseModel().addEventListener("Changed", this._listener.onChanged); } else { //TODO still needed to keep the event order correct (styler before view) - view.addEventListener("ModelChanged", this, this._onModelChanged); + view.addEventListener("ModelChanged", this._listener.onChanged); } - view.addEventListener("Destroy", this, this._onDestroy); - view.addEventListener("LineStyle", this, this._onLineStyle); + view.addEventListener("Selection", this._listener.onSelection); + view.addEventListener("Destroy", this._listener.onDestroy); + view.addEventListener("LineStyle", this._listener.onLineStyle); this._computeComments (); this._computeFolding(); view.redrawLines(); @@ -9340,13 +10853,13 @@ examples.textview.TextStyler = (function() { if (view) { var model = view.getModel(); if (model.getBaseModel) { - model.getBaseModel().removeListener(this._baseModelListener); + model.getBaseModel().removeEventListener("Changed", this._listener.onChanged); } else { - view.removeEventListener("ModelChanged", this, this._onModelChanged); + view.removeEventListener("ModelChanged", this._listener.onChanged); } - view.removeEventListener("Selection", this, this._onSelection); - view.removeEventListener("Destroy", this, this._onDestroy); - view.removeEventListener("LineStyle", this, this._onLineStyle); + view.removeEventListener("Selection", this._listener.onSelection); + view.removeEventListener("Destroy", this._listener.onDestroy); + view.removeEventListener("LineStyle", this._listener.onLineStyle); this.view = null; } }, @@ -9415,7 +10928,7 @@ examples.textview.TextStyler = (function() { if (startLine === endLine) { return null; } - return new orion.textview.FoldingAnnotation(viewModel, "orion.annotation.folding", start, end, + return new mAnnotations.FoldingAnnotation(viewModel, "orion.annotation.folding", start, end, "
", {styleClass: "annotation expanded"}, "", {styleClass: "annotation collapsed"}); }, @@ -9428,8 +10941,12 @@ examples.textview.TextStyler = (function() { if (viewModel.getBaseModel) { baseModel = viewModel.getBaseModel(); } var annotations = annotationModel.getAnnotations(commentStart, commentEnd); var remove = []; + var annotationType = "orion.annotation.task"; while (annotations.hasNext()) { - remove.push(annotations.next()); + var annotation = annotations.next(); + if (annotation.type === annotationType) { + remove.push(annotation); + } } var add = []; var scanner = this._commentScanner; @@ -9445,11 +10962,12 @@ examples.textview.TextStyler = (function() { add.push({ start: tokenStart, end: end, - type: "orion.annotation.task", + type: annotationType, title: baseModel.getText(tokenStart, end), style: {styleClass: "annotation task"}, html: "
", - overviewStyle: {styleClass: "annotationOverview task"} + overviewStyle: {styleClass: "annotationOverview task"}, + rangeStyle: {styleClass: "annotationRange task"} }); } } @@ -9513,39 +11031,35 @@ examples.textview.TextStyler = (function() { while ((token = scanner.nextToken())) { var tokenStart = scanner.getStartOffset() + offset; var style = null; - if (tokenStart === this._matchingBracket) { - style = bracketStyle; - } else { - switch (token) { - case KEYWORD: style = keywordStyle; break; - case STRING: - if (this.whitespacesVisible) { - this._parseString(scanner.getData(), tokenStart, styles, stringStyle); - continue; - } else { - style = stringStyle; - } - break; - case DOC_COMMENT: - this._parseComment(scanner.getData(), tokenStart, styles, docCommentStyle, token); + switch (token) { + case KEYWORD: style = keywordStyle; break; + case STRING: + if (this.whitespacesVisible) { + this._parseString(scanner.getData(), tokenStart, styles, stringStyle); continue; - case SINGLELINE_COMMENT: - this._parseComment(scanner.getData(), tokenStart, styles, singleCommentStyle, token); - continue; - case MULTILINE_COMMENT: - this._parseComment(scanner.getData(), tokenStart, styles, multiCommentStyle, token); - continue; - case WHITE_TAB: - if (this.whitespacesVisible) { - style = tabStyle; - } - break; - case WHITE_SPACE: - if (this.whitespacesVisible) { - style = spaceStyle; - } - break; - } + } else { + style = stringStyle; + } + break; + case DOC_COMMENT: + this._parseComment(scanner.getData(), tokenStart, styles, docCommentStyle, token); + continue; + case SINGLELINE_COMMENT: + this._parseComment(scanner.getData(), tokenStart, styles, singleCommentStyle, token); + continue; + case MULTILINE_COMMENT: + this._parseComment(scanner.getData(), tokenStart, styles, multiCommentStyle, token); + continue; + case WHITE_TAB: + if (this.whitespacesVisible) { + style = tabStyle; + } + break; + case WHITE_SPACE: + if (this.whitespacesVisible) { + style = spaceStyle; + } + break; } styles.push({start: tokenStart, end: scanner.getOffset() + offset, style: style}); } @@ -9684,7 +11198,6 @@ examples.textview.TextStyler = (function() { return result; }, _findMatchingBracket: function(model, offset) { - if (model.getBaseModel) { model = model.getBaseModel(); } var brackets = "{}()[]<>"; var bracket = model.getText(offset, offset + 1); var bracketIndex = brackets.indexOf(bracket, 0); @@ -9809,13 +11322,6 @@ examples.textview.TextStyler = (function() { var view = this.view; var model = view.getModel(); var lineIndex; - var bracket = this._matchingBracket; - if (bracket !== undefined) { - if (model.getBaseModel) { bracket = model.mapOffset(bracket, true); } - lineIndex = model.getLineAtOffset(bracket); - view.redrawLines(lineIndex, lineIndex + 1); - this._matchingBracket = this._currentBracket = undefined; - } if (this.highlightCaretLine) { var oldLineIndex = model.getLineAtOffset(oldSelection.start); lineIndex = model.getLineAtOffset(newSelection.start); @@ -9830,31 +11336,44 @@ examples.textview.TextStyler = (function() { } } } - if (newSelection.start !== newSelection.end || newSelection.start === 0) { - return; - } - var caret = view.getCaretOffset() - 1; - if (caret < 0) { return; } - var mapCaret = caret; - if (model.getBaseModel) { - mapCaret = model.mapOffset(caret); - } - bracket = this._findMatchingBracket(model, mapCaret); - if (bracket !== -1) { - this._currentBracket = mapCaret; - this._matchingBracket = bracket; - if (model.getBaseModel) { bracket = model.mapOffset(bracket, true); } - lineIndex = model.getLineAtOffset(bracket); - view.redrawLines(lineIndex, lineIndex + 1); + if (!this.annotationModel) { return; } + var remove = this._bracketAnnotations, add, caret; + if (newSelection.start === newSelection.end && (caret = view.getCaretOffset()) > 0) { + var mapCaret = caret - 1; + if (model.getBaseModel) { + mapCaret = model.mapOffset(mapCaret); + model = model.getBaseModel(); + } + var bracket = this._findMatchingBracket(model, mapCaret); + if (bracket !== -1) { + add = [{ + start: bracket, + end: bracket + 1, + type: "orion.annotation.matchingBracket", + title: "Matching Bracket", + html: "
", + overviewStyle: {styleClass: "annotationOverview matchingBracket"}, + rangeStyle: {styleClass: "annotationRange matchingBracket"} + }, + { + start: mapCaret, + end: mapCaret + 1, + type: "orion.annotation.currentBracket", + title: "Current Bracket", + html: "
", + overviewStyle: {styleClass: "annotationOverview currentBracket"}, + rangeStyle: {styleClass: "annotationRange currentBracket"} + }]; + } } + this._bracketAnnotations = add; + this.annotationModel.replaceAnnotations(remove, add); }, _onModelChanged: function(e) { var start = e.start; var removedCharCount = e.removedCharCount; var addedCharCount = e.addedCharCount; var changeCount = addedCharCount - removedCharCount; - if (this._matchingBracket && start < this._matchingBracket) { this._matchingBracket += changeCount; } - if (this._currentBracket && start < this._currentBracket) { this._currentBracket += changeCount; } var view = this.view; var viewModel = view.getModel(); var baseModel = viewModel.getBaseModel ? viewModel.getBaseModel() : viewModel; @@ -9870,7 +11389,11 @@ examples.textview.TextStyler = (function() { ts = this.comments[commentStart].start; if (ts > start) { ts += changeCount; } } else { - ts = lineStart; + if (commentStart === commentCount && commentCount > 0 && charCount - changeCount === this.comments[commentCount - 1].end) { + ts = this.comments[commentCount - 1].start; + } else { + ts = lineStart; + } } var te; if (commentEnd < commentCount) { @@ -9971,11 +11494,6 @@ examples.textview.TextStyler = (function() { } } }; - return TextStyler; -}()); - -if (typeof window !== "undefined" && typeof window.define !== "undefined") { - define([], function() { - return examples.textview; - }); -} + + return {TextStyler: TextStyler}; +}, "examples/textview"); diff --git a/browser/devtools/sourceeditor/source-editor-orion.jsm b/browser/devtools/sourceeditor/source-editor-orion.jsm index b78b1c04e78d..15daa7fab329 100644 --- a/browser/devtools/sourceeditor/source-editor-orion.jsm +++ b/browser/devtools/sourceeditor/source-editor-orion.jsm @@ -57,8 +57,7 @@ const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"; * SourceEditor.THEMES to Orion CSS files. */ const ORION_THEMES = { - textmate: ["chrome://browser/content/orion.css", - "chrome://browser/content/orion-mozilla.css"], + mozilla: ["chrome://browser/content/orion-mozilla.css"], }; /** @@ -71,6 +70,13 @@ const ORION_EVENTS = { Selection: "Selection", }; +/** + * Known Orion annotation types. + */ +const ORION_ANNOTATION_TYPES = { + currentBracket: "orion.annotation.currentBracket", + matchingBracket: "orion.annotation.matchingBracket", +}; var EXPORTED_SYMBOLS = ["SourceEditor"]; @@ -95,12 +101,17 @@ function SourceEditor() { SourceEditor.prototype = { _view: null, _iframe: null, + _model: null, _undoStack: null, - _lines_ruler: null, + _linesRuler: null, _styler: null, + _annotationStyler: null, + _annotationModel: null, + _dragAndDrop: null, _mode: null, _expandTab: null, _tabSize: null, + _iframeWindow: null, /** * The editor container element. @@ -152,11 +163,7 @@ SourceEditor.prototype = { let onIframeLoad = (function() { this._iframe.removeEventListener("load", onIframeLoad, true); - - Services.scriptloader.loadSubScript(ORION_SCRIPT, - this._iframe.contentWindow.wrappedJSObject, "utf8"); - - this._onLoad(aCallback); + this._onIframeLoad(); }).bind(this); this._iframe.addEventListener("load", onIframeLoad, true); @@ -166,20 +173,23 @@ SourceEditor.prototype = { aElement.appendChild(this._iframe); this.parentElement = aElement; this._config = aConfig; + this._onReadyCallback = aCallback; }, /** * The editor iframe load event handler. - * * @private - * @param function [aCallback] - * Optional function invoked when the editor completes loading. */ - _onLoad: function SE__onLoad(aCallback) + _onIframeLoad: function SE__onIframeLoad() { + this._iframeWindow = this._iframe.contentWindow.wrappedJSObject; + let window = this._iframeWindow; let config = this._config; - let window = this._iframe.contentWindow.wrappedJSObject; - let textview = window.orion.textview; + + Services.scriptloader.loadSubScript(ORION_SCRIPT, window, "utf8"); + + let TextModel = window.require("orion/textview/textModel").TextModel; + let TextView = window.require("orion/textview/textView").TextView; this._expandTab = typeof config.expandTab != "undefined" ? config.expandTab : SourceEditor.DEFAULTS.EXPAND_TAB; @@ -188,73 +198,74 @@ SourceEditor.prototype = { let theme = config.theme || SourceEditor.DEFAULTS.THEME; let stylesheet = theme in ORION_THEMES ? ORION_THEMES[theme] : theme; - this._view = new textview.TextView({ - model: new textview.TextModel(config.placeholderText), + this._model = new TextModel(config.placeholderText); + this._view = new TextView({ + model: this._model, parent: "editor", stylesheet: stylesheet, tabSize: this._tabSize, + expandTab: this._expandTab, readonly: config.readOnly, + themeClass: "mozilla" + (config.readOnly ? " readonly" : ""), }); + let onOrionLoad = function() { + this._view.removeEventListener("Load", onOrionLoad); + this._onOrionLoad(); + }.bind(this); + + this._view.addEventListener("Load", onOrionLoad); + + let KeyBinding = window.require("orion/textview/keyBinding").KeyBinding; + let TextDND = window.require("orion/textview/textDND").TextDND; + let LineNumberRuler = window.require("orion/textview/rulers").LineNumberRuler; + let UndoStack = window.require("orion/textview/undoStack").UndoStack; + let AnnotationModel = window.require("orion/textview/annotations").AnnotationModel; + + this._annotationModel = new AnnotationModel(this._model); + if (config.showLineNumbers) { - this._lines_ruler = new textview.LineNumberRuler(null, "left", + this._linesRuler = new LineNumberRuler(this._annotationModel, "left", {styleClass: "rulerLines"}, {styleClass: "rulerLine odd"}, {styleClass: "rulerLine even"}); - this._view.addRuler(this._lines_ruler); + this._view.addRuler(this._linesRuler); } this.setMode(config.mode || SourceEditor.DEFAULTS.MODE); - this._undoStack = new textview.UndoStack(this._view, + this._undoStack = new UndoStack(this._view, config.undoLimit || SourceEditor.DEFAULTS.UNDO_LIMIT); - this._initEditorFeatures(); + this._dragAndDrop = new TextDND(this._view, this._undoStack); + + this._view.setAction("tab", this._doTab.bind(this)); + + let shiftTabKey = new KeyBinding(Ci.nsIDOMKeyEvent.DOM_VK_TAB, false, true); + this._view.setAction("Unindent Lines", this._doUnindentLines.bind(this)); + this._view.setKeyBinding(shiftTabKey, "Unindent Lines"); + this._view.setAction("enter", this._doEnter.bind(this)); (config.keys || []).forEach(function(aKey) { - let binding = new textview.KeyBinding(aKey.code, aKey.accel, aKey.shift, - aKey.alt); + let binding = new KeyBinding(aKey.code, aKey.accel, aKey.shift, aKey.alt); this._view.setKeyBinding(binding, aKey.action); if (aKey.callback) { this._view.setAction(aKey.action, aKey.callback); } }, this); - - if (aCallback) { - let iframe = this._view._frame; - let document = iframe.contentDocument; - if (!document || document.readyState != "complete") { - let onIframeLoad = function () { - iframe.contentWindow.removeEventListener("load", onIframeLoad, false); - aCallback(this); - }.bind(this); - iframe.contentWindow.addEventListener("load", onIframeLoad, false); - } else { - aCallback(this); - } - } }, /** - * Initialize the custom Orion editor features. + * The Orion "Load" event handler. This is called when the Orion editor + * completes the initialization. * @private */ - _initEditorFeatures: function SE__initEditorFeatures() + _onOrionLoad: function SE__onOrionLoad() { - let window = this._iframe.contentWindow.wrappedJSObject; - let textview = window.orion.textview; - - this._view.setAction("tab", this._doTab.bind(this)); - - let shiftTabKey = new textview.KeyBinding(Ci.nsIDOMKeyEvent.DOM_VK_TAB, - false, true); - this._view.setAction("Unindent Lines", this._doUnindentLines.bind(this)); - this._view.setKeyBinding(shiftTabKey, "Unindent Lines"); - this._view.setAction("enter", this._doEnter.bind(this)); - - if (this._expandTab) { - this._view.setAction("deletePrevious", this._doDeletePrevious.bind(this)); + if (this._onReadyCallback) { + this._onReadyCallback(this); + this._onReadyCallback = null; } }, @@ -302,34 +313,9 @@ SourceEditor.prototype = { this._view.setSelection(newSelectionStart, newSelectionEnd); this.endCompoundChange(); - } else { - this.setText(indent, selection.start, selection.end); + return true; } - return true; - }, - - /** - * The "deletePrevious" editor action implementation. This adds unindentation - * support to the Backspace key implementation. - * @private - */ - _doDeletePrevious: function SE__doDeletePrevious() - { - let selection = this.getSelection(); - if (selection.start == selection.end && this._expandTab) { - let model = this._model; - let lineIndex = model.getLineAtOffset(selection.start); - let lineStart = model.getLineStart(lineIndex); - let offset = selection.start - lineStart; - if (offset >= this._tabSize && (offset % this._tabSize) == 0) { - let text = this.getText(lineStart, selection.start); - if (!/[^ ]/.test(text)) { - this.setText("", selection.start - this._tabSize, selection.end); - return true; - } - } - } return false; }, @@ -357,7 +343,7 @@ SourceEditor.prototype = { for (let line, i = firstLine; i <= lastLine; i++) { line = model.getLine(i, true); if (line.indexOf(indent) != 0) { - return false; + return true; } lines.push(line.substring(indent.length)); } @@ -425,15 +411,6 @@ SourceEditor.prototype = { return true; }, - /** - * Get the Orion Model, the TextModel object instance we use. - * @private - * @type object - */ - get _model() { - return this._view.getModel(); - }, - /** * Get the editor element. * @@ -453,15 +430,12 @@ SourceEditor.prototype = { * The event type you want to listen for. * @param function aCallback * The function you want executed when the event is triggered. - * @param mixed [aData] - * Optional data to pass to the callback when the event is triggered. */ addEventListener: - function SE_addEventListener(aEventType, aCallback, aData) + function SE_addEventListener(aEventType, aCallback) { if (aEventType in ORION_EVENTS) { - this._view.addEventListener(ORION_EVENTS[aEventType], true, - aCallback, aData); + this._view.addEventListener(ORION_EVENTS[aEventType], aCallback); } else { throw new Error("SourceEditor.addEventListener() unknown event " + "type " + aEventType); @@ -478,15 +452,12 @@ SourceEditor.prototype = { * The event type you have a listener for. * @param function aCallback * The function you have as the event handler. - * @param mixed [aData] - * The optional data passed to the callback. */ removeEventListener: - function SE_removeEventListener(aEventType, aCallback, aData) + function SE_removeEventListener(aEventType, aCallback) { if (aEventType in ORION_EVENTS) { - this._view.removeEventListener(ORION_EVENTS[aEventType], true, - aCallback, aData); + this._view.removeEventListener(ORION_EVENTS[aEventType], aCallback); } else { throw new Error("SourceEditor.removeEventListener() unknown event " + "type " + aEventType); @@ -587,7 +558,7 @@ SourceEditor.prototype = { */ hasFocus: function SE_hasFocus() { - return this._iframe.ownerDocument.activeElement === this._iframe; + return this._view.hasFocus(); }, /** @@ -726,21 +697,41 @@ SourceEditor.prototype = { this._styler.destroy(); this._styler = null; } + if (this._annotationStyler) { + this._annotationStyler.destroy(); + this._annotationStyler = null; + } - let window = this._iframe.contentWindow.wrappedJSObject; - let TextStyler = window.examples.textview.TextStyler; - let TextMateStyler = window.orion.editor.TextMateStyler; - let HtmlGrammar = window.orion.editor.HtmlGrammar; + let window = this._iframeWindow; switch (aMode) { case SourceEditor.MODES.JAVASCRIPT: case SourceEditor.MODES.CSS: - this._styler = new TextStyler(this._view, aMode); + let TextStyler = + window.require("examples/textview/textStyler").TextStyler; + + this._styler = new TextStyler(this._view, aMode, this._annotationModel); + this._styler.setFoldingEnabled(false); + this._styler.setHighlightCaretLine(true); + + let AnnotationStyler = + window.require("orion/textview/annotations").AnnotationStyler; + + this._annotationStyler = + new AnnotationStyler(this._view, this._annotationModel); + this._annotationStyler. + addAnnotationType(ORION_ANNOTATION_TYPES.matchingBracket); + this._annotationStyler. + addAnnotationType(ORION_ANNOTATION_TYPES.currentBracket); break; case SourceEditor.MODES.HTML: case SourceEditor.MODES.XML: - this._styler = new TextMateStyler(this._view, HtmlGrammar.grammar); + let TextMateStyler = + window.require("orion/editor/textMateStyler").TextMateStyler; + let HtmlGrammar = + window.require("orion/editor/htmlGrammar").HtmlGrammar; + this._styler = new TextMateStyler(this._view, new HtmlGrammar().grammar); break; } @@ -765,7 +756,10 @@ SourceEditor.prototype = { */ set readOnly(aValue) { - this._view.readonly = aValue; + this._view.setOptions({ + readonly: aValue, + themeClass: "mozilla" + (aValue ? " readonly" : ""), + }); }, /** @@ -774,7 +768,7 @@ SourceEditor.prototype = { */ get readOnly() { - return this._view.readonly; + return this._view.getOptions("readonly"); }, /** @@ -785,11 +779,16 @@ SourceEditor.prototype = { this._view.destroy(); this.parentElement.removeChild(this._iframe); this.parentElement = null; + this._iframeWindow = null; this._iframe = null; this._undoStack = null; this._styler = null; - this._lines_ruler = null; + this._linesRuler = null; + this._dragAndDrop = null; + this._annotationModel = null; + this._annotationStyler = null; this._view = null; + this._model = null; this._config = null; }, }; diff --git a/browser/devtools/sourceeditor/source-editor-textarea.jsm b/browser/devtools/sourceeditor/source-editor-textarea.jsm index 07417df8fe12..07879c44e614 100644 --- a/browser/devtools/sourceeditor/source-editor-textarea.jsm +++ b/browser/devtools/sourceeditor/source-editor-textarea.jsm @@ -231,7 +231,7 @@ SourceEditor.prototype = { let listeners = this._listeners[SourceEditor.EVENTS.SELECTION] || []; listeners.forEach(function(aListener) { - aListener.callback.call(null, sendEvent, aListener.data); + aListener.callback.call(null, sendEvent); }, this); this._lastSelection = selection; @@ -256,7 +256,7 @@ SourceEditor.prototype = { { let listeners = this._listeners[SourceEditor.EVENTS.TEXT_CHANGED] || []; listeners.forEach(function(aListener) { - aListener.callback.call(null, aEvent, aListener.data); + aListener.callback.call(null, aEvent); }, this); }, @@ -279,16 +279,13 @@ SourceEditor.prototype = { * The event type you want to listen for. * @param function aCallback * The function you want executed when the event is triggered. - * @param mixed [aData] - * Optional data to pass to the callback when the event is triggered. */ addEventListener: - function SE_addEventListener(aEventType, aCallback, aData) + function SE_addEventListener(aEventType, aCallback) { const EVENTS = SourceEditor.EVENTS; let listener = { type: aEventType, - data: aData, callback: aCallback, }; @@ -316,24 +313,20 @@ SourceEditor.prototype = { * The event type you have a listener for. * @param function aCallback * The function you have as the event handler. - * @param mixed [aData] - * The optional data passed to the callback. */ removeEventListener: - function SE_removeEventListener(aEventType, aCallback, aData) + function SE_removeEventListener(aEventType, aCallback) { let listeners = this._listeners[aEventType]; if (!listeners) { - throw new Error("SourceEditor.removeEventListener() called for an " + - "unknown event."); + return; } const EVENTS = SourceEditor.EVENTS; this._listeners[aEventType] = listeners.filter(function(aListener) { let isSameListener = aListener.type == aEventType && - aListener.callback === aCallback && - aListener.data === aData; + aListener.callback === aCallback; if (isSameListener && aListener.domType) { aListener.target.removeEventListener(aListener.domType, aListener.handler, false); @@ -366,7 +359,7 @@ SourceEditor.prototype = { }; aDOMEvent.preventDefault(); - aListener.callback.call(null, sendEvent, aListener.data); + aListener.callback.call(null, sendEvent); }, /** diff --git a/browser/devtools/sourceeditor/source-editor.jsm b/browser/devtools/sourceeditor/source-editor.jsm index c1eb05363b77..37cbfdc098ec 100644 --- a/browser/devtools/sourceeditor/source-editor.jsm +++ b/browser/devtools/sourceeditor/source-editor.jsm @@ -90,7 +90,7 @@ SourceEditor.MODES = { * Predefined themes for syntax highlighting. */ SourceEditor.THEMES = { - TEXTMATE: "textmate", + MOZILLA: "mozilla", }; /** @@ -98,7 +98,7 @@ SourceEditor.THEMES = { */ SourceEditor.DEFAULTS = { MODE: SourceEditor.MODES.TEXT, - THEME: SourceEditor.THEMES.TEXTMATE, + THEME: SourceEditor.THEMES.MOZILLA, UNDO_LIMIT: 200, TAB_SIZE: 4, // overriden by pref EXPAND_TAB: true, // overriden by pref diff --git a/browser/devtools/sourceeditor/test/Makefile.in b/browser/devtools/sourceeditor/test/Makefile.in index f72df2710988..6880e106451f 100644 --- a/browser/devtools/sourceeditor/test/Makefile.in +++ b/browser/devtools/sourceeditor/test/Makefile.in @@ -50,6 +50,8 @@ _BROWSER_TEST_FILES = \ browser_bug687573_vscroll.js \ browser_bug687568_pagescroll.js \ browser_bug687580_drag_and_drop.js \ + browser_bug695035_middle_click_paste.js \ + head.js \ libs:: $(_BROWSER_TEST_FILES) $(INSTALL) $(foreach f,$^,"$f") $(DEPTH)/_tests/testing/mochitest/browser/$(relativesrcdir) diff --git a/browser/devtools/sourceeditor/test/browser_bug687580_drag_and_drop.js b/browser/devtools/sourceeditor/test/browser_bug687580_drag_and_drop.js index e3c0370230cd..1bb9518282e8 100644 --- a/browser/devtools/sourceeditor/test/browser_bug687580_drag_and_drop.js +++ b/browser/devtools/sourceeditor/test/browser_bug687580_drag_and_drop.js @@ -16,7 +16,6 @@ function test() ok(true, "skip test for bug 687580: only applicable for Orion"); return; // Testing for the fix requires direct Orion API access. } - waitForExplicitFinish(); const windowUrl = "data:application/vnd.mozilla.xul+xml," + @@ -71,7 +70,7 @@ function editorLoaded() let ds = Cc["@mozilla.org/widget/dragservice;1"]. getService(Ci.nsIDragService); - let target = view._dragNode; + let target = view._clientDiv; let targetWin = target.ownerDocument.defaultView; let dataTransfer = null; @@ -93,7 +92,9 @@ function editorLoaded() target.removeEventListener("drop", onDrop, false); let selection = editor.getSelection(); - is(selection.start, selection.end, "selection is collapsed"); + is(selection.end - selection.start, + initialSelection.end - initialSelection.start, + "selection is correct"); is(editor.getText(0, 2), "l3", "drag and drop worked"); let offset = editor.getCaretOffset(); diff --git a/browser/devtools/sourceeditor/test/browser_bug695035_middle_click_paste.js b/browser/devtools/sourceeditor/test/browser_bug695035_middle_click_paste.js new file mode 100644 index 000000000000..71e1501e109d --- /dev/null +++ b/browser/devtools/sourceeditor/test/browser_bug695035_middle_click_paste.js @@ -0,0 +1,84 @@ +/* vim: set ts=2 et sw=2 tw=80: */ +/* Any copyright is dedicated to the Public Domain. + http://creativecommons.org/publicdomain/zero/1.0/ */ + +"use strict"; + +Cu.import("resource:///modules/source-editor.jsm"); + +let testWin; +let editor; + +function test() +{ + if (Services.appinfo.OS != "Linux") { + ok(true, "this test only applies to Linux, skipping.") + return; + } + + waitForExplicitFinish(); + + const windowUrl = "data:text/xml," + + ""; + const windowFeatures = "chrome,titlebar,toolbar,centerscreen,resizable,dialog=no"; + + testWin = Services.ww.openWindow(null, windowUrl, "_blank", windowFeatures, null); + testWin.addEventListener("load", function onWindowLoad() { + testWin.removeEventListener("load", onWindowLoad, false); + waitForFocus(initEditor, testWin); + }, false); +} + +function initEditor() +{ + let hbox = testWin.document.querySelector("hbox"); + + editor = new SourceEditor(); + editor.init(hbox, {}, editorLoaded); +} + +function editorLoaded() +{ + editor.focus(); + + let initialText = "initial text!"; + + editor.setText(initialText); + + let expectedString = "foobarBug695035-" + Date.now(); + + let doCopy = function() { + let clipboardHelper = Cc["@mozilla.org/widget/clipboardhelper;1"]. + getService(Ci.nsIClipboardHelper); + clipboardHelper.copyStringToClipboard(expectedString, + Ci.nsIClipboard.kSelectionClipboard); + }; + + let onCopy = function() { + editor.addEventListener(SourceEditor.EVENTS.TEXT_CHANGED, onPaste); + + EventUtils.synthesizeMouse(editor.editorElement, 2, 2, {}, testWin); + EventUtils.synthesizeMouse(editor.editorElement, 3, 3, {button: 1}, testWin); + }; + + let onPaste = function() { + editor.removeEventListener(SourceEditor.EVENTS.TEXT_CHANGED, onPaste); + + is(editor.getText(), expectedString + initialText, "middle-click paste works"); + + executeSoon(testEnd); + }; + + waitForSelection(expectedString, doCopy, onCopy, testEnd); +} + +function testEnd() +{ + editor.destroy(); + testWin.close(); + + testWin = editor = null; + + waitForFocus(finish, window); +} diff --git a/browser/devtools/sourceeditor/test/browser_sourceeditor_initialization.js b/browser/devtools/sourceeditor/test/browser_sourceeditor_initialization.js index db05396075c7..7ffc964f2d80 100644 --- a/browser/devtools/sourceeditor/test/browser_sourceeditor_initialization.js +++ b/browser/devtools/sourceeditor/test/browser_sourceeditor_initialization.js @@ -412,16 +412,16 @@ function testEclipseBug362107() editor.setCaretOffset(16); EventUtils.synthesizeKey("VK_UP", {ctrlKey: true}, testWin); - is(editor.getCaretOffset(), 7, "Ctrl-Up works"); + is(editor.getCaretOffset(), 9, "Ctrl-Up works"); EventUtils.synthesizeKey("VK_UP", {ctrlKey: true}, testWin); - is(editor.getCaretOffset(), 0, "Ctrl-Up works twice"); + is(editor.getCaretOffset(), 2, "Ctrl-Up works twice"); EventUtils.synthesizeKey("VK_DOWN", {ctrlKey: true}, testWin); - is(editor.getCaretOffset(), 13, "Ctrl-Down works"); + is(editor.getCaretOffset(), 9, "Ctrl-Down works"); EventUtils.synthesizeKey("VK_DOWN", {ctrlKey: true}, testWin); - is(editor.getCaretOffset(), 20, "Ctrl-Down works twice"); + is(editor.getCaretOffset(), 16, "Ctrl-Down works twice"); } function testBug687577() diff --git a/browser/devtools/sourceeditor/test/head.js b/browser/devtools/sourceeditor/test/head.js new file mode 100644 index 000000000000..635bbaa69b6a --- /dev/null +++ b/browser/devtools/sourceeditor/test/head.js @@ -0,0 +1,107 @@ +/* vim: set ts=2 et sw=2 tw=80: */ +/* Any copyright is dedicated to the Public Domain. + http://creativecommons.org/publicdomain/zero/1.0/ */ + +"use strict"; + +/* + * Polls the X11 primary selection buffer waiting for the expected value. A known + * value different than the expected value is put on the clipboard first (and + * also polled for) so we can be sure the value we get isn't just the expected + * value because it was already in the buffer. + * + * @param aExpectedStringOrValidatorFn + * The string value that is expected to be in the X11 primary selection buffer + * or a validator function getting clipboard data and returning a bool. + * @param aSetupFn + * A function responsible for setting the primary selection buffer to the + * expected value, called after the known value setting succeeds. + * @param aSuccessFn + * A function called when the expected value is found in the primary + * selection buffer. + * @param aFailureFn + * A function called if the expected value isn't found in the primary + * selection buffer within 5s. It can also be called if the known value + * can't be found. + * @param aFlavor [optional] The flavor to look for. Defaults to "text/unicode". + */ +function waitForSelection(aExpectedStringOrValidatorFn, aSetupFn, + aSuccessFn, aFailureFn, aFlavor) { + let requestedFlavor = aFlavor || "text/unicode"; + + // Build a default validator function for common string input. + var inputValidatorFn = typeof(aExpectedStringOrValidatorFn) == "string" + ? function(aData) aData == aExpectedStringOrValidatorFn + : aExpectedStringOrValidatorFn; + + let clipboard = Cc["@mozilla.org/widget/clipboard;1"]. + getService(Ci.nsIClipboard); + + // reset for the next use + function reset() { + waitForSelection._polls = 0; + } + + function wait(validatorFn, successFn, failureFn, flavor) { + if (++waitForSelection._polls > 50) { + // Log the failure. + ok(false, "Timed out while polling the X11 primary selection buffer."); + reset(); + failureFn(); + return; + } + + let transferable = Cc["@mozilla.org/widget/transferable;1"]. + createInstance(Ci.nsITransferable); + transferable.addDataFlavor(requestedFlavor); + + clipboard.getData(transferable, clipboard.kSelectionClipboard); + + let str = {}; + let strLength = {}; + + transferable.getTransferData(requestedFlavor, str, strLength); + + let data = null; + if (str.value) { + let strValue = str.value.QueryInterface(Ci.nsISupportsString); + data = strValue.data.substring(0, strLength.value / 2); + } + + if (validatorFn(data)) { + // Don't show the success message when waiting for preExpectedVal + if (preExpectedVal) { + preExpectedVal = null; + } else { + ok(true, "The X11 primary selection buffer has the correct value"); + } + reset(); + successFn(); + } else { + setTimeout(function() wait(validatorFn, successFn, failureFn, flavor), 100); + } + } + + // First we wait for a known value different from the expected one. + var preExpectedVal = waitForSelection._monotonicCounter + + "-waitForSelection-known-value"; + + let clipboardHelper = Cc["@mozilla.org/widget/clipboardhelper;1"]. + getService(Ci.nsIClipboardHelper); + clipboardHelper.copyStringToClipboard(preExpectedVal, + Ci.nsIClipboard.kSelectionClipboard); + + wait(function(aData) aData == preExpectedVal, + function() { + // Call the original setup fn + aSetupFn(); + wait(inputValidatorFn, aSuccessFn, aFailureFn, requestedFlavor); + }, aFailureFn, "text/unicode"); +} + +waitForSelection._polls = 0; +waitForSelection.__monotonicCounter = 0; +waitForSelection.__defineGetter__("_monotonicCounter", function () { + return waitForSelection.__monotonicCounter++; +}); +