#include "sequence.h" #include "node.h" #include "scanner.h" #include "token.h" namespace YAML { Sequence::Sequence() { } Sequence::~Sequence() { Clear(); } void Sequence::Clear() { for(unsigned i=0;iPeekNextToken(); switch(pToken->type) { case TT_BLOCK_SEQ_START: ParseBlock(pScanner); break; case TT_BLOCK_ENTRY: ParseImplicit(pScanner); break; case TT_FLOW_SEQ_START: ParseFlow(pScanner); break; } } void Sequence::ParseBlock(Scanner *pScanner) { // eat start token pScanner->EatNextToken(); while(1) { Token *pToken = pScanner->PeekNextToken(); if(!pToken) break; // TODO: throw? if(pToken->type != TT_BLOCK_ENTRY && pToken->type != TT_BLOCK_END) break; // TODO: throw? pScanner->PopNextToken(); if(pToken->type == TT_BLOCK_END) break; Node *pNode = new Node; m_data.push_back(pNode); pNode->Parse(pScanner); } } void Sequence::ParseImplicit(Scanner *pScanner) { while(1) { Token *pToken = pScanner->PeekNextToken(); // we're actually *allowed* to have no tokens at some point if(!pToken) break; // and we end at anything other than a block entry if(pToken->type != TT_BLOCK_ENTRY) break; pScanner->PopNextToken(); Node *pNode = new Node; m_data.push_back(pNode); pNode->Parse(pScanner); } } void Sequence::ParseFlow(Scanner *pScanner) { // eat start token pScanner->EatNextToken(); while(1) { Token *pToken = pScanner->PeekNextToken(); if(!pToken) break; // TODO: throw? // first check for end if(pToken->type == TT_FLOW_SEQ_END) { pScanner->PopNextToken(); break; } // then read the node Node *pNode = new Node; m_data.push_back(pNode); pNode->Parse(pScanner); // now eat the separator (or could be a sequence end, which we ignore - but if it's neither, then it's a bad node) pToken = pScanner->PeekNextToken(); if(pToken->type == TT_FLOW_ENTRY) pScanner->EatNextToken(); else if(pToken->type != TT_FLOW_SEQ_END) break; // TODO: throw? } } void Sequence::Write(std::ostream& out, int indent) { for(int i=0;iWrite(out, indent + 1); } }