Blame src/node.cpp

Packit Service 7770af
#include "sass.hpp"
Packit Service 7770af
#include <vector>
Packit Service 7770af
Packit Service 7770af
#include "node.hpp"
Packit Service 7770af
#include "context.hpp"
Packit Service 7770af
#include "parser.hpp"
Packit Service 7770af
Packit Service 7770af
namespace Sass {
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  Node Node::createCombinator(const Complex_Selector::Combinator& combinator) {
Packit Service 7770af
    NodeDequePtr null;
Packit Service 7770af
    return Node(COMBINATOR, combinator, NULL /*pSelector*/, null /*pCollection*/);
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  Node Node::createSelector(const Complex_Selector& pSelector) {
Packit Service 7770af
    NodeDequePtr null;
Packit Service 7770af
Packit Service 7770af
    Complex_Selector_Ptr pStripped = SASS_MEMORY_COPY(&pSelector);
Packit Service 7770af
    pStripped->tail(NULL);
Packit Service 7770af
    pStripped->combinator(Complex_Selector::ANCESTOR_OF);
Packit Service 7770af
Packit Service 7770af
    Node n(SELECTOR, Complex_Selector::ANCESTOR_OF, pStripped, null /*pCollection*/);
Packit Service 7770af
    n.got_line_feed = pSelector.has_line_feed();
Packit Service 7770af
    return n;
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  Node Node::createCollection() {
Packit Service 7770af
    NodeDequePtr pEmptyCollection = std::make_shared<NodeDeque>();
Packit Service 7770af
    return Node(COLLECTION, Complex_Selector::ANCESTOR_OF, NULL /*pSelector*/, pEmptyCollection);
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  Node Node::createCollection(const NodeDeque& values) {
Packit Service 7770af
    NodeDequePtr pShallowCopiedCollection = std::make_shared<NodeDeque>(values);
Packit Service 7770af
    return Node(COLLECTION, Complex_Selector::ANCESTOR_OF, NULL /*pSelector*/, pShallowCopiedCollection);
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  Node Node::createNil() {
Packit Service 7770af
    NodeDequePtr null;
Packit Service 7770af
    return Node(NIL, Complex_Selector::ANCESTOR_OF, NULL /*pSelector*/, null /*pCollection*/);
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  Node::Node(const TYPE& type, Complex_Selector::Combinator combinator, Complex_Selector_Ptr pSelector, NodeDequePtr& pCollection)
Packit Service 7770af
  : got_line_feed(false), mType(type), mCombinator(combinator), mpSelector(pSelector), mpCollection(pCollection)
Packit Service 7770af
  { if (pSelector) got_line_feed = pSelector->has_line_feed(); }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  Node Node::klone() const {
Packit Service 7770af
    NodeDequePtr pNewCollection = std::make_shared<NodeDeque>();
Packit Service 7770af
    if (mpCollection) {
Packit Service 7770af
      for (NodeDeque::iterator iter = mpCollection->begin(), iterEnd = mpCollection->end(); iter != iterEnd; iter++) {
Packit Service 7770af
        Node& toClone = *iter;
Packit Service 7770af
        pNewCollection->push_back(toClone.klone());
Packit Service 7770af
      }
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    Node n(mType, mCombinator, mpSelector ? SASS_MEMORY_COPY(mpSelector) : NULL, pNewCollection);
Packit Service 7770af
    n.got_line_feed = got_line_feed;
Packit Service 7770af
    return n;
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  bool Node::contains(const Node& potentialChild) const {
Packit Service 7770af
    bool found = false;
Packit Service 7770af
Packit Service 7770af
    for (NodeDeque::iterator iter = mpCollection->begin(), iterEnd = mpCollection->end(); iter != iterEnd; iter++) {
Packit Service 7770af
      Node& toTest = *iter;
Packit Service 7770af
Packit Service 7770af
      if (toTest == potentialChild) {
Packit Service 7770af
        found = true;
Packit Service 7770af
        break;
Packit Service 7770af
      }
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    return found;
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  bool Node::operator==(const Node& rhs) const {
Packit Service 7770af
    if (this->type() != rhs.type()) {
Packit Service 7770af
      return false;
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    if (this->isCombinator()) {
Packit Service 7770af
Packit Service 7770af
      return this->combinator() == rhs.combinator();
Packit Service 7770af
Packit Service 7770af
    } else if (this->isNil()) {
Packit Service 7770af
Packit Service 7770af
      return true; // no state to check
Packit Service 7770af
Packit Service 7770af
    } else if (this->isSelector()){
Packit Service 7770af
Packit Service 7770af
      return *this->selector() == *rhs.selector();
Packit Service 7770af
Packit Service 7770af
    } else if (this->isCollection()) {
Packit Service 7770af
Packit Service 7770af
      if (this->collection()->size() != rhs.collection()->size()) {
Packit Service 7770af
        return false;
Packit Service 7770af
      }
Packit Service 7770af
Packit Service 7770af
      for (NodeDeque::iterator lhsIter = this->collection()->begin(), lhsIterEnd = this->collection()->end(),
Packit Service 7770af
           rhsIter = rhs.collection()->begin(); lhsIter != lhsIterEnd; lhsIter++, rhsIter++) {
Packit Service 7770af
Packit Service 7770af
        if (*lhsIter != *rhsIter) {
Packit Service 7770af
          return false;
Packit Service 7770af
        }
Packit Service 7770af
Packit Service 7770af
      }
Packit Service 7770af
Packit Service 7770af
      return true;
Packit Service 7770af
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    // We shouldn't get here.
Packit Service 7770af
    throw "Comparing unknown node types. A new type was probably added and this method wasn't implemented for it.";
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  void Node::plus(Node& rhs) {
Packit Service 7770af
    if (!this->isCollection() || !rhs.isCollection()) {
Packit Service 7770af
      throw "Both the current node and rhs must be collections.";
Packit Service 7770af
    }
Packit Service 7770af
    this->collection()->insert(this->collection()->end(), rhs.collection()->begin(), rhs.collection()->end());
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
#ifdef DEBUG
Packit Service 7770af
  std::ostream& operator<<(std::ostream& os, const Node& node) {
Packit Service 7770af
Packit Service 7770af
    if (node.isCombinator()) {
Packit Service 7770af
Packit Service 7770af
      switch (node.combinator()) {
Packit Service 7770af
        case Complex_Selector::ANCESTOR_OF: os << "\" \""; break;
Packit Service 7770af
        case Complex_Selector::PARENT_OF:   os << "\">\""; break;
Packit Service 7770af
        case Complex_Selector::PRECEDES:    os << "\"~\""; break;
Packit Service 7770af
        case Complex_Selector::ADJACENT_TO: os << "\"+\""; break;
Packit Service 7770af
        case Complex_Selector::REFERENCE: os    << "\"/\""; break;
Packit Service 7770af
      }
Packit Service 7770af
Packit Service 7770af
    } else if (node.isNil()) {
Packit Service 7770af
Packit Service 7770af
      os << "nil";
Packit Service 7770af
Packit Service 7770af
    } else if (node.isSelector()){
Packit Service 7770af
Packit Service 7770af
      os << node.selector()->head()->to_string();
Packit Service 7770af
Packit Service 7770af
    } else if (node.isCollection()) {
Packit Service 7770af
Packit Service 7770af
      os << "[";
Packit Service 7770af
Packit Service 7770af
      for (NodeDeque::iterator iter = node.collection()->begin(), iterBegin = node.collection()->begin(), iterEnd = node.collection()->end(); iter != iterEnd; iter++) {
Packit Service 7770af
        if (iter != iterBegin) {
Packit Service 7770af
          os << ", ";
Packit Service 7770af
        }
Packit Service 7770af
Packit Service 7770af
        os << (*iter);
Packit Service 7770af
      }
Packit Service 7770af
Packit Service 7770af
      os << "]";
Packit Service 7770af
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    return os;
Packit Service 7770af
Packit Service 7770af
  }
Packit Service 7770af
#endif
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  Node complexSelectorToNode(Complex_Selector_Ptr pToConvert) {
Packit Service 7770af
    if (pToConvert == NULL) {
Packit Service 7770af
      return Node::createNil();
Packit Service 7770af
    }
Packit Service 7770af
    Node node = Node::createCollection();
Packit Service 7770af
    node.got_line_feed = pToConvert->has_line_feed();
Packit Service 7770af
    bool has_lf = pToConvert->has_line_feed();
Packit Service 7770af
Packit Service 7770af
    // unwrap the selector from parent ref
Packit Service 7770af
    if (pToConvert->head() && pToConvert->head()->has_parent_ref()) {
Packit Service 7770af
      Complex_Selector_Obj tail = pToConvert->tail();
Packit Service 7770af
      if (tail) tail->has_line_feed(pToConvert->has_line_feed());
Packit Service 7770af
      pToConvert = tail;
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    while (pToConvert) {
Packit Service 7770af
Packit Service 7770af
      bool empty_parent_ref = pToConvert->head() && pToConvert->head()->is_empty_reference();
Packit Service 7770af
Packit Service 7770af
      if (pToConvert->head() || empty_parent_ref) {
Packit Service 7770af
      }
Packit Service 7770af
Packit Service 7770af
      // the first Complex_Selector may contain a dummy head pointer, skip it.
Packit Service 7770af
      if (pToConvert->head() && !empty_parent_ref) {
Packit Service 7770af
        node.collection()->push_back(Node::createSelector(*pToConvert));
Packit Service 7770af
        if (has_lf) node.collection()->back().got_line_feed = has_lf;
Packit Service 7770af
        has_lf = false;
Packit Service 7770af
      }
Packit Service 7770af
Packit Service 7770af
      if (pToConvert->combinator() != Complex_Selector::ANCESTOR_OF) {
Packit Service 7770af
        node.collection()->push_back(Node::createCombinator(pToConvert->combinator()));
Packit Service 7770af
        if (has_lf) node.collection()->back().got_line_feed = has_lf;
Packit Service 7770af
        has_lf = false;
Packit Service 7770af
      }
Packit Service 7770af
Packit Service 7770af
      if (pToConvert && empty_parent_ref && pToConvert->tail()) {
Packit Service 7770af
        // pToConvert->tail()->has_line_feed(pToConvert->has_line_feed());
Packit Service 7770af
      }
Packit Service 7770af
Packit Service 7770af
      pToConvert = pToConvert->tail();
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    return node;
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
  Complex_Selector_Ptr nodeToComplexSelector(const Node& toConvert) {
Packit Service 7770af
    if (toConvert.isNil()) {
Packit Service 7770af
      return NULL;
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
    if (!toConvert.isCollection()) {
Packit Service 7770af
      throw "The node to convert to a Complex_Selector_Ptr must be a collection type or nil.";
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
Packit Service 7770af
    NodeDeque& childNodes = *toConvert.collection();
Packit Service 7770af
Packit Service 7770af
    std::string noPath("");
Packit Service 7770af
    Complex_Selector_Obj pFirst = SASS_MEMORY_NEW(Complex_Selector, ParserState("[NODE]"), Complex_Selector::ANCESTOR_OF, NULL, NULL);
Packit Service 7770af
Packit Service 7770af
    Complex_Selector_Obj pCurrent = pFirst;
Packit Service 7770af
Packit Service 7770af
    if (toConvert.isSelector()) pFirst->has_line_feed(toConvert.got_line_feed);
Packit Service 7770af
    if (toConvert.isCombinator()) pFirst->has_line_feed(toConvert.got_line_feed);
Packit Service 7770af
Packit Service 7770af
    for (NodeDeque::iterator childIter = childNodes.begin(), childIterEnd = childNodes.end(); childIter != childIterEnd; childIter++) {
Packit Service 7770af
Packit Service 7770af
      Node& child = *childIter;
Packit Service 7770af
Packit Service 7770af
      if (child.isSelector()) {
Packit Service 7770af
        // JMA - need to clone the selector, because they can end up getting shared across Node
Packit Service 7770af
        // collections, and can result in an infinite loop during the call to parentSuperselector()
Packit Service 7770af
        pCurrent->tail(SASS_MEMORY_COPY(child.selector()));
Packit Service 7770af
        // if (child.got_line_feed) pCurrent->has_line_feed(child.got_line_feed);
Packit Service 7770af
        pCurrent = pCurrent->tail();
Packit Service 7770af
      } else if (child.isCombinator()) {
Packit Service 7770af
        pCurrent->combinator(child.combinator());
Packit Service 7770af
        if (child.got_line_feed) pCurrent->has_line_feed(child.got_line_feed);
Packit Service 7770af
Packit Service 7770af
        // if the next node is also a combinator, create another Complex_Selector to hold it so it doesn't replace the current combinator
Packit Service 7770af
        if (childIter+1 != childIterEnd) {
Packit Service 7770af
          Node& nextNode = *(childIter+1);
Packit Service 7770af
          if (nextNode.isCombinator()) {
Packit Service 7770af
            pCurrent->tail(SASS_MEMORY_NEW(Complex_Selector, ParserState("[NODE]"), Complex_Selector::ANCESTOR_OF, NULL, NULL));
Packit Service 7770af
            if (nextNode.got_line_feed) pCurrent->tail()->has_line_feed(nextNode.got_line_feed);
Packit Service 7770af
            pCurrent = pCurrent->tail();
Packit Service 7770af
          }
Packit Service 7770af
        }
Packit Service 7770af
      } else {
Packit Service 7770af
        throw "The node to convert's children must be only combinators or selectors.";
Packit Service 7770af
      }
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    // Put the dummy Compound_Selector in the first position, for consistency with the rest of libsass
Packit Service 7770af
    Compound_Selector_Ptr fakeHead = SASS_MEMORY_NEW(Compound_Selector, ParserState("[NODE]"), 1);
Packit Service 7770af
    Parent_Selector_Ptr selectorRef = SASS_MEMORY_NEW(Parent_Selector, ParserState("[NODE]"));
Packit Service 7770af
    fakeHead->elements().push_back(selectorRef);
Packit Service 7770af
    if (toConvert.got_line_feed) pFirst->has_line_feed(toConvert.got_line_feed);
Packit Service 7770af
    // pFirst->has_line_feed(pFirst->has_line_feed() || pFirst->tail()->has_line_feed() || toConvert.got_line_feed);
Packit Service 7770af
    pFirst->head(fakeHead);
Packit Service 7770af
    return SASS_MEMORY_COPY(pFirst);
Packit Service 7770af
  }
Packit Service 7770af
Packit Service 7770af
  // A very naive trim function, which removes duplicates in a node
Packit Service 7770af
  // This is only used in Complex_Selector::unify_with for now, may need modifications to fit other needs
Packit Service 7770af
  Node Node::naiveTrim(Node& seqses) {
Packit Service 7770af
Packit Service 7770af
    std::vector<Node*> res;
Packit Service 7770af
    std::vector<Complex_Selector_Obj> known;
Packit Service 7770af
Packit Service 7770af
    NodeDeque::reverse_iterator seqsesIter = seqses.collection()->rbegin(),
Packit Service 7770af
                                seqsesIterEnd = seqses.collection()->rend();
Packit Service 7770af
Packit Service 7770af
    for (; seqsesIter != seqsesIterEnd; ++seqsesIter)
Packit Service 7770af
    {
Packit Service 7770af
      Node& seqs1 = *seqsesIter;
Packit Service 7770af
      if( seqs1.isSelector() ) {
Packit Service 7770af
        Complex_Selector_Obj sel = seqs1.selector();
Packit Service 7770af
        std::vector<Complex_Selector_Obj>::iterator it;
Packit Service 7770af
        bool found = false;
Packit Service 7770af
        for (it = known.begin(); it != known.end(); ++it) {
Packit Service 7770af
          if (**it == *sel) { found = true; break; }
Packit Service 7770af
        }
Packit Service 7770af
        if( !found ) {
Packit Service 7770af
          known.push_back(seqs1.selector());
Packit Service 7770af
          res.push_back(&seqs1);
Packit Service 7770af
        }
Packit Service 7770af
      } else {
Packit Service 7770af
        res.push_back(&seqs1);
Packit Service 7770af
      }
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    Node result = Node::createCollection();
Packit Service 7770af
Packit Service 7770af
    for (size_t i = res.size() - 1; i != std::string::npos; --i) {
Packit Service 7770af
      result.collection()->push_back(*res[i]);
Packit Service 7770af
    }
Packit Service 7770af
Packit Service 7770af
    return result;
Packit Service 7770af
  }
Packit Service 7770af
}