[Cherry-pick][Text Autosizing] prevent oscillation of font sizes during autosizing
authorcommit-queue@webkit.org <commit-queue@webkit.org@268f45cc-cd09-0410-ab3c-d52691b4dbfc>
Tue, 19 Feb 2013 18:57:31 +0000 (18:57 +0000)
committerJaehun Lim <ljaehun.lim@samsung.com>
Thu, 4 Apr 2013 02:45:17 +0000 (11:45 +0900)
Text Autosizing: prevent oscillation of font sizes during autosizing
https://bugs.webkit.org/show_bug.cgi?id=108205

Patch by Tim Volodine <timvolodine@chromium.org> on 2013-02-19
Reviewed by Kenneth Rohde Christiansen.

Source/WebCore:

On some websites autosized font-sizes oscillate due to layouts caused by
hovering or incremental page loading (and on other sites font sizes do
eventually stabilize, but it takes many layouts before they reach a steady
size). To prevent all these cases, we no longer allow the autosizing
multiplier to change after it has been set (to a value other than 1).

This won't always give exactly the same results, but testing on 2000 top
sites shows that this makes little difference in practice, and it prevents
these very jarring cases. As a happy side-effect, this speeds up layouts
as font sizes change less.

Test: fast/text-autosizing/oscillation-javascript-fontsize-change.html

* page/FrameView.cpp:
(WebCore::FrameView::setFrameRect):
* page/Settings.cpp:
(WebCore::Settings::setTextAutosizingFontScaleFactor):
* rendering/TextAutosizer.cpp:
(WebCore::TextAutosizer::recalculateMultipliers):
(WebCore):
(WebCore::TextAutosizer::processContainer):
* rendering/TextAutosizer.h:
(TextAutosizer):

LayoutTests:

Added overflow-y:hidden to some existing tests, since previously those tests
would start off with incorrect multipliers (because mainFrame->view()-layoutSize()
is initially 785 instead of 800 as ScrollView wrongly guesses a scrollbar will
be needed), and then the multipliers would get corrected on a subsequent layout.
Now that we don't allow the multiplier to change after being set, it needs to be
right first time.
Also added specific oscillation test triggered by javascript.

* fast/text-autosizing/constrained-height-body-expected.html:
* fast/text-autosizing/constrained-height-body.html:
* fast/text-autosizing/constrained-then-float-ancestors-expected.html:
* fast/text-autosizing/constrained-then-float-ancestors.html:
* fast/text-autosizing/constrained-then-position-fixed-ancestors-expected.html:
* fast/text-autosizing/constrained-then-position-fixed-ancestors.html:
* fast/text-autosizing/nested-em-line-height-expected.html:
* fast/text-autosizing/nested-em-line-height.html:
* fast/text-autosizing/oscillation-javascript-fontsize-change-expected.html: Added.
* fast/text-autosizing/oscillation-javascript-fontsize-change.html: Added.
* fast/text-autosizing/simple-paragraph-expected.html:
* fast/text-autosizing/simple-paragraph.html:
* fast/text-autosizing/span-child-expected.html:
* fast/text-autosizing/span-child.html:
* fast/text-autosizing/unwrappable-blocks-expected.html:
* fast/text-autosizing/unwrappable-blocks.html:
* fast/text-autosizing/unwrappable-inlines-expected.html:
* fast/text-autosizing/unwrappable-inlines.html:

git-svn-id: http://svn.webkit.org/repository/webkit/trunk@143356 268f45cc-cd09-0410-ab3c-d52691b4dbfc

Conflicts:

LayoutTests/ChangeLog
Source/WebCore/ChangeLog

