blob: 67dc6e0a419aa0389484f60d67a66980520bbe25 [file] [log] [blame]
/*
* (C) 1999-2003 Lars Knoll (knoll@kde.org)
* Copyright (C) 2004, 2005, 2006, 2007, 2008 Apple Inc. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "config.h"
#include "CSSStyleDeclaration.h"
#include "CSSMutableStyleDeclaration.h"
#include "CSSMutableValue.h"
#include "CSSParser.h"
#include "CSSProperty.h"
#include "CSSPropertyNames.h"
#include "CSSRule.h"
#include "Node.h"
#include "SVGElement.h"
#include <wtf/ASCIICType.h>
using namespace WTF;
namespace WebCore {
CSSStyleDeclaration::CSSStyleDeclaration(CSSRule* parent)
: StyleBase(parent)
{
}
PassRefPtr<CSSValue> CSSStyleDeclaration::getPropertyCSSValue(const String& propertyName)
{
int propID = cssPropertyID(propertyName);
if (!propID)
return 0;
// Short-cut, not involving any change to the refcount.
if (!isMutableStyleDeclaration())
return getPropertyCSSValue(propID);
// Slow path.
RefPtr<CSSValue> value = getPropertyCSSValue(propID);
if (!value || !value->isMutableValue())
return value.release();
Node* node = static_cast<CSSMutableStyleDeclaration*>(this)->node();
if (!node || !node->isStyledElement())
return value.release();
Node* associatedNode = static_cast<CSSMutableValue*>(value.get())->node();
if (associatedNode) {
ASSERT(associatedNode == node);
return value.release();
}
static_cast<CSSMutableValue*>(value.get())->setNode(node);
return value.release();
}
String CSSStyleDeclaration::getPropertyValue(const String &propertyName)
{
int propID = cssPropertyID(propertyName);
if (!propID)
return String();
return getPropertyValue(propID);
}
String CSSStyleDeclaration::getPropertyPriority(const String& propertyName)
{
int propID = cssPropertyID(propertyName);
if (!propID)
return String();
return getPropertyPriority(propID) ? "important" : "";
}
String CSSStyleDeclaration::getPropertyShorthand(const String& propertyName)
{
int propID = cssPropertyID(propertyName);
if (!propID)
return String();
int shorthandID = getPropertyShorthand(propID);
if (!shorthandID)
return String();
return getPropertyName(static_cast<CSSPropertyID>(shorthandID));
}
bool CSSStyleDeclaration::isPropertyImplicit(const String& propertyName)
{
int propID = cssPropertyID(propertyName);
if (!propID)
return false;
return isPropertyImplicit(propID);
}
void CSSStyleDeclaration::setProperty(const String& propertyName, const String& value, ExceptionCode& ec)
{
size_t important = value.find("!important", 0, false);
int propertyID = cssPropertyID(propertyName);
if (!propertyID)
return;
if (important == notFound)
setProperty(propertyID, value, false, ec);
else
setProperty(propertyID, value.left(important - 1), true, ec);
}
void CSSStyleDeclaration::setProperty(const String& propertyName, const String& value, const String& priority, ExceptionCode& ec)
{
int propID = cssPropertyID(propertyName);
if (!propID) {
// FIXME: Should we raise an exception here?
return;
}
bool important = priority.find("important", 0, false) != notFound;
setProperty(propID, value, important, ec);
}
String CSSStyleDeclaration::removeProperty(const String& propertyName, ExceptionCode& ec)
{
int propID = cssPropertyID(propertyName);
if (!propID)
return String();
return removeProperty(propID, ec);
}
bool CSSStyleDeclaration::isPropertyName(const String& propertyName)
{
return cssPropertyID(propertyName);
}
CSSRule* CSSStyleDeclaration::parentRule() const
{
return (parent() && parent()->isRule()) ? static_cast<CSSRule*>(parent()) : 0;
}
bool CSSStyleDeclaration::cssPropertyMatches(const CSSProperty* property) const
{
RefPtr<CSSValue> value = getPropertyCSSValue(property->id());
return value && value->cssText() == property->value()->cssText();
}
void CSSStyleDeclaration::diff(CSSMutableStyleDeclaration* style) const
{
if (!style)
return;
Vector<int> propertiesToRemove;
{
CSSMutableStyleDeclaration::const_iterator end = style->end();
for (CSSMutableStyleDeclaration::const_iterator it = style->begin(); it != end; ++it) {
const CSSProperty& property = *it;
if (cssPropertyMatches(&property))
propertiesToRemove.append(property.id());
}
}
// FIXME: This should use mass removal.
for (unsigned i = 0; i < propertiesToRemove.size(); i++)
style->removeProperty(propertiesToRemove[i]);
}
PassRefPtr<CSSMutableStyleDeclaration> CSSStyleDeclaration::copyPropertiesInSet(const int* set, unsigned length) const
{
Vector<CSSProperty> list;
list.reserveInitialCapacity(length);
for (unsigned i = 0; i < length; i++) {
RefPtr<CSSValue> value = getPropertyCSSValue(set[i]);
if (value)
list.append(CSSProperty(set[i], value.release(), false));
}
return CSSMutableStyleDeclaration::create(list);
}
} // namespace WebCore