Skip to content

Commit 8074814

Browse files
authored
Merge pull request #673 from LLK/revert-660-revert-489
Revert "Revert "Merge pull request #489 from adroitwhiz/touching-white-fixes""
2 parents 0babe1e + b234625 commit 8074814

File tree

5 files changed

+98
-20
lines changed

5 files changed

+98
-20
lines changed

src/RenderWebGL.js

+75-16
Original file line numberDiff line numberDiff line change
@@ -188,9 +188,23 @@ class RenderWebGL extends EventEmitter {
188188
/** @type {function} */
189189
this._exitRegion = null;
190190

191+
/** @type {object} */
192+
this._backgroundDrawRegionId = {
193+
enter: () => this._enterDrawBackground(),
194+
exit: () => this._exitDrawBackground()
195+
};
196+
191197
/** @type {Array.<snapshotCallback>} */
192198
this._snapshotCallbacks = [];
193199

200+
/** @type {Array<number>} */
201+
// Don't set this directly-- use setBackgroundColor so it stays in sync with _backgroundColor3b
202+
this._backgroundColor4f = [0, 0, 0, 1];
203+
204+
/** @type {Uint8ClampedArray} */
205+
// Don't set this directly-- use setBackgroundColor so it stays in sync with _backgroundColor4f
206+
this._backgroundColor3b = new Uint8ClampedArray(3);
207+
194208
this._createGeometry();
195209

196210
this.on(RenderConstants.Events.NativeSizeChanged, this.onNativeSizeChanged);
@@ -250,7 +264,14 @@ class RenderWebGL extends EventEmitter {
250264
* @param {number} blue The blue component for the background.
251265
*/
252266
setBackgroundColor (red, green, blue) {
253-
this._backgroundColor = [red, green, blue, 1];
267+
this._backgroundColor4f[0] = red;
268+
this._backgroundColor4f[1] = green;
269+
this._backgroundColor4f[2] = blue;
270+
271+
this._backgroundColor3b[0] = red * 255;
272+
this._backgroundColor3b[1] = green * 255;
273+
this._backgroundColor3b[2] = blue * 255;
274+
254275
}
255276

256277
/**
@@ -629,7 +650,7 @@ class RenderWebGL extends EventEmitter {
629650

630651
twgl.bindFramebufferInfo(gl, null);
631652
gl.viewport(0, 0, gl.canvas.width, gl.canvas.height);
632-
gl.clearColor.apply(gl, this._backgroundColor);
653+
gl.clearColor.apply(gl, this._backgroundColor4f);
633654
gl.clear(gl.COLOR_BUFFER_BIT);
634655

635656
this._drawThese(this._drawList, ShaderManager.DRAW_MODE.default, this._projection);
@@ -745,12 +766,20 @@ class RenderWebGL extends EventEmitter {
745766
*/
746767
isTouchingColor (drawableID, color3b, mask3b) {
747768
const candidates = this._candidatesTouching(drawableID, this._visibleDrawList);
748-
if (candidates.length === 0) {
769+
770+
let bounds;
771+
if (colorMatches(color3b, this._backgroundColor3b, 0)) {
772+
// If the color we're checking for is the background color, don't confine the check to
773+
// candidate drawables' bounds--since the background spans the entire stage, we must check
774+
// everything that lies inside the drawable.
775+
bounds = this._touchingBounds(drawableID);
776+
} else if (candidates.length === 0) {
777+
// If not checking for the background color, we can return early if there are no candidate drawables.
749778
return false;
779+
} else {
780+
bounds = this._candidatesBounds(candidates);
750781
}
751782

752-
const bounds = this._candidatesBounds(candidates);
753-
754783
const maxPixelsForCPU = this._getMaxPixelsForCPU();
755784

756785
const debugCanvasContext = this._debugCanvas && this._debugCanvas.getContext('2d');
@@ -811,6 +840,19 @@ class RenderWebGL extends EventEmitter {
811840
}
812841
}
813842

843+
_enterDrawBackground () {
844+
const gl = this.gl;
845+
const currentShader = this._shaderManager.getShader(ShaderManager.DRAW_MODE.background, 0);
846+
gl.disable(gl.BLEND);
847+
gl.useProgram(currentShader.program);
848+
twgl.setBuffersAndAttributes(gl, currentShader, this._bufferInfo);
849+
}
850+
851+
_exitDrawBackground () {
852+
const gl = this.gl;
853+
gl.enable(gl.BLEND);
854+
}
855+
814856
_isTouchingColorGpuStart (drawableID, candidateIDs, bounds, color3b, mask3b) {
815857
this._doExitDrawRegion();
816858

@@ -822,15 +864,8 @@ class RenderWebGL extends EventEmitter {
822864
gl.viewport(0, 0, bounds.width, bounds.height);
823865
const projection = twgl.m4.ortho(bounds.left, bounds.right, bounds.top, bounds.bottom, -1, 1);
824866

825-
let fillBackgroundColor = this._backgroundColor;
826-
827-
// When using masking such that the background fill color will showing through, ensure we don't
828-
// fill using the same color that we are trying to detect!
829-
if (color3b[0] > 196 && color3b[1] > 196 && color3b[2] > 196) {
830-
fillBackgroundColor = [0, 0, 0, 255];
831-
}
832-
833-
gl.clearColor.apply(gl, fillBackgroundColor);
867+
// Clear the query buffer to fully transparent. This will be the color of pixels that fail the stencil test.
868+
gl.clearColor(0, 0, 0, 0);
834869
gl.clear(gl.COLOR_BUFFER_BIT | gl.STENCIL_BUFFER_BIT);
835870

836871
let extraUniforms;
@@ -842,6 +877,9 @@ class RenderWebGL extends EventEmitter {
842877
}
843878

844879
try {
880+
// Using the stencil buffer, mask out the drawing to either the drawable's alpha channel
881+
// or pixels of the drawable which match the mask color, depending on whether a mask color is given.
882+
// Masked-out pixels will not be checked.
845883
gl.enable(gl.STENCIL_TEST);
846884
gl.stencilFunc(gl.ALWAYS, 1, 1);
847885
gl.stencilOp(gl.KEEP, gl.KEEP, gl.REPLACE);
@@ -862,12 +900,25 @@ class RenderWebGL extends EventEmitter {
862900
gl.stencilOp(gl.KEEP, gl.KEEP, gl.KEEP);
863901
gl.colorMask(true, true, true, true);
864902

903+
// Draw the background as a quad. Drawing a background with gl.clear will not mask to the stenciled area.
904+
this.enterDrawRegion(this._backgroundDrawRegionId);
905+
906+
const uniforms = {
907+
u_backgroundColor: this._backgroundColor4f
908+
};
909+
910+
const currentShader = this._shaderManager.getShader(ShaderManager.DRAW_MODE.background, 0);
911+
twgl.setUniforms(currentShader, uniforms);
912+
twgl.drawBufferInfo(gl, this._bufferInfo, gl.TRIANGLES);
913+
914+
// Draw the candidate drawables on top of the background.
865915
this._drawThese(candidateIDs, ShaderManager.DRAW_MODE.default, projection,
866916
{idFilterFunc: testID => testID !== drawableID}
867917
);
868918
} finally {
869919
gl.colorMask(true, true, true, true);
870920
gl.disable(gl.STENCIL_TEST);
921+
this._doExitDrawRegion();
871922
}
872923
}
873924

@@ -886,7 +937,8 @@ class RenderWebGL extends EventEmitter {
886937
}
887938

888939
for (let pixelBase = 0; pixelBase < pixels.length; pixelBase += 4) {
889-
if (colorMatches(color3b, pixels, pixelBase)) {
940+
// Transparent pixels are masked (either by the drawable's alpha channel or color mask).
941+
if (pixels[pixelBase + 3] !== 0 && colorMatches(color3b, pixels, pixelBase)) {
890942
return true;
891943
}
892944
}
@@ -1321,7 +1373,7 @@ class RenderWebGL extends EventEmitter {
13211373
gl.viewport(0, 0, bounds.width, bounds.height);
13221374
const projection = twgl.m4.ortho(bounds.left, bounds.right, bounds.top, bounds.bottom, -1, 1);
13231375

1324-
gl.clearColor.apply(gl, this._backgroundColor);
1376+
gl.clearColor.apply(gl, this._backgroundColor4f);
13251377
gl.clear(gl.COLOR_BUFFER_BIT);
13261378
this._drawThese(this._drawList, ShaderManager.DRAW_MODE.default, projection);
13271379

@@ -1409,6 +1461,13 @@ class RenderWebGL extends EventEmitter {
14091461
// Update the CPU position data
14101462
drawable.updateCPURenderAttributes();
14111463
const candidateBounds = drawable.getFastBounds();
1464+
1465+
// Push bounds out to integers. If a drawable extends out into half a pixel, that half-pixel still
1466+
// needs to be tested. Plus, in some areas we construct another rectangle from the union of these,
1467+
// and iterate over its pixels (width * height). Turns out that doesn't work so well when the
1468+
// width/height aren't integers.
1469+
candidateBounds.snapToInt();
1470+
14121471
if (bounds.intersects(candidateBounds)) {
14131472
result.push({
14141473
id,

src/ShaderManager.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -176,7 +176,12 @@ ShaderManager.DRAW_MODE = {
176176
/**
177177
* Draw a line with caps.
178178
*/
179-
line: 'line'
179+
line: 'line',
180+
181+
/**
182+
* Draw the background in a certain color. Must sometimes be used instead of gl.clear.
183+
*/
184+
background: 'background'
180185
};
181186

182187
module.exports = ShaderManager;

src/shaders/sprite.frag

+14-2
Original file line numberDiff line numberDiff line change
@@ -39,9 +39,15 @@ uniform float u_lineThickness;
3939
uniform float u_lineLength;
4040
#endif // DRAW_MODE_line
4141

42+
#ifdef DRAW_MODE_background
43+
uniform vec4 u_backgroundColor;
44+
#endif // DRAW_MODE_background
45+
4246
uniform sampler2D u_skin;
4347

48+
#ifndef DRAW_MODE_background
4449
varying vec2 v_texCoord;
50+
#endif
4551

4652
// Add this to divisors to prevent division by 0, which results in NaNs propagating through calculations.
4753
// Smaller values can cause problems on some mobile devices.
@@ -109,7 +115,7 @@ const vec2 kCenter = vec2(0.5, 0.5);
109115

110116
void main()
111117
{
112-
#ifndef DRAW_MODE_line
118+
#if !(defined(DRAW_MODE_line) || defined(DRAW_MODE_background))
113119
vec2 texcoord0 = v_texCoord;
114120

115121
#ifdef ENABLE_mosaic
@@ -215,7 +221,9 @@ void main()
215221
gl_FragColor.rgb /= gl_FragColor.a + epsilon;
216222
#endif
217223

218-
#else // DRAW_MODE_line
224+
#endif // !(defined(DRAW_MODE_line) || defined(DRAW_MODE_background))
225+
226+
#ifdef DRAW_MODE_line
219227
// Maaaaagic antialiased-line-with-round-caps shader.
220228

221229
// "along-the-lineness". This increases parallel to the line.
@@ -234,4 +242,8 @@ void main()
234242
// the closer we are to the line, invert it.
235243
gl_FragColor = u_lineColor * clamp(1.0 - line, 0.0, 1.0);
236244
#endif // DRAW_MODE_line
245+
246+
#ifdef DRAW_MODE_background
247+
gl_FragColor = u_backgroundColor;
248+
#endif
237249
}

src/shaders/sprite.vert

+3-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ uniform vec4 u_penPoints;
1414
const float epsilon = 1e-3;
1515
#endif
1616

17-
#ifndef DRAW_MODE_line
17+
#if !(defined(DRAW_MODE_line) || defined(DRAW_MODE_background))
1818
uniform mat4 u_projectionMatrix;
1919
uniform mat4 u_modelMatrix;
2020
attribute vec2 a_texCoord;
@@ -66,6 +66,8 @@ void main() {
6666
// 4. Apply view transform
6767
position *= 2.0 / u_stageSize;
6868
gl_Position = vec4(position, 0, 1);
69+
#elif defined(DRAW_MODE_background)
70+
gl_Position = vec4(a_position * 2.0, 0, 1);
6971
#else
7072
gl_Position = u_projectionMatrix * u_modelMatrix * vec4(a_position, 0, 1);
7173
v_texCoord = a_texCoord;
54.6 KB
Binary file not shown.

0 commit comments

Comments
 (0)