22 files changed:
LayoutTests/fast/text-autosizing/constrained-height-body-expected.html
LayoutTests/fast/text-autosizing/constrained-height-body.html
LayoutTests/fast/text-autosizing/constrained-then-float-ancestors-expected.html
LayoutTests/fast/text-autosizing/constrained-then-float-ancestors.html
LayoutTests/fast/text-autosizing/constrained-then-position-fixed-ancestors-expected.html
LayoutTests/fast/text-autosizing/constrained-then-position-fixed-ancestors.html
LayoutTests/fast/text-autosizing/nested-em-line-height-expected.html
LayoutTests/fast/text-autosizing/nested-em-line-height.html
LayoutTests/fast/text-autosizing/oscillation-javascript-fontsize-change-expected.html [new file with mode: 0644]
LayoutTests/fast/text-autosizing/oscillation-javascript-fontsize-change.html [new file with mode: 0644]
LayoutTests/fast/text-autosizing/simple-paragraph-expected.html
LayoutTests/fast/text-autosizing/simple-paragraph.html
LayoutTests/fast/text-autosizing/span-child-expected.html
LayoutTests/fast/text-autosizing/span-child.html
LayoutTests/fast/text-autosizing/unwrappable-blocks-expected.html
LayoutTests/fast/text-autosizing/unwrappable-blocks.html
LayoutTests/fast/text-autosizing/unwrappable-inlines-expected.html
LayoutTests/fast/text-autosizing/unwrappable-inlines.html
Source/WebCore/page/FrameView.cpp
Source/WebCore/page/Settings.cpp
Source/WebCore/rendering/TextAutosizer.cpp
Source/WebCore/rendering/TextAutosizer.h

index 59a1080..be3cd2f 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 </head>
index 26ec70a..bc41fbd 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 <script>
index 1e67107..a7e6626 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 </head>
index 83c9bb2..b82d185 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 <script>
index 783a512..3b24b4e 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 </head>
index 24c77d4..06383d5 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 <script>
index caffb68..32638c4 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 </head>
index 2baa55e..2aa345e 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 <script>
diff --git a/LayoutTests/fast/text-autosizing/oscillation-javascript-fontsize-change-expected.html b/LayoutTests/fast/text-autosizing/oscillation-javascript-fontsize-change-expected.html
new file mode 100644 (file)
index 0000000..04a7946
--- /dev/null
@@ -0,0 +1,31 @@
+<!DOCTYPE html>
+<html style="font-size: 16px">
+<head>
+
+<meta name="viewport" content="width=800">
+<style>
+  body {
+    width: 800px;
+    margin: 0;
+    overflow-y: hidden;
+  }
+</style>
+
+</head>
+<body>
+
+<div style="top:50px;position:absolute;font-size:19.8px">
+    . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .<br/>
+    This paragraph should be autosized to 19.8px<br/>
+    because it contains line breaks.<br/>
+    This test is intended to check<br/>
+    that there are no oscillations due to javascript<br/>
+    briefly increasing the font size of a<br/>
+    small paragraph below.
+</div>
+<div id="sizechanging">
+    This text changes size using javascript below.
+</div>
+
+</body>
+</html>
diff --git a/LayoutTests/fast/text-autosizing/oscillation-javascript-fontsize-change.html b/LayoutTests/fast/text-autosizing/oscillation-javascript-fontsize-change.html
new file mode 100644 (file)
index 0000000..190cce4
--- /dev/null
@@ -0,0 +1,53 @@
+<!DOCTYPE html>
+<html style="font-size: 16px">
+<head>
+
+<meta name="viewport" content="width=800">
+<style>
+  body {
+    width: 800px;
+    margin: 0;
+    overflow-y: hidden;
+  }
+  .largersize{font-size: 1.1em}
+</style>
+
+<script>
+if (window.internals) {
+    window.internals.settings.setTextAutosizingEnabled(true);
+    window.internals.settings.setTextAutosizingWindowSizeOverride(320, 480);
+} else if (window.console && console.warn) {
+    console.warn("This test depends on the Text Autosizing setting being true, so run it in DumpRenderTree, or manually enable Text Autosizing, and either use a mobile device with 320px device-width (like Nexus S or iPhone), or define HACK_FORCE_TEXT_AUTOSIZING_ON_DESKTOP.");
+}
+</script>
+
+</head>
+<body>
+
+<div style="top:50px;position:absolute;">
+    . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .<br/>
+    This paragraph should be autosized to 19.8px<br/>
+    because it contains line breaks.<br/>
+    This test is intended to check<br/>
+    that there are no oscillations due to javascript<br/>
+    briefly increasing the font size of a<br/>
+    small paragraph below.
+</div>
+<div id="sizechanging">
+    This text changes size using javascript below.
+</div>
+
+<script>
+element = document.getElementById("sizechanging");
+if (element.offsetHeight) {
+    // force layout (computation of offsetHeight triggers reflow)
+}
+element.className = 'largersize';
+if (element.offsetHeight) {}
+element.className = '';
+if (element.offsetHeight) {}
+</script>
+<noscript>fail (no support for javascript)</noscript>
+
+</body>
+</html>
index 3fd4a3a..d15ab21 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 </head>
index 850916e..c974c4b 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 <script>
index cc44216..c02991a 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 </head>
index 825bf44..7754063 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 </style>
 
 <script>
