WsParser_VS/libSyntax/syntax_novel.cpp

116 lines
5.0 KiB
C++
Raw Normal View History

2024-03-17 07:58:28 +00:00
#include "syntax_novel.h"
#include <tokens_novel.h>
using namespace lib_syntax;
using namespace example_novel;
using namespace lib_token;
using namespace ast_basic;
// token-avaliable ==========================================================================
auto leftb = std::make_shared<LeftBracket>(); // {
auto rightb = std::make_shared<RightBracket>(); // }
auto refers = std::make_shared<ReferMark>(); // @
2024-05-02 11:25:21 +00:00
auto story_key = std::make_shared<Keywords>(u8"<EFBFBD><EFBFBD><EFBFBD><EFBFBD>", u8"story-mark"); // <20><><EFBFBD><EFBFBD>
2024-03-17 07:58:28 +00:00
auto numbers = std::make_shared<Numbers>(); // [0-9]+
2024-05-02 11:25:21 +00:00
auto frag_key = std::make_shared<Keywords>(u8"<EFBFBD><EFBFBD><EFBFBD><EFBFBD>", u8"fragment-mark"); // <20><><EFBFBD><EFBFBD>
auto volume_key = std::make_shared<Keywords>(u8"<EFBFBD>־<EFBFBD>", u8"volume-mark"); // <20>־<EFBFBD>
auto article_key = std::make_shared<Keywords>(u8"<EFBFBD>½<EFBFBD>", u8"article-mark"); // <20>½<EFBFBD>
auto split_mark = std::make_shared<Split>(); // &
2024-03-17 07:58:28 +00:00
auto vtext = std::make_shared<VTextSection>(); // ^([^\\{\\}\\n@&]+)
2024-05-02 11:25:21 +00:00
auto name_text = std::make_shared<NameSection>(); // ^([^:\\{\\}\\n@&][^\\{\\}\\n@&]*)
2024-03-17 07:58:28 +00:00
auto newl = std::make_shared<NewLine>();
// rule-parts ===============================================================================
// MatchRule
#define MR(x) std::make_shared<const TokenMatch>(x)
// Buffer
2024-05-02 11:11:03 +00:00
#define Rules QList<std::shared_ptr<const BaseRule>>
2024-03-17 07:58:28 +00:00
// Option
#define OptMulT(token) std::make_shared<const Rept>(MR(token), 0, INT_MAX)
#define OptMulR(rule) std::make_shared<const Rept>(rule, 0, INT_MAX)
// multi+
#define MultiR(rule) std::make_shared<const Rept>(rule, 1, INT_MAX)
2024-05-02 11:11:03 +00:00
QList<std::shared_ptr<const BaseRule>> LinesMerge(const QList<std::shared_ptr<const BaseRule>>& mbrs) {
QList<std::shared_ptr<const BaseRule>> values_ret;
for (auto& item : mbrs) {
values_ret << item << OptMulT(newl);
}
return values_ret;
}
QList<std::shared_ptr<const BaseRule>> LinesMerge(std::shared_ptr<const BaseRule> item) {
return QList<std::shared_ptr<const BaseRule>>() << item << OptMulT(newl);
}
2024-03-17 07:58:28 +00:00
// remove-return
auto remove_nl = [](const ExprRule::TokenSeqs& p)->ExprRule::TokenSeqs {
ExprRule::TokenSeqs result;
for (auto& n : p) {
if (n->isLeaf()) {
auto target_token = n->tokens().first();
if (target_token->define()->name() == newl->name())
continue;
}
result.append(n);
}
return result;
};
2024-05-02 11:25:21 +00:00
auto decl_comp = std::make_shared<const Any>(Rules{MR(numbers), MR(vtext), MR(name_text), MR(split_mark)});
2024-05-02 11:11:03 +00:00
auto decl_expr = ExprRule(u8"decl_section", (int)NovelExprs::DESC_SECTION).reloadRule(remove_nl,
std::make_shared<const Seqs>(LinesMerge(MultiR(decl_comp))
));
auto fragment_decl = ExprRule(u8"fragment_define", (int)NovelExprs::FRAG_DEFINES).reloadRule(remove_nl, std::make_shared<const Seqs>(
2024-05-02 11:25:21 +00:00
LinesMerge(Rules{MR(leftb), MR(frag_key), MR(name_text)}) <<
2024-05-02 11:11:03 +00:00
OptMulR(decl_expr) <<
LinesMerge(MR(rightb))
));
auto fragment_refer = ExprRule(u8"fragment_refer", (int)NovelExprs::FRAG_REFERS).reloadRule(remove_nl, std::make_shared<const Seqs>(
2024-05-02 11:25:21 +00:00
LinesMerge(Rules{MR(leftb), MR(refers), MR(frag_key), MR(name_text), MR(split_mark), MR(name_text)}) <<
2024-05-02 11:11:03 +00:00
OptMulR(decl_expr) <<
LinesMerge(MR(rightb))
));
auto fragment_comp = std::make_shared<const Any>(Rules{fragment_decl, fragment_refer, decl_expr});
auto story_define = ExprRule(u8"story_define", (int)NovelExprs::STORY_DEFINES).reloadRule(remove_nl, std::make_shared<const Seqs>(
2024-05-02 11:25:21 +00:00
LinesMerge(Rules{MR(leftb), MR(story_key), MR(numbers), MR(name_text)}) <<
2024-05-02 11:11:03 +00:00
OptMulR(fragment_comp) <<
LinesMerge(MR(rightb))
));
2024-03-17 07:58:28 +00:00
// ===================================================================
2024-05-02 11:11:03 +00:00
auto article_decl = ExprRule(u8"article_define", (int)NovelExprs::ARTICLE_DEFINE).reloadRule(remove_nl, std::make_shared<const Seqs>(
2024-05-02 11:25:21 +00:00
LinesMerge(Rules{MR(leftb), MR(article_key), MR(name_text)}) <<
2024-05-02 11:11:03 +00:00
OptMulR(std::make_shared<const Any>(Rules{ fragment_refer, decl_expr })) <<
LinesMerge(MR(rightb))
));
auto volume_decl = ExprRule(u8"volume_define", (int)NovelExprs::VOLUME_DEFINE).reloadRule(remove_nl, std::make_shared<const Seqs>(
2024-05-02 11:25:21 +00:00
LinesMerge(Rules{MR(leftb), MR(volume_key), MR(name_text)}) <<
2024-05-02 11:11:03 +00:00
OptMulR(std::make_shared<const Any>(Rules{ decl_expr, article_decl })) <<
LinesMerge(MR(rightb))
));
auto document_define = ExprRule(u8"decls-doc", (int)NovelExprs::DOC_DEFINES).reloadRule(remove_nl, std::make_shared<const Seqs>(
Rules{
OptMulT(newl),
MultiR(std::make_shared<const Any>(Rules{story_define, volume_decl}))
}
));
2024-03-17 07:58:28 +00:00
std::shared_ptr<const ExprRule> NovalSyntax::getParseTree() { return document_define; }
std::shared_ptr<const TokenReader> NovalSyntax::getLexReader()
{
auto inst = std::make_shared<lib_token::TokenReader>(QList<std::shared_ptr<const lib_token::TokenDefine>>()
2024-05-02 11:25:21 +00:00
<< leftb << rightb << refers << split_mark << story_key << frag_key << volume_key << article_key << numbers << name_text << vtext << newl);
2024-03-17 07:58:28 +00:00
return inst;
}