make Mandelbrot work nicely inside a tab in Fennec 4 - some remaining problems/confli...
[mandelbrot.git] / xulapp / chrome / mandelbrot / content / mandelbrot.js
index 1ead31e1b2cb4e0423355e5690929c69dfcf864d..72154191907dd4592a2f881ac53280fcb87f9d85 100644 (file)
@@ -41,14 +41,66 @@ var gPref = Components.classes["@mozilla.org/preferences-service;1"]
                       .getBranch(null);
 var gStartTime = 0;
 var gMbrotBundle;
+var gCurrentImageData;
 
 function Startup() {
   updateIterMenu();
+  updateAlgoMenu();
   updatePaletteMenu();
   gMbrotBundle = document.getElementById("mbrotBundle");
   document.getElementById("statusLabel").value = gMbrotBundle.getString("statusEmpty");
 }
 
+function adjustCoordsAndDraw(aC_min, aC_max) {
+  let iWidth = 0;
+  try {
+    iWidth = gPref.getIntPref("mandelbrot.image.width");
+  }
+  catch (e) { }
+  if ((iWidth < 10) || (iWidth > 5000)) {
+    iWidth = 300;
+    gPref.setIntPref("mandelbrot.image.width", iWidth);
+  }
+  let iHeight = 0;
+  try {
+    iHeight = gPref.getIntPref("mandelbrot.image.height");
+  }
+  catch (e) { }
+  if ((iHeight < 10) || (iHeight > 5000)) {
+    iHeight = 300;
+    gPref.setIntPref("mandelbrot.image.height", iHeight);
+  }
+
+  // correct coordinates
+  if (aC_min.r < -2)
+    aC_min.r = -2;
+  if (aC_max.r > 2)
+    aC_max.r = 2;
+  if ((aC_min.r > 2) || (aC_max.r < -2) || (aC_min.r >= aC_max.r)) {
+    aC_min.r = -2.0; aC_max.r = 1.0;
+  }
+  if (aC_min.i < -2)
+    aC_min.i = -2;
+  if (aC_max.i > 2)
+    aC_max.i = 2;
+  if ((aC_min.i > 2) || (aC_max.i < -2) || (aC_min.i >= aC_max.i)) {
+    aC_min.i = -1.3; aC_max.i = 1.3;
+  }
+
+  let CWidth = aC_max.r - aC_min.r;
+  let CHeight = aC_max.i - aC_min.i;
+  let C_mid = new complex(aC_min.r + CWidth / 2, aC_min.i + CHeight / 2);
+
+  let CRatio = Math.max(CWidth / iWidth, CHeight / iHeight);
+
+  gPref.setCharPref("mandelbrot.last_image.Cr_min", C_mid.r - iWidth * CRatio / 2);
+  gPref.setCharPref("mandelbrot.last_image.Cr_max", C_mid.r + iWidth * CRatio / 2);
+  gPref.setCharPref("mandelbrot.last_image.Ci_min", C_mid.i - iHeight * CRatio / 2);
+  gPref.setCharPref("mandelbrot.last_image.Ci_max", C_mid.i + iHeight * CRatio / 2);
+
+  drawImage();
+}
+
 function drawImage() {
   let canvas = document.getElementById("mbrotImage");
   let context = canvas.getContext("2d");
@@ -64,8 +116,8 @@ function drawImage() {
     Cr_max = parseFloat(gPref.getCharPref("mandelbrot.last_image.Cr_max"));
   }
   catch (e) { }
-  if ((Cr_min < -2) || (Cr_min > 2) ||
-      (Cr_max < -2) || (Cr_max > 2) || (Cr_min >= Cr_max)) {
+  if ((Cr_min < -3) || (Cr_min > 2) ||
+      (Cr_max < -3) || (Cr_max > 2) || (Cr_min >= Cr_max)) {
     Cr_min = -2.0; Cr_max = 1.0;
   }
   gPref.setCharPref("mandelbrot.last_image.Cr_min", Cr_min);
@@ -78,9 +130,9 @@ function drawImage() {
     Ci_max = parseFloat(gPref.getCharPref("mandelbrot.last_image.Ci_max"));
   }
   catch (e) { }
-  if ((Ci_min < -2) || (Ci_min > 2) ||
-      (Ci_max < -2) || (Ci_max > 2) || (Ci_min >= Ci_max)) {
-    Ci_min = -2.0; Ci_max = 1.0;
+  if ((Ci_min < -2.5) || (Ci_min > 2.5) ||
+      (Ci_max < -2.5) || (Ci_max > 2.5) || (Ci_min >= Ci_max)) {
+    Ci_min = -1.5; Ci_max = 1.5;
   }
   gPref.setCharPref("mandelbrot.last_image.Ci_min", Ci_min);
   gPref.setCharPref("mandelbrot.last_image.Ci_max", Ci_max);
@@ -107,6 +159,14 @@ function drawImage() {
     gPref.setIntPref("mandelbrot.image.height", iHeight);
   }
 
+  gCurrentImageData = {
+    C_min: new complex(Cr_min, Ci_min),
+    C_max: new complex(Cr_max, Ci_max),
+    iWidth: iWidth,
+    iHeight: iHeight,
+    iterMax: iterMax
+  };
+
   canvas.width = iWidth;
   canvas.height = iHeight;
 
@@ -120,27 +180,27 @@ function drawImage() {
 }
 
 function drawLine(line, dimensions, canvas, context, iterMax, algorithm) {
-    let Cr_min = dimensions[0];
-    let Cr_max = dimensions[1];
-    let Cr_scale = Cr_max - Cr_min;
-
-    let Ci_min = dimensions[2];
-    let Ci_max = dimensions[3];
-    let Ci_scale = Ci_max - Ci_min;
-
-    let pixels = [];
-    for (var img_y = line; img_y < canvas.height && img_y < line+8; img_y++)
-      for (let img_x = 0; img_x < canvas.width; img_x++) {
-        let C = new complex(Cr_min + (img_x / canvas.width) * Cr_scale,
-                            Ci_min + (img_y / canvas.height) * Ci_scale);
-        pixels.push.apply(pixels, drawPoint(context, img_x, img_y, C, iterMax, algorithm));
-      }
-    context.putImageData({width: canvas.width, height: pixels.length/4/canvas.width, data: pixels}, 0, line);
+  let Cr_min = dimensions[0];
+  let Cr_max = dimensions[1];
+  let Cr_scale = Cr_max - Cr_min;
+
+  let Ci_min = dimensions[2];
+  let Ci_max = dimensions[3];
+  let Ci_scale = Ci_max - Ci_min;
+
+  let pixels = [];
+  for (var img_y = line; img_y < canvas.height && img_y < line+8; img_y++)
+    for (let img_x = 0; img_x < canvas.width; img_x++) {
+      let C = new complex(Cr_min + (img_x / canvas.width) * Cr_scale,
+                          Ci_min + (img_y / canvas.height) * Ci_scale);
+      pixels.push.apply(pixels, drawPoint(context, img_x, img_y, C, iterMax, algorithm));
+    }
+  context.putImageData({width: canvas.width, height: pixels.length/4/canvas.width, data: pixels}, 0, line);
 
-    if (img_y < canvas.height)
-      setTimeout(drawLine, 0, img_y, dimensions, canvas, context, iterMax, algorithm);
-    else if (gStartTime)
-      EndCalc();
+  if (img_y < canvas.height)
+    setTimeout(drawLine, 0, img_y, dimensions, canvas, context, iterMax, algorithm);
+  else if (gStartTime)
+    EndCalc();
 }
 
 function EndCalc() {
@@ -342,74 +402,123 @@ function drawPoint(context, img_x, img_y, C, iterMax, algorithm) {
 /***** pure UI functions *****/
 
 var zoomstart;
+var imgBackup;
 
 function mouseevent(etype, event) {
   let canvas = document.getElementById("mbrotImage");
+  let context = canvas.getContext("2d");
   switch (etype) {
     case 'down':
-      if (event.button == 0)
+      if (event.button == 0) {
         // left button - start dragzoom
         zoomstart = {x: event.clientX - canvas.offsetLeft,
                      y: event.clientY - canvas.offsetTop};
+        imgBackup = context.getImageData(0, 0, canvas.width, canvas.height);
+      }
       break;
     case 'up':
-      if (event.button == 0) {
-        var prompts = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
+      if (event.button == 0 && zoomstart) {
+        context.putImageData(imgBackup, 0, 0);
+        let zoomend = {x: event.clientX - canvas.offsetLeft,
+                       y: event.clientY - canvas.offsetTop};
+
+        // make sure zoomend is bigger than zoomstart
+        if ((zoomend.x == zoomstart.x) || (zoomend.y == zoomstart.y)) {
+          // cannot zoom what has no area, discard it
+          zoomstart = undefined;
+          return;
+        }
+        if (zoomend.x < zoomstart.x)
+          [zoomend.x, zoomstart.x] = [zoomstart.x, zoomend.x];
+        if (zoomend.y < zoomstart.y)
+          [zoomend.y, zoomstart.y] = [zoomstart.y, zoomend.y];
+
+        let prompts = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
                                 .getService(Components.interfaces.nsIPromptService);
-        var ok = prompts.confirm(null, gMbrotBundle.getString("zoomConfirmTitle"),
-            gMbrotBundle.getString("zoomConfirmLabel") + ' --- ' +
-            zoomstart.x + ',' + zoomstart.y + '-' +
-            (event.clientX - canvas.offsetLeft) + ',' +
-            (event.clientY - canvas.offsetTop));
+        let ok = prompts.confirm(null, gMbrotBundle.getString("zoomConfirmTitle"),
+                                 gMbrotBundle.getString("zoomConfirmLabel"));
         // ok is now true if OK was clicked, and false if cancel was clicked
+        if (ok) {
+          // determine new "coordinates"
+          let CWidth = gCurrentImageData.C_max.r - gCurrentImageData.C_min.r;
+          let CHeight = gCurrentImageData.C_max.i - gCurrentImageData.C_min.i;
+          let newC_min = new complex(
+              gCurrentImageData.C_min.r + zoomstart.x / gCurrentImageData.iWidth * CWidth,
+              gCurrentImageData.C_min.i + zoomstart.y / gCurrentImageData.iHeight * CHeight);
+          let newC_max = new complex(
+              gCurrentImageData.C_min.r + zoomend.x / gCurrentImageData.iWidth * CWidth,
+              gCurrentImageData.C_min.i + zoomend.y / gCurrentImageData.iHeight * CHeight);
+
+          adjustCoordsAndDraw(newC_min, newC_max);
+        }
       }
       zoomstart = undefined;
+      break;
+    case 'move':
+      if (event.button == 0 && zoomstart) {
+        context.putImageData(imgBackup, 0, 0);
+        context.strokeStyle = "rgb(255,255,31)";
+        context.strokeRect(zoomstart.x, zoomstart.y,
+                           event.clientX - canvas.offsetLeft - zoomstart.x,
+                           event.clientY - canvas.offsetTop - zoomstart.y);
+      }
     break;
   }
 }
 
 function saveImage() {
   const nsIFilePicker = Components.interfaces.nsIFilePicker;
-  var fp = null;
+  let fp = null;
   try {
     fp = Components.classes["@mozilla.org/filepicker;1"]
                    .createInstance(nsIFilePicker);
   } catch (e) {}
   if (!fp) return;
-  var promptString = gMbrotBundle.getString("savePrompt");
+  let promptString = gMbrotBundle.getString("savePrompt");
   fp.init(window, promptString, nsIFilePicker.modeSave);
   fp.appendFilter(gMbrotBundle.getString("pngFilterName"), "*.png");
   fp.defaultString = "mandelbrot.png";
 
-  var fpResult = fp.show();
+  let fpResult = fp.show();
   if (fpResult != nsIFilePicker.returnCancel) {
     saveCanvas(document.getElementById("mbrotImage"), fp.file);
   }
 }
 
+function exitMandelbrot() {
+  var appInfo = Components.classes["@mozilla.org/xre/app-info;1"]
+                          .getService(Components.interfaces.nsIXULAppInfo);
+  if (appInfo.ID == "mandelbrot@kairo.at")
+    quitApp(false);
+  else
+    window.close();
+}
+
 function updateBookmarkMenu(aParent) {
   document.getElementById("bookmarkSave").disabled =
     (!document.getElementById("drawButton").hidden || (gStartTime > 0));
 
   while (aParent.hasChildNodes() &&
-         aParent.lastChild.id != 'bookmarkSeparator')
+         aParent.lastChild.id != "bookmarkSeparator")
     aParent.removeChild(aParent.lastChild);
 
-  var file = Components.classes["@mozilla.org/file/directory_service;1"]
+  let file = Components.classes["@mozilla.org/file/directory_service;1"]
                        .getService(Components.interfaces.nsIProperties)
                        .get("ProfD", Components.interfaces.nsIFile);
   file.append("mandelbookmarks.sqlite");
   if (file.exists()) {
-    var connection = Components.classes["@mozilla.org/storage/service;1"]
+    let connection = Components.classes["@mozilla.org/storage/service;1"]
                                .getService(Components.interfaces.mozIStorageService)
                                .openDatabase(file);
     try {
       if (connection.tableExists("bookmarks")) {
-        var statement = connection.createStatement(
-            "SELECT name FROM bookmarks ORDER BY ROWID DESC");
-        while (statement.executeStep())
-          aParent.appendChild(document.createElement("menuitem"))
-                 .setAttribute("label", statement.getString(0));
+        let statement = connection.createStatement(
+            "SELECT name,ROWID FROM bookmarks ORDER BY ROWID ASC");
+        while (statement.executeStep()) {
+          let newItem = aParent.appendChild(document.createElement("menuitem"));
+          newItem.setAttribute("label", statement.getString(0));
+          newItem.setAttribute("bmRowID", statement.getString(1));
+        }
         statement.reset();
         statement.finalize();
         return;
@@ -419,32 +528,67 @@ function updateBookmarkMenu(aParent) {
     }
   }
   // Create the "Nothing Available" Menu item and disable it.
-  var na = aParent.appendChild(document.createElement("menuitem"));
+  let na = aParent.appendChild(document.createElement("menuitem"));
   na.setAttribute("label", gMbrotBundle.getString("noBookmarks"));
   na.setAttribute("disabled", "true");
 }
 
 function callBookmark(evtarget) {
+  if (evtarget.id == "bookmarkSave" || evtarget.id == "bookmarkSeparator")
+    return;
+  if (evtarget.id == "bookmarkOverview") {
+    adjustCoordsAndDraw(new complex(0,0), new complex(0,0));
+    return;
+  }
+
+  if (evtarget.getAttribute('bmRowID')) {
+    let iterMax = 0;
+    let C_min = null;
+    let C_max = null;
+
+    let file = Components.classes["@mozilla.org/file/directory_service;1"]
+                         .getService(Components.interfaces.nsIProperties)
+                         .get("ProfD", Components.interfaces.nsIFile);
+    file.append("mandelbookmarks.sqlite");
+    let connection = Components.classes["@mozilla.org/storage/service;1"]
+                               .getService(Components.interfaces.mozIStorageService)
+                               .openDatabase(file);
+    let statement = connection.createStatement(
+        "SELECT iteration_max,Cr_min,Cr_max,Ci_min,Ci_max FROM bookmarks WHERE ROWID=?1");
+    statement.bindStringParameter(0, evtarget.getAttribute('bmRowID'));
+    while (statement.executeStep()) {
+      iterMax = statement.getInt32(0);
+      C_min = new complex(statement.getDouble(1), statement.getDouble(3));
+      C_max = new complex(statement.getDouble(2), statement.getDouble(4));
+    }
+    statement.finalize();
+    connection.close();
+
+    if (iterMax && C_min && C_max) {
+      gPref.setIntPref("mandelbrot.iteration_max", iterMax);
+      adjustCoordsAndDraw(C_min, C_max);
+    }
+  }
 }
 
 function saveBookmark() {
   // retrieve wanted bookmark name with a prompt
-  var prompts = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
+  let prompts = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
                           .getService(Components.interfaces.nsIPromptService);
-  var input = {value: ""}; // empty default value
-  var ok = prompts.prompt(null, gMbrotBundle.getString("saveBookmarkTitle"), gMbrotBundle.getString("saveBookmarkLabel"), input, null, {});
+  let input = {value: ""}; // empty default value
+  let ok = prompts.prompt(null, gMbrotBundle.getString("saveBookmarkTitle"), gMbrotBundle.getString("saveBookmarkLabel"), input, null, {});
   // ok is true if OK is pressed, false if Cancel. input.value holds the value of the edit field if "OK" was pressed.
   if (!ok || !input.value)
     return
 
-  var bmName = input.value;
+  let bmName = input.value;
 
   // Open or create the bookmarks database.
-  var file = Components.classes["@mozilla.org/file/directory_service;1"]
+  let file = Components.classes["@mozilla.org/file/directory_service;1"]
                        .getService(Components.interfaces.nsIProperties)
                        .get("ProfD", Components.interfaces.nsIFile);
   file.append("mandelbookmarks.sqlite");
-  var connection = Components.classes["@mozilla.org/storage/service;1"]
+  let connection = Components.classes["@mozilla.org/storage/service;1"]
                              .getService(Components.interfaces.mozIStorageService)
                              .openDatabase(file);
   connection.beginTransaction();
@@ -456,14 +600,14 @@ function saveBookmark() {
   // TEXT. The value is a text string, stored using the database encoding (UTF-8, UTF-16BE or UTF-16-LE).
 
   // Put value of the current image into the bookmarks table
-  var statement = connection.createStatement(
+  let statement = connection.createStatement(
       "INSERT INTO bookmarks (name,iteration_max,Cr_min,Cr_max,Ci_min,Ci_max) VALUES (?1,?2,?3,?4,?5,?6)");
   statement.bindStringParameter(0, bmName);
-  statement.bindStringParameter(1, gPref.getIntPref("mandelbrot.iteration_max"));
-  statement.bindStringParameter(2, parseFloat(gPref.getCharPref("mandelbrot.last_image.Cr_min")));
-  statement.bindStringParameter(3, parseFloat(gPref.getCharPref("mandelbrot.last_image.Cr_max")));
-  statement.bindStringParameter(4, parseFloat(gPref.getCharPref("mandelbrot.last_image.Ci_min")));
-  statement.bindStringParameter(5, parseFloat(gPref.getCharPref("mandelbrot.last_image.Ci_max")));
+  statement.bindStringParameter(1, gCurrentImageData.iterMax);
+  statement.bindStringParameter(2, gCurrentImageData.C_min.r);
+  statement.bindStringParameter(3, gCurrentImageData.C_max.r);
+  statement.bindStringParameter(4, gCurrentImageData.C_min.i);
+  statement.bindStringParameter(5, gCurrentImageData.C_max.i);
   statement.execute();
   statement.finalize();
   connection.commitTransaction();
@@ -471,19 +615,18 @@ function saveBookmark() {
 }
 
 function updateIterMenu() {
+  let currentIter = 0;
   try {
-    var currentIter = gPref.getIntPref("mandelbrot.iteration_max");
-  }
-  catch(e) {
-    var currentIter = 0;
+    currentIter = gPref.getIntPref("mandelbrot.iteration_max");
   }
+  catch(e) { }
   if (currentIter < 10) {
     currentIter = 500;
     setIter(currentIter);
   }
 
-  var popup = document.getElementById("menu_iterPopup");
-  var item = popup.firstChild;
+  let popup = document.getElementById("menu_iterPopup");
+  let item = popup.firstChild;
   while (item) {
     if (item.getAttribute("name") == "iter") {
       if (item.getAttribute("value") == currentIter)
@@ -500,12 +643,11 @@ function setIter(aIter) {
 }
 
 function updatePaletteMenu() {
+  let currentPalette = '';
   try {
-    var currentPalette = gPref.getCharPref("mandelbrot.color_palette");
-  }
-  catch(e) {
-    var currentPalette = '';
+    currentPalette = gPref.getCharPref("mandelbrot.color_palette");
   }
+  catch(e) { }
   if (!currentPalette.length) {
     currentPalette = 'kairo';
     setPalette(currentPalette);
@@ -513,8 +655,8 @@ function updatePaletteMenu() {
   if (!gColorPalette || !gColorPalette.length)
     gColorPalette = getColorPalette(currentPalette);
 
-  var popup = document.getElementById("menu_palettePopup");
-  var item = popup.firstChild;
+  let popup = document.getElementById("menu_palettePopup");
+  let item = popup.firstChild;
   while (item) {
     if (item.getAttribute("name") == "palette") {
       if (item.getAttribute("value") == currentPalette)
@@ -536,30 +678,34 @@ function imgSettings() {
 }
 
 function updateDebugMenu() {
-  var jitMenuItem = document.getElementById("jitEnabled");
-  jitMenuItem.setAttribute("checked", gPref.getBoolPref("javascript.options.jit.chrome"));
+  let scope = (document.getElementById("mandelbrotWindow").nodeName == "page") ? "content" : "chrome";
+  for each (let type in ["tracejit", "methodjit"]) {
+    let jitMenuItem = document.getElementById(type + "Enabled");
+    jitMenuItem.setAttribute("checked", gPref.getBoolPref("javascript.options." + type + "." + scope));
+  }
 }
 
-function toggleJITState(jitMenuItem) {
-  var jitEnabled = !gPref.getBoolPref("javascript.options.jit.chrome");
-  gPref.setBoolPref("javascript.options.jit.chrome", jitEnabled)
-  jitMenuItem.setAttribute("checked", jitEnabled? "true" : "false");
+function toggleJITState(jitMenuItem, jittype) {
+  let scope = (document.getElementById("mandelbrotWindow").nodeName == "page") ? "content" : "chrome";
+  let jitpref = "javascript.options." + jittype + "jit." + scope;
+  let jitEnabled = !gPref.getBoolPref(jitpref);
+  gPref.setBoolPref(jitpref, jitEnabled)
+  jitMenuItem.setAttribute("checked", jitEnabled ? "true" : "false");
 }
 
 function updateAlgoMenu() {
+  let currentAlgo = '';
   try {
-    var currentAlgo = gPref.getCharPref("mandelbrot.use_algorithm");
-  }
-  catch(e) {
-    var currentAlgo = '';
+    currentAlgo = gPref.getCharPref("mandelbrot.use_algorithm");
   }
+  catch(e) { }
   if (!currentAlgo.length) {
     currentAlgo = 'numeric';
     setAlgorithm(currentAlgo);
   }
 
-  var popup = document.getElementById("menu_algoPopup");
-  var item = popup.firstChild;
+  let popup = document.getElementById("menu_algoPopup");
+  let item = popup.firstChild;
   while (item) {
     if (item.getAttribute("name") == "algorithm") {
       if (item.getAttribute("value") == currentAlgo)
@@ -576,7 +722,7 @@ function setAlgorithm(algoID) {
 }
 
 function addonsManager(aPane) {
-  var theEM = Components.classes["@mozilla.org/appshell/window-mediator;1"]
+  let theEM = Components.classes["@mozilla.org/appshell/window-mediator;1"]
                         .getService(Components.interfaces.nsIWindowMediator)
                         .getMostRecentWindow("Extension:Manager");
   if (theEM) {