GfmlLxr_.java 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. /*
  2. XOWA: the XOWA Offline Wiki Application
  3. Copyright (C) 2012-2017 gnosygnu@gmail.com
  4. XOWA is licensed under the terms of the General Public License (GPL) Version 3,
  5. or alternatively under the terms of the Apache License Version 2.0.
  6. You may use XOWA according to either of these licenses as is most appropriate
  7. for your project on a case-by-case basis.
  8. The terms of each license can be found in the source code repository:
  9. GPLv3 License: https://github.com/gnosygnu/xowa/blob/master/LICENSE-GPLv3.txt
  10. Apache License: https://github.com/gnosygnu/xowa/blob/master/LICENSE-APACHE2.txt
  11. */
  12. package gplx.gfml;
  13. import gplx.types.commons.KeyVal;
  14. import gplx.types.errs.Err;
  15. import gplx.core.texts.*; /*CharStream*/
  16. import gplx.frameworks.invks.GfoMsg;
  17. import gplx.frameworks.invks.Gfo_invk;
  18. import gplx.frameworks.invks.Gfo_invk_;
  19. import gplx.frameworks.evts.Gfo_evt_itm;
  20. import gplx.frameworks.evts.Gfo_evt_mgr;
  21. import gplx.frameworks.evts.Gfo_evt_mgr_;
  22. import gplx.frameworks.invks.GfsCtx;
  23. import gplx.types.basics.utls.StringUtl;
  24. import gplx.types.commons.String_bldr;
  25. import gplx.types.commons.String_bldr_;
  26. import gplx.types.errs.ErrUtl;
  27. public class GfmlLxr_ {
  28. public static GfmlLxr general_(String key, GfmlTkn protoTkn) {return GfmlLxr_general.new_(key, protoTkn);}
  29. public static GfmlLxr solo_(String key, GfmlTkn singletonTkn) {return GfmlLxr_singleton.new_(key, singletonTkn.Raw(), singletonTkn);}
  30. public static GfmlLxr range_(String key, String[] ary, GfmlTkn protoTkn, boolean ignoreOutput) {return GfmlLxr_group.new_(key, ary, protoTkn, ignoreOutput);}
  31. public static GfmlLxr symbol_(String key, String raw, String val, GfmlBldrCmd cmd) {
  32. GfmlTkn tkn = GfmlTkn_.singleton_(key, raw, val, cmd);
  33. return GfmlLxr_.solo_(key, tkn);
  34. }
  35. public static GfmlLxr frame_(String key, GfmlFrame frame, String bgn, String end) {return GfmlLxr_frame.new_(key, frame, bgn, end, GfmlBldrCmd_pendingTkns_add.Instance, GfmlBldrCmd_frameEnd.data_());}
  36. public static final GfmlLxr Null = new GfmlLxr_null();
  37. public static final String CmdTknChanged_evt = "Changed";
  38. public static GfmlLxr as_(Object obj) {return obj instanceof GfmlLxr ? (GfmlLxr)obj : null;}
  39. public static GfmlLxr cast(Object obj) {try {return (GfmlLxr)obj;} catch(Exception exc) {throw ErrUtl.NewCast(exc, GfmlLxr.class, obj);}}
  40. }
  41. class GfmlLxr_null implements GfmlLxr {
  42. public String Key() {return "gfml.nullLxr";}
  43. public Gfo_evt_mgr Evt_mgr() {if (evt_mgr == null) evt_mgr = new Gfo_evt_mgr(this); return evt_mgr;} Gfo_evt_mgr evt_mgr;
  44. public Object Invk(GfsCtx ctx, int ikey, String k, GfoMsg m) {return Gfo_invk_.Rv_unhandled;}
  45. public GfmlTkn CmdTkn() {return GfmlTkn_.Null;} public void CmdTkn_set(GfmlTkn val) {}
  46. public String[] Hooks() {return StringUtl.AryEmpty;}
  47. public GfmlTkn MakeTkn(CharStream stream, int hookLength) {return GfmlTkn_.Null;}
  48. public void SubLxr_Add(GfmlLxr... lexer) {}
  49. public GfmlLxr SubLxr() {return this;}
  50. }
  51. class GfmlLxr_singleton implements GfmlLxr, Gfo_evt_itm {
  52. public Gfo_evt_mgr Evt_mgr() {if (evt_mgr == null) evt_mgr = new Gfo_evt_mgr(this); return evt_mgr;} Gfo_evt_mgr evt_mgr;
  53. public Object Invk(GfsCtx ctx, int ikey, String k, GfoMsg m) {return Gfo_invk_.Rv_unhandled;}
  54. public String Key() {return key;} private String key;
  55. public GfmlTkn CmdTkn() {return singletonTkn;} GfmlTkn singletonTkn;
  56. public void CmdTkn_set(GfmlTkn val) {
  57. String oldRaw = singletonTkn.Raw();
  58. singletonTkn = val;
  59. hooks = StringUtl.Ary(val.Raw());
  60. Gfo_evt_mgr_.Pub_vals(this, GfmlLxr_.CmdTknChanged_evt, KeyVal.NewStr("old", oldRaw), KeyVal.NewStr("new", val.Raw()), KeyVal.NewStr("lxr", this));
  61. }
  62. public String[] Hooks() {return hooks;} private String[] hooks;
  63. public GfmlTkn MakeTkn(CharStream stream, int hookLength) {
  64. stream.MoveNextBy(hookLength);
  65. return singletonTkn;
  66. }
  67. public GfmlLxr SubLxr() {return subLxr;} GfmlLxr subLxr;
  68. public void SubLxr_Add(GfmlLxr... lexer) {subLxr.SubLxr_Add(lexer);}
  69. public static GfmlLxr_singleton new_(String key, String hook, GfmlTkn singletonTkn) {
  70. GfmlLxr_singleton rv = new GfmlLxr_singleton();
  71. rv.ctor_(key, hook, singletonTkn, GfmlLxr_.Null);
  72. return rv;
  73. } protected GfmlLxr_singleton() {}
  74. public void ctor_(String key, String hook, GfmlTkn singletonTkn, GfmlLxr subLxr) {
  75. this.key = key;
  76. this.hooks = StringUtl.Ary(hook);
  77. this.subLxr = subLxr;
  78. this.singletonTkn = singletonTkn;
  79. }
  80. }
  81. class GfmlLxr_group implements GfmlLxr {
  82. public String Key() {return key;} private String key;
  83. public Gfo_evt_mgr Evt_mgr() {if (evt_mgr == null) evt_mgr = new Gfo_evt_mgr(this); return evt_mgr;} Gfo_evt_mgr evt_mgr;
  84. public Object Invk(GfsCtx ctx, int ikey, String k, GfoMsg m) {return Gfo_invk_.Rv_unhandled;}
  85. public GfmlTkn CmdTkn() {return outputTkn;} public void CmdTkn_set(GfmlTkn val) {} GfmlTkn outputTkn;
  86. public String[] Hooks() {return trie.Symbols();}
  87. public GfmlTkn MakeTkn(CharStream stream, int hookLength) {
  88. while (stream.AtMid()) {
  89. if (!ignoreOutput)
  90. sb.AddMidLen(stream.Ary(), stream.Pos(), hookLength);
  91. stream.MoveNextBy(hookLength);
  92. String found = StringUtl.Cast(trie.FindMatch(stream));
  93. if (found == null) break;
  94. hookLength = trie.LastMatchCount;
  95. }
  96. if (ignoreOutput) return GfmlTkn_.IgnoreOutput;
  97. String raw = sb.ToStrAndClear();
  98. return outputTkn.MakeNew(raw, raw);
  99. }
  100. public GfmlLxr SubLxr() {throw Err_sublxr();}
  101. public void SubLxr_Add(GfmlLxr... lexer) {throw Err_sublxr();}
  102. Err Err_sublxr() {return ErrUtl.NewUnimplemented("group lxr does not have subLxrs", "key", key, "output_tkn", outputTkn.Raw());}
  103. GfmlTrie trie = GfmlTrie.new_(); String_bldr sb = String_bldr_.new_(); boolean ignoreOutput;
  104. public static GfmlLxr_group new_(String key, String[] hooks, GfmlTkn outputTkn, boolean ignoreOutput) {
  105. GfmlLxr_group rv = new GfmlLxr_group();
  106. rv.key = key;
  107. for (String hook : hooks)
  108. rv.trie.Add(hook, hook);
  109. rv.outputTkn = outputTkn; rv.ignoreOutput = ignoreOutput;
  110. return rv;
  111. } GfmlLxr_group() {}
  112. }
  113. class GfmlLxr_general implements GfmlLxr, Gfo_invk {
  114. public Gfo_evt_mgr Evt_mgr() {if (evt_mgr == null) evt_mgr = new Gfo_evt_mgr(this); return evt_mgr;} Gfo_evt_mgr evt_mgr;
  115. public String Key() {return key;} private String key;
  116. public GfmlTkn CmdTkn() {return txtTkn;} public void CmdTkn_set(GfmlTkn val) {} GfmlTkn txtTkn;
  117. public String[] Hooks() {return symTrie.Symbols();}
  118. public GfmlTkn MakeTkn(CharStream stream, int firstTknLength) {
  119. GfmlTkn rv = null;
  120. if (symLxr != null) { // symLxr has something; produce
  121. rv = MakeTkn_symLxr(stream);
  122. if (rv != GfmlTkn_.IgnoreOutput) return rv;
  123. }
  124. while (stream.AtMid()) { // keep moving til (a) symChar or (b) endOfStream
  125. Object result = symTrie.FindMatch(stream);
  126. symTknLen = symTrie.LastMatchCount;
  127. if (result == null) { // no match; must be txtChar;
  128. txtBfr.Add(stream);
  129. stream.MoveNext();
  130. }
  131. else { // symChar
  132. symLxr = (GfmlLxr)result; // set symLxr for next pass
  133. if (txtBfr.Has()) // txtBfr has something: gen txtTkn
  134. rv = txtBfr.MakeTkn(stream, txtTkn);
  135. else { // txtBfr empty: gen symbol
  136. rv = MakeTkn_symLxr(stream);
  137. if (rv == GfmlTkn_.IgnoreOutput) continue;
  138. }
  139. return rv;
  140. }
  141. }
  142. if (txtBfr.Has()) // endOfStream, but txtBfr has chars
  143. return txtBfr.MakeTkn(stream, txtTkn);
  144. return GfmlTkn_.EndOfStream;
  145. }
  146. public void SubLxr_Add(GfmlLxr... lxrs) {
  147. for (GfmlLxr lxr : lxrs) {
  148. for (String hook : lxr.Hooks())
  149. symTrie.Add(hook, lxr);
  150. Gfo_evt_mgr_.Sub_same(lxr, GfmlLxr_.CmdTknChanged_evt, this);
  151. }
  152. }
  153. public GfmlLxr SubLxr() {return this;}
  154. GfmlTkn MakeTkn_symLxr(CharStream stream) {
  155. GfmlLxr lexer = symLxr; symLxr = null;
  156. int length = symTknLen; symTknLen = 0;
  157. return lexer.MakeTkn(stream, length);
  158. }
  159. GfmlLxr_general_txtBfr txtBfr = new GfmlLxr_general_txtBfr(); GfmlTrie symTrie = GfmlTrie.new_(); GfmlLxr symLxr; int symTknLen;
  160. public static GfmlLxr_general new_(String key, GfmlTkn txtTkn) {
  161. GfmlLxr_general rv = new GfmlLxr_general();
  162. rv.key = key; rv.txtTkn = txtTkn;
  163. return rv;
  164. } protected GfmlLxr_general() {}
  165. public Object Invk(GfsCtx ctx, int ikey, String k, GfoMsg m) {
  166. if (ctx.Match(k, GfmlLxr_.CmdTknChanged_evt)) {
  167. symTrie.Del(m.ReadStr("old"));
  168. symTrie.Add(m.ReadStr("new"), m.CastObj("lxr"));
  169. }
  170. else return Gfo_invk_.Rv_unhandled;
  171. return this;
  172. }
  173. }
  174. class GfmlLxr_general_txtBfr {
  175. public int Bgn = NullPos;
  176. public int Len;
  177. public boolean Has() {return Bgn != NullPos;}
  178. public void Add(CharStream stream) {
  179. if (Bgn == NullPos) Bgn = stream.Pos();
  180. Len++;
  181. } static final int NullPos = -1;
  182. public GfmlTkn MakeTkn(CharStream stream, GfmlTkn textTkn) {
  183. String raw = StringUtl.NewCharAry(stream.Ary(), Bgn, Len);
  184. Bgn = -1; Len = 0;
  185. return textTkn.MakeNew(raw, raw);
  186. }
  187. }
  188. class GfmlLxr_frame extends GfmlLxr_singleton { GfmlFrame frame; GfmlLxr endLxr, txtLxr;
  189. public void BgnRaw_set(String val) {// needed for lxr pragma
  190. GfmlBldrCmd_frameBgn bgnCmd = GfmlBldrCmd_frameBgn.new_(frame, txtLxr);
  191. GfmlTkn bgnTkn = GfmlTkn_.singleton_(this.Key() + "_bgn", val, GfmlTkn_.NullVal, bgnCmd);
  192. this.CmdTkn_set(bgnTkn);
  193. }
  194. public void EndRaw_set(String val) {// needed for lxr pragma
  195. GfmlBldrCmd_frameEnd endCmd = GfmlBldrCmd_frameEnd.data_();
  196. GfmlTkn endTkn = GfmlTkn_.singleton_(this.Key() + "_end", val, GfmlTkn_.NullVal, endCmd);
  197. endLxr.CmdTkn_set(endTkn);
  198. }
  199. public static GfmlLxr new_(String key, GfmlFrame frame, String bgn, String end, GfmlBldrCmd txtCmd, GfmlBldrCmd endCmd) {
  200. GfmlLxr_frame rv = new GfmlLxr_frame();
  201. GfmlTkn txtTkn = frame.FrameType() == GfmlFrame_.Type_comment
  202. ? GfmlTkn_.valConst_(key + "_txt", GfmlTkn_.NullVal, txtCmd)
  203. : GfmlTkn_.cmd_(key + "_txt", txtCmd)
  204. ;
  205. GfmlLxr txtLxr = GfmlLxr_.general_(key + "_txt", txtTkn);
  206. GfmlTkn bgnTkn = GfmlTkn_.singleton_(key + "_bgn", bgn, GfmlTkn_.NullVal, GfmlBldrCmd_frameBgn.new_(frame, txtLxr));
  207. rv.ctor_(key, bgn, bgnTkn, txtLxr);
  208. GfmlTkn endTkn = GfmlTkn_.singleton_(key + "_end", end, GfmlTkn_.NullVal, endCmd);
  209. GfmlLxr endLxr = GfmlLxr_.solo_(key + "_end", endTkn);
  210. rv.SubLxr_Add(endLxr);
  211. rv.frame = frame;
  212. rv.endLxr = endLxr;
  213. rv.txtLxr = txtLxr;
  214. return rv;
  215. } GfmlLxr_frame() {}
  216. public static GfmlLxr_frame as_(Object obj) {return obj instanceof GfmlLxr_frame ? (GfmlLxr_frame)obj : null;}
  217. public static GfmlLxr_frame cast(Object obj) {try {return (GfmlLxr_frame)obj;} catch(Exception exc) {throw ErrUtl.NewCast(exc, GfmlLxr_frame.class, obj);}}
  218. }