index 48d3b88..7ed18df 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 pre { margin: 0; }
 </style>
 
index b6603ae..7929454 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 pre { margin: 0; }
 </style>
 
index 48fc903..a718814 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 pre { margin: 0; }
 </style>
 
index 9b0732e..16434c9 100644 (file)
@@ -5,7 +5,7 @@
 <meta name="viewport" content="width=800">
 <style>
 html { font-size: 16px; }
-body { width: 800px; margin: 0; }
+body { width: 800px; margin: 0; overflow-y: hidden; }
 pre { margin: 0; }
 </style>
 
index 20596c0..f388d21 100755 (executable)
@@ -419,6 +419,17 @@ void FrameView::setFrameRect(const IntRect& newRect)
     if (newRect == oldRect)
         return;
 
+#if ENABLE(TEXT_AUTOSIZING)
+    // Autosized font sizes depend on the width of the viewing area.
+    if (newRect.width() != oldRect.width()) {
+        Page* page = m_frame ? m_frame->page() : 0;
+        if (page && page->mainFrame() == m_frame && page->settings()->textAutosizingEnabled()) {
+            for (Frame* frame = page->mainFrame(); frame; frame = frame->tree()->traverseNext())
+                m_frame->document()->textAutosizer()->recalculateMultipliers();
+        }
+    }
+#endif
+
     ScrollView::setFrameRect(newRect);
 
     updateScrollableAreaSet();
index 7035cd2..158f9d8 100755 (executable)
@@ -40,6 +40,7 @@
 #include "PageCache.h"
 #include "ResourceHandle.h"
 #include "StorageMap.h"
+#include "TextAutosizer.h"
 #include <limits>
 
 using namespace std;
@@ -457,6 +458,11 @@ void Settings::setTextAutosizingWindowSizeOverride(const IntSize& textAutosizing
 void Settings::setTextAutosizingFontScaleFactor(float fontScaleFactor)
 {
     m_textAutosizingFontScaleFactor = fontScaleFactor;
+
+    // FIXME: I wonder if this needs to traverse frames like in WebViewImpl::resize, or whether there is only one document per Settings instance?
+    for (Frame* frame = m_page->mainFrame(); frame; frame = frame->tree()->traverseNext())
+        frame->document()->textAutosizer()->recalculateMultipliers();
+
     m_page->setNeedsRecalcStyleInAllFrames();
 }
 
index 43f2a1a..900ba55 100644 (file)
@@ -90,6 +90,16 @@ TextAutosizer::~TextAutosizer()
 {
 }
 
+void TextAutosizer::recalculateMultipliers()
+{
+    RenderObject* renderer = m_document->renderer();
+    while (renderer) {
+        if (renderer->style() && renderer->style()->textAutosizingMultiplier() != 1)
+            setMultiplier(renderer, 1);
+        renderer = renderer->nextInPreOrder();
+    }
+}
+
 bool TextAutosizer::processSubtree(RenderObject* layoutRoot)
 {
     // FIXME: Text Autosizing should only be enabled when m_document->page()->mainFrame()->view()->useFixedLayout()
@@ -188,7 +198,7 @@ void TextAutosizer::processContainer(float multiplier, RenderBlock* container, T
     RenderObject* descendant = nextInPreOrderSkippingDescendantsOfContainers(subtreeRoot, subtreeRoot);
     while (descendant) {
         if (descendant->isText()) {
-            if (localMultiplier != descendant->style()->textAutosizingMultiplier()) {
+            if (localMultiplier != 1 && descendant->style()->textAutosizingMultiplier() == 1) {
                 setMultiplier(descendant, localMultiplier);
                 setMultiplier(descendant->parent(), localMultiplier); // Parent does line spacing.
             }
index bb38baa..1f7f7b7 100644 (file)
@@ -50,6 +50,7 @@ public:
     virtual ~TextAutosizer();
 
     bool processSubtree(RenderObject* layoutRoot);
+    void recalculateMultipliers();
 
     static float computeAutosizedFontSize(float specifiedSize, float multiplier);