• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2008 Apple Inc. All Rights Reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24  */
25 
26 #include "config.h"
27 #include "CSSVariablesDeclaration.h"
28 
29 #include "CSSParser.h"
30 #include "CSSRule.h"
31 #include "CSSValueList.h"
32 #include "Document.h"
33 #include "ExceptionCode.h"
34 
35 namespace WebCore {
36 
CSSVariablesDeclaration(StyleBase * parent,const Vector<String> & names,const Vector<RefPtr<CSSValue>> & values)37 CSSVariablesDeclaration::CSSVariablesDeclaration(StyleBase* parent, const Vector<String>& names, const Vector<RefPtr<CSSValue> >& values)
38     : StyleBase(parent)
39 {
40     m_variableNames = names;
41     ASSERT(names.size() == values.size());
42     unsigned s = names.size();
43     for (unsigned i = 0; i < s; ++i)
44         addParsedVariable(names[i], values[i], false);
45 }
46 
~CSSVariablesDeclaration()47 CSSVariablesDeclaration::~CSSVariablesDeclaration()
48 {
49 }
50 
getVariableValue(const String & variableName)51 String CSSVariablesDeclaration::getVariableValue(const String& variableName)
52 {
53     CSSValue* val = m_variablesMap.get(variableName).get();
54     if (val)
55         return val->cssText();
56     return "";
57 }
58 
removeVariable(const String & variableName,ExceptionCode &)59 String CSSVariablesDeclaration::removeVariable(const String& variableName, ExceptionCode&)
60 {
61     // FIXME: The spec has this method taking an exception code but no exceptions are
62     // specified as being thrown.
63     RefPtr<CSSValue> val = m_variablesMap.take(variableName);
64     String result = val ? val->cssText() : "";
65     if (val) {
66         int s = m_variableNames.size();
67         for (int i = 0; i < s; ++i) {
68             if (m_variableNames[i] == variableName) {
69                 m_variableNames.remove(i);
70                 i--;
71                 s--;
72             }
73         }
74 
75         setNeedsStyleRecalc();
76     }
77 
78     // FIXME: Communicate this change so that the document will update.
79     return result;
80 }
81 
setVariable(const String & variableName,const String & variableValue,ExceptionCode & excCode)82 void CSSVariablesDeclaration::setVariable(const String& variableName, const String& variableValue, ExceptionCode& excCode)
83 {
84     // Try to parse the variable value into a Value*.  If it fails we throw an exception.
85     CSSParser parser(useStrictParsing());
86     if (!parser.parseVariable(this, variableName, variableValue)) // If the parse succeeds, it will call addParsedVariable (our internal method for doing the add) with the parsed Value*.
87         excCode = SYNTAX_ERR;
88     else
89         setNeedsStyleRecalc();
90 }
91 
addParsedVariable(const String & variableName,PassRefPtr<CSSValue> variableValue,bool updateNamesList)92 void CSSVariablesDeclaration::addParsedVariable(const String& variableName, PassRefPtr<CSSValue> variableValue, bool updateNamesList)
93 {
94 // FIXME: Disabling declarations as variable values for now since they no longer have a common base class with CSSValues.
95 #if 0
96     variableValue->setParent(this); // Needed to connect variables that are CSSMutableStyleDeclarations, since the parent couldn't be set until now.
97 #endif
98 
99     // Don't leak duplicates.  For multiple variables with the same name, the last one
100     // declared will win.
101     CSSValue* current = m_variablesMap.take(variableName).get();
102     if (!current && updateNamesList)
103         m_variableNames.append(variableName);
104     m_variablesMap.set(variableName, variableValue);
105 
106     // FIXME: Communicate this change so the document will update.
107 }
108 
getParsedVariable(const String & variableName)109 CSSValueList* CSSVariablesDeclaration::getParsedVariable(const String& variableName)
110 {
111     CSSValue* result = m_variablesMap.get(variableName).get();
112     if (result->isValueList())
113         return static_cast<CSSValueList*>(result);
114     return 0;
115 }
116 
getParsedVariableDeclarationBlock(const String &)117 CSSMutableStyleDeclaration* CSSVariablesDeclaration::getParsedVariableDeclarationBlock(const String&)
118 {
119 // FIXME: Disabling declarations as variable values for now since they no longer have a common base class with CSSValues.
120 #if 0
121     StyleBase* result = m_variablesMap.get(variableName).get();
122 
123     if (result->isMutableStyleDeclaration())
124         return static_cast<CSSMutableStyleDeclaration*>(result);
125 #endif
126     return 0;
127 }
128 
length() const129 unsigned CSSVariablesDeclaration::length() const
130 {
131     return m_variableNames.size();
132 }
133 
item(unsigned index)134 String CSSVariablesDeclaration::item(unsigned index)
135 {
136     if (index >= m_variableNames.size())
137         return "";
138     return m_variableNames[index];
139 }
140 
parentRule()141 CSSRule* CSSVariablesDeclaration::parentRule()
142 {
143     return (parent() && parent()->isRule()) ? static_cast<CSSRule*>(parent()) : 0;
144 }
145 
cssText() const146 String CSSVariablesDeclaration::cssText() const
147 {
148     String result = "{ ";
149     unsigned s = m_variableNames.size();
150     for (unsigned i = 0; i < s; ++i) {
151         result += m_variableNames[i] + ": ";
152         result += m_variablesMap.get(m_variableNames[i])->cssText();
153         if (i < s - 1)
154             result += "; ";
155     }
156     result += " }";
157     return result;
158 }
159 
setCssText(const String &)160 void CSSVariablesDeclaration::setCssText(const String&)
161 {
162     // FIXME: It's not clear if this is actually settable.
163 }
164 
setNeedsStyleRecalc()165 void CSSVariablesDeclaration::setNeedsStyleRecalc()
166 {
167     // FIXME: Make this much better (it has the same problem CSSMutableStyleDeclaration does).
168     StyleBase* root = this;
169     while (StyleBase* parent = root->parent())
170         root = parent;
171     if (root->isCSSStyleSheet())
172         static_cast<CSSStyleSheet*>(root)->doc()->updateStyleSelector();
173 }
174 
175 }
176