pluotsorbet/midp/gfx.js

1188 строки
45 KiB
JavaScript

/* -*- Mode: Java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/* vim: set shiftwidth=4 tabstop=4 autoindent cindent expandtab: */
'use strict';
(function(Native) {
Native.create("com/sun/midp/lcdui/DisplayDeviceContainer.getDisplayDevicesIds0.()[I", function() {
var ids = util.newPrimitiveArray("I", 1);
ids[0] = 1;
return ids;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.getDisplayName0.(I)Ljava/lang/String;", function(id) {
return null;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.isDisplayPrimary0.(I)Z", function(id) {
console.warn("DisplayDevice.isDisplayPrimary0.(I)Z not implemented (" + id + ")");
return true;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.isbuildInDisplay0.(I)Z", function(id) {
return true;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.getDisplayCapabilities0.(I)I", function(id) {
return 0x3ff;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.isDisplayPenSupported0.(I)Z", function(id) {
return true;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.isDisplayPenMotionSupported0.(I)Z", function(id) {
return true;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.getReverseOrientation0.(I)Z", function(id) {
return false;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.getScreenWidth0.(I)I", function(id) {
return MIDP.Context2D.canvas.width;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.getScreenHeight0.(I)I", function(id) {
return MIDP.Context2D.canvas.height;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.displayStateChanged0.(II)V", function(hardwareId, state) {
console.warn("DisplayDevice.displayStateChanged0.(II)V not implemented (" + hardwareId + ", " + state + ")");
});
Native.create("com/sun/midp/lcdui/DisplayDevice.setFullScreen0.(IIZ)V", function(hardwareId, displayId, mode) {
console.warn("DisplayDevice.setFullScreen0.(IIZ)V not implemented (" +
hardwareId + ", " + displayId + ", " + mode + ")");
});
Native.create("com/sun/midp/lcdui/DisplayDevice.gainedForeground0.(II)V", function(hardwareId, displayId) {
console.warn("DisplayDevice.gainedForeground0.(II)V not implemented (" + hardwareId + ", " + displayId + ")");
});
Native.create("com/sun/midp/lcdui/DisplayDeviceAccess.vibrate0.(IZ)Z", function(displayId, on) {
return true;
});
Native.create("com/sun/midp/lcdui/DisplayDeviceAccess.isBacklightSupported0.(I)Z", function(displayId) {
return true;
});
Native.create("com/sun/midp/lcdui/DisplayDevice.refresh0.(IIIIII)V", function(hardwareId, displayId, x1, y1, x2, y2) {
console.warn("DisplayDevice.refresh0.(IIIIII)V not implemented (" +
hardwareId + ", " + displayId + ", " + x1 + ", " + y1 + ", " + x2 + ", " + y2 + ")");
});
function swapRB(pixel) {
return (pixel & 0xff00ff00) | ((pixel >> 16) & 0xff) | ((pixel & 0xff) << 16);
}
function swapRBAndSetAlpha(pixel) {
return swapRB(pixel) | 0xff000000;
}
/**
* Extract the image data from `context` and place it in `rgbData`.
*/
function contextToRgbData(context, rgbData, offset, scanlength, x, y, width, height, converterFunc) {
var pixels = new Uint32Array(context.getImageData(x, y, width, height).data.buffer);
var i = 0;
for (var y1 = y; y1 < y + height; y1++) {
for (var x1 = x; x1 < x + width; x1++) {
rgbData[offset + (x1 - x) + (y1 - y) * scanlength] = converterFunc(pixels[i++]);
}
}
}
/**
* Insert `rgbData` into `context`.
*/
function rgbDataToContext(context, rgbData, offset, scanlength, converterFunc) {
var width = context.canvas.width;
var height = context.canvas.height;
var imageData = context.createImageData(width, height);
var pixels = new Uint32Array(imageData.data.buffer);
var i = 0;
for (var y = 0; y < height; ++y) {
var j = offset + y * scanlength;
for (var x = 0; x < width; ++x) {
pixels[i++] = converterFunc(rgbData[j++]);
}
}
context.putImageData(imageData, 0, 0);
}
function createContext2d(width, height) {
var canvas = document.createElement("canvas");
canvas.width = width;
canvas.height = height;
return canvas.getContext("2d");
}
function setImageData(imageData, width, height, data) {
imageData.klass.classInfo.getField("I.width.I").set(imageData, width);
imageData.klass.classInfo.getField("I.height.I").set(imageData, height);
imageData.nativeImageData = data;
}
/**
* Ensure the nativeImageData of the given image points to a
* Canvas Context2D, converting (and saving) it if necessary.
*
* @return {CanvasRenderingContext2D} context
*/
function convertNativeImageData(imageData) {
var data = imageData.nativeImageData;
if (!(data instanceof CanvasRenderingContext2D)) {
// Assume it's an image.
var context = createContext2d(data.width, data.height);
context.drawImage(data, 0, 0);
imageData.nativeImageData = data = context;
}
return data;
}
Native.create("javax/microedition/lcdui/ImageDataFactory.createImmutableImageDecodeImage.(Ljavax/microedition/lcdui/ImageData;[BII)V",
function(imageData, bytes, offset, length) {
return new Promise(function(resolve, reject) {
var blob = new Blob([bytes.subarray(offset, offset + length)], { type: "image/png" });
var img = new Image();
img.src = URL.createObjectURL(blob);
img.onload = function() {
setImageData(imageData, img.naturalWidth, img.naturalHeight, img);
resolve();
}
img.onerror = function(e) {
reject(new JavaException("java/lang/IllegalArgumentException", "error decoding image"));
}
});
}, true);
Native.create("javax/microedition/lcdui/ImageDataFactory.createImmutableImageDataRegion.(Ljavax/microedition/lcdui/ImageData;Ljavax/microedition/lcdui/ImageData;IIIIIZ)V",
function(dataDest, dataSource, x, y, width, height, transform, isMutable) {
var context = createContext2d(width, height);
if (transform === TRANS_MIRROR || transform === TRANS_MIRROR_ROT180) {
context.scale(-1, 1);
} else if (transform === TRANS_MIRROR_ROT90 || transform === TRANS_MIRROR_ROT270) {
context.scale(1, -1);
} else if (transform === TRANS_ROT90 || transform === TRANS_MIRROR_ROT90) {
context.rotate(Math.PI / 2);
} else if (transform === TRANS_ROT180 || transform === TRANS_MIRROR_ROT180) {
context.rotate(Math.PI);
} else if (transform === TRANS_ROT270 || transform === TRANS_MIRROR_ROT270) {
context.rotate(1.5 * Math.PI);
}
context.drawImage(dataSource.nativeImageData, x, y, width, height, 0, 0, width, height);
setImageData(dataDest, width, height, context);
dataDest.klass.classInfo.getField("I.isMutable.Z").set(dataDest, isMutable);
});
Native.create("javax/microedition/lcdui/ImageDataFactory.createMutableImageData.(Ljavax/microedition/lcdui/ImageData;II)V",
function(imageData, width, height) {
var context = createContext2d(width, height);
context.fillStyle = "rgb(255,255,255)"; // white
context.fillRect(0, 0, width, height);
setImageData(imageData, width, height, context);
});
Native.create("javax/microedition/lcdui/ImageDataFactory.createImmutableImageDecodeRGBImage.(Ljavax/microedition/lcdui/ImageData;[IIIZ)V",
function(imageData, rgbData, width, height, processAlpha) {
var context = createContext2d(width, height);
rgbDataToContext(context, rgbData, 0, width, processAlpha ? swapRB : swapRBAndSetAlpha);
setImageData(imageData, width, height, context);
});
Native.create("javax/microedition/lcdui/ImageData.getRGB.([IIIIIII)V", function(rgbData, offset, scanlength, x, y, width, height) {
contextToRgbData(convertNativeImageData(this), rgbData, offset, scanlength, x, y, width, height, swapRB);
});
Native.create("com/nokia/mid/ui/DirectUtils.makeMutable.(Ljavax/microedition/lcdui/Image;)V", function(image) {
var imageData = image.klass.classInfo.getField("I.imageData.Ljavax/microedition/lcdui/ImageData;").get(image);
imageData.klass.classInfo.getField("I.isMutable.Z").set(imageData, 1);
});
Native.create("com/nokia/mid/ui/DirectUtils.setPixels.(Ljavax/microedition/lcdui/Image;I)V", function(image, argb) {
var width = image.klass.classInfo.getField("I.width.I").get(image);
var height = image.klass.classInfo.getField("I.height.I").get(image);
var imageData = image.klass.classInfo.getField("I.imageData.Ljavax/microedition/lcdui/ImageData;").get(image);
var ctx = createContext2d(width, height);
setImageData(imageData, width, height, ctx);
var ctxImageData = ctx.createImageData(width, height);
var pixels = new Uint32Array(ctxImageData.data.buffer);
var color = swapRB(argb);
var i = 0;
for (var y = 0; y < height; ++y) {
for (var x = 0; x < width; ++x) {
pixels[i++] = color;
}
}
ctx.putImageData(ctxImageData, 0, 0);
});
var FACE_SYSTEM = 0;
var FACE_MONOSPACE = 32;
var FACE_PROPORTIONAL = 64;
var STYLE_PLAIN = 0;
var STYLE_BOLD = 1;
var STYLE_ITALIC = 2;
var STYLE_UNDERLINED = 4;
var SIZE_SMALL = 8;
var SIZE_MEDIUM = 0;
var SIZE_LARGE = 16;
Native.create("javax/microedition/lcdui/Font.init.(III)V", function(face, style, size) {
var defaultSize = urlParams.fontSize ? urlParams.fontSize : Math.max(10, (MIDP.Context2D.canvas.height / 35) | 0);
if (size & SIZE_SMALL)
size = defaultSize / 1.25;
else if (size & SIZE_LARGE)
size = defaultSize * 1.25;
else
size = defaultSize;
size |= 0;
if (style & STYLE_BOLD)
style = "bold";
else if (style & STYLE_ITALIC)
style = "italic";
else
style = "";
if (face & FACE_MONOSPACE)
face = "monospace";
else if (face & FACE_PROPORTIONAL)
face = "sans-serif";
else
face = "Arial, Helvetica, sans-serif";
this.klass.classInfo.getField("I.baseline.I").set(this, size | 0);
this.klass.classInfo.getField("I.height.I").set(this, (size * 1.3)|0);
this.css = style + " " + size + "pt " + face;
});
Native.create("javax/microedition/lcdui/Font.stringWidth.(Ljava/lang/String;)I", function(str) {
return withFont(this, MIDP.Context2D, util.fromJavaString(str));
});
Native.create("javax/microedition/lcdui/Font.charWidth.(C)I", function(char) {
return withFont(this, MIDP.Context2D, String.fromCharCode(char));
});
Native.create("javax/microedition/lcdui/Font.charsWidth.([CII)I", function(str, offset, len) {
return withFont(this, MIDP.Context2D, util.fromJavaChars(str).slice(offset, offset + len));
});
Native.create("javax/microedition/lcdui/Font.substringWidth.(Ljava/lang/String;II)I", function(str, offset, len) {
return withFont(this, MIDP.Context2D, util.fromJavaString(str).slice(offset, offset + len));
});
var HCENTER = 1;
var VCENTER = 2;
var LEFT = 4;
var RIGHT = 8;
var TOP = 16;
var BOTTOM = 32;
var BASELINE = 64;
function withGraphics(g, cb) {
var img = g.klass.classInfo.getField("I.img.Ljavax/microedition/lcdui/Image;").get(g),
c = null;
if (img === null) {
c = MIDP.Context2D;
} else {
var imgData = img.klass.classInfo.getField("I.imageData.Ljavax/microedition/lcdui/ImageData;").get(img),
c = imgData.nativeImageData;
}
cb(c);
}
function withClip(g, c, x, y, cb) {
var clipX1 = g.klass.classInfo.getField("I.clipX1.S").get(g),
clipY1 = g.klass.classInfo.getField("I.clipY1.S").get(g),
clipX2 = g.klass.classInfo.getField("I.clipX2.S").get(g),
clipY2 = g.klass.classInfo.getField("I.clipY2.S").get(g),
clipped = g.klass.classInfo.getField("I.clipped.Z").get(g),
transX = g.klass.classInfo.getField("I.transX.I").get(g),
transY = g.klass.classInfo.getField("I.transY.I").get(g);
c.save();
if (clipped) {
c.beginPath();
c.rect(clipX1, clipY1, clipX2 - clipX1, clipY2 - clipY1);
c.clip();
}
c.translate(transX, transY);
cb(x, y);
c.restore();
}
function withAnchor(g, c, anchor, x, y, w, h, cb) {
withClip(g, c, x, y, function(x, y) {
if (anchor & RIGHT)
x -= w;
if (anchor & HCENTER)
x -= (w/2)|0;
if (anchor & BOTTOM)
y -= h;
if (anchor & VCENTER)
y -= (h/2)|0;
cb(x, y);
});
}
function withFont(font, c, str) {
c.font = font.css;
return c.measureText(str).width | 0;
}
function withTextAnchor(g, c, anchor, x, y, str, cb) {
withClip(g, c, x, y, function(x, y) {
var w = withFont(g.klass.classInfo.getField("I.currentFont.Ljavax/microedition/lcdui/Font;").get(g), c, str);
c.textAlign = "left";
c.textBaseline = "top";
if (anchor & RIGHT) {
x -= w;
} else if (anchor & HCENTER) {
x -= (w >>> 1) | 0;
}
if (anchor & BOTTOM) {
c.textBaseline = "bottom";
} else if (anchor & BASELINE) {
c.textBaseline = "alphabetic";
} else if (anchor & VCENTER) {
throw new JavaException("java/lang/IllegalArgumentException", "VCENTER not allowed with text");
}
cb(x, y, w);
});
}
function abgrIntToCSS(pixel) {
var a = (pixel >> 24) & 0xff;
var b = (pixel >> 16) & 0xff;
var g = (pixel >> 8) & 0xff;
var r = pixel & 0xff;
return "rgba(" + r + "," + g + "," + b + "," + (a/255) + ")";
};
function withPixel(g, c, cb) {
var pixel = g.klass.classInfo.getField("I.pixel.I").get(g);
c.save();
c.fillStyle = c.strokeStyle = abgrIntToCSS(pixel);
cb();
c.restore();
}
/**
* create the outline of an elliptical arc
* covering the specified rectangle.
* @param x the x-coordinate of the center of the ellipse.
* @param y y-coordinate of the center of the ellipse.
* @param rw the horizontal radius of the arc.
* @param rh the vertical radius of the arc.
* @param arcStart the beginning angle
* @param arcEnd the ending angle
* @param closed if true, draw a closed arc sector.
*/
function createEllipticalArc(c, x, y, rw, rh, arcStart, arcEnd, closed) {
c.save();
c.translate(x, y);
if (closed) {
c.moveTo(0, 0);
}
// draw circle arc which will be stretched into an oval arc
c.scale(1, rh / rw);
c.arc(0, 0, rw, arcStart, arcEnd, false);
if (closed) {
c.lineTo(0, 0);
}
c.restore();
}
/**
* Create a round rectangle path.
* @param x the x coordinate of the rectangle
* @param y the y coordinate of the rectangle
* @param width the width of the rectangle
* @param height the height of the rectangle
* @param arcWidth the horizontal diameter of the arc at the four corners
* @param arcHeight the vertical diameter of the arc at the four corners
*/
function createRoundRect(c, x, y, width, height, arcWidth, arcHeight) {
var rw = arcWidth / 2;
var rh = arcHeight / 2;
c.moveTo(x + rw, y);
c.lineTo(x + width - rw, y);
createEllipticalArc(c, x + width - rw, y + rh, rw, rh, 1.5 * Math.PI, 2 * Math.PI, false);
c.lineTo(x + width, y + height - rh);
createEllipticalArc(c, x + width - rw, y + height - rh, rw, rh, 0, 0.5 * Math.PI, false);
c.lineTo(x + rw, y + height);
createEllipticalArc(c, x + rw, y + height - rh, rw, rh, 0.5 * Math.PI, Math.PI, false);
c.lineTo(x, y + rh);
createEllipticalArc(c, x + rw, y + rh, rw, rh, Math.PI, 1.5 * Math.PI, false);
}
/**
* Like withPixel, but ignores alpha channel, setting the alpha value to 1.
* Useful when you suspect that the caller is specifying the alpha channel
* incorrectly, although we should actually figure out why that's happening.
*/
function withOpaquePixel(g, c, cb) {
var pixel = g.klass.classInfo.getField("I.pixel.I").get(g);
c.save();
var b = (pixel >> 16) & 0xff;
var g = (pixel >> 8) & 0xff;
var r = pixel & 0xff;
var style = "rgba(" + r + "," + g + "," + b + "," + 1 + ")";
c.fillStyle = c.strokeStyle = style;
cb();
c.restore();
}
function withSize(dx, dy, cb) {
if (!dx)
dx = 1;
if (!dy)
dy = 1;
cb(dx, dy);
}
function renderImage(graphics, image, x, y, anchor) {
var texture = image.klass.classInfo.getField("I.imageData.Ljavax/microedition/lcdui/ImageData;").get(image).nativeImageData;
if (!texture) {
console.warn("Graphics.render: image missing native data");
return;
}
if (texture instanceof CanvasRenderingContext2D) {
texture = texture.canvas; // Render the canvas, not the context.
}
withGraphics(graphics, function(c) {
withAnchor(graphics, c, anchor, x, y, texture.width, texture.height, function(x, y) {
c.drawImage(texture, x, y);
});
});
}
Override.create("com/sun/midp/chameleon/CGraphicsUtil.draw9pcsBackground.(Ljavax/microedition/lcdui/Graphics;IIII[Ljavax/microedition/lcdui/Image;)V",
function(g, x, y, w, h, image) {
if (image == null || image.length != 9) {
return;
}
var transX = g.klass.classInfo.getField("I.transX.I");
var transY = g.klass.classInfo.getField("I.transY.I");
transX.set(g, transX.get(g) + x);
transY.set(g, transY.get(g) + y);
// Top Border
var iW = image[1].klass.classInfo.getField("I.width.I").get(image[1]);
renderImage(g, image[0], 0, 0, LEFT | TOP);
w -= image[2].klass.classInfo.getField("I.width.I").get(image[2]);
for (var i = image[0].klass.classInfo.getField("I.width.I").get(image[0]); i < w; i += iW) {
renderImage(g, image[1], i, 0, LEFT | TOP);
}
w += image[2].klass.classInfo.getField("I.width.I").get(image[2]);
renderImage(g, image[2], w, 0, RIGHT | TOP);
// Tile middle rows
if (image[4] != null) {
iW = image[4].klass.classInfo.getField("I.width.I").get(image[4]);
}
var iH = image[3].klass.classInfo.getField("I.height.I").get(image[3]);
h -= image[6].klass.classInfo.getField("I.height.I").get(image[6]);
w -= image[5].klass.classInfo.getField("I.width.I").get(image[5]);
for (var i = image[0].klass.classInfo.getField("I.height.I").get(image[0]); i <= h; i += iH) {
renderImage(g, image[3], 0, i, LEFT | TOP);
for (var j = image[3].klass.classInfo.getField("I.width.I").get(image[3]); j <= w; j += iW) {
renderImage(g, image[4], j, i, LEFT | TOP);
}
renderImage(g, image[5], w + image[5].klass.classInfo.getField("I.width.I").get(image[5]), i,
RIGHT | TOP);
}
w += image[5].klass.classInfo.getField("I.width.I").get(image[5]);
h += image[6].klass.classInfo.getField("I.height.I").get(image[6]);
// Bottom border
iW = image[7].klass.classInfo.getField("I.width.I").get(image[7]);
renderImage(g, image[6], 0, h, LEFT | BOTTOM);
w -= image[8].klass.classInfo.getField("I.width.I").get(image[8]);
for (var i = image[6].klass.classInfo.getField("I.width.I").get(image[6]); i < w; i += iW) {
renderImage(g, image[7], i, h, LEFT | BOTTOM);
}
w += image[8].klass.classInfo.getField("I.width.I").get(image[8]);
renderImage(g, image[8], w, h, RIGHT | BOTTOM);
transX.set(g, transX.get(g) - x);
transY.set(g, transY.get(g) - y);
});
Native.create("javax/microedition/lcdui/Graphics.getDisplayColor.(I)I", function(color) {
return color;
});
Native.create("javax/microedition/lcdui/Graphics.getPixel.(IIZ)I", function(rgb, gray, isGray) {
return swapRB(rgb) | 0xff000000;
});
// DirectGraphics constants
var TYPE_USHORT_4444_ARGB = 4444;
var TYPE_USHORT_565_RGB = 565;
Native.create("com/nokia/mid/ui/DirectGraphicsImp.setARGBColor.(I)V", function(rgba) {
var g = this.klass.classInfo.getField("I.graphics.Ljavax/microedition/lcdui/Graphics;").get(this);
var red = (rgba >> 16) & 0xff;
var green = (rgba >> 8) & 0xff;
var blue = rgba & 0xff;
g.klass.classInfo.getField("I.pixel.I").set(g, swapRB(rgba));
g.klass.classInfo.getField("I.rgbColor.I").set(g, rgba & 0x00ffffff);
// Conversion matches Graphics#grayVal(int, int, int).
g.klass.classInfo.getField("I.gray.I").set(g, (red * 76 + green * 150 + blue * 29) >> 8);
});
Native.create("com/nokia/mid/ui/DirectGraphicsImp.getAlphaComponent.()I", function() {
var g = this.klass.classInfo.getField("I.graphics.Ljavax/microedition/lcdui/Graphics;").get(this);
var pixel = g.klass.classInfo.getField("I.pixel.I").get(g);
return (pixel >> 24) & 0xff;
});
Native.create("com/nokia/mid/ui/DirectGraphicsImp.getPixels.([SIIIIIII)V",
function(pixels, offset, scanlength, x, y, width, height, format) {
if (pixels == null) {
throw new JavaException("java/lang/NullPointerException", "Pixels array is null");
}
var converterFunc = null;
if (format == TYPE_USHORT_4444_ARGB) {
converterFunc = function(abgr) {
var a = (abgr & 0xF0000000) >>> 16;
var r = (abgr & 0x000000F0) << 4;
var g = (abgr & 0x0000F000) >> 8;
var b = (abgr & 0x00F00000) >>> 20;
return (a | r | g | b);
};
} else if (format == TYPE_USHORT_565_RGB) {
converterFunc = function(abgr) {
var r = (abgr & 0b000000000000000011111000) << 8;
var g = (abgr & 0b000000001111110000000000) >>> 5;
var b = (abgr & 0b111110000000000000000000) >>> 19;
return (r | g | b);
};
} else {
throw new JavaException("java/lang/IllegalArgumentException", "Format unsupported");
}
var graphics = this.klass.classInfo.getField("I.graphics.Ljavax/microedition/lcdui/Graphics;").get(this);
var image = graphics.klass.classInfo.getField("I.img.Ljavax/microedition/lcdui/Image;").get(graphics);
if (!image) {
throw new JavaException("java/lang/IllegalArgumentException", "getPixels with no image not yet supported");
}
var imageData = image.klass.classInfo.getField("I.imageData.Ljavax/microedition/lcdui/ImageData;").get(image);
contextToRgbData(convertNativeImageData(imageData), pixels, offset, scanlength, x, y, width, height, converterFunc);
});
Native.create("com/nokia/mid/ui/DirectGraphicsImp.drawPixels.([SZIIIIIIII)V",
function(pixels, transparency, offset, scanlength, x, y, width, height, manipulation, format) {
if (pixels == null) {
throw new JavaException("java/lang/NullPointerException", "Pixels array is null");
}
var converterFunc = null;
if (format == TYPE_USHORT_4444_ARGB && transparency && !manipulation) {
converterFunc = function(argb) {
var a = (argb & 0xF000) << 16;
var r = (argb & 0x0F00) >>> 4;
var g = (argb & 0x00F0) << 8;
var b = (argb & 0x000F) << 20;
return (a | b | g | r);
};
} else {
throw new JavaException("java/lang/IllegalArgumentException", "Format unsupported");
}
var graphics = this.klass.classInfo.getField("I.graphics.Ljavax/microedition/lcdui/Graphics;").get(this);
var context = createContext2d(width, height);
rgbDataToContext(context, pixels, offset, scanlength, converterFunc);
withGraphics(graphics, function(c) {
withClip(graphics, c, x, y, function(x, y) {
c.drawImage(context.canvas, x, y);
});
});
});
Native.create("javax/microedition/lcdui/Graphics.render.(Ljavax/microedition/lcdui/Image;III)Z", function(image, x, y, anchor) {
renderImage(this, image, x, y, anchor);
return true;
});
Native.create("javax/microedition/lcdui/Graphics.drawString.(Ljava/lang/String;III)V", function(jStr, x, y, anchor) {
var str = util.fromJavaString(jStr);
var g = this;
withGraphics(g, function(c) {
withTextAnchor(g, c, anchor, x, y, str, function(x, y) {
withOpaquePixel(g, c, function() {
c.fillText(str, x, y);
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.drawSubstring.(Ljava/lang/String;IIIII)V",
function(jStr, offset, len, x, y, anchor) {
var str = util.fromJavaString(jStr).substr(offset, len);
var g = this;
withGraphics(g, function(c) {
withTextAnchor(g, c, anchor, x, y, str, function(x, y) {
withPixel(g, c, function() {
c.fillText(str, x, y);
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.drawChars.([CIIIII)V", function(data, offset, len, x, y, anchor) {
var str = util.fromJavaChars(data, offset, len);
var g = this;
withGraphics(g, function(c) {
withTextAnchor(g, c, anchor, x, y, str, function(x, y) {
withPixel(g, c, function() {
c.fillText(str, x, y);
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.drawChar.(CIII)V", function(jChr, x, y, anchor) {
var chr = String.fromCharCode(jChr);
var g = this;
withGraphics(g, function(c) {
withTextAnchor(g, c, anchor, x, y, chr, function(x, y) {
withPixel(g, c, function() {
c.fillText(chr, x, y);
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.fillTriangle.(IIIIII)V", function(x1, y1, x2, y2, x3, y3) {
var g = this;
withGraphics(g, function(c) {
withClip(g, c, x1, y1, function(x, y) {
withPixel(g, c, function() {
withSize(x2 - x1, y2 - y1, function(dx1, dy1) {
withSize(x3 - x1, y3 - y1, function(dx2, dy2) {
c.beginPath();
c.moveTo(x, y);
c.lineTo(x + dx1, y + dy1);
c.lineTo(x + dx2, y + dy2);
c.closePath();
c.fill();
});
});
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.drawRect.(IIII)V", function(x, y, w, h) {
if (w < 0 || h < 0) {
return;
}
var g = this;
withGraphics(g, function(c) {
withClip(g, c, x, y, function(x, y) {
withPixel(g, c, function() {
withSize(w, h, function(w, h) {
c.strokeRect(x, y, w, h);
});
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.drawRoundRect.(IIIIII)V", function(x, y, w, h, arcWidth, arcHeight) {
if (w < 0 || h < 0) {
return;
}
var g = this;
withGraphics(g, function(c) {
withClip(g, c, x, y, function(x, y) {
withPixel(g, c, function() {
withSize(w, h, function(w, h) {
c.beginPath();
createRoundRect(c, x, y, w, h, arcWidth, arcHeight);
c.stroke();
});
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.fillRect.(IIII)V", function(x, y, w, h) {
if (w <= 0 || h <= 0) {
return;
}
var g = this;
withGraphics(g, function(c) {
withClip(g, c, x, y, function(x, y) {
withPixel(g, c, function() {
withSize(w, h, function(w, h) {
c.fillRect(x, y, w, h);
});
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.fillRoundRect.(IIIIII)V", function(x, y, w, h, arcWidth, arcHeight) {
if (w <= 0 || h <= 0) {
return;
}
var g = this;
withGraphics(g, function(c) {
withClip(g, c, x, y, function(x, y) {
withPixel(g, c, function() {
withSize(w, h, function(w, h) {
c.beginPath();
createRoundRect(c, x, y, w, h, arcWidth, arcHeight);
c.fill();
});
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.drawArc.(IIIIII)V", function(x, y, width, height, startAngle, arcAngle) {
if (width < 0 || height < 0) {
return;
}
var g = this;
withGraphics(g, function(c) {
withPixel(g, c, function() {
var endRad = -startAngle * 0.0175;
var startRad = endRad - arcAngle * 0.0175;
c.beginPath();
createEllipticalArc(c, x, y, width / 2, height / 2, startRad, endRad, false);
c.stroke();
});
});
});
Native.create("javax/microedition/lcdui/Graphics.fillArc.(IIIIII)V", function(x, y, width, height, startAngle, arcAngle) {
if (width <= 0 || height <= 0) {
return;
}
var g = this;
withGraphics(g, function(c) {
withPixel(g, c, function() {
var endRad = -startAngle * 0.0175;
var startRad = endRad - arcAngle * 0.0175;
c.beginPath();
c.moveTo(x, y);
createEllipticalArc(c, x, y, width / 2, height / 2, startRad, endRad, true);
c.moveTo(x, y);
c.fill();
});
});
});
var TRANS_NONE = 0;
var TRANS_MIRROR_ROT180 = 1;
var TRANS_MIRROR = 2;
var TRANS_ROT180 = 3;
var TRANS_MIRROR_ROT270 = 4;
var TRANS_ROT90 = 5;
var TRANS_ROT270 = 6;
var TRANS_MIRROR_ROT90 = 7;
Override.create("javax/microedition/lcdui/Graphics.drawRegion.(Ljavax/microedition/lcdui/Image;IIIIIIII)V",
function(image, sx, sy, sw, sh, transform, x, y, anchor) {
if (!image) {
throw new JavaException("java/lang/NullPointerException", "src image is null");
}
var imgData = image.klass.classInfo.getField("I.imageData.Ljavax/microedition/lcdui/ImageData;").get(image),
texture = imgData.nativeImageData;
if (texture instanceof CanvasRenderingContext2D) {
texture = texture.canvas; // Render the canvas, not the context.
}
var g = this;
withGraphics(g, function(c) {
withAnchor(g, c, anchor, x, y, sw, sh, function(x, y) {
c.translate(x, y);
if (transform === TRANS_MIRROR || transform === TRANS_MIRROR_ROT180)
c.scale(-1, 1);
if (transform === TRANS_MIRROR_ROT90 || transform === TRANS_MIRROR_ROT270)
c.scale(1, -1);
if (transform === TRANS_ROT90 || transform === TRANS_MIRROR_ROT90)
c.rotate(Math.PI / 2);
if (transform === TRANS_ROT180 || transform === TRANS_MIRROR_ROT180)
c.rotate(Math.PI);
if (transform === TRANS_ROT270 || transform === TRANS_MIRROR_ROT270)
c.rotate(1.5 * Math.PI);
c.drawImage(texture, sx, sy, sw, sh, 0, 0, sw, sh);
});
});
});
Native.create("javax/microedition/lcdui/Graphics.drawLine.(IIII)V", function(x1, y1, x2, y2) {
var dx = x2 - x1, dy = y2 - y1;
var g = this;
withGraphics(g, function(c) {
withClip(g, c, x1, y1, function(x, y) {
withSize(dx, dy, function(dx, dy) {
withPixel(g, c, function() {
c.beginPath();
c.moveTo(x, y);
c.lineTo(x + dx, y + dy);
c.stroke();
c.closePath();
});
});
});
});
});
Native.create("javax/microedition/lcdui/Graphics.drawRGB.([IIIIIIIZ)V",
function(rgbData, offset, scanlength, x, y, width, height, processAlpha) {
var context = createContext2d(width, height);
rgbDataToContext(context, rgbData, offset, scanlength, processAlpha ? swapRB : swapRBAndSetAlpha);
var g = this;
withGraphics(g, function(c) {
withClip(g, c, x, y, function(x, y) {
c.drawImage(context.canvas, x, y);
});
});
});
var textEditorId = 0,
textEditorResolve = null,
dirtyEditors = [];
function wakeTextEditorThread(id) {
dirtyEditors.push(id);
if (textEditorResolve) {
textEditorResolve();
textEditorResolve = null;
}
}
Native.create("com/nokia/mid/ui/TextEditor.init.(Ljava/lang/String;IIII)I",
function(text, maxSize, constraints, width, height) {
if (constraints != 0) {
console.warn("TextEditor.constraints not implemented");
}
this.textEditorId = ++textEditorId;
this.textEditor = TextEditorProvider.createEditor(constraints);
this.visible = false;
this.focused = false;
this.backgroundColor = 0xFFFFFFFF | 0; // opaque white
this.foregroundColor = 0xFF000000 | 0; // opaque black
this.textEditor.setStyle("border", "none");
this.textEditor.setStyle("resize", "none");
this.textEditor.setStyle("backgroundColor", abgrIntToCSS(this.backgroundColor));
this.textEditor.setStyle("color", abgrIntToCSS(this.foregroundColor));
this.getCaretPosition = function() {
if (this.textEditor.getParent()) {
return this.textEditor.getSelectionStart();
}
if (this.caretPosition !== null) {
return this.caretPosition;
}
return 0;
};
this.setCaretPosition = function(index) {
if (this.textEditor.getParent()) {
this.textEditor.setSelectionRange(index, index);
} else {
this.caretPosition = index;
}
};
this.textEditor.setContent(util.fromJavaString(text));
this.setCaretPosition(this.textEditor.getContent().length);
this.textEditor.setAttribute("maxlength", maxSize);
this.textEditor.setStyle("width", width + "px");
this.textEditor.setStyle("height", height + "px");
this.textEditor.setStyle("position", "absolute");
this.textEditor.setVisible(false);
this.textEditor.oninput(function(e) {
wakeTextEditorThread(this.textEditorId);
}.bind(this));
return textEditorId;
});
Native.create("com/nokia/mid/ui/CanvasItem.attachNativeImpl.()V", function() {
this.textEditor.setParent(document.getElementById("display"));
if (this.caretPosition !== null) {
this.textEditor.setSelectionRange(this.caretPosition, this.caretPosition);
this.caretPosition = null;
}
});
Native.create("com/nokia/mid/ui/CanvasItem.detachNativeImpl.()V", function() {
this.caretPosition = this.textEditor.getSelectionStart();
this.textEditor.setParent(null);
});
Native.create("com/nokia/mid/ui/CanvasItem.setSize.(II)V", function(width, height) {
this.textEditor.setStyle("width", width + "px");
this.textEditor.setStyle("height", height + "px");
});
Native.create("com/nokia/mid/ui/CanvasItem.setVisible.(Z)V", function(visible) {
if (visible && !this.visible) {
this.textEditor.setVisible(true);
} else if (!visible && this.visible) {
this.textEditor.setVisible(false);
}
this.visible = visible;
});
Native.create("com/nokia/mid/ui/CanvasItem.getWidth.()I", function() {
return parseInt(this.textEditor.getStyle("width"));
});
Native.create("com/nokia/mid/ui/CanvasItem.getHeight.()I", function() {
return parseInt(this.textEditor.getStyle("height"));
});
Native.create("com/nokia/mid/ui/CanvasItem.setPosition0.(II)V", function(x, y) {
this.textEditor.setStyle("left", x + "px");
this.textEditor.setStyle("top", y + "px");
});
Native.create("com/nokia/mid/ui/CanvasItem.getPositionX.()I", function() {
return parseInt(this.textEditor.getStyle("left")) || 0;
});
Native.create("com/nokia/mid/ui/CanvasItem.getPositionY.()I", function() {
return parseInt(this.textEditor.getStyle("top")) || 0;
});
Native.create("com/nokia/mid/ui/CanvasItem.isVisible.()Z", function() {
return this.visible;
});
Native.create("com/nokia/mid/ui/TextEditor.setConstraints.(I)V", function(constraints) {
this.textEditor.setConstraints(constraints);
});
Native.create("com/nokia/mid/ui/TextEditor.getConstraints.()I", function() {
return this.textEditor.getConstraints();
});
Native.create("com/nokia/mid/ui/TextEditor.setFocus.(Z)V", function(focused) {
if (focused && !this.focused) {
this.textEditor.focus();
} else if (!focused && this.focused) {
this.textEditor.blur();
}
this.focused = focused;
});
Native.create("com/nokia/mid/ui/TextEditor.hasFocus.()Z", function() {
return this.focused;
});
Native.create("com/nokia/mid/ui/TextEditor.setCaret.(I)V", function(index) {
this.setCaretPosition(index);
});
Native.create("com/nokia/mid/ui/TextEditor.getCaretPosition.()I", function() {
return this.getCaretPosition();
});
Native.create("com/nokia/mid/ui/TextEditor.getBackgroundColor.()I", function() {
return this.backgroundColor;
});
Native.create("com/nokia/mid/ui/TextEditor.getForegroundColor.()I", function() {
return this.foregroundColor;
});
Native.create("com/nokia/mid/ui/TextEditor.setBackgroundColor.(I)V", function(backgroundColor) {
this.backgroundColor = backgroundColor;
this.textEditor.setStyle("backgroundColor", abgrIntToCSS(backgroundColor));
});
Native.create("com/nokia/mid/ui/TextEditor.setForegroundColor.(I)V", function(foregroundColor) {
this.foregroundColor = foregroundColor;
this.textEditor.setStyle("color", abgrIntToCSS(foregroundColor));
});
Native.create("com/nokia/mid/ui/TextEditor.getContent.()Ljava/lang/String;", function() {
return this.textEditor.getContent();
});
Native.create("com/nokia/mid/ui/TextEditor.setContent.(Ljava/lang/String;)V", function(jStr) {
var str = util.fromJavaString(jStr);
this.textEditor.setContent(str);
this.setCaretPosition(str.length);
});
Native.create("com/nokia/mid/ui/TextEditor.insert.(Ljava/lang/String;I)V", function(jStr, pos) {
var old = this.textEditor.getContent();
var str = util.fromJavaString(jStr);
this.textEditor.setContent(old.slice(0, pos) + str + old.slice(pos));
this.setCaretPosition(pos + str.length);
});
Native.create("com/nokia/mid/ui/TextEditor.delete.(II)V", function(offset, length) {
var old = this.textEditor.getContent();
if (offset < 0 || offset > old.length || length < 0 || offset + length > old.length) {
throw new JavaException("java.lang.StringIndexOutOfBoundsException", "offset/length invalid");
}
this.textEditor.setContent(old.slice(0, offset) + old.slice(offset + length));
this.setCaretPosition(offset);
});
Native.create("com/nokia/mid/ui/TextEditor.getMaxSize.()I", function() {
return parseInt(this.textEditor.getAttribute("maxlength"));
});
Native.create("com/nokia/mid/ui/TextEditor.setMaxSize.(I)I", function(maxSize) {
if (this.textEditor.getContent().length > maxSize) {
this.textEditor.setContent(this.textEditor.getContent().substring(0, maxSize));
if (this.getCaretPosition() > maxSize) {
this.setCaretPosition(maxSize);
}
}
this.textEditor.setAttribute("maxlength", maxSize);
// The return value is the assigned size, which could be less than
// the size that was requested, although in this case we always set it
// to the requested size.
return maxSize;
});
Native.create("com/nokia/mid/ui/TextEditor.size.()I", function() {
return this.textEditor.getContent().length;
});
Native.create("com/nokia/mid/ui/TextEditorThread.sleep.()V", function() {
return new Promise(function(resolve, reject) {
if (!dirtyEditors.length) {
textEditorResolve = resolve;
} else {
resolve();
}
});
}, true);
Native.create("com/nokia/mid/ui/TextEditorThread.getNextDirtyEditor.()I", function() {
if (!dirtyEditors.length) {
console.error("ERROR: getNextDirtyEditor called but no dirty editors");
return 0;
}
return dirtyEditors.shift();
});
var initialWindowHeight = window.innerHeight;
var isVKVisible = false;
var keyboardHeight = 0;
var pendingShowNotify = false;
var pendingHideNotify = false;
var keyboardVisibilityListenerResolve;
window.addEventListener("resize", function(evt) {
if (window.innerHeight < initialWindowHeight) {
if (isVKVisible) {
console.warn("Window shrunk but we thought the keyboard was already visible!");
}
isVKVisible = true;
keyboardHeight = initialWindowHeight - window.innerHeight;
if (pendingHideNotify) {
pendingHideNotify = false;
return;
} else if (keyboardVisibilityListenerResolve) {
keyboardVisibilityListenerResolve(true);
keyboardVisibilityListenerResolve = null;
} else {
pendingShowNotify = true;
}
} else if (window.innerHeight >= initialWindowHeight) {
if (window.innerHeight > initialWindowHeight) {
console.warn("Window grew beyond initial size!");
initialWindowHeight = window.innerHeight;
}
if (!isVKVisible) {
console.warn("Window grew but we thought the keyboard was already hidden!");
}
isVKVisible = false;
keyboardHeight = 0;
if (pendingShowNotify) {
pendingShowNotify = false;
return;
} else if (keyboardVisibilityListenerResolve) {
keyboardVisibilityListenerResolve(false);
keyboardVisibilityListenerResolve = null;
} else {
pendingHideNotify = true;
}
}
});
Native.create("com/nokia/mid/ui/VirtualKeyboard.isVisible.()Z", function() {
return isVKVisible;
});
Native.create("com/nokia/mid/ui/VirtualKeyboard.getXPosition.()I", function() {
return 0;
});
Native.create("com/nokia/mid/ui/VirtualKeyboard.getYPosition.()I", function() {
// We should return the number of pixels between the top of the
// screen and the top of the keyboard
return window.innerHeight;
});
Native.create("com/nokia/mid/ui/VirtualKeyboard.getWidth.()I", function() {
return window.innerWidth;
});
Native.create("com/nokia/mid/ui/VirtualKeyboard.getHeight.()I", function() {
return keyboardHeight;
});
Native.create("com/nokia/mid/ui/VKVisibilityNotificationRunnable.sleepUntilVKVisibilityChange.()Z", function() {
return new Promise(function(resolve, reject) {
if (pendingShowNotify) {
resolve(true);
pendingShowNotify = false;
} else if (pendingHideNotify) {
resolve(false);
pendingHideNotify = false;
} else {
keyboardVisibilityListenerResolve = resolve;
}
});
}, true);
})(Native);