github.com/gop9/olt@v0.0.0-20200202132135-d956aad50b08/gio/app/internal/gpu/path.go (about)

     1  // SPDX-License-Identifier: Unlicense OR MIT
     2  
     3  package gpu
     4  
     5  // GPU accelerated path drawing using the algorithms from
     6  // Pathfinder (https://github.com/servo/pathfinder).
     7  
     8  import (
     9  	"image"
    10  	"unsafe"
    11  
    12  	"github.com/gop9/olt/gio/app/internal/gl"
    13  	"github.com/gop9/olt/gio/f32"
    14  	"github.com/gop9/olt/gio/internal/path"
    15  )
    16  
    17  type pather struct {
    18  	ctx *context
    19  
    20  	viewport image.Point
    21  
    22  	stenciler *stenciler
    23  	coverer   *coverer
    24  }
    25  
    26  type coverer struct {
    27  	ctx  *context
    28  	prog [2]gl.Program
    29  	vars [2]struct {
    30  		z                             gl.Uniform
    31  		uScale, uOffset               gl.Uniform
    32  		uUVScale, uUVOffset           gl.Uniform
    33  		uCoverUVScale, uCoverUVOffset gl.Uniform
    34  		uColor                        gl.Uniform
    35  	}
    36  }
    37  
    38  type stenciler struct {
    39  	ctx                *context
    40  	defFBO             gl.Framebuffer
    41  	indexBufQuads      int
    42  	prog               gl.Program
    43  	iprog              gl.Program
    44  	fbos               fboSet
    45  	intersections      fboSet
    46  	uScale, uOffset    gl.Uniform
    47  	uPathOffset        gl.Uniform
    48  	uIntersectUVOffset gl.Uniform
    49  	uIntersectUVScale  gl.Uniform
    50  	indexBuf           gl.Buffer
    51  }
    52  
    53  type fboSet struct {
    54  	fbos []stencilFBO
    55  }
    56  
    57  type stencilFBO struct {
    58  	size image.Point
    59  	fbo  gl.Framebuffer
    60  	tex  gl.Texture
    61  }
    62  
    63  type pathData struct {
    64  	ncurves int
    65  	data    gl.Buffer
    66  }
    67  
    68  var (
    69  	pathAttribs                = []string{"corner", "maxy", "from", "ctrl", "to"}
    70  	attribPathCorner gl.Attrib = 0
    71  	attribPathMaxY   gl.Attrib = 1
    72  	attribPathFrom   gl.Attrib = 2
    73  	attribPathCtrl   gl.Attrib = 3
    74  	attribPathTo     gl.Attrib = 4
    75  
    76  	intersectAttribs = []string{"pos", "uv"}
    77  )
    78  
    79  func newPather(ctx *context) *pather {
    80  	return &pather{
    81  		ctx:       ctx,
    82  		stenciler: newStenciler(ctx),
    83  		coverer:   newCoverer(ctx),
    84  	}
    85  }
    86  
    87  func newCoverer(ctx *context) *coverer {
    88  	prog, err := createColorPrograms(ctx, coverVSrc, coverFSrc)
    89  	if err != nil {
    90  		panic(err)
    91  	}
    92  	c := &coverer{
    93  		ctx:  ctx,
    94  		prog: prog,
    95  	}
    96  	for i, prog := range prog {
    97  		ctx.UseProgram(prog)
    98  		switch materialType(i) {
    99  		case materialTexture:
   100  			uTex := gl.GetUniformLocation(ctx.Functions, prog, "tex")
   101  			ctx.Uniform1i(uTex, 0)
   102  			c.vars[i].uUVScale = gl.GetUniformLocation(ctx.Functions, prog, "uvScale")
   103  			c.vars[i].uUVOffset = gl.GetUniformLocation(ctx.Functions, prog, "uvOffset")
   104  		case materialColor:
   105  			c.vars[i].uColor = gl.GetUniformLocation(ctx.Functions, prog, "color")
   106  		}
   107  		uCover := gl.GetUniformLocation(ctx.Functions, prog, "cover")
   108  		ctx.Uniform1i(uCover, 1)
   109  		c.vars[i].z = gl.GetUniformLocation(ctx.Functions, prog, "z")
   110  		c.vars[i].uScale = gl.GetUniformLocation(ctx.Functions, prog, "scale")
   111  		c.vars[i].uOffset = gl.GetUniformLocation(ctx.Functions, prog, "offset")
   112  		c.vars[i].uCoverUVScale = gl.GetUniformLocation(ctx.Functions, prog, "uvCoverScale")
   113  		c.vars[i].uCoverUVOffset = gl.GetUniformLocation(ctx.Functions, prog, "uvCoverOffset")
   114  	}
   115  	return c
   116  }
   117  
   118  func newStenciler(ctx *context) *stenciler {
   119  	defFBO := gl.Framebuffer(ctx.GetBinding(gl.FRAMEBUFFER_BINDING))
   120  	prog, err := gl.CreateProgram(ctx.Functions, stencilVSrc, stencilFSrc, pathAttribs)
   121  	if err != nil {
   122  		panic(err)
   123  	}
   124  	ctx.UseProgram(prog)
   125  	iprog, err := gl.CreateProgram(ctx.Functions, intersectVSrc, intersectFSrc, intersectAttribs)
   126  	if err != nil {
   127  		panic(err)
   128  	}
   129  	coverLoc := gl.GetUniformLocation(ctx.Functions, iprog, "cover")
   130  	ctx.UseProgram(iprog)
   131  	ctx.Uniform1i(coverLoc, 0)
   132  	return &stenciler{
   133  		ctx:                ctx,
   134  		defFBO:             defFBO,
   135  		prog:               prog,
   136  		iprog:              iprog,
   137  		uScale:             gl.GetUniformLocation(ctx.Functions, prog, "scale"),
   138  		uOffset:            gl.GetUniformLocation(ctx.Functions, prog, "offset"),
   139  		uPathOffset:        gl.GetUniformLocation(ctx.Functions, prog, "pathOffset"),
   140  		uIntersectUVScale:  gl.GetUniformLocation(ctx.Functions, iprog, "uvScale"),
   141  		uIntersectUVOffset: gl.GetUniformLocation(ctx.Functions, iprog, "uvOffset"),
   142  		indexBuf:           ctx.CreateBuffer(),
   143  	}
   144  }
   145  
   146  func (s *fboSet) resize(ctx *context, sizes []image.Point) {
   147  	// Add fbos.
   148  	for i := len(s.fbos); i < len(sizes); i++ {
   149  		tex := ctx.CreateTexture()
   150  		ctx.BindTexture(gl.TEXTURE_2D, tex)
   151  		ctx.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE)
   152  		ctx.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE)
   153  		ctx.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST)
   154  		ctx.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST)
   155  		fbo := ctx.CreateFramebuffer()
   156  		s.fbos = append(s.fbos, stencilFBO{
   157  			fbo: fbo,
   158  			tex: tex,
   159  		})
   160  	}
   161  	// Resize fbos.
   162  	for i, sz := range sizes {
   163  		f := &s.fbos[i]
   164  		// Resizing or recreating FBOs can introduce rendering stalls.
   165  		// Avoid if the space waste is not too high.
   166  		resize := sz.X > f.size.X || sz.Y > f.size.Y
   167  		waste := float32(sz.X*sz.Y) / float32(f.size.X*f.size.Y)
   168  		resize = resize || waste > 1.2
   169  		if resize {
   170  			f.size = sz
   171  			ctx.BindTexture(gl.TEXTURE_2D, f.tex)
   172  			tt := ctx.caps.floatTriple
   173  			ctx.TexImage2D(gl.TEXTURE_2D, 0, tt.internalFormat, sz.X, sz.Y, tt.format, tt.typ, nil)
   174  			ctx.BindFramebuffer(gl.FRAMEBUFFER, f.fbo)
   175  			ctx.FramebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, f.tex, 0)
   176  		}
   177  	}
   178  	// Delete extra fbos.
   179  	s.delete(ctx, len(sizes))
   180  }
   181  
   182  func (s *fboSet) invalidate(ctx *context) {
   183  	for _, f := range s.fbos {
   184  		ctx.BindFramebuffer(gl.FRAMEBUFFER, f.fbo)
   185  		ctx.InvalidateFramebuffer(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0)
   186  	}
   187  }
   188  
   189  func (s *fboSet) delete(ctx *context, idx int) {
   190  	for i := idx; i < len(s.fbos); i++ {
   191  		f := s.fbos[i]
   192  		ctx.DeleteFramebuffer(f.fbo)
   193  		ctx.DeleteTexture(f.tex)
   194  	}
   195  	s.fbos = s.fbos[:idx]
   196  }
   197  
   198  func (s *stenciler) release() {
   199  	s.fbos.delete(s.ctx, 0)
   200  	s.ctx.DeleteProgram(s.prog)
   201  	s.ctx.DeleteBuffer(s.indexBuf)
   202  }
   203  
   204  func (p *pather) release() {
   205  	p.stenciler.release()
   206  	p.coverer.release()
   207  }
   208  
   209  func (c *coverer) release() {
   210  	for _, p := range c.prog {
   211  		c.ctx.DeleteProgram(p)
   212  	}
   213  }
   214  
   215  func buildPath(ctx *context, p []byte) *pathData {
   216  	buf := ctx.CreateBuffer()
   217  	ctx.BindBuffer(gl.ARRAY_BUFFER, buf)
   218  	ctx.BufferData(gl.ARRAY_BUFFER, p, gl.STATIC_DRAW)
   219  	return &pathData{
   220  		ncurves: len(p) / path.VertStride,
   221  		data:    buf,
   222  	}
   223  }
   224  
   225  func (p *pathData) release(ctx *context) {
   226  	ctx.DeleteBuffer(p.data)
   227  }
   228  
   229  func (p *pather) begin(sizes []image.Point) {
   230  	p.stenciler.begin(sizes)
   231  }
   232  
   233  func (p *pather) end() {
   234  	p.stenciler.end()
   235  }
   236  
   237  func (p *pather) stencilPath(bounds image.Rectangle, offset f32.Point, uv image.Point, data *pathData) {
   238  	p.stenciler.stencilPath(bounds, offset, uv, data)
   239  }
   240  
   241  func (s *stenciler) beginIntersect(sizes []image.Point) {
   242  	s.ctx.ActiveTexture(gl.TEXTURE1)
   243  	s.ctx.BindTexture(gl.TEXTURE_2D, gl.Texture{})
   244  	s.ctx.ActiveTexture(gl.TEXTURE0)
   245  	s.ctx.BlendFunc(gl.DST_COLOR, gl.ZERO)
   246  	// 8 bit coverage is enough, but OpenGL ES only supports single channel
   247  	// floating point formats. Replace with GL_RGB+GL_UNSIGNED_BYTE if
   248  	// no floating point support is available.
   249  	s.intersections.resize(s.ctx, sizes)
   250  	s.ctx.ClearColor(1.0, 0.0, 0.0, 0.0)
   251  	s.ctx.UseProgram(s.iprog)
   252  }
   253  
   254  func (s *stenciler) endIntersect() {
   255  	s.ctx.BindFramebuffer(gl.FRAMEBUFFER, s.defFBO)
   256  }
   257  
   258  func (s *stenciler) invalidateFBO() {
   259  	s.intersections.invalidate(s.ctx)
   260  	s.fbos.invalidate(s.ctx)
   261  	s.ctx.BindFramebuffer(gl.FRAMEBUFFER, s.defFBO)
   262  }
   263  
   264  func (s *stenciler) cover(idx int) stencilFBO {
   265  	return s.fbos.fbos[idx]
   266  }
   267  
   268  func (s *stenciler) begin(sizes []image.Point) {
   269  	s.ctx.ActiveTexture(gl.TEXTURE1)
   270  	s.ctx.BindTexture(gl.TEXTURE_2D, gl.Texture{})
   271  	s.ctx.ActiveTexture(gl.TEXTURE0)
   272  	s.ctx.BlendFunc(gl.ONE, gl.ONE)
   273  	s.fbos.resize(s.ctx, sizes)
   274  	s.ctx.ClearColor(0.0, 0.0, 0.0, 0.0)
   275  	s.ctx.UseProgram(s.prog)
   276  	s.ctx.EnableVertexAttribArray(attribPathCorner)
   277  	s.ctx.EnableVertexAttribArray(attribPathMaxY)
   278  	s.ctx.EnableVertexAttribArray(attribPathFrom)
   279  	s.ctx.EnableVertexAttribArray(attribPathCtrl)
   280  	s.ctx.EnableVertexAttribArray(attribPathTo)
   281  	s.ctx.BindBuffer(gl.ELEMENT_ARRAY_BUFFER, s.indexBuf)
   282  }
   283  
   284  func (s *stenciler) stencilPath(bounds image.Rectangle, offset f32.Point, uv image.Point, data *pathData) {
   285  	s.ctx.BindBuffer(gl.ARRAY_BUFFER, data.data)
   286  	s.ctx.Viewport(uv.X, uv.Y, bounds.Dx(), bounds.Dy())
   287  	// Transform UI coordinates to OpenGL coordinates.
   288  	texSize := f32.Point{X: float32(bounds.Dx()), Y: float32(bounds.Dy())}
   289  	scale := f32.Point{X: 2 / texSize.X, Y: 2 / texSize.Y}
   290  	orig := f32.Point{X: -1 - float32(bounds.Min.X)*2/texSize.X, Y: -1 - float32(bounds.Min.Y)*2/texSize.Y}
   291  	s.ctx.Uniform2f(s.uScale, scale.X, scale.Y)
   292  	s.ctx.Uniform2f(s.uOffset, orig.X, orig.Y)
   293  	s.ctx.Uniform2f(s.uPathOffset, offset.X, offset.Y)
   294  	// Draw in batches that fit in uint16 indices.
   295  	start := 0
   296  	nquads := data.ncurves / 4
   297  	for start < nquads {
   298  		batch := nquads - start
   299  		if max := int(^uint16(0)) / 6; batch > max {
   300  			batch = max
   301  		}
   302  		// Enlarge VBO if necessary.
   303  		if batch > s.indexBufQuads {
   304  			indices := make([]uint16, batch*6)
   305  			for i := 0; i < batch; i++ {
   306  				i := uint16(i)
   307  				indices[i*6+0] = i*4 + 0
   308  				indices[i*6+1] = i*4 + 1
   309  				indices[i*6+2] = i*4 + 2
   310  				indices[i*6+3] = i*4 + 2
   311  				indices[i*6+4] = i*4 + 1
   312  				indices[i*6+5] = i*4 + 3
   313  			}
   314  			s.ctx.BufferData(gl.ELEMENT_ARRAY_BUFFER, gl.BytesView(indices), gl.STATIC_DRAW)
   315  			s.indexBufQuads = batch
   316  		}
   317  		off := path.VertStride * start * 4
   318  		s.ctx.VertexAttribPointer(attribPathCorner, 2, gl.SHORT, false, path.VertStride, off+int(unsafe.Offsetof((*(*path.Vertex)(nil)).CornerX)))
   319  		s.ctx.VertexAttribPointer(attribPathMaxY, 1, gl.FLOAT, false, path.VertStride, off+int(unsafe.Offsetof((*(*path.Vertex)(nil)).MaxY)))
   320  		s.ctx.VertexAttribPointer(attribPathFrom, 2, gl.FLOAT, false, path.VertStride, off+int(unsafe.Offsetof((*(*path.Vertex)(nil)).FromX)))
   321  		s.ctx.VertexAttribPointer(attribPathCtrl, 2, gl.FLOAT, false, path.VertStride, off+int(unsafe.Offsetof((*(*path.Vertex)(nil)).CtrlX)))
   322  		s.ctx.VertexAttribPointer(attribPathTo, 2, gl.FLOAT, false, path.VertStride, off+int(unsafe.Offsetof((*(*path.Vertex)(nil)).ToX)))
   323  		s.ctx.DrawElements(gl.TRIANGLES, batch*6, gl.UNSIGNED_SHORT, 0)
   324  		start += batch
   325  	}
   326  }
   327  
   328  func (s *stenciler) end() {
   329  	s.ctx.DisableVertexAttribArray(attribPathCorner)
   330  	s.ctx.DisableVertexAttribArray(attribPathMaxY)
   331  	s.ctx.DisableVertexAttribArray(attribPathFrom)
   332  	s.ctx.DisableVertexAttribArray(attribPathCtrl)
   333  	s.ctx.DisableVertexAttribArray(attribPathTo)
   334  	s.ctx.BindFramebuffer(gl.FRAMEBUFFER, s.defFBO)
   335  }
   336  
   337  func (p *pather) cover(z float32, mat materialType, col [4]float32, scale, off, uvScale, uvOff, coverScale, coverOff f32.Point) {
   338  	p.coverer.cover(z, mat, col, scale, off, uvScale, uvOff, coverScale, coverOff)
   339  }
   340  
   341  func (c *coverer) cover(z float32, mat materialType, col [4]float32, scale, off, uvScale, uvOff, coverScale, coverOff f32.Point) {
   342  	c.ctx.UseProgram(c.prog[mat])
   343  	switch mat {
   344  	case materialColor:
   345  		c.ctx.Uniform4f(c.vars[mat].uColor, col[0], col[1], col[2], col[3])
   346  	case materialTexture:
   347  		c.ctx.Uniform2f(c.vars[mat].uUVScale, uvScale.X, uvScale.Y)
   348  		c.ctx.Uniform2f(c.vars[mat].uUVOffset, uvOff.X, uvOff.Y)
   349  	}
   350  	c.ctx.Uniform1f(c.vars[mat].z, z)
   351  	c.ctx.Uniform2f(c.vars[mat].uScale, scale.X, scale.Y)
   352  	c.ctx.Uniform2f(c.vars[mat].uOffset, off.X, off.Y)
   353  	c.ctx.Uniform2f(c.vars[mat].uCoverUVScale, coverScale.X, coverScale.Y)
   354  	c.ctx.Uniform2f(c.vars[mat].uCoverUVOffset, coverOff.X, coverOff.Y)
   355  	c.ctx.DrawArrays(gl.TRIANGLE_STRIP, 0, 4)
   356  }
   357  
   358  const stencilVSrc = `
   359  #version 100
   360  
   361  precision highp float;
   362  
   363  uniform vec2 scale;
   364  uniform vec2 offset;
   365  uniform vec2 pathOffset;
   366  
   367  attribute vec2 corner;
   368  attribute float maxy;
   369  attribute vec2 from;
   370  attribute vec2 ctrl;
   371  attribute vec2 to;
   372  
   373  varying vec2 vFrom;
   374  varying vec2 vCtrl;
   375  varying vec2 vTo;
   376  
   377  void main() {
   378  	// Add a one pixel overlap so curve quads cover their
   379  	// entire curves. Could use conservative rasterization
   380  	// if available.
   381  	vec2 from = from + pathOffset;
   382  	vec2 ctrl = ctrl + pathOffset;
   383  	vec2 to = to + pathOffset;
   384  	float maxy = maxy + pathOffset.y;
   385  	vec2 pos;
   386  	if (corner.x > 0.0) {
   387  		// East.
   388  		pos.x = max(max(from.x, ctrl.x), to.x)+1.0;
   389  	} else {
   390  		// West.
   391  		pos.x = min(min(from.x, ctrl.x), to.x)-1.0;
   392  	}
   393  	if (corner.y > 0.0) {
   394  		// North.
   395  		pos.y = maxy + 1.0;
   396  	} else {
   397  		// South.
   398  		pos.y = min(min(from.y, ctrl.y), to.y) - 1.0;
   399  	}
   400  	vFrom = from-pos;
   401  	vCtrl = ctrl-pos;
   402  	vTo = to-pos;
   403      pos *= scale;
   404      pos += offset;
   405      gl_Position = vec4(pos, 1, 1);
   406  }
   407  `
   408  
   409  const stencilFSrc = `
   410  #version 100
   411  
   412  precision mediump float;
   413  
   414  varying vec2 vFrom;
   415  varying vec2 vCtrl;
   416  varying vec2 vTo;
   417  
   418  uniform sampler2D areaLUT;
   419  
   420  void main() {
   421  	float dx = vTo.x - vFrom.x;
   422  	// Sort from and to in increasing order so the root below
   423  	// is always the positive square root, if any.
   424  	// We need the direction of the curve below, so this can't be
   425  	// done from the vertex shader.
   426  	bool increasing = vTo.x >= vFrom.x;
   427  	vec2 left = increasing ? vFrom : vTo;
   428  	vec2 right = increasing ? vTo : vFrom;
   429  
   430  	// The signed horizontal extent of the fragment.
   431  	vec2 extent = clamp(vec2(vFrom.x, vTo.x), -0.5, 0.5);
   432  	// Find the t where the curve crosses the middle of the
   433  	// extent, x₀.
   434  	// Given the Bézier curve with x coordinates P₀, P₁, P₂
   435  	// where P₀ is at the origin, its x coordinate in t
   436  	// is given by:
   437  	//
   438  	// x(t) = 2(1-t)tP₁ + t²P₂
   439  	// 
   440  	// Rearranging:
   441  	//
   442  	// x(t) = (P₂ - 2P₁)t² + 2P₁t
   443  	//
   444  	// Setting x(t) = x₀ and using Muller's quadratic formula ("Citardauq")
   445  	// for robustnesss,
   446  	//
   447  	// t = 2x₀/(2P₁±√(4P₁²+4(P₂-2P₁)x₀))
   448  	//
   449  	// which simplifies to
   450  	//
   451  	// t = x₀/(P₁±√(P₁²+(P₂-2P₁)x₀))
   452  	//
   453  	// Setting v = P₂-P₁,
   454  	//
   455  	// t = x₀/(P₁±√(P₁²+(v-P₁)x₀))
   456  	//
   457  	// t lie in [0; 1]; P₂ ≥ P₁ and P₁ ≥ 0 since we split curves where
   458  	// the control point lies before the start point or after the end point.
   459  	// It can then be shown that only the positive square root is valid.
   460  	float midx = mix(extent.x, extent.y, 0.5);
   461  	float x0 = midx - left.x;
   462  	vec2 p1 = vCtrl - left;
   463  	vec2 v = right - vCtrl;
   464  	float t = x0/(p1.x+sqrt(p1.x*p1.x+(v.x-p1.x)*x0));
   465  	// Find y(t) on the curve.
   466  	float y = mix(mix(left.y, vCtrl.y, t), mix(vCtrl.y, right.y, t), t);
   467  	// And the slope.
   468  	vec2 d_half = mix(p1, v, t);
   469  	float dy = d_half.y/d_half.x;
   470  	// Together, y and dy form a line approximation.
   471  
   472  	// Compute the fragment area above the line.
   473  	// The area is symmetric around dy = 0. Scale slope with extent width.
   474  	float width = extent.y - extent.x;
   475  	dy = abs(dy*width);
   476  
   477  	vec4 sides = vec4(dy*+0.5 + y, dy*-0.5 + y, (+0.5-y)/dy, (-0.5-y)/dy);
   478  	sides = clamp(sides+0.5, 0.0, 1.0);
   479  
   480  	float area = 0.5*(sides.z - sides.z*sides.y + 1.0 - sides.x+sides.x*sides.w);
   481  	area *= width;
   482  
   483  	// Work around issue #13.
   484  	if (width == 0.0)
   485  		area = 0.0;
   486  
   487  	gl_FragColor.r = area;
   488  }
   489  `
   490  
   491  const coverVSrc = `
   492  #version 100
   493  
   494  precision highp float;
   495  
   496  uniform float z;
   497  uniform vec2 scale;
   498  uniform vec2 offset;
   499  uniform vec2 uvScale;
   500  uniform vec2 uvOffset;
   501  uniform vec2 uvCoverScale;
   502  uniform vec2 uvCoverOffset;
   503  
   504  attribute vec2 pos;
   505  
   506  varying vec2 vCoverUV;
   507  
   508  attribute vec2 uv;
   509  varying vec2 vUV;
   510  
   511  void main() {
   512      gl_Position = vec4(pos*scale + offset, z, 1);
   513  	vUV = uv*uvScale + uvOffset;
   514  	vCoverUV = uv*uvCoverScale+uvCoverOffset;
   515  }
   516  `
   517  
   518  const coverFSrc = `
   519  #version 100
   520  
   521  precision mediump float;
   522  
   523  // Use high precision to be pixel accurate for
   524  // large cover atlases.
   525  varying highp vec2 vCoverUV;
   526  uniform sampler2D cover;
   527  varying vec2 vUV;
   528  
   529  HEADER
   530  
   531  void main() {
   532      gl_FragColor = GET_COLOR;
   533  	float cover = abs(texture2D(cover, vCoverUV).r);
   534  	gl_FragColor *= cover;
   535  }
   536  `
   537  
   538  const intersectVSrc = `
   539  #version 100
   540  
   541  precision highp float;
   542  
   543  attribute vec2 pos;
   544  attribute vec2 uv;
   545  
   546  uniform vec2 uvScale;
   547  uniform vec2 uvOffset;
   548  
   549  varying vec2 vUV;
   550  
   551  void main() {
   552  	vec2 p = pos;
   553  	p.y = -p.y;
   554  	gl_Position = vec4(p, 0, 1);
   555  	vUV = uv*uvScale + uvOffset;
   556  }
   557  `
   558  
   559  const intersectFSrc = `
   560  #version 100
   561  
   562  precision mediump float;
   563  
   564  // Use high precision to be pixel accurate for
   565  // large cover atlases.
   566  varying highp vec2 vUV;
   567  uniform sampler2D cover;
   568  
   569  void main() {
   570  	float cover = abs(texture2D(cover, vUV).r);
   571      gl_FragColor.r = cover;
   572  }
   573  `