Merge pull request #12869 from calixteman/lw
Fix zoom issue with too thin lines
This commit is contained in:
		
						commit
						4142001fc2
					
				| @ -923,6 +923,10 @@ const CanvasGraphics = (function CanvasGraphicsClosure() { | |||||||
|       this.ctx.transform.apply(this.ctx, viewport.transform); |       this.ctx.transform.apply(this.ctx, viewport.transform); | ||||||
| 
 | 
 | ||||||
|       this.baseTransform = this.ctx.mozCurrentTransform.slice(); |       this.baseTransform = this.ctx.mozCurrentTransform.slice(); | ||||||
|  |       this._combinedScaleFactor = Math.hypot( | ||||||
|  |         this.baseTransform[0], | ||||||
|  |         this.baseTransform[2] | ||||||
|  |       ); | ||||||
| 
 | 
 | ||||||
|       if (this.imageLayer) { |       if (this.imageLayer) { | ||||||
|         this.imageLayer.beginLayout(); |         this.imageLayer.beginLayout(); | ||||||
| @ -1357,22 +1361,23 @@ const CanvasGraphics = (function CanvasGraphicsClosure() { | |||||||
|           const scale = Util.singularValueDecompose2dScale(transform)[0]; |           const scale = Util.singularValueDecompose2dScale(transform)[0]; | ||||||
|           ctx.strokeStyle = strokeColor.getPattern(ctx, this); |           ctx.strokeStyle = strokeColor.getPattern(ctx, this); | ||||||
|           const lineWidth = this.getSinglePixelWidth(); |           const lineWidth = this.getSinglePixelWidth(); | ||||||
|           if (lineWidth === -1) { |           const scaledLineWidth = this.current.lineWidth * scale; | ||||||
|  |           if (lineWidth < 0 && -lineWidth >= scaledLineWidth) { | ||||||
|             ctx.resetTransform(); |             ctx.resetTransform(); | ||||||
|             ctx.lineWidth = 1; |             ctx.lineWidth = Math.round(this._combinedScaleFactor); | ||||||
|           } else { |           } else { | ||||||
|             ctx.lineWidth = Math.max(lineWidth, this.current.lineWidth * scale); |             ctx.lineWidth = Math.max(lineWidth, scaledLineWidth); | ||||||
|           } |           } | ||||||
|           ctx.stroke(); |           ctx.stroke(); | ||||||
|           ctx.restore(); |           ctx.restore(); | ||||||
|         } else { |         } else { | ||||||
|           const lineWidth = this.getSinglePixelWidth(); |           const lineWidth = this.getSinglePixelWidth(); | ||||||
|           if (lineWidth === -1) { |           if (lineWidth < 0 && -lineWidth >= this.current.lineWidth) { | ||||||
|             // The current transform will transform a square pixel into a
 |             // The current transform will transform a square pixel into a
 | ||||||
|             // parallelogram where both heights are lower than 1 and not equal.
 |             // parallelogram where both heights are lower than 1 and not equal.
 | ||||||
|             ctx.save(); |             ctx.save(); | ||||||
|             ctx.resetTransform(); |             ctx.resetTransform(); | ||||||
|             ctx.lineWidth = 1; |             ctx.lineWidth = Math.round(this._combinedScaleFactor); | ||||||
|             ctx.stroke(); |             ctx.stroke(); | ||||||
|             ctx.restore(); |             ctx.restore(); | ||||||
|           } else { |           } else { | ||||||
| @ -1618,7 +1623,7 @@ const CanvasGraphics = (function CanvasGraphicsClosure() { | |||||||
|         ) { |         ) { | ||||||
|           if (resetLineWidthToOne) { |           if (resetLineWidthToOne) { | ||||||
|             ctx.resetTransform(); |             ctx.resetTransform(); | ||||||
|             ctx.lineWidth = 1; |             ctx.lineWidth = Math.round(this._combinedScaleFactor); | ||||||
|           } |           } | ||||||
|           ctx.stroke(); |           ctx.stroke(); | ||||||
|         } |         } | ||||||
| @ -1638,7 +1643,7 @@ const CanvasGraphics = (function CanvasGraphicsClosure() { | |||||||
|             ctx.save(); |             ctx.save(); | ||||||
|             ctx.moveTo(x, y); |             ctx.moveTo(x, y); | ||||||
|             ctx.resetTransform(); |             ctx.resetTransform(); | ||||||
|             ctx.lineWidth = 1; |             ctx.lineWidth = Math.round(this._combinedScaleFactor); | ||||||
|             ctx.strokeText(character, 0, 0); |             ctx.strokeText(character, 0, 0); | ||||||
|             ctx.restore(); |             ctx.restore(); | ||||||
|           } else { |           } else { | ||||||
| @ -1741,7 +1746,7 @@ const CanvasGraphics = (function CanvasGraphicsClosure() { | |||||||
|         ) { |         ) { | ||||||
|           this._cachedGetSinglePixelWidth = null; |           this._cachedGetSinglePixelWidth = null; | ||||||
|           lineWidth = this.getSinglePixelWidth(); |           lineWidth = this.getSinglePixelWidth(); | ||||||
|           resetLineWidthToOne = lineWidth === -1; |           resetLineWidthToOne = lineWidth < 0; | ||||||
|         } |         } | ||||||
|       } else { |       } else { | ||||||
|         lineWidth /= scale; |         lineWidth /= scale; | ||||||
| @ -2661,25 +2666,31 @@ const CanvasGraphics = (function CanvasGraphicsClosure() { | |||||||
|         // This is equivalent to:
 |         // This is equivalent to:
 | ||||||
|         //  h = max(|line_1_inv(M)|, |line_2_inv(M)|)
 |         //  h = max(|line_1_inv(M)|, |line_2_inv(M)|)
 | ||||||
|         const m = this.ctx.mozCurrentTransform; |         const m = this.ctx.mozCurrentTransform; | ||||||
|         const sqDet = (m[0] * m[3] - m[2] * m[1]) ** 2; | 
 | ||||||
|  |         const absDet = Math.abs(m[0] * m[3] - m[2] * m[1]); | ||||||
|         const sqNorm1 = m[0] ** 2 + m[2] ** 2; |         const sqNorm1 = m[0] ** 2 + m[2] ** 2; | ||||||
|         const sqNorm2 = m[1] ** 2 + m[3] ** 2; |         const sqNorm2 = m[1] ** 2 + m[3] ** 2; | ||||||
|         if (sqNorm1 !== sqNorm2 && (sqNorm1 > sqDet || sqNorm2 > sqDet)) { |         const pixelHeight = Math.sqrt(Math.max(sqNorm1, sqNorm2)) / absDet; | ||||||
|           // The parallelogram isn't a losange and at least one height
 |         if ( | ||||||
|  |           sqNorm1 !== sqNorm2 && | ||||||
|  |           this._combinedScaleFactor * pixelHeight > 1 | ||||||
|  |         ) { | ||||||
|  |           // The parallelogram isn't a square and at least one height
 | ||||||
|           // is lower than 1 so the resulting line width must be 1
 |           // is lower than 1 so the resulting line width must be 1
 | ||||||
|           // but it cannot be achieved with one scale: when scaling a pixel
 |           // but it cannot be achieved with one scale: when scaling a pixel
 | ||||||
|           // we'll get a rectangle (see issue #12295).
 |           // we'll get a rectangle (see issue #12295).
 | ||||||
|           // For example with matrix [0.001 0, 0, 100], a pixel is transformed
 |           // For example with matrix [0.001 0, 0, 100], a pixel is transformed
 | ||||||
|           // in a rectangle 0.001x100. If we just scale by 1000 (to have a 1)
 |           // in a rectangle 0.001x100. If we just scale by 1000 (to have a 1)
 | ||||||
|           // then we'll get a rectangle 1x1e5 which is wrong.
 |           // then we'll get a rectangle 1x1e5 which is wrong.
 | ||||||
|           // In this case, we must reset the transform and set linewidth to 1
 |           // In this case, we must reset the transform, set linewidth to 1
 | ||||||
|           // and then stroke.
 |           // and then stroke.
 | ||||||
|           this._cachedGetSinglePixelWidth = -1; |           this._cachedGetSinglePixelWidth = -( | ||||||
|         } else if (sqDet > Number.EPSILON ** 2) { |             this._combinedScaleFactor * pixelHeight | ||||||
|  |           ); | ||||||
|  |         } else if (absDet > Number.EPSILON) { | ||||||
|           // The multiplication by the constant 1.0000001 is here to have
 |           // The multiplication by the constant 1.0000001 is here to have
 | ||||||
|           // a number slightly greater than what we "exactly" want.
 |           // a number slightly greater than what we "exactly" want.
 | ||||||
|           this._cachedGetSinglePixelWidth = |           this._cachedGetSinglePixelWidth = pixelHeight * 1.0000001; | ||||||
|             Math.sqrt(Math.max(sqNorm1, sqNorm2) / sqDet) * 1.0000001; |  | ||||||
|         } else { |         } else { | ||||||
|           // Matrix is non-invertible.
 |           // Matrix is non-invertible.
 | ||||||
|           this._cachedGetSinglePixelWidth = 1; |           this._cachedGetSinglePixelWidth = 1; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user