Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 /* | 1 /* |
| 2 * Copyright (C) 2004, 2008, 2009, 2010 Apple Inc. All rights reserved. | 2 * Copyright (C) 2004, 2008, 2009, 2010 Apple Inc. All rights reserved. |
| 3 * | 3 * |
| 4 * Redistribution and use in source and binary forms, with or without | 4 * Redistribution and use in source and binary forms, with or without |
| 5 * modification, are permitted provided that the following conditions | 5 * modification, are permitted provided that the following conditions |
| 6 * are met: | 6 * are met: |
| 7 * 1. Redistributions of source code must retain the above copyright | 7 * 1. Redistributions of source code must retain the above copyright |
| 8 * notice, this list of conditions and the following disclaimer. | 8 * notice, this list of conditions and the following disclaimer. |
| 9 * 2. Redistributions in binary form must reproduce the above copyright | 9 * 2. Redistributions in binary form must reproduce the above copyright |
| 10 * notice, this list of conditions and the following disclaimer in the | 10 * notice, this list of conditions and the following disclaimer in the |
| (...skipping 29 matching lines...) Expand all Loading... | |
| 40 #include "platform/graphics/GraphicsLayer.h" | 40 #include "platform/graphics/GraphicsLayer.h" |
| 41 #include "platform/graphics/paint/DrawingRecorder.h" | 41 #include "platform/graphics/paint/DrawingRecorder.h" |
| 42 | 42 |
| 43 namespace blink { | 43 namespace blink { |
| 44 | 44 |
| 45 CaretBase::CaretBase() = default; | 45 CaretBase::CaretBase() = default; |
| 46 CaretBase::~CaretBase() = default; | 46 CaretBase::~CaretBase() = default; |
| 47 | 47 |
| 48 DEFINE_TRACE(CaretBase) {} | 48 DEFINE_TRACE(CaretBase) {} |
| 49 | 49 |
| 50 void CaretBase::clearCaretRect() { | |
| 51 m_caretLocalRect = LayoutRect(); | |
| 52 } | |
| 53 | |
| 54 static inline bool caretRendersInsideNode(Node* node) { | 50 static inline bool caretRendersInsideNode(Node* node) { |
| 55 return node && !isDisplayInsideTable(node) && !editingIgnoresContent(*node); | 51 return node && !isDisplayInsideTable(node) && !editingIgnoresContent(*node); |
| 56 } | 52 } |
| 57 | 53 |
| 58 LayoutBlock* CaretBase::caretLayoutObject(Node* node) { | 54 LayoutBlock* CaretBase::caretLayoutObject(Node* node) { |
| 59 if (!node) | 55 if (!node) |
| 60 return nullptr; | 56 return nullptr; |
| 61 | 57 |
| 62 LayoutObject* layoutObject = node->layoutObject(); | 58 LayoutObject* layoutObject = node->layoutObject(); |
| 63 if (!layoutObject) | 59 if (!layoutObject) |
| 64 return nullptr; | 60 return nullptr; |
| 65 | 61 |
| 66 // if caretNode is a block and caret is inside it then caret should be painted | 62 // if caretNode is a block and caret is inside it then caret should be painted |
| 67 // by that block | 63 // by that block |
| 68 bool paintedByBlock = | 64 bool paintedByBlock = |
| 69 layoutObject->isLayoutBlock() && caretRendersInsideNode(node); | 65 layoutObject->isLayoutBlock() && caretRendersInsideNode(node); |
| 70 // TODO(yoichio): This function is called at least | 66 // TODO(yoichio): This function is called at least |
| 71 // DocumentLifeCycle::LayoutClean but caretRendersInsideNode above can | 67 // DocumentLifeCycle::LayoutClean but caretRendersInsideNode above can |
| 72 // layout. Thus |node->layoutObject()| can be changed then this is bad | 68 // layout. Thus |node->layoutObject()| can be changed then this is bad |
| 73 // design. We should make caret painting algorithm clean. | 69 // design. We should make caret painting algorithm clean. |
| 74 CHECK_EQ(layoutObject, node->layoutObject()) | 70 CHECK_EQ(layoutObject, node->layoutObject()) |
| 75 << "Layout tree should not changed"; | 71 << "Layout tree should not changed"; |
| 76 return paintedByBlock ? toLayoutBlock(layoutObject) | 72 return paintedByBlock ? toLayoutBlock(layoutObject) |
| 77 : layoutObject->containingBlock(); | 73 : layoutObject->containingBlock(); |
| 78 } | 74 } |
| 79 | 75 |
| 80 static void mapCaretRectToCaretPainter(LayoutItem caretLayoutItem, | 76 static LayoutRect mapCaretRectToCaretPainter( |
| 81 LayoutBlockItem caretPainterItem, | 77 LayoutItem caretLayoutItem, |
| 82 LayoutRect& caretRect) { | 78 LayoutBlockItem caretPainterItem, |
| 79 const LayoutRect& passedCaretRect) { | |
| 83 // FIXME: This shouldn't be called on un-rooted subtrees. | 80 // FIXME: This shouldn't be called on un-rooted subtrees. |
| 84 // FIXME: This should probably just use mapLocalToAncestor. | 81 // FIXME: This should probably just use mapLocalToAncestor. |
| 85 // Compute an offset between the caretLayoutItem and the caretPainterItem. | 82 // Compute an offset between the caretLayoutItem and the caretPainterItem. |
| 86 | 83 |
| 87 DCHECK(caretLayoutItem.isDescendantOf(caretPainterItem)); | 84 DCHECK(caretLayoutItem.isDescendantOf(caretPainterItem)); |
| 88 | 85 |
| 89 bool unrooted = false; | 86 LayoutRect caretRect = passedCaretRect; |
| 90 while (caretLayoutItem != caretPainterItem) { | 87 while (caretLayoutItem != caretPainterItem) { |
| 91 LayoutItem containerItem = caretLayoutItem.container(); | 88 LayoutItem containerItem = caretLayoutItem.container(); |
| 92 if (containerItem.isNull()) { | 89 if (containerItem.isNull()) |
| 93 unrooted = true; | 90 return LayoutRect(); |
| 94 break; | |
| 95 } | |
| 96 caretRect.move(caretLayoutItem.offsetFromContainer(containerItem)); | 91 caretRect.move(caretLayoutItem.offsetFromContainer(containerItem)); |
| 97 caretLayoutItem = containerItem; | 92 caretLayoutItem = containerItem; |
| 98 } | 93 } |
| 99 | 94 return caretRect; |
| 100 if (unrooted) | |
| 101 caretRect = LayoutRect(); | |
| 102 } | 95 } |
| 103 | 96 |
| 104 void CaretBase::updateCaretRect(const PositionWithAffinity& caretPosition) { | 97 LayoutRect CaretBase::computeCaretRect( |
| 105 m_caretLocalRect = LayoutRect(); | 98 const PositionWithAffinity& caretPosition) { |
| 106 | |
| 107 if (caretPosition.isNull()) | 99 if (caretPosition.isNull()) |
| 108 return; | 100 return LayoutRect(); |
| 109 | 101 |
| 110 DCHECK(caretPosition.anchorNode()->layoutObject()); | 102 DCHECK(caretPosition.anchorNode()->layoutObject()); |
| 111 | 103 |
| 112 // First compute a rect local to the layoutObject at the selection start. | 104 // First compute a rect local to the layoutObject at the selection start. |
| 113 LayoutObject* layoutObject; | 105 LayoutObject* layoutObject; |
| 114 m_caretLocalRect = localCaretRectOfPosition(caretPosition, layoutObject); | 106 LayoutRect caretLocalRect = |
|
yoichio
2017/01/12 09:22:22
nit: const LayoutRect
| |
| 107 localCaretRectOfPosition(caretPosition, layoutObject); | |
| 115 | 108 |
| 116 // Get the layoutObject that will be responsible for painting the caret | 109 // Get the layoutObject that will be responsible for painting the caret |
| 117 // (which is either the layoutObject we just found, or one of its containers). | 110 // (which is either the layoutObject we just found, or one of its containers). |
| 118 LayoutBlockItem caretPainterItem = | 111 LayoutBlockItem caretPainterItem = |
| 119 LayoutBlockItem(caretLayoutObject(caretPosition.anchorNode())); | 112 LayoutBlockItem(caretLayoutObject(caretPosition.anchorNode())); |
| 120 | 113 |
| 121 mapCaretRectToCaretPainter(LayoutItem(layoutObject), caretPainterItem, | 114 return mapCaretRectToCaretPainter(LayoutItem(layoutObject), caretPainterItem, |
| 122 m_caretLocalRect); | 115 caretLocalRect); |
| 123 } | 116 } |
| 124 | 117 |
| 125 void CaretBase::updateCaretRect(const VisiblePosition& caretPosition) { | 118 LayoutRect CaretBase::computeCaretRect(const VisiblePosition& caretPosition) { |
| 126 updateCaretRect(caretPosition.toPositionWithAffinity()); | 119 return computeCaretRect(caretPosition.toPositionWithAffinity()); |
| 127 } | 120 } |
| 128 | 121 |
| 129 IntRect CaretBase::absoluteBoundsForLocalRect(Node* node, | 122 IntRect CaretBase::absoluteBoundsForLocalRect(Node* node, |
| 130 const LayoutRect& rect) const { | 123 const LayoutRect& rect) const { |
| 131 LayoutBlock* caretPainter = caretLayoutObject(node); | 124 LayoutBlock* caretPainter = caretLayoutObject(node); |
| 132 if (!caretPainter) | 125 if (!caretPainter) |
| 133 return IntRect(); | 126 return IntRect(); |
| 134 | 127 |
| 135 LayoutRect localRect(rect); | 128 LayoutRect localRect(rect); |
| 136 caretPainter->flipForWritingMode(localRect); | 129 caretPainter->flipForWritingMode(localRect); |
| (...skipping 24 matching lines...) Expand all Loading... | |
| 161 | 154 |
| 162 bool CaretBase::shouldRepaintCaret(Node& node) const { | 155 bool CaretBase::shouldRepaintCaret(Node& node) const { |
| 163 // If PositionAnchorType::BeforeAnchor or PositionAnchorType::AfterAnchor, | 156 // If PositionAnchorType::BeforeAnchor or PositionAnchorType::AfterAnchor, |
| 164 // carets need to be repainted not only when the node is contentEditable but | 157 // carets need to be repainted not only when the node is contentEditable but |
| 165 // also when its parentNode() is contentEditable. | 158 // also when its parentNode() is contentEditable. |
| 166 node.document().updateStyleAndLayoutTree(); | 159 node.document().updateStyleAndLayoutTree(); |
| 167 return hasEditableStyle(node) || | 160 return hasEditableStyle(node) || |
| 168 (node.parentNode() && hasEditableStyle(*node.parentNode())); | 161 (node.parentNode() && hasEditableStyle(*node.parentNode())); |
| 169 } | 162 } |
| 170 | 163 |
| 171 void CaretBase::invalidateCaretRect(Node* node) { | 164 void CaretBase::invalidateCaretRect(Node* node, |
| 165 const LayoutRect& caretLocalRect) { | |
| 172 node->document().updateStyleAndLayoutTree(); | 166 node->document().updateStyleAndLayoutTree(); |
| 173 if (hasEditableStyle(*node)) | 167 if (hasEditableStyle(*node)) |
| 174 invalidateLocalCaretRect(node, localCaretRectWithoutUpdate()); | 168 invalidateLocalCaretRect(node, caretLocalRect); |
| 175 } | 169 } |
| 176 | 170 |
| 177 void CaretBase::paintCaret(Node* node, | 171 void CaretBase::paintCaret(Node* node, |
| 178 GraphicsContext& context, | 172 GraphicsContext& context, |
| 173 const DisplayItemClient& client, | |
| 174 const LayoutRect& caretLocalRect, | |
| 179 const LayoutPoint& paintOffset, | 175 const LayoutPoint& paintOffset, |
| 180 DisplayItem::Type displayItemType) const { | 176 DisplayItem::Type displayItemType) { |
| 181 if (DrawingRecorder::useCachedDrawingIfPossible(context, *this, | 177 if (DrawingRecorder::useCachedDrawingIfPossible(context, client, |
| 182 displayItemType)) | 178 displayItemType)) |
| 183 return; | 179 return; |
| 184 | 180 |
| 185 LayoutRect drawingRect = localCaretRectWithoutUpdate(); | 181 LayoutRect drawingRect = caretLocalRect; |
| 186 if (LayoutBlock* layoutObject = caretLayoutObject(node)) | 182 if (LayoutBlock* layoutObject = caretLayoutObject(node)) |
| 187 layoutObject->flipForWritingMode(drawingRect); | 183 layoutObject->flipForWritingMode(drawingRect); |
| 188 drawingRect.moveBy(paintOffset); | 184 drawingRect.moveBy(paintOffset); |
| 189 | 185 |
| 190 const Color caretColor = | 186 const Color caretColor = |
| 191 node->layoutObject()->resolveColor(CSSPropertyCaretColor); | 187 node->layoutObject()->resolveColor(CSSPropertyCaretColor); |
| 192 IntRect paintRect = pixelSnappedIntRect(drawingRect); | 188 IntRect paintRect = pixelSnappedIntRect(drawingRect); |
| 193 DrawingRecorder drawingRecorder(context, *this, DisplayItem::kCaret, | 189 DrawingRecorder drawingRecorder(context, client, DisplayItem::kCaret, |
| 194 paintRect); | 190 paintRect); |
| 195 context.fillRect(paintRect, caretColor); | 191 context.fillRect(paintRect, caretColor); |
| 196 } | 192 } |
| 197 | 193 |
| 198 String CaretBase::debugName() const { | 194 String CaretBase::debugName() const { |
| 199 return "Caret"; | 195 return "Caret"; |
| 200 } | 196 } |
| 201 | 197 |
| 202 LayoutRect CaretBase::visualRect() const { | 198 LayoutRect CaretBase::visualRect() const { |
| 203 return m_visualRect; | 199 return m_visualRect; |
| 204 } | 200 } |
| 205 | 201 |
| 206 } // namespace blink | 202 } // namespace blink |
| OLD | NEW |