slist.tin 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455
  1. #nop vim: set filetype=tt:;
  2. /*
  3. 本文件属于 PaoTin++ 的一部分。
  4. PaoTin++ © 2020~2023 的所有版权均由担子炮(dzp <danzipao@gmail.com>) 享有并保留一切法律权利
  5. 你可以在遵照 GPLv3 协议的基础之上使用、修改及重新分发本程序。
  6. */
  7. #nop 本文件是 xtintin 的一部分,实现了一些字符串列表处理函数;
  8. ///=== {
  9. ///// 字符串列表处理函数:
  10. //
  11. // #@ slist.FromList <列表>
  12. // 将列表转换为字符串列表。
  13. // <119>注意<299>,由于 TinTin++ 自身语法的缺陷,字符串列表的第一个元素只能是
  14. // 简单字符串,不能是列表、字符串列表、表格,其余元素不受此限制。
  15. // };
  16. #func {slist.FromList} {
  17. #local list {%0};
  18. #if { &list[] == 0 } {
  19. #return {};
  20. };
  21. #local output {};
  22. #local idx {};
  23. #loop {1} {&list[]} {idx} {
  24. #local elem {$list[$idx]};
  25. #local complex 0;
  26. #local isSlist 0;
  27. #if { @slist.Size{$elem} > 1 } {
  28. #local isSlist 1;
  29. };
  30. #if { $isSlist || {$elem} == {{\{.*\}}} } {
  31. #local complex 1;
  32. };
  33. #if { $idx == 1 } {
  34. #if { $complex } {
  35. #line sub {var;escapes;secure} #cat output {$elem};
  36. };
  37. #else {
  38. #cat output {$elem};
  39. };
  40. };
  41. #elseif { $complex } {
  42. #cat output {;{$elem}};
  43. };
  44. #else {
  45. #cat output {;$elem};
  46. };
  47. };
  48. #return {$output};
  49. };
  50. ///=== {
  51. // #@ slist.Size <字符串列表>
  52. // 计算字符串列表中一共有几项内容。
  53. // };
  54. #func {slist.Size} {
  55. #local l {};
  56. #list l create {%0};
  57. #list l size result;
  58. };
  59. ///=== {
  60. // #@ slist.Append <字符串列表> <字符串>
  61. // 往字符串列表里追加新内容。
  62. // <119>注意<299>,由于 TinTin++ 自身语法的缺陷,字符串列表的第一个元素只能是
  63. // 简单字符串,不能是列表、字符串列表、表格,其余元素不受此限制。
  64. //
  65. // EXAMPLE: \@slist.Append{{a;b;c};X}
  66. // RESULT: {a;b;c;X}
  67. //
  68. // EXAMPLE: \@slist.Append{{a;b;c};{X;Y}}
  69. // RESULT: {a;b;c;{X;Y}}
  70. //
  71. // EXAMPLE: \@slist.Append{{a;b;c};{{X}{xxx}{Y}{yyy}}}
  72. // RESULT: {a;b;c;{{X}{xxx}{Y}{yyy}}}
  73. //
  74. // EXAMPLE: \@slist.Append{{};X}
  75. // RESULT: {X}
  76. // };
  77. #func {slist.Append} {
  78. #local list {%1};
  79. #local str {%2};
  80. #if { {$list} === {} } {
  81. #if { {$str} != {} } {
  82. #line sub {var;secure} #cat list $str;
  83. };
  84. };
  85. #elseif { {$str} == {%*;%*} } {
  86. #cat list {;{$str}};
  87. };
  88. #elseif { {$str} == {\{%*\}} } {
  89. #cat list {;{$str}};
  90. };
  91. #else {
  92. #cat list {;$str};
  93. };
  94. #return {$list};
  95. };
  96. ///=== {
  97. // #@ slist.Insert <字符串列表> <位置> <字符串>
  98. // 在字符串列表的指定位置插入新内容,元素位置从 1 开始计。
  99. // <119>注意<299>,由于 TinTin++ 自身语法的缺陷,字符串列表的第一个元素只能是
  100. // 简单字符串,不能是列表、字符串列表、表格,其余元素不受此限制。
  101. // };
  102. #func {slist.Insert} {
  103. #local list {%1};
  104. #local index {%2};
  105. #local str {%3};
  106. #local l {};
  107. #list l create {%1};
  108. #local l {@list.Extend{{$l};@math.Eval{$index - 1}}};
  109. #if { $index > &l[] } {
  110. #list l add {{$str}};
  111. };
  112. #else {
  113. #if { $index == 1 } {
  114. #line sub {var;secure} #format str {%s} $str;
  115. };
  116. #list l insert {$index} {$str};
  117. };
  118. #return {@slist.FromList{$l}};
  119. };
  120. ///=== {
  121. // #@ slist.Remove <字符串列表> <位置> [<N>]
  122. // 删除字符串列表中,从指定位置开始的连续 N 个元素,元素位置从 1 开始计。
  123. // N 为可选值,默认为 1。
  124. // };
  125. #func {slist.Remove} {
  126. #local index {%2};
  127. #local amount {@defaultNum{%3;1}};
  128. #local l {};
  129. #list l create {%1};
  130. #list l delete {$index} {$amount};
  131. #return {@slist.FromList{$l}};
  132. };
  133. ///=== {
  134. // #@ slist.IndexOf <字符串列表> <字符串>
  135. // 从字符串列表中查找某个字符串,并返回其位置(从 1 开始计),0 表示没找到。
  136. // };
  137. #func {slist.IndexOf} {
  138. #local list {%1};
  139. #local str {%2};
  140. #local idx {0};
  141. #local elem {};
  142. #foreach {$list} {elem} {
  143. #math idx {$idx + 1};
  144. #if { {$elem} === {$str} } {
  145. #return $idx;
  146. };
  147. };
  148. #return 0;
  149. };
  150. ///=== {
  151. // #@ slist.Find <字符串列表> <正则表达式>
  152. // 从字符串列表中按顺序查找某个模式,并返回第一个符合模式的元素的位置(从 1 开始计),0 表示没找到。
  153. // };
  154. #func {slist.Find} {
  155. #local list {%1};
  156. #local str {%2};
  157. #local idx {0};
  158. #local elem {};
  159. #foreach {$list} {elem} {
  160. #math idx {$idx + 1};
  161. #if { "$elem" == "$str" } {
  162. #return $idx;
  163. };
  164. };
  165. #return 0;
  166. };
  167. ///=== {
  168. // #@ slist.Filter <字符串列表> <正则表达式>
  169. // 遍历字符串列表,根据正则表达式过滤元素并生成新字符串列表。如果条件为真则收集,否则跳过。
  170. // 本函数与 select 作用类似。但 select 支持条件表达式,可进行逻辑运算,本函数仅支持模式匹配。
  171. // };
  172. #func {slist.Filter} {
  173. #local list {%1};
  174. #local regex {%2};
  175. #local output {};
  176. #local elem {};
  177. #foreach {$list} {elem} {
  178. #if { "$elem" == "$regex" } {
  179. #local output {@slist.Append{{$output};{$elem}}};
  180. };
  181. };
  182. #return {$output};
  183. };
  184. ///=== {
  185. // #@ slist.Contains <字符串列表> <字符串>
  186. // 判断字符串是否在字符串列表中。
  187. // };
  188. #func {slist.Contains} {#return {@math.Eval{@slist.IndexOf{{%1};{%2}} > 0 }}};
  189. ///=== {
  190. // #@ slist.Sort <字符串列表>
  191. // 按照字母顺序重新排序字符串列表中的各项。
  192. //
  193. // #@ slist.Order <字符串列表>
  194. // 按照数值顺序重新排序字符串列表中的各项。
  195. // };
  196. #func {slist.Sort} {#return {@slist.FromList{@list.Sort{@list.FromSlist{%0}}}}};
  197. #func {slist.Order} {#return {@slist.FromList{@list.Order{@list.FromSlist{%0}}}}};
  198. ///=== {
  199. // #@ slist.Reverse <字符串列表>
  200. // 按照倒序重新排列字符串列表内容。
  201. // };
  202. #func {slist.Reverse} {
  203. #return {@slist.FromList{@list.Reverse{@list.FromSlist{%0}}}};
  204. };
  205. ///=== {
  206. // #@ slist.Get <字符串列表> <下标>
  207. // 从字符串列表中按照下标取出一个值。
  208. // };
  209. #func {slist.Get} {
  210. #local list {%1};
  211. #local index {%2};
  212. #local index {@math.Floor{@defaultNum{$index;0}}};
  213. #local n {0};
  214. #local elem {};
  215. #local selected {};
  216. #foreach {$list} {elem} {
  217. #math n {$n + 1};
  218. #if { $index > 0 } {
  219. #if { $n == $index } {
  220. #local selected {$elem};
  221. #if { $index == 1 } {
  222. #line sub {var;escapes} #format selected {%s} {$selected};
  223. };
  224. #break;
  225. };
  226. };
  227. #else {
  228. #math random {1 d $n};
  229. #if { $random == 1 } {
  230. #local selected {$elem};
  231. };
  232. };
  233. };
  234. #return {$selected};
  235. };
  236. ///=== {
  237. // #@ slist.RandomGet <字符串列表>
  238. // 从字符串列表中随机取出一个值。
  239. // };
  240. #func {slist.RandomGet} {
  241. #return {@slist.Get{{%0};}};
  242. };
  243. ///=== {
  244. // #@ slist.Range <开始值> <结束值> [<步长>]
  245. // 生成等差数列,从开始值开始,一直到不超过结束值为止。结果用字符串列表表示。
  246. // 可选的步长用来决定公差,默认为 1。
  247. // 如果开始值小于结束值,则步长自动取正值。
  248. // 如果开始值大于结束值,则步长自动取其负值(如果参数为正则取相反数否则不变)。
  249. // };
  250. #func {slist.Range} {
  251. #local begin {%1};
  252. #local end {%2};
  253. #local delta {@defaultNum{%3;1}};
  254. #local op {<=};
  255. #local delta {@math.Abs{$delta}};
  256. #if { $begin > $end } {
  257. #local op {>=};
  258. #local delta {@math.Eval{- $delta}};
  259. };
  260. #local list {$begin};
  261. #math begin {$begin + $delta};
  262. #while { $begin $op $end } {
  263. #cat list {;$begin};
  264. #math begin {$begin + $delta};
  265. };
  266. #return {$list};
  267. };
  268. ///=== {
  269. // #@ slist.Join <字符串列表> <分隔符> [<是否保留空元素>]
  270. // 将字符串列表的每个元素用分隔符连接在一起。返回连接后的字符串。
  271. // 第三个参数是可选的,用来指定是否保留字符串列表中的空元素。
  272. // 默认不保留,如果想要保留请设置为 true。
  273. // };
  274. #func {slist.Join} {#return {@slist.JoinAs{{%1};;{%2};%3}}};
  275. ///=== {
  276. // #@ slist.JoinAs <字符串列表> <格式串> <分隔符> [<是否保留空元素>]
  277. // 将字符串列表的每个元素用格式串渲染之后,再用分隔符连接在一起。
  278. // 返回连接后的字符串。
  279. // 格式串默认为 %s。
  280. // 第四个参数是可选的,用来指定是否保留字符串列表中的空元素。
  281. // 默认不保留,如果想要保留请设置为 true。
  282. // };
  283. #func {slist.JoinAs} {
  284. #local strList {%1};
  285. #local format {@default{%2;%s}};
  286. #local sep {%3};
  287. #local keepNull {@isTrue{@default{%4;false}}};
  288. #local str {};
  289. #local first {1};
  290. #local elem {};
  291. #foreach {$strList} {elem} {
  292. #if { @isEmpty{$elem} && !$keepNull } {
  293. #continue;
  294. };
  295. #format elem {$format} {$elem};
  296. #if $first {
  297. #local str {$elem};
  298. #local first {0};
  299. };
  300. #else {
  301. #cat str {$sep$elem};
  302. };
  303. };
  304. #return {$str};
  305. };
  306. ///=== {
  307. // #@ slist.Zip <列表1> <列表2> [...]
  308. // 将多个列表合并成一个列表。结果列表的每个元素分别对应参数列表中对应位置的元素。
  309. // 各个参数列表中的元素将按顺序一一对应,以分号分隔连接到一起成为结果列表的元素。
  310. // };
  311. #func {slist.Zip} {
  312. #info arguments save;
  313. #unvar info[ARGUMENTS][0];
  314. #local argv {$info[ARGUMENTS]};
  315. #if { &argv[] < 2 } {
  316. #return {%1};
  317. };
  318. #local i {};
  319. #local maxSize {0};
  320. #loop {1} {&argv[]} {i} {
  321. #local list {};
  322. #list list create {$argv[+$i]};
  323. #local {argv[+$i]} {$list};
  324. #local size {&list[]};
  325. #if { $size > $maxSize } {
  326. #local maxSize $size;
  327. };
  328. };
  329. #if { $maxSize < 1 } {
  330. #return {};
  331. };
  332. #local zipped {};
  333. #loop {1} {$maxSize} {i} {
  334. #local n {};
  335. #local l {};
  336. #loop {1} {&argv[]} {n} {
  337. #if { $i > &argv[+$n][] } {
  338. #list l add {{}};
  339. };
  340. #else {
  341. #list l add {{$argv[+$n][$i]}};
  342. };
  343. };
  344. #list zipped {add} {{@slist.FromList{$l}}};
  345. };
  346. #return {@slist.FromList{$zipped}};
  347. };
  348. ///=== {
  349. // #@ slist.ZipAs <格式串> <列表1> <列表2> [...]
  350. // 将多个列表合并成一个列表。结果列表的每个元素分别对应参数列表中对应位置的元素。
  351. // 合并时使用格式串对所有参数列表中的元素进行格式化合并。
  352. // };
  353. #func {slist.ZipAs} {
  354. #info arguments save;
  355. #unvar info[ARGUMENTS][0];
  356. #local argv {$info[ARGUMENTS]};
  357. #local format {$argv[1]};
  358. #list argv delete {1};
  359. #if { &argv[] < 1 } {
  360. #return {};
  361. };
  362. #local i {};
  363. #local maxSize {0};
  364. #loop {1} {&argv[]} {i} {
  365. #local list {};
  366. #list list create {$argv[+$i]};
  367. #local {argv[+$i]} {$list};
  368. #local size {&list[]};
  369. #if { $size > $maxSize } {
  370. #local maxSize $size;
  371. };
  372. };
  373. #if { $maxSize < 1 } {
  374. #return {};
  375. };
  376. #local zipped {};
  377. #loop {1} {$maxSize} {i} {
  378. #local n {};
  379. #local l {};
  380. #loop {1} {&argv[]} {n} {
  381. #if { $i > &argv[+$n][] } {
  382. #list l add {{}};
  383. };
  384. #else {
  385. #list l add {{$argv[+$n][$i]}};
  386. };
  387. };
  388. #local value {};
  389. #line sub var #format value {$format} $l[];
  390. #list zipped {add} {{$value}};
  391. };
  392. #return {@slist.FromList{$zipped}};
  393. };