Skip to content

Commit

Permalink
Fix WebGL-based canvas source flipping bug + optimize non-animated ca…
Browse files Browse the repository at this point in the history
…nvas rerenders
  • Loading branch information
Lauren Budorick authored Sep 18, 2017
1 parent 6ab53dc commit 0601904
Showing 1 changed file with 20 additions and 8 deletions.
28 changes: 20 additions & 8 deletions src/source/canvas_source.js
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ class CanvasSource extends ImageSource {
secondaryContext: ?CanvasRenderingContext2D;
width: number;
height: number;
canvasData: ?ImageData;
play: () => void;
pause: () => void;

Expand Down Expand Up @@ -114,21 +115,27 @@ class CanvasSource extends ImageSource {
*/
// setCoordinates inherited from ImageSource

readCanvas() {
readCanvas(resize: boolean) {
// We *should* be able to use a pure HTMLCanvasElement in
// texImage2D/texSubImage2D (in ImageSource#_prepareImage), but for
// some reason this breaks the map on certain GPUs (see #4262).

if (this.context instanceof CanvasRenderingContext2D) {
return this.context.getImageData(0, 0, this.width, this.height);
this.canvasData = this.context.getImageData(0, 0, this.width, this.height);
} else if (this.context instanceof WebGLRenderingContext) {
const gl = this.context;
const data = new Uint8Array(this.width * this.height * 4);
gl.readPixels(0, 0, this.width, this.height, gl.RGBA, gl.UNSIGNED_BYTE, data);

if (!this.secondaryContext) this.secondaryContext = window.document.createElement('canvas').getContext('2d');
const imageData = this.secondaryContext.createImageData(this.width, this.height);
imageData.data.set(data);
return imageData;
if (!this.canvasData || resize) {
this.canvasData = this.secondaryContext.createImageData(this.width, this.height);
}

// WebGL reads pixels bottom to top, but for our ImageData object we need top to bottom: flip here
for (let i = this.height - 1, j = 0; i >= 0; i--, j++) {
this.canvasData.data.set(data.subarray(i * this.width * 4, (i + 1) * this.width * 4), j * this.width * 4);
}
}
}

Expand All @@ -145,12 +152,17 @@ class CanvasSource extends ImageSource {
if (this._hasInvalidDimensions()) return;

if (Object.keys(this.tiles).length === 0) return; // not enough data for current position
const canvasData = this.readCanvas();
if (!canvasData) {

const reread = this.animate || !this.canvasData || resize;
if (reread) {
this.readCanvas(resize);
}

if (!this.canvasData) {
this.fire('error', new Error('Could not read canvas data.'));
return;
}
this._prepareImage(this.map.painter.gl, canvasData, resize);
this._prepareImage(this.map.painter.gl, this.canvasData, resize);
}

serialize(): Object {
Expand Down

0 comments on commit 0601904

Please sign in to comment.