#define SASS_AST #include #include #include #include #include #include #include #ifdef __clang__ /* * There are some overloads used here that trigger the clang overload * hiding warning. Specifically: * * Type type() which hides string type() from Expression * * and * * Block* block() which hides virtual Block* block() from Statement * */ #pragma clang diagnostic push #pragma clang diagnostic ignored "-Woverloaded-virtual" #endif #ifndef SASS_CONSTANTS #include "constants.hpp" #endif #ifndef SASS_OPERATION #include "operation.hpp" #endif #ifndef SASS_TOKEN #include "token.hpp" #endif #ifndef SASS_ENVIRONMENT #include "environment.hpp" #endif #include "sass.h" #include "sass_values.h" #include "sass_functions.h" #include "units.hpp" #ifndef SASS_ERROR_HANDLING #include "error_handling.hpp" #endif #include "ast_def_macros.hpp" #include "inspect.hpp" #include #include #include #ifndef SASS_POSITION #include "position.hpp" #endif namespace Sass { using namespace std; ////////////////////////////////////////////////////////// // Abstract base class for all abstract syntax tree nodes. ////////////////////////////////////////////////////////// class Block; class Statement; class Expression; class Selector; class AST_Node { ADD_PROPERTY(string, path); ADD_PROPERTY(Position, position); public: AST_Node(string path, Position position) : path_(path), position_(position) { } virtual ~AST_Node() = 0; // virtual Block* block() { return 0; } ATTACH_OPERATIONS(); }; inline AST_Node::~AST_Node() { } ////////////////////////////////////////////////////////////////////// // Abstract base class for expressions. This side of the AST hierarchy // represents elements in value contexts, which exist primarily to be // evaluated and returned. ////////////////////////////////////////////////////////////////////// class Expression : public AST_Node { public: enum Concrete_Type { NONE, BOOLEAN, NUMBER, COLOR, STRING, LIST, MAP, NULL_VAL, NUM_TYPES }; private: // expressions in some contexts shouldn't be evaluated ADD_PROPERTY(bool, is_delayed); ADD_PROPERTY(bool, is_expanded); ADD_PROPERTY(bool, is_interpolant); ADD_PROPERTY(Concrete_Type, concrete_type); public: Expression(string path, Position position, bool d = false, bool e = false, bool i = false, Concrete_Type ct = NONE) : AST_Node(path, position), is_delayed_(d), is_expanded_(d), is_interpolant_(i), concrete_type_(ct) { } virtual operator bool() { return true; } virtual ~Expression() { }; virtual string type() { return ""; /* TODO: raise an error? */ } virtual bool is_invisible() { return false; } static string type_name() { return ""; } virtual bool is_false() { return false; } virtual bool operator==( Expression& rhs) const { return false; } virtual size_t hash() { return 0; } }; } ///////////////////////////////////////////////////////////////////////////// // Hash method specializations for unordered_map to work with Sass::Expression ///////////////////////////////////////////////////////////////////////////// namespace std { template<> struct hash { size_t operator()(Sass::Expression* s) const { return s->hash(); } }; template<> struct equal_to { bool operator()( Sass::Expression* lhs, Sass::Expression* rhs) const { return *lhs == *rhs; } }; } namespace Sass { using namespace std; ///////////////////////////////////////////////////////////////////////////// // Mixin class for AST nodes that should behave like vectors. Uses the // "Template Method" design pattern to allow subclasses to adjust their flags // when certain objects are pushed. ///////////////////////////////////////////////////////////////////////////// template class Vectorized { vector elements_; protected: size_t hash_; void reset_hash() { hash_ = 0; } virtual void adjust_after_pushing(T element) { } public: Vectorized(size_t s = 0) : elements_(vector()) { elements_.reserve(s); } virtual ~Vectorized() = 0; size_t length() const { return elements_.size(); } bool empty() const { return elements_.empty(); } T& operator[](size_t i) { return elements_[i]; } const T& operator[](size_t i) const { return elements_[i]; } Vectorized& operator<<(T element) { reset_hash(); elements_.push_back(element); adjust_after_pushing(element); return *this; } Vectorized& operator+=(Vectorized* v) { for (size_t i = 0, L = v->length(); i < L; ++i) *this << (*v)[i]; return *this; } vector& elements() { return elements_; } const vector& elements() const { return elements_; } vector& elements(vector& e) { elements_ = e; return elements_; } }; template inline Vectorized::~Vectorized() { } ///////////////////////////////////////////////////////////////////////////// // Mixin class for AST nodes that should behave like ahash table. Uses an // extra internally to maintain insertion order for interation. ///////////////////////////////////////////////////////////////////////////// class Hashed { private: unordered_map elements_; vector list_; protected: size_t hash_; Expression* duplicate_key_; void reset_hash() { hash_ = 0; } void reset_duplicate_key() { duplicate_key_ = 0; } virtual void adjust_after_pushing(std::pair p) { } public: Hashed(size_t s = 0) : elements_(unordered_map(s)), list_(vector()) { elements_.reserve(s); list_.reserve(s); reset_duplicate_key(); } virtual ~Hashed(); size_t length() const { return list_.size(); } bool empty() const { return list_.empty(); } bool has(Expression* k) const { return elements_.count(k) == 1; } Expression* at(Expression* k) const { return elements_.at(k); } bool has_duplicate_key() const { return duplicate_key_ != 0; } Expression* get_duplicate_key() const { return duplicate_key_; } Hashed& operator<<(pair p) { reset_hash(); if (!has(p.first)) list_.push_back(p.first); else if (!duplicate_key_) duplicate_key_ = p.first; elements_[p.first] = p.second; adjust_after_pushing(p); return *this; } Hashed& operator+=(Hashed* h) { if (length() == 0) { this->elements_ = h->elements_; this->list_ = h->list_; return *this; } for (auto key : h->keys()) { *this << make_pair(key, h->at(key)); } reset_duplicate_key(); return *this; } const unordered_map& pairs() const { return elements_; } const vector& keys() const { return list_; } }; inline Hashed::~Hashed() { } ///////////////////////////////////////////////////////////////////////// // Abstract base class for statements. This side of the AST hierarchy // represents elements in expansion contexts, which exist primarily to be // rewritten and macro-expanded. ///////////////////////////////////////////////////////////////////////// class Statement : public AST_Node { public: Statement(string path, Position position) : AST_Node(path, position) { } virtual ~Statement() = 0; // needed for rearranging nested rulesets during CSS emission virtual bool is_hoistable() { return false; } virtual Block* block() { return 0; } }; inline Statement::~Statement() { } //////////////////////// // Blocks of statements. //////////////////////// class Block : public Statement, public Vectorized { ADD_PROPERTY(bool, is_root); // needed for properly formatted CSS emission ADD_PROPERTY(bool, has_hoistable); ADD_PROPERTY(bool, has_non_hoistable); protected: void adjust_after_pushing(Statement* s) { if (s->is_hoistable()) has_hoistable_ = true; else has_non_hoistable_ = true; }; public: Block(string path, Position position, size_t s = 0, bool r = false) : Statement(path, position), Vectorized(s), is_root_(r), has_hoistable_(false), has_non_hoistable_(false) { } Block* block() { return this; } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////////////////////// // Abstract base class for statements that contain blocks of statements. //////////////////////////////////////////////////////////////////////// class Has_Block : public Statement { ADD_PROPERTY(Block*, block); public: Has_Block(string path, Position position, Block* b) : Statement(path, position), block_(b) { } virtual ~Has_Block() = 0; }; inline Has_Block::~Has_Block() { } ///////////////////////////////////////////////////////////////////////////// // Rulesets (i.e., sets of styles headed by a selector and containing a block // of style declarations. ///////////////////////////////////////////////////////////////////////////// class Selector; class Ruleset : public Has_Block { ADD_PROPERTY(Selector*, selector); public: Ruleset(string path, Position position, Selector* s, Block* b) : Has_Block(path, position, b), selector_(s) { } // nested rulesets need to be hoisted out of their enclosing blocks bool is_hoistable() { return true; } ATTACH_OPERATIONS(); }; ///////////////////////////////////////////////////////// // Nested declaration sets (i.e., namespaced properties). ///////////////////////////////////////////////////////// class String; class Propset : public Has_Block { ADD_PROPERTY(String*, property_fragment); public: Propset(string path, Position position, String* pf, Block* b = 0) : Has_Block(path, position, b), property_fragment_(pf) { } ATTACH_OPERATIONS(); }; ///////////////// // Media queries. ///////////////// class List; class Media_Block : public Has_Block { ADD_PROPERTY(List*, media_queries); ADD_PROPERTY(Selector*, selector); public: Media_Block(string path, Position position, List* mqs, Block* b) : Has_Block(path, position, b), media_queries_(mqs), selector_(0) { } bool is_hoistable() { return true; } ATTACH_OPERATIONS(); }; /////////////////// // Feature queries. /////////////////// class Feature_Block : public Has_Block { ADD_PROPERTY(Feature_Query*, feature_queries); ADD_PROPERTY(Selector*, selector); public: Feature_Block(string path, Position position, Feature_Query* fqs, Block* b) : Has_Block(path, position, b), feature_queries_(fqs), selector_(0) { } bool is_hoistable() { return true; } ATTACH_OPERATIONS(); }; /////////////////////////////////////////////////////////////////////// // At-rules -- arbitrary directives beginning with "@" that may have an // optional statement block. /////////////////////////////////////////////////////////////////////// class At_Rule : public Has_Block { ADD_PROPERTY(string, keyword); ADD_PROPERTY(Selector*, selector); ADD_PROPERTY(Expression*, value); public: At_Rule(string path, Position position, string kwd, Selector* sel = 0, Block* b = 0) : Has_Block(path, position, b), keyword_(kwd), selector_(sel), value_(0) // set value manually if needed { } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////////////////////// // Declarations -- style rules consisting of a property name and values. //////////////////////////////////////////////////////////////////////// class Declaration : public Statement { ADD_PROPERTY(String*, property); ADD_PROPERTY(Expression*, value); ADD_PROPERTY(bool, is_important); public: Declaration(string path, Position position, String* prop, Expression* val, bool i = false) : Statement(path, position), property_(prop), value_(val), is_important_(i) { } ATTACH_OPERATIONS(); }; ///////////////////////////////////// // Assignments -- variable and value. ///////////////////////////////////// class Variable; class Expression; class Assignment : public Statement { ADD_PROPERTY(string, variable); ADD_PROPERTY(Expression*, value); ADD_PROPERTY(bool, is_guarded); ADD_PROPERTY(bool, is_global); public: Assignment(string path, Position position, string var, Expression* val, bool guarded = false, bool global = false) : Statement(path, position), variable_(var), value_(val), is_guarded_(guarded), is_global_(global) { } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////////////////////////// // Import directives. CSS and Sass import lists can be intermingled, so it's // necessary to store a list of each in an Import node. //////////////////////////////////////////////////////////////////////////// class Import : public Statement { vector files_; vector urls_; public: Import(string path, Position position) : Statement(path, position), files_(vector()), urls_(vector()) { } vector& files() { return files_; } vector& urls() { return urls_; } ATTACH_OPERATIONS(); }; class Import_Stub : public Statement { ADD_PROPERTY(string, file_name); public: Import_Stub(string path, Position position, string f) : Statement(path, position), file_name_(f) { } ATTACH_OPERATIONS(); }; ////////////////////////////// // The Sass `@warn` directive. ////////////////////////////// class Warning : public Statement { ADD_PROPERTY(Expression*, message); public: Warning(string path, Position position, Expression* msg) : Statement(path, position), message_(msg) { } ATTACH_OPERATIONS(); }; /////////////////////////////// // The Sass `@error` directive. /////////////////////////////// class Error : public Statement { ADD_PROPERTY(Expression*, message); public: Error(string path, Position position, Expression* msg) : Statement(path, position), message_(msg) { } ATTACH_OPERATIONS(); }; /////////////////////////////// // The Sass `@debug` directive. /////////////////////////////// class Debug : public Statement { ADD_PROPERTY(Expression*, value); public: Debug(string path, Position position, Expression* val) : Statement(path, position), value_(val) { } ATTACH_OPERATIONS(); }; /////////////////////////////////////////// // CSS comments. These may be interpolated. /////////////////////////////////////////// class Comment : public Statement { ADD_PROPERTY(String*, text); public: Comment(string path, Position position, String* txt) : Statement(path, position), text_(txt) { } ATTACH_OPERATIONS(); }; //////////////////////////////////// // The Sass `@if` control directive. //////////////////////////////////// class If : public Statement { ADD_PROPERTY(Expression*, predicate); ADD_PROPERTY(Block*, consequent); ADD_PROPERTY(Block*, alternative); public: If(string path, Position position, Expression* pred, Block* con, Block* alt = 0) : Statement(path, position), predicate_(pred), consequent_(con), alternative_(alt) { } ATTACH_OPERATIONS(); }; ///////////////////////////////////// // The Sass `@for` control directive. ///////////////////////////////////// class For : public Has_Block { ADD_PROPERTY(string, variable); ADD_PROPERTY(Expression*, lower_bound); ADD_PROPERTY(Expression*, upper_bound); ADD_PROPERTY(bool, is_inclusive); public: For(string path, Position position, string var, Expression* lo, Expression* hi, Block* b, bool inc) : Has_Block(path, position, b), variable_(var), lower_bound_(lo), upper_bound_(hi), is_inclusive_(inc) { } ATTACH_OPERATIONS(); }; ////////////////////////////////////// // The Sass `@each` control directive. ////////////////////////////////////// class Each : public Has_Block { ADD_PROPERTY(vector, variables); ADD_PROPERTY(Expression*, list); public: Each(string path, Position position, vector vars, Expression* lst, Block* b) : Has_Block(path, position, b), variables_(vars), list_(lst) { } ATTACH_OPERATIONS(); }; /////////////////////////////////////// // The Sass `@while` control directive. /////////////////////////////////////// class While : public Has_Block { ADD_PROPERTY(Expression*, predicate); public: While(string path, Position position, Expression* pred, Block* b) : Has_Block(path, position, b), predicate_(pred) { } ATTACH_OPERATIONS(); }; ///////////////////////////////////////////////////////////// // The @return directive for use inside SassScript functions. ///////////////////////////////////////////////////////////// class Return : public Statement { ADD_PROPERTY(Expression*, value); public: Return(string path, Position position, Expression* val) : Statement(path, position), value_(val) { } ATTACH_OPERATIONS(); }; //////////////////////////////// // The Sass `@extend` directive. //////////////////////////////// class Extension : public Statement { ADD_PROPERTY(Selector*, selector); public: Extension(string path, Position position, Selector* s) : Statement(path, position), selector_(s) { } ATTACH_OPERATIONS(); }; ///////////////////////////////////////////////////////////////////////////// // Definitions for both mixins and functions. The two cases are distinguished // by a type tag. ///////////////////////////////////////////////////////////////////////////// struct Context; struct Backtrace; class Parameters; typedef Environment Env; typedef const char* Signature; typedef Expression* (*Native_Function)(Env&, Env&, Context&, Signature, const string&, Position, Backtrace*); typedef const char* Signature; class Definition : public Has_Block { public: enum Type { MIXIN, FUNCTION }; ADD_PROPERTY(string, name); ADD_PROPERTY(Parameters*, parameters); ADD_PROPERTY(Env*, environment); ADD_PROPERTY(Type, type); ADD_PROPERTY(Native_Function, native_function); ADD_PROPERTY(Sass_C_Function, c_function); ADD_PROPERTY(void*, cookie); ADD_PROPERTY(bool, is_overload_stub); ADD_PROPERTY(Signature, signature); public: Definition(string path, Position position, string n, Parameters* params, Block* b, Type t) : Has_Block(path, position, b), name_(n), parameters_(params), environment_(0), type_(t), native_function_(0), c_function_(0), cookie_(0), is_overload_stub_(false), signature_(0) { } Definition(string path, Position position, Signature sig, string n, Parameters* params, Native_Function func_ptr, bool overload_stub = false) : Has_Block(path, position, 0), name_(n), parameters_(params), environment_(0), type_(FUNCTION), native_function_(func_ptr), c_function_(0), cookie_(0), is_overload_stub_(overload_stub), signature_(sig) { } Definition(string path, Position position, Signature sig, string n, Parameters* params, Sass_C_Function func_ptr, void* cookie, bool whatever, bool whatever2) : Has_Block(path, position, 0), name_(n), parameters_(params), environment_(0), type_(FUNCTION), native_function_(0), c_function_(func_ptr), cookie_(cookie), is_overload_stub_(false), signature_(sig) { } ATTACH_OPERATIONS(); }; ////////////////////////////////////// // Mixin calls (i.e., `@include ...`). ////////////////////////////////////// class Arguments; class Mixin_Call : public Has_Block { ADD_PROPERTY(string, name); ADD_PROPERTY(Arguments*, arguments); public: Mixin_Call(string path, Position position, string n, Arguments* args, Block* b = 0) : Has_Block(path, position, b), name_(n), arguments_(args) { } ATTACH_OPERATIONS(); }; /////////////////////////////////////////////////// // The @content directive for mixin content blocks. /////////////////////////////////////////////////// class Content : public Statement { public: Content(string path, Position position) : Statement(path, position) { } ATTACH_OPERATIONS(); }; /////////////////////////////////////////////////////////////////////// // Lists of values, both comma- and space-separated (distinguished by a // type-tag.) Also used to represent variable-length argument lists. /////////////////////////////////////////////////////////////////////// class List : public Expression, public Vectorized { void adjust_after_pushing(Expression* e) { is_expanded(false); } public: enum Separator { SPACE, COMMA }; private: ADD_PROPERTY(Separator, separator); ADD_PROPERTY(bool, is_arglist); public: List(string path, Position position, size_t size = 0, Separator sep = SPACE, bool argl = false) : Expression(path, position), Vectorized(size), separator_(sep), is_arglist_(argl) { concrete_type(LIST); } string type() { return is_arglist_ ? "arglist" : "list"; } static string type_name() { return "list"; } bool is_invisible() { return !length(); } Expression* value_at_index(size_t i); virtual bool operator==(Expression& rhs) const { try { List& l = dynamic_cast(rhs); if (!(l && length() == l.length() && separator() == l.separator())) return false; for (size_t i = 0, L = l.length(); i < L; ++i) if (!(*(elements()[i]) == *(l[i]))) return false; return true; } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ > 0) return hash_; hash_ = std::hash()(separator() == COMMA ? "comma" : "space"); for (size_t i = 0, L = length(); i < L; ++i) hash_ ^= (elements()[i])->hash(); return hash_; } ATTACH_OPERATIONS(); }; /////////////////////////////////////////////////////////////////////// // Key value paris. /////////////////////////////////////////////////////////////////////// class Map : public Expression, public Hashed { void adjust_after_pushing(std::pair p) { is_expanded(false); } public: Map(string path, Position position, size_t size = 0) : Expression(path, position), Hashed(size) { concrete_type(MAP); } string type() { return "map"; } static string type_name() { return "map"; } bool is_invisible() { return !length(); } virtual bool operator==(Expression& rhs) const { try { Map& m = dynamic_cast(rhs); if (!(m && length() == m.length())) return false; for (auto key : keys()) if (!(*at(key) == *m.at(key))) return false; return true; } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ > 0) return hash_; for (auto key : keys()) hash_ ^= key->hash() ^ at(key)->hash(); return hash_; } ATTACH_OPERATIONS(); }; ////////////////////////////////////////////////////////////////////////// // Binary expressions. Represents logical, relational, and arithmetic // operations. Templatized to avoid large switch statements and repetitive // subclassing. ////////////////////////////////////////////////////////////////////////// class Binary_Expression : public Expression { public: enum Type { AND, OR, // logical connectives EQ, NEQ, GT, GTE, LT, LTE, // arithmetic relations ADD, SUB, MUL, DIV, MOD, // arithmetic functions NUM_OPS // so we know how big to make the op table }; private: ADD_PROPERTY(Type, type); ADD_PROPERTY(Expression*, left); ADD_PROPERTY(Expression*, right); public: Binary_Expression(string path, Position position, Type t, Expression* lhs, Expression* rhs) : Expression(path, position), type_(t), left_(lhs), right_(rhs) { } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////////////////////////// // Arithmetic negation (logical negation is just an ordinary function call). //////////////////////////////////////////////////////////////////////////// class Unary_Expression : public Expression { public: enum Type { PLUS, MINUS, NOT }; private: ADD_PROPERTY(Type, type); ADD_PROPERTY(Expression*, operand); public: Unary_Expression(string path, Position position, Type t, Expression* o) : Expression(path, position), type_(t), operand_(o) { } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////////// // Individual argument objects for mixin and function calls. //////////////////////////////////////////////////////////// class Argument : public Expression { ADD_PROPERTY(Expression*, value); ADD_PROPERTY(string, name); ADD_PROPERTY(bool, is_rest_argument); ADD_PROPERTY(bool, is_keyword_argument); size_t hash_; public: Argument(string p, Position pos, Expression* val, string n = "", bool rest = false, bool keyword = false) : Expression(p, pos), value_(val), name_(n), is_rest_argument_(rest), is_keyword_argument_(keyword), hash_(0) { if (!name_.empty() && is_rest_argument_) { error("variable-length argument may not be passed by name", path(), position()); } } virtual bool operator==(Expression& rhs) const { try { Argument& m = dynamic_cast(rhs); if (!(m && name() == m.name())) return false; return *value() == *value(); } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ > 0) return hash_; hash_ = std::hash()(name()) ^ value()->hash(); return hash_; } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////////////////////// // Argument lists -- in their own class to facilitate context-sensitive // error checking (e.g., ensuring that all ordinal arguments precede all // named arguments). //////////////////////////////////////////////////////////////////////// class Arguments : public Expression, public Vectorized { ADD_PROPERTY(bool, has_named_arguments); ADD_PROPERTY(bool, has_rest_argument); ADD_PROPERTY(bool, has_keyword_argument); protected: void adjust_after_pushing(Argument* a) { if (!a->name().empty()) { if (has_rest_argument_ || has_keyword_argument_) { error("named arguments must precede variable-length argument", a->path(), a->position()); } has_named_arguments_ = true; } else if (a->is_rest_argument()) { if (has_rest_argument_) { error("functions and mixins may only be called with one variable-length argument", a->path(), a->position()); } if (has_keyword_argument_) { error("only keyword arguments may follow variable arguments", a->path(), a->position()); } has_rest_argument_ = true; } else if (a->is_keyword_argument()) { if (has_keyword_argument_) { error("functions and mixins may only be called with one keyword argument", a->path(), a->position()); } has_keyword_argument_ = true; } else { if (has_rest_argument_) { error("ordinal arguments must precede variable-length arguments", a->path(), a->position()); } if (has_named_arguments_) { error("ordinal arguments must precede named arguments", a->path(), a->position()); } } } public: Arguments(string path, Position position) : Expression(path, position), Vectorized(), has_named_arguments_(false), has_rest_argument_(false), has_keyword_argument_(false) { } ATTACH_OPERATIONS(); }; ////////////////// // Function calls. ////////////////// class Function_Call : public Expression { ADD_PROPERTY(string, name); ADD_PROPERTY(Arguments*, arguments); ADD_PROPERTY(void*, cookie); size_t hash_; public: Function_Call(string path, Position position, string n, Arguments* args, void* cookie) : Expression(path, position), name_(n), arguments_(args), cookie_(cookie), hash_(0) { concrete_type(STRING); } Function_Call(string path, Position position, string n, Arguments* args) : Expression(path, position), name_(n), arguments_(args), cookie_(0), hash_(0) { concrete_type(STRING); } virtual bool operator==(Expression& rhs) const { try { Function_Call& m = dynamic_cast(rhs); if (!(m && name() == m.name())) return false; if (!(m && arguments()->length() == m.arguments()->length())) return false; for (size_t i =0, L = arguments()->length(); i < L; ++i) if (!((*arguments())[i] == (*m.arguments())[i])) return false; return true; } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ > 0) return hash_; hash_ = std::hash()(name()); for (auto argument : arguments()->elements()) hash_ ^= argument->hash(); return hash_; } ATTACH_OPERATIONS(); }; ///////////////////////// // Function call schemas. ///////////////////////// class Function_Call_Schema : public Expression { ADD_PROPERTY(String*, name); ADD_PROPERTY(Arguments*, arguments); public: Function_Call_Schema(string path, Position position, String* n, Arguments* args) : Expression(path, position), name_(n), arguments_(args) { concrete_type(STRING); } ATTACH_OPERATIONS(); }; /////////////////////// // Variable references. /////////////////////// class Variable : public Expression { ADD_PROPERTY(string, name); public: Variable(string path, Position position, string n) : Expression(path, position), name_(n) { } virtual bool operator==(Expression& rhs) const { try { Variable& e = dynamic_cast(rhs); return e && name() == e.name(); } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { return std::hash()(name()); } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////////////////////////// // Textual (i.e., unevaluated) numeric data. Variants are distinguished with // a type tag. //////////////////////////////////////////////////////////////////////////// class Textual : public Expression { public: enum Type { NUMBER, PERCENTAGE, DIMENSION, HEX }; private: ADD_PROPERTY(Type, type); ADD_PROPERTY(string, value); size_t hash_; public: Textual(string path, Position position, Type t, string val) : Expression(path, position, true), type_(t), value_(val), hash_(0) { } virtual bool operator==(Expression& rhs) const { try { Textual& e = dynamic_cast(rhs); return e && value() == e.value() && type() == e.type(); } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ == 0) hash_ = std::hash()(value_) ^ std::hash()(type_); return hash_; } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////// // Numbers, percentages, dimensions, and colors. //////////////////////////////////////////////// class Number : public Expression { ADD_PROPERTY(double, value); ADD_PROPERTY(bool, zero); vector numerator_units_; vector denominator_units_; size_t hash_; public: Number(string path, Position position, double val, string u = "", bool zero = true) : Expression(path, position), value_(val), zero_(zero), numerator_units_(vector()), denominator_units_(vector()), hash_(0) { if (!u.empty()) numerator_units_.push_back(u); concrete_type(NUMBER); } bool zero() { return zero_; } vector& numerator_units() { return numerator_units_; } vector& denominator_units() { return denominator_units_; } string type() { return "number"; } static string type_name() { return "number"; } string unit() const { stringstream u; for (size_t i = 0, S = numerator_units_.size(); i < S; ++i) { if (i) u << '*'; u << numerator_units_[i]; } if (!denominator_units_.empty()) u << '/'; for (size_t i = 0, S = denominator_units_.size(); i < S; ++i) { if (i) u << '*'; u << denominator_units_[i]; } return u.str(); } bool is_unitless() { return numerator_units_.empty() && denominator_units_.empty(); } void normalize(string to = "") { // (multiple passes because I'm too tired to think up something clever) // Find a unit to convert everything to, if one isn't provided. if (to.empty()) { for (size_t i = 0, S = numerator_units_.size(); i < S; ++i) { string u(numerator_units_[i]); if (string_to_unit(u) == INCOMMENSURABLE) { continue; } else { to = u; break; } } } if (to.empty()) { for (size_t i = 0, S = denominator_units_.size(); i < S; ++i) { string u(denominator_units_[i]); if (string_to_unit(u) == INCOMMENSURABLE) { continue; } else { to = u; break; } } } // Now loop through again and do all the conversions. for (size_t i = 0, S = numerator_units_.size(); i < S; ++i) { string from(numerator_units_[i]); if (string_to_unit(from) == INCOMMENSURABLE) continue; value_ *= conversion_factor(from, to); numerator_units_[i] = to; } for (size_t i = 0, S = denominator_units_.size(); i < S; ++i) { string from(denominator_units_[i]); if (string_to_unit(from) == INCOMMENSURABLE) continue; value_ /= conversion_factor(from, to); denominator_units_[i] = to; } // Now divide out identical units in the numerator and denominator. vector ncopy; ncopy.reserve(numerator_units_.size()); for (vector::iterator n = numerator_units_.begin(); n != numerator_units_.end(); ++n) { vector::iterator d = find(denominator_units_.begin(), denominator_units_.end(), *n); if (d != denominator_units_.end()) { denominator_units_.erase(d); } else { ncopy.push_back(*n); } } numerator_units_ = ncopy; // Sort the units to make them pretty and, well, normal. sort(numerator_units_.begin(), numerator_units_.end()); sort(denominator_units_.begin(), denominator_units_.end()); } // useful for making one number compatible with another string find_convertible_unit() const { for (size_t i = 0, S = numerator_units_.size(); i < S; ++i) { string u(numerator_units_[i]); if (string_to_unit(u) != INCOMMENSURABLE) return u; } for (size_t i = 0, S = denominator_units_.size(); i < S; ++i) { string u(denominator_units_[i]); if (string_to_unit(u) != INCOMMENSURABLE) return u; } return string(); } virtual bool operator==(Expression& rhs) const { try { Number& e(dynamic_cast(rhs)); if (!e) return false; e.normalize(find_convertible_unit()); return unit() == e.unit() && value() == e.value(); } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ == 0) hash_ = std::hash()(value_); return hash_; } ATTACH_OPERATIONS(); }; ////////// // Colors. ////////// class Color : public Expression { ADD_PROPERTY(double, r); ADD_PROPERTY(double, g); ADD_PROPERTY(double, b); ADD_PROPERTY(double, a); ADD_PROPERTY(bool, sixtuplet); ADD_PROPERTY(string, disp); size_t hash_; public: Color(string path, Position position, double r, double g, double b, double a = 1, bool sixtuplet = true, const string disp = "") : Expression(path, position), r_(r), g_(g), b_(b), a_(a), sixtuplet_(sixtuplet), disp_(disp), hash_(0) { concrete_type(COLOR); } string type() { return "color"; } static string type_name() { return "color"; } virtual bool operator==(Expression& rhs) const { try { Color& c = (dynamic_cast(rhs)); return c && r() == c.r() && g() == c.g() && b() == c.b() && a() == c.a(); } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ == 0) hash_ = std::hash()(r_) ^ std::hash()(g_) ^ std::hash()(b_) ^ std::hash()(a_); return hash_; } ATTACH_OPERATIONS(); }; //////////// // Booleans. //////////// class Boolean : public Expression { ADD_PROPERTY(bool, value); size_t hash_; public: Boolean(string path, Position position, bool val) : Expression(path, position), value_(val), hash_(0) { concrete_type(BOOLEAN); } virtual operator bool() { return value_; } string type() { return "bool"; } static string type_name() { return "bool"; } virtual bool is_false() { return !value_; } virtual bool operator==(Expression& rhs) const { try { Boolean& e = dynamic_cast(rhs); return e && value() == e.value(); } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ == 0) hash_ = std::hash()(value_); return hash_; } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////////////////////// // Abstract base class for Sass string values. Includes interpolated and // "flat" strings. //////////////////////////////////////////////////////////////////////// class String : public Expression { ADD_PROPERTY(bool, needs_unquoting); public: String(string path, Position position, bool unq = false, bool delayed = false) : Expression(path, position, delayed), needs_unquoting_(unq) { concrete_type(STRING); } static string type_name() { return "string"; } virtual ~String() = 0; ATTACH_OPERATIONS(); }; inline String::~String() { }; /////////////////////////////////////////////////////////////////////// // Interpolated strings. Meant to be reduced to flat strings during the // evaluation phase. /////////////////////////////////////////////////////////////////////// class String_Schema : public String, public Vectorized { ADD_PROPERTY(char, quote_mark); ADD_PROPERTY(bool, has_interpolants); size_t hash_; public: String_Schema(string path, Position position, size_t size = 0, bool unq = false, char qm = '\0', bool i = false) : String(path, position, unq), Vectorized(size), quote_mark_(qm), has_interpolants_(i), hash_(0) { } string type() { return "string"; } static string type_name() { return "string"; } virtual bool operator==(Expression& rhs) const { try { String_Schema& e = dynamic_cast(rhs); if (!(e && length() == e.length())) return false; for (size_t i = 0, L = length(); i < L; ++i) if (!((*this)[i] == e[i])) return false; return true; } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ > 0) return hash_; for (auto string : elements()) hash_ ^= string->hash(); return hash_; } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////// // Flat strings -- the lowest level of raw textual data. //////////////////////////////////////////////////////// class String_Constant : public String { ADD_PROPERTY(string, value); string unquoted_; size_t hash_; public: String_Constant(string path, Position position, string val, bool unq = false) : String(path, position, unq, true), value_(val), hash_(0) { unquoted_ = unquote(value_); } String_Constant(string path, Position position, const char* beg, bool unq = false) : String(path, position, unq, true), value_(string(beg)), hash_(0) { unquoted_ = unquote(value_); } String_Constant(string path, Position position, const char* beg, const char* end, bool unq = false) : String(path, position, unq, true), value_(string(beg, end-beg)), hash_(0) { unquoted_ = unquote(value_); } String_Constant(string path, Position position, const Token& tok, bool unq = false) : String(path, position, unq, true), value_(string(tok.begin, tok.end)), hash_(0) { unquoted_ = unquote(value_); } string type() { return "string"; } static string type_name() { return "string"; } virtual bool operator==(Expression& rhs) const { try { String_Constant& e = dynamic_cast(rhs); return e && unquoted_ == e.unquoted_; } catch (std::bad_cast&) { return false; } catch (...) { throw; } } virtual size_t hash() { if (hash_ == 0) hash_ = std::hash()(unquoted_); return hash_; } static char single_quote() { return '\''; } static char double_quote() { return '"'; } bool is_quoted() { return value_.length() && (value_[0] == '"' || value_[0] == '\''); } char quote_mark() { return is_quoted() ? value_[0] : '\0'; } ATTACH_OPERATIONS(); }; ///////////////// // Media queries. ///////////////// class Media_Query : public Expression, public Vectorized { ADD_PROPERTY(String*, media_type); ADD_PROPERTY(bool, is_negated); ADD_PROPERTY(bool, is_restricted); public: Media_Query(string path, Position position, String* t = 0, size_t s = 0, bool n = false, bool r = false) : Expression(path, position), Vectorized(s), media_type_(t), is_negated_(n), is_restricted_(r) { } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////// // Media expressions (for use inside media queries). //////////////////////////////////////////////////// class Media_Query_Expression : public Expression { ADD_PROPERTY(Expression*, feature); ADD_PROPERTY(Expression*, value); ADD_PROPERTY(bool, is_interpolated); public: Media_Query_Expression(string path, Position position, Expression* f, Expression* v, bool i = false) : Expression(path, position), feature_(f), value_(v), is_interpolated_(i) { } ATTACH_OPERATIONS(); }; /////////////////// // Feature queries. /////////////////// class Feature_Query : public Expression, public Vectorized { public: Feature_Query(string path, Position position, size_t s = 0) : Expression(path, position), Vectorized(s) { } ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////// // Feature expressions (for use inside feature queries). //////////////////////////////////////////////////////// class Feature_Query_Condition : public Expression, public Vectorized { public: enum Operand { NONE, AND, OR, NOT }; private: ADD_PROPERTY(String*, feature); ADD_PROPERTY(Expression*, value); ADD_PROPERTY(Operand, operand); ADD_PROPERTY(bool, is_root); public: Feature_Query_Condition(string path, Position position, size_t s = 0, String* f = 0, Expression* v = 0, Operand o = NONE, bool r = false) : Expression(path, position), Vectorized(s), feature_(f), value_(v), operand_(o), is_root_(r) { } ATTACH_OPERATIONS(); }; ////////////////// // The null value. ////////////////// class Null : public Expression { public: Null(string path, Position position) : Expression(path, position) { concrete_type(NULL_VAL); } string type() { return "null"; } static string type_name() { return "null"; } bool is_invisible() { return true; } operator bool() { return false; } bool is_false() { return true; } virtual bool operator==(Expression& rhs) const { return rhs.concrete_type() == NULL_VAL; } virtual size_t hash() { return 0; } ATTACH_OPERATIONS(); }; ///////////////////////////////// // Thunks for delayed evaluation. ///////////////////////////////// class Thunk : public Expression { ADD_PROPERTY(Expression*, expression); ADD_PROPERTY(Env*, environment); public: Thunk(string path, Position position, Expression* exp, Env* env = 0) : Expression(path, position), expression_(exp), environment_(env) { } }; ///////////////////////////////////////////////////////// // Individual parameter objects for mixins and functions. ///////////////////////////////////////////////////////// class Parameter : public AST_Node { ADD_PROPERTY(string, name); ADD_PROPERTY(Expression*, default_value); ADD_PROPERTY(bool, is_rest_parameter); public: Parameter(string p, Position pos, string n, Expression* def = 0, bool rest = false) : AST_Node(p, pos), name_(n), default_value_(def), is_rest_parameter_(rest) { if (default_value_ && is_rest_parameter_) { error("variable-length parameter may not have a default value", path(), position()); } } ATTACH_OPERATIONS(); }; ///////////////////////////////////////////////////////////////////////// // Parameter lists -- in their own class to facilitate context-sensitive // error checking (e.g., ensuring that all optional parameters follow all // required parameters). ///////////////////////////////////////////////////////////////////////// class Parameters : public AST_Node, public Vectorized { ADD_PROPERTY(bool, has_optional_parameters); ADD_PROPERTY(bool, has_rest_parameter); protected: void adjust_after_pushing(Parameter* p) { if (p->default_value()) { if (has_rest_parameter_) { error("optional parameters may not be combined with variable-length parameters", p->path(), p->position()); } has_optional_parameters_ = true; } else if (p->is_rest_parameter()) { if (has_rest_parameter_) { error("functions and mixins cannot have more than one variable-length parameter", p->path(), p->position()); } if (has_optional_parameters_) { error("optional parameters may not be combined with variable-length parameters", p->path(), p->position()); } has_rest_parameter_ = true; } else { if (has_rest_parameter_) { error("required parameters must precede variable-length parameters", p->path(), p->position()); } if (has_optional_parameters_) { error("required parameters must precede optional parameters", p->path(), p->position()); } } } public: Parameters(string path, Position position) : AST_Node(path, position), Vectorized(), has_optional_parameters_(false), has_rest_parameter_(false) { } ATTACH_OPERATIONS(); }; ////////////////////////////////////////////////////////////////////////////////////////// // Additional method on Lists to retrieve values directly or from an encompassed Argument. ////////////////////////////////////////////////////////////////////////////////////////// inline Expression* List::value_at_index(size_t i) { return is_arglist_ ? ((Argument*)(*this)[i])->value() : (*this)[i]; } ///////////////////////////////////////// // Abstract base class for CSS selectors. ///////////////////////////////////////// class Selector : public AST_Node { ADD_PROPERTY(bool, has_reference); ADD_PROPERTY(bool, has_placeholder); public: Selector(string path, Position position, bool r = false, bool h = false) : AST_Node(path, position), has_reference_(r), has_placeholder_(h) { } virtual ~Selector() = 0; virtual Selector_Placeholder* find_placeholder(); virtual int specificity() { return Constants::SPECIFICITY_BASE; } }; inline Selector::~Selector() { } ///////////////////////////////////////////////////////////////////////// // Interpolated selectors -- the interpolated String will be expanded and // re-parsed into a normal selector class. ///////////////////////////////////////////////////////////////////////// class Selector_Schema : public Selector { ADD_PROPERTY(String*, contents); public: Selector_Schema(string path, Position position, String* c) : Selector(path, position), contents_(c) { } ATTACH_OPERATIONS(); }; //////////////////////////////////////////// // Abstract base class for simple selectors. //////////////////////////////////////////// class Simple_Selector : public Selector { public: Simple_Selector(string path, Position position) : Selector(path, position) { } virtual ~Simple_Selector() = 0; virtual Compound_Selector* unify_with(Compound_Selector*, Context&); virtual bool is_pseudo_element() { return false; } bool operator==(const Simple_Selector& rhs) const; inline bool operator!=(const Simple_Selector& rhs) const { return !(*this == rhs); } bool operator<(const Simple_Selector& rhs) const; }; inline Simple_Selector::~Simple_Selector() { } ///////////////////////////////////// // Parent references (i.e., the "&"). ///////////////////////////////////// class Selector_Reference : public Simple_Selector { ADD_PROPERTY(Selector*, selector); public: Selector_Reference(string path, Position position, Selector* r = 0) : Simple_Selector(path, position), selector_(r) { has_reference(true); } virtual int specificity() { if (selector()) return selector()->specificity(); else return 0; } ATTACH_OPERATIONS(); }; ///////////////////////////////////////////////////////////////////////// // Placeholder selectors (e.g., "%foo") for use in extend-only selectors. ///////////////////////////////////////////////////////////////////////// class Selector_Placeholder : public Simple_Selector { ADD_PROPERTY(string, name); public: Selector_Placeholder(string path, Position position, string n) : Simple_Selector(path, position), name_(n) { has_placeholder(true); } virtual Selector_Placeholder* find_placeholder(); ATTACH_OPERATIONS(); }; ///////////////////////////////////////////////////////////////////// // Type selectors (and the universal selector) -- e.g., div, span, *. ///////////////////////////////////////////////////////////////////// class Type_Selector : public Simple_Selector { ADD_PROPERTY(string, name); public: Type_Selector(string path, Position position, string n) : Simple_Selector(path, position), name_(n) { } virtual int specificity() { if (name() == "*") return 0; else return 1; } virtual Compound_Selector* unify_with(Compound_Selector*, Context&); ATTACH_OPERATIONS(); }; //////////////////////////////////////////////// // Selector qualifiers -- i.e., classes and ids. //////////////////////////////////////////////// class Selector_Qualifier : public Simple_Selector { ADD_PROPERTY(string, name); public: Selector_Qualifier(string path, Position position, string n) : Simple_Selector(path, position), name_(n) { } virtual int specificity() { if (name()[0] == '#') return Constants::SPECIFICITY_BASE * Constants::SPECIFICITY_BASE; else return Constants::SPECIFICITY_BASE; } virtual Compound_Selector* unify_with(Compound_Selector*, Context&); ATTACH_OPERATIONS(); }; /////////////////////////////////////////////////// // Attribute selectors -- e.g., [src*=".jpg"], etc. /////////////////////////////////////////////////// class Attribute_Selector : public Simple_Selector { ADD_PROPERTY(string, name); ADD_PROPERTY(string, matcher); ADD_PROPERTY(String*, value); // might be interpolated public: Attribute_Selector(string path, Position position, string n, string m, String* v) : Simple_Selector(path, position), name_(n), matcher_(m), value_(v) { } ATTACH_OPERATIONS(); }; ////////////////////////////////////////////////////////////////// // Pseudo selectors -- e.g., :first-child, :nth-of-type(...), etc. ////////////////////////////////////////////////////////////////// class Pseudo_Selector : public Simple_Selector { ADD_PROPERTY(string, name); ADD_PROPERTY(String*, expression); public: Pseudo_Selector(string path, Position position, string n, String* expr = 0) : Simple_Selector(path, position), name_(n), expression_(expr) { } virtual int specificity() { // TODO: clean up the pseudo-element checking if (name() == ":before" || name() == "::before" || name() == ":after" || name() == "::after" || name() == ":first-line" || name() == "::first-line" || name() == ":first-letter" || name() == "::first-letter") return 1; else return Constants::SPECIFICITY_BASE; } virtual bool is_pseudo_element() { if (name() == ":before" || name() == "::before" || name() == ":after" || name() == "::after" || name() == ":first-line" || name() == "::first-line" || name() == ":first-letter" || name() == "::first-letter") { return true; } else { // If it's not a known pseudo-element, check whether it looks like one. This is similar to the type method on the Pseudo class in ruby sass. return name().find("::") == 0; } } virtual Compound_Selector* unify_with(Compound_Selector*, Context&); ATTACH_OPERATIONS(); }; ///////////////////////////////////////////////// // Wrapped selector -- pseudo selector that takes a list of selectors as argument(s) e.g., :not(:first-of-type), :-moz-any(ol p.blah, ul, menu, dir) ///////////////////////////////////////////////// class Wrapped_Selector : public Simple_Selector { ADD_PROPERTY(string, name); ADD_PROPERTY(Selector*, selector); public: Wrapped_Selector(string path, Position position, string n, Selector* sel) : Simple_Selector(path, position), name_(n), selector_(sel) { } ATTACH_OPERATIONS(); }; struct Complex_Selector_Pointer_Compare { bool operator() (const Complex_Selector* const pLeft, const Complex_Selector* const pRight) const; }; //////////////////////////////////////////////////////////////////////////// // Simple selector sequences. Maintains flags indicating whether it contains // any parent references or placeholders, to simplify expansion. //////////////////////////////////////////////////////////////////////////// typedef set SourcesSet; class Compound_Selector : public Selector, public Vectorized { private: SourcesSet sources_; protected: void adjust_after_pushing(Simple_Selector* s) { if (s->has_reference()) has_reference(true); if (s->has_placeholder()) has_placeholder(true); } public: Compound_Selector(string path, Position position, size_t s = 0) : Selector(path, position), Vectorized(s) { } Compound_Selector* unify_with(Compound_Selector* rhs, Context& ctx); virtual Selector_Placeholder* find_placeholder(); Simple_Selector* base() { // Implement non-const in terms of const. Safe to const_cast since this method is non-const return const_cast(static_cast(this)->base()); } const Simple_Selector* base() const { if (length() > 0 && typeid(*(*this)[0]) == typeid(Type_Selector)) return (*this)[0]; return 0; } bool is_superselector_of(Compound_Selector* rhs); virtual int specificity() { int sum = 0; for (size_t i = 0, L = length(); i < L; ++i) { sum += (*this)[i]->specificity(); } return sum; } bool is_empty_reference() { return length() == 1 && typeid(*(*this)[0]) == typeid(Selector_Reference) && !static_cast((*this)[0])->selector(); } vector to_str_vec(); // sometimes need to convert to a flat "by-value" data structure bool operator<(const Compound_Selector& rhs) const; bool operator==(const Compound_Selector& rhs) const; inline bool operator!=(const Compound_Selector& rhs) const { return !(*this == rhs); } SourcesSet& sources() { return sources_; } void clearSources() { sources_.clear(); } void mergeSources(SourcesSet& sources, Context& ctx); Compound_Selector* clone(Context&) const; // does not clone the Simple_Selector*s Compound_Selector* minus(Compound_Selector* rhs, Context& ctx); ATTACH_OPERATIONS(); }; //////////////////////////////////////////////////////////////////////////// // General selectors -- i.e., simple sequences combined with one of the four // CSS selector combinators (">", "+", "~", and whitespace). Essentially a // linked list. //////////////////////////////////////////////////////////////////////////// struct Context; class Complex_Selector : public Selector { public: enum Combinator { ANCESTOR_OF, PARENT_OF, PRECEDES, ADJACENT_TO }; private: ADD_PROPERTY(Combinator, combinator); ADD_PROPERTY(Compound_Selector*, head); ADD_PROPERTY(Complex_Selector*, tail); public: Complex_Selector(string path, Position position, Combinator c, Compound_Selector* h, Complex_Selector* t) : Selector(path, position), combinator_(c), head_(h), tail_(t) { if ((h && h->has_reference()) || (t && t->has_reference())) has_reference(true); if ((h && h->has_placeholder()) || (t && t->has_placeholder())) has_placeholder(true); } Compound_Selector* base(); Complex_Selector* context(Context&); Complex_Selector* innermost(); size_t length(); bool is_superselector_of(Compound_Selector*); bool is_superselector_of(Complex_Selector*); virtual Selector_Placeholder* find_placeholder(); Combinator clear_innermost(); void set_innermost(Complex_Selector*, Combinator); virtual int specificity() const { int sum = 0; if (head()) sum += head()->specificity(); if (tail()) sum += tail()->specificity(); return sum; } bool operator<(const Complex_Selector& rhs) const; bool operator==(const Complex_Selector& rhs) const; inline bool operator!=(const Complex_Selector& rhs) const { return !(*this == rhs); } SourcesSet sources() { //s = Set.new //seq.map {|sseq_or_op| s.merge sseq_or_op.sources if sseq_or_op.is_a?(SimpleSequence)} //s SourcesSet srcs; Compound_Selector* pHead = head(); Complex_Selector* pTail = tail(); if (pHead) { SourcesSet& headSources = pHead->sources(); srcs.insert(headSources.begin(), headSources.end()); } if (pTail) { SourcesSet tailSources = pTail->sources(); srcs.insert(tailSources.begin(), tailSources.end()); } return srcs; } void addSources(SourcesSet& sources, Context& ctx) { // members.map! {|m| m.is_a?(SimpleSequence) ? m.with_more_sources(sources) : m} Complex_Selector* pIter = this; while (pIter) { Compound_Selector* pHead = pIter->head(); if (pHead) { pHead->mergeSources(sources, ctx); } pIter = pIter->tail(); } } void clearSources() { Complex_Selector* pIter = this; while (pIter) { Compound_Selector* pHead = pIter->head(); if (pHead) { pHead->clearSources(); } pIter = pIter->tail(); } } Complex_Selector* clone(Context&) const; // does not clone Compound_Selector*s Complex_Selector* cloneFully(Context&) const; // clones Compound_Selector*s vector to_vector(); ATTACH_OPERATIONS(); }; typedef deque ComplexSelectorDeque; /////////////////////////////////// // Comma-separated selector groups. /////////////////////////////////// class Selector_List : public Selector, public Vectorized { #ifdef DEBUG ADD_PROPERTY(string, mCachedSelector); #endif protected: void adjust_after_pushing(Complex_Selector* c); public: Selector_List(string path, Position position, size_t s = 0) : Selector(path, position), Vectorized(s) { } virtual Selector_Placeholder* find_placeholder(); virtual int specificity() { int sum = 0; for (size_t i = 0, L = length(); i < L; ++i) { sum += (*this)[i]->specificity(); } return sum; } // vector members() { return elements_; } ATTACH_OPERATIONS(); }; template bool selectors_equal(const SelectorType& one, const SelectorType& two, bool simpleSelectorOrderDependent) { // Test for equality among selectors while differentiating between checks that demand the underlying Simple_Selector // ordering to be the same or not. This works because operator< (which doesn't make a whole lot of sense for selectors, but // is required for proper stl collection ordering) is implemented using string comparision. This gives stable sorting // behavior, and can be used to determine if the selectors would have exactly idential output. operator== matches the // ruby sass implementations for eql, which sometimes perform order independent comparisions (like set comparisons of the // members of a SimpleSequence (Compound_Selector)). // // Due to the reliance on operator== and operater< behavior, this templated method is currently only intended for // use with Compound_Selector and Complex_Selector objects. if (simpleSelectorOrderDependent) { return !(one < two) && !(two < one); } else { return one == two; } } } #ifdef __clang__ #pragma clang diagnostic pop #endif