250 lines
10 KiB
C++
250 lines
10 KiB
C++
#include "syntax_novel.h"
|
||
#include <tokens_novel.h>
|
||
#include "ast_novel.h"
|
||
|
||
using namespace lib_syntax;
|
||
using namespace example_novel;
|
||
using namespace lib_token;
|
||
using namespace ast_basic;
|
||
using namespace std;
|
||
|
||
// rule-parts ===============================================================================
|
||
#include "syntax_templets.h"
|
||
using namespace lib_composit;
|
||
|
||
void point_nm_set(std::shared_ptr<PointDefines> inst, std::shared_ptr<const lib_token::IToken> token) {
|
||
inst->addToken(token);
|
||
inst->setName(token->content());
|
||
}
|
||
void ref_story_set(std::shared_ptr<PointRefers> inst, std::shared_ptr<const lib_token::IToken> token) {
|
||
inst->addToken(token);
|
||
inst->setStoryRefer(token->content());
|
||
}
|
||
void ref_slice_set(std::shared_ptr<PointRefers> inst, std::shared_ptr<const lib_token::IToken> token) {
|
||
inst->addToken(token);
|
||
inst->setSliceRefer(token->content());
|
||
}
|
||
void ref_point_set(std::shared_ptr<PointRefers> inst, std::shared_ptr<const lib_token::IToken> token) {
|
||
inst->addToken(token);
|
||
inst->setPointRefer(token->content());
|
||
}
|
||
void slice_nm_set(std::shared_ptr<FragmentSlice> inst, std::shared_ptr<const lib_token::IToken> token) {
|
||
inst->addToken(token);
|
||
inst->setName(token->content());
|
||
}
|
||
void story_nm_set(std::shared_ptr<StoryDefine> inst, std::shared_ptr<const lib_token::IToken> token) {
|
||
inst->addToken(token);
|
||
inst->setName(token->content());
|
||
}
|
||
void article_nm_set(std::shared_ptr<ArticleDefine>inst, std::shared_ptr<const lib_token::IToken> token) {
|
||
inst->addToken(token);
|
||
inst->setName(token->content());
|
||
}
|
||
void volume_nm_set(std::shared_ptr<VolumeDefine> inst, std::shared_ptr<const lib_token::IToken> token) {
|
||
inst->addToken(token);
|
||
inst->setName(token->content());
|
||
}
|
||
void rank_set(std::shared_ptr<RankDeclare> inst, std::shared_ptr<const lib_token::IToken> token) {
|
||
inst->addToken(token);
|
||
inst->setRank(token->content().toInt());
|
||
}
|
||
|
||
auto content_extract = [](std::shared_ptr<const lib_token::IActionToken> token) {
|
||
QString content;
|
||
while (token) {
|
||
if (token->defines())
|
||
content.prepend(token->content() + " ");
|
||
token = token->prevToken();
|
||
}
|
||
return content;
|
||
};
|
||
|
||
using TextDeclsSyntaxDef = Any<Match<Numbers>, Match<NormalText>, Match<ReferMk>, Match<SplitMk>>;
|
||
class DeclSyntax : public ElementRule<TextSection, (int) NovelNode::TextSection, TextDeclsSyntaxDef> {
|
||
public:
|
||
DeclSyntax() : ElementRule<TextSection, (int) NovelNode::TextSection, TextDeclsSyntaxDef>("decl_section") { }
|
||
|
||
// 通过 ElementRule 继承
|
||
QList<std::shared_ptr<const MatchCursor>> expr_rule_parse(std::shared_ptr<const MatchCursor> cursor, QList<std::shared_ptr<const MatchCursor>>& out) const override {
|
||
auto syntax_text = this->present();
|
||
auto current_rst = content_extract(cursor->currentToken());
|
||
auto rst = _children_store->parse(cursor, out);
|
||
for (auto r : rst)
|
||
std::const_pointer_cast<MatchCursor>(r)->setStop(r->exprsErrorCount());
|
||
return rst;
|
||
}
|
||
};
|
||
|
||
|
||
|
||
using PointSyntaxDef = lib_composit::Seqs<Match<LBracket>, Match<PointWord>, Action<PointDefines, NameText, point_nm_set>,
|
||
OptMulti<DeclSyntax>,
|
||
Match<RBracket>>;
|
||
class PointSyntax : public ElementRule<PointDefines, (int) NovelNode::PointDefines, PointSyntaxDef> {
|
||
public:
|
||
PointSyntax() : ElementRule<PointDefines, (int) NovelNode::PointDefines, PointSyntaxDef>("point_define") { }
|
||
|
||
// 通过 ElementRule 继承
|
||
QList<std::shared_ptr<const MatchCursor>> expr_rule_parse(std::shared_ptr<const MatchCursor> cursor, QList<std::shared_ptr<const MatchCursor>>& out) const override {
|
||
return _children_store->parse(cursor, out);
|
||
}
|
||
};
|
||
|
||
|
||
|
||
using ReferSyntaxDef = lib_composit::Seqs<Match<LBracket>, Match<ReferMk>, Match<PointWord>, Action<PointRefers, NameText, ref_story_set>, Match<ReferMk>, Action<PointRefers, NameText, ref_slice_set>, Match<ReferMk>, Action<PointRefers, NameText, ref_point_set>,
|
||
OptMulti<DeclSyntax>,
|
||
Match<RBracket>>;
|
||
class ReferSyntax : public ElementRule<PointRefers, (int) NovelNode::PointRefers, ReferSyntaxDef> {
|
||
public:
|
||
ReferSyntax() : ElementRule < PointRefers, (int) NovelNode::PointRefers, ReferSyntaxDef>("point_refer") { }
|
||
|
||
// 通过 ElementRule 继承
|
||
QList<std::shared_ptr<const MatchCursor>> expr_rule_parse(std::shared_ptr<const MatchCursor> cursor, QList<std::shared_ptr<const MatchCursor>>& out) const override {
|
||
return _children_store->parse(cursor, out);
|
||
}
|
||
};
|
||
|
||
|
||
|
||
using SliceSyntaxDef = lib_composit::Seqs<Match<LBracket>, Match<SliceWord>, Action<FragmentSlice, NameText, slice_nm_set>,
|
||
lib_composit::OptMulti<Any<PointSyntax, ReferSyntax, DeclSyntax>>,
|
||
Match<RBracket>>;
|
||
class SliceSyntax : public ElementRule<FragmentSlice, (int) NovelNode::FragmentSlice, SliceSyntaxDef> {
|
||
public:
|
||
SliceSyntax() : ElementRule<FragmentSlice, (int) NovelNode::FragmentSlice, SliceSyntaxDef>("slice_define") { }
|
||
|
||
// 通过 ElementRule 继承
|
||
QList<std::shared_ptr<const MatchCursor>> expr_rule_parse(std::shared_ptr<const MatchCursor> cursor, QList<std::shared_ptr<const MatchCursor>>& out) const override {
|
||
return _children_store->parse(cursor, out);
|
||
}
|
||
};
|
||
|
||
|
||
|
||
using StorySyntaxDef = lib_composit::Seqs<Match<LBracket>, Match<StoryWord>, Action<StoryDefine, NameText, story_nm_set>,
|
||
lib_composit::OptMulti<Any<SliceSyntax, DeclSyntax>>,
|
||
Match<RBracket>>;
|
||
class StorySyntax : public ElementRule<StoryDefine, (int) NovelNode::StoryDefine, StorySyntaxDef> {
|
||
public:
|
||
StorySyntax() : ElementRule<StoryDefine, (int) NovelNode::StoryDefine, StorySyntaxDef>("story_define") { }
|
||
|
||
// 通过 ElementRule 继承
|
||
QList<std::shared_ptr<const MatchCursor>> expr_rule_parse(std::shared_ptr<const MatchCursor> cursor, QList<std::shared_ptr<const MatchCursor>>& out) const override {
|
||
auto syntax = this->present();
|
||
auto current_rst = content_extract(cursor->currentToken());
|
||
auto rst = _children_store->parse(cursor, out);
|
||
|
||
QString result_list;
|
||
for (auto c : rst) {
|
||
result_list += content_extract(c->currentToken()) += "\n";
|
||
}
|
||
|
||
return rst;
|
||
}
|
||
};
|
||
|
||
|
||
|
||
using ArticleSyntaxDef = lib_composit::Seqs<Match<LBracket>, Match<ArticleWord>, Action<ArticleDefine, NameText, article_nm_set>,
|
||
lib_composit::OptMulti<Any<ReferSyntax, DeclSyntax>>,
|
||
Match<RBracket>>;
|
||
class ArticleSyntax : public ElementRule<ArticleDefine, (int) NovelNode::ArticleDefine, ArticleSyntaxDef> {
|
||
public:
|
||
ArticleSyntax() : ElementRule<ArticleDefine, (int) NovelNode::ArticleDefine, ArticleSyntaxDef>("article_define") { }
|
||
|
||
// 通过 ElementRule 继承
|
||
QList<std::shared_ptr<const MatchCursor>> expr_rule_parse(std::shared_ptr<const MatchCursor> cursor, QList<std::shared_ptr<const MatchCursor>>& out) const override {
|
||
return _children_store->parse(cursor, out);
|
||
}
|
||
};
|
||
|
||
|
||
|
||
using VolumeSyntaxDef = lib_composit::Seqs<Match<LBracket>, Match<VolumeWord>, Action<VolumeDefine, NameText, volume_nm_set>,
|
||
lib_composit::OptMulti<Any<ArticleSyntax, DeclSyntax>>,
|
||
Match<RBracket>>;
|
||
class VolumeSyntax : public ElementRule<VolumeDefine, (int) NovelNode::VolumeDefine, VolumeSyntaxDef> {
|
||
public:
|
||
VolumeSyntax() : ElementRule<VolumeDefine, (int) NovelNode::VolumeDefine, VolumeSyntaxDef>("volume_define") { }
|
||
|
||
// 通过 ElementRule 继承
|
||
QList<std::shared_ptr<const MatchCursor>> expr_rule_parse(std::shared_ptr<const MatchCursor> cursor, QList<std::shared_ptr<const MatchCursor>>& out) const override {
|
||
return _children_store->parse(cursor, out);
|
||
}
|
||
};
|
||
|
||
|
||
|
||
using RankSyntaxDef = lib_composit::Seqs<Match<DeclareSymbo>, Match<RankWord>, Action<RankDeclare, Numbers, rank_set>>;
|
||
class RankSyntax : public ElementRule<RankDeclare, (int) NovelNode::RankDeclaration, RankSyntaxDef> {
|
||
public:
|
||
RankSyntax() : ElementRule<RankDeclare, (int) NovelNode::RankDeclaration, RankSyntaxDef>("rank_define") { }
|
||
|
||
// 通过 ElementRule 继承
|
||
QList<std::shared_ptr<const MatchCursor>> expr_rule_parse(std::shared_ptr<const MatchCursor> cursor, QList<std::shared_ptr<const MatchCursor>>& out) const override {
|
||
return _children_store->parse(cursor, out);
|
||
}
|
||
};
|
||
|
||
|
||
|
||
using DocSyntaxDef = lib_composit::Seqs<Opt<RankSyntax>, lib_composit::OptMulti<Any<StorySyntax, VolumeSyntax>>>;
|
||
class DocumentSyntax : public ElementRule<Document, (int) NovelNode::Document, DocSyntaxDef> {
|
||
public:
|
||
DocumentSyntax() : ElementRule<Document, (int) NovelNode::Document, DocSyntaxDef>("decls-doc") { }
|
||
|
||
// 通过 ElementRule 继承
|
||
QList<std::shared_ptr<const MatchCursor>> expr_rule_parse(std::shared_ptr<const MatchCursor> cursor, QList<std::shared_ptr<const MatchCursor>>& out) const override {
|
||
auto text = _children_store->present();
|
||
return _children_store->parse(cursor, out);
|
||
}
|
||
};
|
||
|
||
|
||
std::shared_ptr<const lib_syntax::ExprRule> example_novel::NovalSyntax::getSyntaxTree() {
|
||
return std::make_shared<DocumentSyntax>();
|
||
}
|
||
|
||
|
||
|
||
//std::shared_ptr<const ast_gen::SyntaxElement> NovalSyntax::tidy(std::shared_ptr<ast_gen::SyntaxElement> root, QList<std::shared_ptr<ast_gen::SyntaxElement>> children)
|
||
//{
|
||
// build_objecttree(root, children);
|
||
// node_register(root, children);
|
||
// return root;
|
||
//}
|
||
//void NovalSyntax::build_objecttree(std::shared_ptr<ast_gen::SyntaxElement> root, QList<std::shared_ptr<ast_gen::SyntaxElement>> children)
|
||
//{
|
||
// for (auto& cinst : children) {
|
||
// cinst->setParent(root);
|
||
//
|
||
// QList<std::shared_ptr<ast_gen::SyntaxElement>> child_items;
|
||
// for (auto& it : cinst->bindExpression()->children()) {
|
||
// auto const_it = std::dynamic_pointer_cast<const ast_gen::SyntaxElement>(it);
|
||
// child_items.append(std::const_pointer_cast<ast_gen::SyntaxElement>(const_it));
|
||
// }
|
||
//
|
||
// build_objecttree(cinst, child_items);
|
||
// }
|
||
//}
|
||
//void NovalSyntax::node_register(std::shared_ptr<const ast_gen::SyntaxElement> root, QList<std::shared_ptr<ast_gen::SyntaxElement>> children)
|
||
//{
|
||
// for (auto& child : children) {
|
||
// if (!child->isAnonymous()) {
|
||
// auto check_result = ast_gen::GlobalElement::UniquePtr->appendToCache(child);
|
||
// if (check_result)
|
||
// throw new SyntaxException(QString("SyntaxError[0x0004]系统中包含同类型重名命名节点:%1<type:%2>(%3,%4)")
|
||
// .arg(child->signature()).arg(child->typeMark()).arg(child->path()).arg(check_result->path()));
|
||
// }
|
||
//
|
||
// QList<std::shared_ptr<ast_gen::SyntaxElement>> next_child_items;
|
||
// for (auto& it : child->bindExpression()->children()) {
|
||
// auto const_it = std::dynamic_pointer_cast<const ast_gen::SyntaxElement>(it);
|
||
// next_child_items.append(std::const_pointer_cast<ast_gen::SyntaxElement>(const_it));
|
||
// }
|
||
//
|
||
// node_register(child, next_child_items);
|
||
// }
|
||
//}
|