• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 1997 Martin Jones (mjones@kde.org)
3  *           (C) 1997 Torben Weis (weis@kde.org)
4  *           (C) 1998 Waldo Bastian (bastian@kde.org)
5  *           (C) 1999 Lars Knoll (knoll@kde.org)
6  *           (C) 1999 Antti Koivisto (koivisto@kde.org)
7  * Copyright (C) 2003, 2004, 2005, 2006, 2007 Apple Inc. All rights reserved.
8  *
9  * This library is free software; you can redistribute it and/or
10  * modify it under the terms of the GNU Library General Public
11  * License as published by the Free Software Foundation; either
12  * version 2 of the License, or (at your option) any later version.
13  *
14  * This library is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
17  * Library General Public License for more details.
18  *
19  * You should have received a copy of the GNU Library General Public License
20  * along with this library; see the file COPYING.LIB.  If not, write to
21  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
22  * Boston, MA 02110-1301, USA.
23  */
24 
25 #include "config.h"
26 #include "HTMLTableRowElement.h"
27 
28 #include "ExceptionCode.h"
29 #include "HTMLCollection.h"
30 #include "HTMLNames.h"
31 #include "HTMLTableCellElement.h"
32 #include "HTMLTableElement.h"
33 #include "HTMLTableSectionElement.h"
34 #include "NodeList.h"
35 #include "Text.h"
36 
37 namespace WebCore {
38 
39 using namespace HTMLNames;
40 
HTMLTableRowElement(const QualifiedName & tagName,Document * doc)41 HTMLTableRowElement::HTMLTableRowElement(const QualifiedName& tagName, Document* doc)
42     : HTMLTablePartElement(tagName, doc)
43 {
44     ASSERT(hasTagName(trTag));
45 }
46 
checkDTD(const Node * newChild)47 bool HTMLTableRowElement::checkDTD(const Node* newChild)
48 {
49     if (newChild->isTextNode())
50         return static_cast<const Text*>(newChild)->containsOnlyWhitespace();
51     return newChild->hasTagName(tdTag) || newChild->hasTagName(thTag) ||
52            newChild->hasTagName(formTag) || newChild->hasTagName(scriptTag);
53 }
54 
addChild(PassRefPtr<Node> child)55 ContainerNode* HTMLTableRowElement::addChild(PassRefPtr<Node> child)
56 {
57     if (child->hasTagName(formTag)) {
58         // First add the child.
59         HTMLTablePartElement::addChild(child);
60 
61         // Now simply return ourselves as the container to insert into.
62         // This has the effect of demoting the form to a leaf and moving it safely out of the way.
63         return this;
64     }
65 
66     return HTMLTablePartElement::addChild(child);
67 }
68 
rowIndex() const69 int HTMLTableRowElement::rowIndex() const
70 {
71     Node *table = parentNode();
72     if (!table)
73         return -1;
74     table = table->parentNode();
75     if (!table || !table->hasTagName(tableTag))
76         return -1;
77 
78     // To match Firefox, the row indices work like this:
79     //   Rows from the first <thead> are numbered before all <tbody> rows.
80     //   Rows from the first <tfoot> are numbered after all <tbody> rows.
81     //   Rows from other <thead> and <tfoot> elements don't get row indices at all.
82 
83     int rIndex = 0;
84 
85     if (HTMLTableSectionElement* head = static_cast<HTMLTableElement*>(table)->tHead()) {
86         for (Node *row = head->firstChild(); row; row = row->nextSibling()) {
87             if (row == this)
88                 return rIndex;
89             if (row->hasTagName(trTag))
90                 ++rIndex;
91         }
92     }
93 
94     for (Node *node = table->firstChild(); node; node = node->nextSibling()) {
95         if (node->hasTagName(tbodyTag)) {
96             HTMLTableSectionElement* section = static_cast<HTMLTableSectionElement*>(node);
97             for (Node* row = section->firstChild(); row; row = row->nextSibling()) {
98                 if (row == this)
99                     return rIndex;
100                 if (row->hasTagName(trTag))
101                     ++rIndex;
102             }
103         }
104     }
105 
106     if (HTMLTableSectionElement* foot = static_cast<HTMLTableElement*>(table)->tFoot()) {
107         for (Node *row = foot->firstChild(); row; row = row->nextSibling()) {
108             if (row == this)
109                 return rIndex;
110             if (row->hasTagName(trTag))
111                 ++rIndex;
112         }
113     }
114 
115     // We get here for rows that are in <thead> or <tfoot> sections other than the main header and footer.
116     return -1;
117 }
118 
sectionRowIndex() const119 int HTMLTableRowElement::sectionRowIndex() const
120 {
121     int rIndex = 0;
122     const Node *n = this;
123     do {
124         n = n->previousSibling();
125         if (n && n->hasTagName(trTag))
126             rIndex++;
127     }
128     while (n);
129 
130     return rIndex;
131 }
132 
insertCell(int index,ExceptionCode & ec)133 PassRefPtr<HTMLElement> HTMLTableRowElement::insertCell(int index, ExceptionCode& ec)
134 {
135     RefPtr<HTMLCollection> children = cells();
136     int numCells = children ? children->length() : 0;
137     if (index < -1 || index > numCells) {
138         ec = INDEX_SIZE_ERR;
139         return 0;
140     }
141 
142     RefPtr<HTMLTableCellElement> c = new HTMLTableCellElement(tdTag, document());
143     if (index < 0 || index >= numCells)
144         appendChild(c, ec);
145     else {
146         Node* n;
147         if (index < 1)
148             n = firstChild();
149         else
150             n = children->item(index);
151         insertBefore(c, n, ec);
152     }
153     return c.release();
154 }
155 
deleteCell(int index,ExceptionCode & ec)156 void HTMLTableRowElement::deleteCell(int index, ExceptionCode& ec)
157 {
158     RefPtr<HTMLCollection> children = cells();
159     int numCells = children ? children->length() : 0;
160     if (index == -1)
161         index = numCells-1;
162     if (index >= 0 && index < numCells) {
163         RefPtr<Node> cell = children->item(index);
164         HTMLElement::removeChild(cell.get(), ec);
165     } else
166         ec = INDEX_SIZE_ERR;
167 }
168 
cells()169 PassRefPtr<HTMLCollection> HTMLTableRowElement::cells()
170 {
171     return HTMLCollection::create(this, HTMLCollection::TRCells);
172 }
173 
setCells(HTMLCollection *,ExceptionCode & ec)174 void HTMLTableRowElement::setCells(HTMLCollection *, ExceptionCode& ec)
175 {
176     ec = NO_MODIFICATION_ALLOWED_ERR;
177 }
178 
align() const179 String HTMLTableRowElement::align() const
180 {
181     return getAttribute(alignAttr);
182 }
183 
setAlign(const String & value)184 void HTMLTableRowElement::setAlign(const String &value)
185 {
186     setAttribute(alignAttr, value);
187 }
188 
bgColor() const189 String HTMLTableRowElement::bgColor() const
190 {
191     return getAttribute(bgcolorAttr);
192 }
193 
setBgColor(const String & value)194 void HTMLTableRowElement::setBgColor(const String &value)
195 {
196     setAttribute(bgcolorAttr, value);
197 }
198 
ch() const199 String HTMLTableRowElement::ch() const
200 {
201     return getAttribute(charAttr);
202 }
203 
setCh(const String & value)204 void HTMLTableRowElement::setCh(const String &value)
205 {
206     setAttribute(charAttr, value);
207 }
208 
chOff() const209 String HTMLTableRowElement::chOff() const
210 {
211     return getAttribute(charoffAttr);
212 }
213 
setChOff(const String & value)214 void HTMLTableRowElement::setChOff(const String &value)
215 {
216     setAttribute(charoffAttr, value);
217 }
218 
vAlign() const219 String HTMLTableRowElement::vAlign() const
220 {
221     return getAttribute(valignAttr);
222 }
223 
setVAlign(const String & value)224 void HTMLTableRowElement::setVAlign(const String &value)
225 {
226     setAttribute(valignAttr, value);
227 }
228 
229 }
230