RESOLVED FIXED 31507
REGRESSION: [CAIRO] wrong drawing of Gradients and Patterns on texts
https://bugs.webkit.org/show_bug.cgi?id=31507
Summary REGRESSION: [CAIRO] wrong drawing of Gradients and Patterns on texts
Dirk Schulze
Reported 2009-11-14 02:57:33 PST
Created attachment 43226 [details] Test for text with Gradient. The gradient should hav same dimension on the rect as well as the text The rendering of texts (filled/stroked) with gradients or patterns is wrong. This is broken since r44268 http://trac.webkit.org/changeset/44268/trunk/WebCore/platform/graphics/cairo/FontCairo.cpp. Maybe the changes are to Windows specific and should be embraced by if PLATFORM(WIN)??? I got it working after undoing the changes. There is also a LayoutTest or bug number missing on r44268 for a easy check.
Attachments
Test for text with Gradient. The gradient should hav same dimension on the rect as well as the text (673 bytes, image/svg+xml)
2009-11-14 02:57 PST, Dirk Schulze
no flags
Patch (1.91 MB, patch)
2011-02-26 12:42 PST, Martin Robinson
no flags
Patch with Benjamin's suggestions (1.90 MB, patch)
2011-02-27 12:41 PST, Martin Robinson
no flags
Patch updated to ToT and using PlatformContextCairo (1.72 MB, patch)
2011-04-11 13:45 PDT, Martin Robinson
no flags
Patch (1.34 MB, patch)
2011-05-03 22:21 PDT, Martin Robinson
no flags
Patch with Dirk's suggestion (1.34 MB, patch)
2011-05-04 08:39 PDT, Martin Robinson
no flags
Dirk Schulze
Comment 1 2009-11-22 12:46:38 PST
Added Alexander, since it is his patch, that breaks the rendering.
Brent Fulgham
Comment 2 2009-12-02 16:20:30 PST
Note: This renders properly in WinCairo.
Dirk Schulze
Comment 3 2009-12-03 14:44:59 PST
Brent gave me the link of the original bug. It's bug 21492. I unapplied the changes of FontCairo.cpp and everything works. Should we role out the patch?
Martin Robinson
Comment 4 2011-02-25 18:33:47 PST
The issue here seems the be that we are setting the Cairo source with the transformation matrix at the time of the fill, rather than with the transformation matrix at the time the fill was set on the WebCore::GraphicsContext. I should have a patch for this posted within the next couple of days that fixes quite a few SVG test cases.
Martin Robinson
Comment 5 2011-02-26 12:42:31 PST
Benjamin Otte
Comment 6 2011-02-26 14:51:03 PST
View in context: https://bugs.webkit.org/attachment.cgi?id=83948&action=review > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:146 > + cairo_push_group(cr); please add: cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE); > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:173 > + reduceSourceByAlpha(cr, globalAlpha); Maybe add a FIXME here to add a function for reducing color of gradients manually? As I said yesterday, you can create a new gradient pattern with the same colorstops reduced by global alpha. > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:180 > + } I'd factor out lines 163-180 into a common function to be called from here and prepareForStroking. > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:226 > + cairo_restore(cr); Is this correct now that we do global alpha in setPlatformFill()? This should just be a cairo_fill(), no?
Martin Robinson
Comment 7 2011-02-27 12:41:20 PST
Created attachment 83986 [details] Patch with Benjamin's suggestions
Martin Robinson
Comment 8 2011-02-27 12:41:49 PST
(In reply to comment #6) > View in context: https://bugs.webkit.org/attachment.cgi?id=83948&action=review Thanks for the suggestions. > > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:146 > > + cairo_push_group(cr); > > please add: > cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE); Done! > > > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:173 > > + reduceSourceByAlpha(cr, globalAlpha); > > Maybe add a FIXME here to add a function for reducing color of gradients manually? As I said yesterday, you can create a new gradient pattern with the same colorstops reduced by global alpha. Done! > > > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:180 > > + } > > I'd factor out lines 163-180 into a common function to be called from here and prepareForStroking. Done! > > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:226 > > + cairo_restore(cr); > > Is this correct now that we do global alpha in setPlatformFill()? > This should just be a cairo_fill(), no? Since I passed in the NoAdjustment argument here, prepareForFilling didn't adjust the source patterns for global alpha. Since we can clip here, I think that cairo_paint_with_alpha might be faster here.
Brent Fulgham
Comment 9 2011-04-10 22:31:30 PDT
Comment on attachment 83986 [details] Patch with Benjamin's suggestions I like this change a lot. It simplifies a number of rather confusing routines.
Martin Robinson
Comment 10 2011-04-11 13:45:55 PDT
Created attachment 89073 [details] Patch updated to ToT and using PlatformContextCairo
Dirk Schulze
Comment 11 2011-04-26 16:41:12 PDT
Comment on attachment 89073 [details] Patch updated to ToT and using PlatformContextCairo View in context: https://bugs.webkit.org/attachment.cgi?id=89073&action=review Great patch. r=me > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:135 > + cairo_set_source_rgb(shadowContext, 1.0, 0.0, 0.0); use 1 and 0
Eric Seidel (no email)
Comment 12 2011-04-26 16:51:32 PDT
Comment on attachment 89073 [details] Patch updated to ToT and using PlatformContextCairo I feel like this patch may be wrong. Because you should probably be more like CG, and not need to use setPlatformPattern, instead just use the ctm at draw time. But, this passes the test. This stuff is hard, and requires iteration. I think we should land this patch and work from here. Talk to krit.
Dirk Schulze
Comment 13 2011-04-26 17:02:06 PDT
(In reply to comment #12) > (From update of attachment 89073 [details]) > I feel like this patch may be wrong. Because you should probably be more like CG, and not need to use setPlatformPattern, instead just use the ctm at draw time. > > But, this passes the test. This stuff is hard, and requires iteration. I think we should land this patch and work from here. Talk to krit. The reason why we do it the way in CG is different to Cairo. With CG you can transform patterns, gradients and the context idependent of each other. Cairo is able to do so.
Martin Robinson
Comment 14 2011-05-03 21:46:46 PDT
Comment on attachment 89073 [details] Patch updated to ToT and using PlatformContextCairo It seems that I misunderstood the root cause of this issue. It turns out that that we just need to ensure that the pattern/gradient is set on the context before transforming for the final text render. I've uploaded a greatly simplified patch for this issue, which is also based on ToT. Sorry for the patch spam.
Eric Seidel (no email)
Comment 15 2011-05-03 22:18:06 PDT
Comment on attachment 89073 [details] Patch updated to ToT and using PlatformContextCairo View in context: https://bugs.webkit.org/attachment.cgi?id=89073&action=review Hmm... I predict you're gonna rewrite this again... I rewrote the gradient stuff for CG at least 3 times before it got to its current state... > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:786 > + platformContext()->updateStrokeTransform(); I'm still not sure why you need the transform saved at this time and can't just use the CTM when the drawing is done? > Source/WebCore/platform/graphics/cairo/PlatformContextCairo.cpp:39 > + , m_strokeTransform(static_cast<cairo_matrix_t*>(fastMalloc(sizeof(cairo_matrix_t)))) Wow. Really? They don't provide any create function? > Source/WebCore/platform/graphics/cairo/PlatformContextCairo.cpp:48 > + fastFree(m_strokeTransform); Why not just use an OwnPtr if we're just gonna malloc and free? I guess ownptr calls delete... which would be bad... or? > Source/WebCore/platform/graphics/cairo/PlatformContextCairo.cpp:101 > + if (alpha >= 1) What would this even mean?
Martin Robinson
Comment 16 2011-05-03 22:21:22 PDT
Martin Robinson
Comment 17 2011-05-03 22:25:43 PDT
(In reply to comment #15) > (From update of attachment 89073 [details]) > View in context: https://bugs.webkit.org/attachment.cgi?id=89073&action=review > Hmm... I predict you're gonna rewrite this again... I rewrote the gradient stuff for CG at least 3 times before it got to its current state... It's surprisingly tricky! > > > Source/WebCore/platform/graphics/cairo/GraphicsContextCairo.cpp:786 > > + platformContext()->updateStrokeTransform(); > > I'm still not sure why you need the transform saved at this time and can't just use the CTM when the drawing is done? > > > Source/WebCore/platform/graphics/cairo/PlatformContextCairo.cpp:39 > > + , m_strokeTransform(static_cast<cairo_matrix_t*>(fastMalloc(sizeof(cairo_matrix_t)))) > > Wow. Really? They don't provide any create function? > > > Source/WebCore/platform/graphics/cairo/PlatformContextCairo.cpp:48 > > + fastFree(m_strokeTransform); > > Why not just use an OwnPtr if we're just gonna malloc and free? I guess ownptr calls delete... which would be bad... or? Apolgies! There was a delay in uploading the new patch because of a scrollbar race condition. I'm concurrently working on that as well. The code above no longer exists. > > > Source/WebCore/platform/graphics/cairo/PlatformContextCairo.cpp:101 > > + if (alpha >= 1) > What would this even mean? It seemed safer given the range of floats. I can use an ASSERT here instead, if you think that would be better.
Dirk Schulze
Comment 18 2011-05-04 00:27:45 PDT
Comment on attachment 92192 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=92192&action=review > Source/WebCore/platform/graphics/cairo/FontCairo.cpp:51 > + static const float syntheticObliqueSkew = -tanf(14 * acosf(0) / 90); I'd move this to the top of the file. This is common style in webkit for static consts. > Source/WebCore/platform/graphics/cairo/PlatformContextCairo.cpp:102 > + RefPtr<cairo_pattern_t> cairoPattern(adoptRef(pattern->createPlatformPattern(AffineTransform()))); > + cairo_set_source(cr, cairoPattern.get()); Again, it looks like you change the CTM of the context to match the pattern space instead of doing it the other way around. We have to d it in CG, because CG doesn't provide a pattern transform. But cairo has a second transform. Just unapply every change you made to the context to get italic fonts working from the pattern and you're done. No extra changes to the CTM please.
Martin Robinson
Comment 19 2011-05-04 08:24:17 PDT
Comment on attachment 92192 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=92192&action=review >> Source/WebCore/platform/graphics/cairo/FontCairo.cpp:51 >> + static const float syntheticObliqueSkew = -tanf(14 * acosf(0) / 90); > > I'd move this to the top of the file. This is common style in webkit for static consts. Sure thing. I'll upload a new patch. >> Source/WebCore/platform/graphics/cairo/PlatformContextCairo.cpp:102 >> + cairo_set_source(cr, cairoPattern.get()); > > Again, it looks like you change the CTM of the context to match the pattern space instead of doing it the other way around. We have to d it in CG, because CG doesn't provide a pattern transform. But cairo has a second transform. Just unapply every change you made to the context to get italic fonts working from the pattern and you're done. No extra changes to the CTM please. I'm not sure I understand what you mean here. The pattern is transformed with cairo_pattern_set_matrix in Pattern::createPlatformPattern, but not here. If I understand what you're saying correctly, I'm doing what you suggest already. :)
Martin Robinson
Comment 20 2011-05-04 08:39:01 PDT
Created attachment 92248 [details] Patch with Dirk's suggestion
Dirk Schulze
Comment 21 2011-06-18 22:39:16 PDT
Comment on attachment 92248 [details] Patch with Dirk's suggestion r=me
Martin Robinson
Comment 22 2011-06-20 16:52:02 PDT
Martin Robinson
Comment 23 2011-06-20 16:52:33 PDT
Comment on attachment 92248 [details] Patch with Dirk's suggestion Clearing flags after landing. Thanks for the review Dirk!
Note You need to log in before you can comment on or make changes to this bug.