7d9c8505e42e5e6e28d6e5eaee616b1a75b941c2
[WebKit-https.git] / Source / WebCore / rendering / RenderTextControlMultiLine.cpp
1 /**
2  * Copyright (C) 2006, 2007 Apple Inc. All rights reserved.
3  *           (C) 2008 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.com/) 
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Library General Public
7  * License as published by the Free Software Foundation; either
8  * version 2 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Library General Public License for more details.
14  *
15  * You should have received a copy of the GNU Library General Public License
16  * along with this library; see the file COPYING.LIB.  If not, write to
17  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
18  * Boston, MA 02110-1301, USA.
19  *
20  */
21
22 #include "config.h"
23 #include "RenderTextControlMultiLine.h"
24
25 #include "Event.h"
26 #include "EventNames.h"
27 #include "Frame.h"
28 #include "HTMLNames.h"
29 #include "HTMLTextAreaElement.h"
30 #include "HitTestResult.h"
31
32 namespace WebCore {
33
34 RenderTextControlMultiLine::RenderTextControlMultiLine(Node* node, bool placeholderVisible)
35     : RenderTextControl(node, placeholderVisible)
36 {
37 }
38
39 RenderTextControlMultiLine::~RenderTextControlMultiLine()
40 {
41     if (node() && node()->inDocument())
42         static_cast<HTMLTextAreaElement*>(node())->rendererWillBeDestroyed();
43 }
44
45 void RenderTextControlMultiLine::subtreeHasChanged()
46 {
47     RenderTextControl::subtreeHasChanged();
48     HTMLTextAreaElement* textArea = static_cast<HTMLTextAreaElement*>(node());
49     textArea->setChangedSinceLastFormControlChangeEvent(true);
50     textArea->setFormControlValueMatchesRenderer(false);
51     textArea->setNeedsValidityCheck();
52
53     if (!node()->focused())
54         return;
55
56     if (Frame* frame = this->frame())
57         frame->editor()->textDidChangeInTextArea(textArea);
58 }
59
60 bool RenderTextControlMultiLine::nodeAtPoint(const HitTestRequest& request, HitTestResult& result, const IntPoint& pointInContainer, int tx, int ty, HitTestAction hitTestAction)
61 {
62     if (!RenderTextControl::nodeAtPoint(request, result, pointInContainer, tx, ty, hitTestAction))
63         return false;
64
65     if (result.innerNode() == node() || result.innerNode() == innerTextElement())
66         hitInnerTextElement(result, pointInContainer.x(), pointInContainer.y(), tx, ty);
67
68     return true;
69 }
70
71 void RenderTextControlMultiLine::forwardEvent(Event* event)
72 {
73     RenderTextControl::forwardEvent(event);
74 }
75
76 float RenderTextControlMultiLine::getAvgCharWidth(AtomicString family)
77 {
78     // Since Lucida Grande is the default font, we want this to match the width
79     // of Courier New, the default font for textareas in IE, Firefox and Safari Win.
80     // 1229 is the avgCharWidth value in the OS/2 table for Courier New.
81     if (family == AtomicString("Lucida Grande"))
82         return scaleEmToUnits(1229);
83
84     return RenderTextControl::getAvgCharWidth(family);
85 }
86
87 int RenderTextControlMultiLine::preferredContentWidth(float charWidth) const
88 {
89     int factor = static_cast<HTMLTextAreaElement*>(node())->cols();
90     return static_cast<int>(ceilf(charWidth * factor)) + scrollbarThickness();
91 }
92
93 void RenderTextControlMultiLine::adjustControlHeightBasedOnLineHeight(int lineHeight)
94 {
95     setHeight(height() + lineHeight * static_cast<HTMLTextAreaElement*>(node())->rows());
96 }
97
98 int RenderTextControlMultiLine::baselinePosition(FontBaseline baselineType, bool firstLine, LineDirectionMode direction, LinePositionMode linePositionMode) const
99 {
100     return RenderBox::baselinePosition(baselineType, firstLine, direction, linePositionMode);
101 }
102
103 void RenderTextControlMultiLine::updateFromElement()
104 {
105     createSubtreeIfNeeded(0);
106     RenderTextControl::updateFromElement();
107
108     setInnerTextValue(static_cast<HTMLTextAreaElement*>(node())->value());
109 }
110
111 void RenderTextControlMultiLine::cacheSelection(int start, int end)
112 {
113     static_cast<HTMLTextAreaElement*>(node())->cacheSelection(start, end);
114 }
115
116 PassRefPtr<RenderStyle> RenderTextControlMultiLine::createInnerTextStyle(const RenderStyle* startStyle) const
117 {
118     RefPtr<RenderStyle> textBlockStyle = RenderStyle::create();
119     textBlockStyle->inheritFrom(startStyle);
120     adjustInnerTextStyle(startStyle, textBlockStyle.get());
121     textBlockStyle->setDisplay(BLOCK);
122
123     return textBlockStyle.release();
124 }
125
126 RenderStyle* RenderTextControlMultiLine::textBaseStyle() const
127 {
128     return style();
129 }
130     
131 int RenderTextControlMultiLine::textBlockInsetLeft() const
132 {
133     int inset = borderLeft() + paddingLeft();
134     if (HTMLElement* innerText = innerTextElement()) {
135         if (RenderBox* innerTextRenderer = innerText->renderBox())
136             inset += innerTextRenderer->paddingLeft();
137     }
138     return inset;
139 }
140
141 int RenderTextControlMultiLine::textBlockInsetRight() const
142 {
143     int inset = borderRight() + paddingRight();
144     if (HTMLElement* innerText = innerTextElement()) {
145         if (RenderBox* innerTextRenderer = innerText->renderBox())
146             inset += innerTextRenderer->paddingRight();
147     }
148     return inset;
149 }
150
151 int RenderTextControlMultiLine::textBlockInsetTop() const
152 {
153     int inset = borderTop() + paddingTop();
154     if (HTMLElement* innerText = innerTextElement()) {
155         if (RenderBox* innerTextRenderer = innerText->renderBox())
156             inset += innerTextRenderer->paddingTop();
157     }
158     return inset;
159 }
160     
161 }