-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathprooftrees.sty
929 lines (926 loc) · 42.9 KB
/
prooftrees.sty
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
%% Copyright 2016-2024 Clea F. Rees
%%
%% This work may be distributed and/or modified under the
%% conditions of the LaTeX Project Public License, either version 1.3c
%% of this license or (at your option) any later version.
%% The latest version of this license is in
%% https://www.latex-project.org/lppl.txt
%% and version 1.3c or later is part of all distributions of LaTeX
%% version 2008-05-04 or later.
%%
%% This work has the LPPL maintenance status `maintained'.
%%
%% The Current Maintainer of this work is Clea F. Rees.
%%
%% This file may only be distributed together with a copy of the package
%% prooftrees.
%%
%% This work consists of all files listed in manifest.txt.
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\NeedsTeXFormat{LaTeX2e}
\RequirePackage{svn-prov}
\ProvidesPackageSVN{$Id: prooftrees.sty 10522 2024-10-23 16:31:08Z cfrees $}[v0.9 \revinfo]
% define \prooftrees@enw to hold the name of the environment
% default is to name the environment prooftree, this ensures backwards compatibility
\newcommand*\prooftrees@enw{prooftree}
% allow users to change the name to tableau using tableaux
\DeclareOption{tableaux}{\renewcommand*\prooftrees@enw{tableau}}
% just in case
\DeclareOption{tableau}{\renewcommand*\prooftrees@enw{tableau}}
\DeclareOption*{\PassOptionsToPackage{\CurrentOption}{forest}}
% if \prooftree is not yet defined, set the name to prooftree; otherwise, use tableau to avoid conflict with bussproofs (which uses 'prooftree' rather than 'bussproof' as one might expect)
\ifcsname prooftree\endcsname
\renewcommand*\prooftrees@enw{tableau}%
\else
\renewcommand*\prooftrees@enw{prooftree}%
\fi
% \ifundef\prooftree{\renewcommand*\prooftrees@enw{prooftree}}{\renewcommand*\prooftrees@enw{tableau}}
% let users override the default prooftree in case they need to load bussproofs later
\ProcessOptions
\RequirePackage{forest}[2016/12/04]
\RequirePackage{amssymb,amstext}
\newcommand*\linenumberstyle[1]{#1.}
% currently, keys starting 'proof tree' and macros starting 'prooftree' or 'prooftree@' are intended for internal use only
% this does not apply to the environment prooftree
% other keys and macros are intended for use in documents
% in particular, the style 'proof tree' is **NOT** intended to be used directly by the user and its direct use is **ABSOLUTELY NOT SUPPORTED IN ANY WAY, SHAPE OR FORM**; it is intended only for implicit use when the prooftree environment calls it
\forestset{% don't use @ in register/option names - the documentation is lying when it says non-alphanumerics will be converted to underscores when forming pgfmath functions ;)
declare boolean register={line numbering},% line numbers
line numbering,% default is for line numbers
declare boolean register={justifications},% line justifications
not justifications,% default is for no line justifications (b/c there's no point in enabling this if the user doesn't specify any content)
declare boolean register={single branches},% single branches: explicitly drawn branches and a normal level distance between lone children and their parents
not single branches,% default is for lone children to be grouped with their parents
declare boolean register={auto move},% ble mae'n bosibl, symud pethau'n awtomatig
auto move,% default: symud yn awtomatig
declare dimen register={line no width},% default will be set to the width of 99 wrapped in the line numbering style
line no width'=0pt,% fallback default is 0pt
declare dimen register={just sep},% amount by which to shift justifications away from the main tree
just sep'=1.5em,% default is 1.5em
declare dimen register={just dist},% distance of justifications from centre of inner tree; overrides just sep
just dist'=0pt,
declare dimen register={line no sep},% amount by which to shift line numbers away from the main tree
line no sep'=1.5em,
declare dimen register={line no dist},% distance of line nos. from centre of inner tree; overrides line no sep
line no dist'=0pt,
declare dimen register={close sep},% distance between closure symbols and any following annotation
close sep'=.75\baselineskip,
declare dimen register={proof tree line no x},
proof tree line no x'=0pt,
declare dimen register={proof tree justification x},
proof tree justification x'=0pt,
declare dimen register={proof tree inner proof width},
proof tree inner proof width'=0pt,
declare dimen register={proof tree inner proof midpoint},
proof tree inner proof midpoint'=0pt,
declare count register={proof tree rhif lefelau},% count the levels in the proof tree
proof tree rhif lefelau'=0,
declare count register={proof tree lcount},% count the line numbers (on the left)
proof tree lcount'=0,
declare count register={proof tree jcount},% count the justifications (on the right)
proof tree jcount'=0,
declare count register={line no shift},% adjustment for line numbering
line no shift'=0,
declare count register={proof tree aros},
proof tree aros'=0,
declare toks register={check with},
check with={\ensuremath{\checkmark}},
declare boolean register={check right},
check right,
check left/.style={not check right},
declare toks register={subs with},
subs with={\ensuremath{\backslash}},
declare boolean register={subs right},
subs right,
subs left/.style={not subs right},
declare toks register={close with},
close with={\ensuremath{\otimes}},
declare keylist register={close format},
close format={font=\scriptsize},
declare keylist register={close with format},
close with format={},
declare toks register={merge delimiter},
merge delimiter={\text{; }},
declare boolean register={just refs left},
just refs left,
just refs right/.style={not just refs left},
declare keylist register={just format},
just format={},
declare keylist register={line no format},
line no format={},
declare autowrapped toks register={highlight format},
highlight format={draw=gray, rounded corners},
declare keylist register={proof statement format},
proof statement format={},
declare keylist register={wff format},
wff format={},
declare boolean={proof tree justification}{0},
declare boolean={proof tree line number}{0},
declare boolean={grouped}{0},
declare boolean={proof tree phantom}{0},
declare boolean={highlight wff}{0},
declare boolean={highlight just}{0},
declare boolean={highlight line no}{0},
declare boolean={highlight line}{0},
Autoforward={highlight line}{highlight just, highlight wff, highlight line no},
declare boolean={proof tree toing}{0},
declare boolean={proof tree toing with}{0},
declare boolean={proof tree rhiant cymysg}{0},
declare boolean={proof tree rhifo}{1},
declare boolean={proof tree arweinydd}{0},
declare autowrapped toks={just}{},
declare toks={proof tree rhestr rhifau llinellau}{},
declare toks={proof tree close}{},
declare toks={proof tree rhestr rhifau llinellau cau}{},
declare autowrapped toks={just options}{},
declare autowrapped toks={line no options}{},
declare autowrapped toks={wff options}{},
declare autowrapped toks={line options}{},
Autoforward={line options}{just options={#1}, line no options={#1}, wff options={#1}},
declare count={proof tree toing by}{0},
declare count={proof tree cadw toing by}{0},
declare count={proof tree toooing}{0},
declare count={proof tree proof line no}{0},
% keylists for internal storage
declare keylist={proof tree jrefs}{},
declare keylist={proof tree crefs}{},
% keylists for use in stages
declare keylist={proof tree ffurf}{},
declare keylist={proof tree symud awto}{},
declare keylist={proof tree creu nodiadau}{},
declare keylist={proof tree nodiadau}{},
% > not documented yet, I think
% > now indicates use of process when it is the first token, preceding a list of instructions as opposed to pgfmath stuff
define long step={proof tree symud}{}{%
root,sort by={>{O}{level},>{_O<}{1}{n children}},sort'=descendants
},
define long step={proof tree cywiro symud}{}{%
root,if line numbering={n=2}{n=1},sort by={>{O}{level},>{_O<}{1}{n children}},sort'=descendants
},
define long step={proof tree camau}{}{% updated version of defn. from saso's code (forest2-saso-ptsz.tex) & http://chat.stackexchange.com/transcript/message/28321501#28321501
root,sort by={>{O}{y},>{Ow1+d}{x}{-##1}},sort'={filter={descendants}{>{OO!&}{proof tree rhifo}{proof tree phantom}}}% angen +d - gweler http://chat.stackexchange.com/transcript/message/28607212#28607212
},
define long step={proof tree wffs}{}{% coeden brif yn unig ar ôl i greu nodiadau
fake=root,if line numbering={n=2}{n=1},tree
},
checked/.style={% mark discharge with optional name substituted into existential
delay={%
if check right={%
content+'={\ \forestregister{check with}#1},
}{%
+content'={\forestregister{check with}#1\ },
},
},
},
subs/.style={% mark substitution of name into universal
delay={%
if subs right={%
content+'={\ \forestregister{subs with}#1},
}{%
+content'={\forestregister{subs with}#1\ },
},
},
},
close/.style={% this now uses nodes rather than a label to accommodate annotations; closing must be done before packing the tree to ensure that sufficient space is allowed for the symbol and any following annotation; the annotations must be processed before anything is moved to ensure that the correct line numbers are used later, even if the references are given as relative node names
if={%
>{__=}{#1}{}%
}{}{%
temptoksb={},
temptoksa={#1},
split register={temptoksa}{:}{proof tree close,temptoksb},
if temptoksb={}{}{%
split register={temptoksb}{,}{proof tree cref},
},
},
delay={%
append={% this node holds the closure symbol
[\forestregister{close with},
not proof tree rhifo,
proof tree phantom,
grouped,
no edge,
process keylist register=close with format,
before computing xy={% adjust the distance between the closure symbol and any annotation
delay={%
l'=\baselineskip,% cywiro? fel arall, bydda'r peth byth yn cael ei wneud achos proof tree phantom? dim yn siwr o gwbl
for children={%
l/.register=close sep,
},
},
},
before drawing tree={%
if={>{RR|}{line numbering}{justifications}}{%
proof tree proof line no/.option=!parent.proof tree proof line no,
}{},
},
if={%
>{__=}{#1}{}%
}{}{% don't create a second node if there's no annotation
delay={%
append={% this node holds the annotation, possibly including cross-references which will be relative to the node's grandparent
[,
not proof tree rhifo,
proof tree phantom,
grouped,
no edge,
process keylist register=close format,
if={%
>{O_=}{!parent,parent.proof tree close}{}%
}{}{content/.option=!{parent,parent}.proof tree close},
proof tree crefs/.option=!{parent,parent}.proof tree crefs,
delay={%
!{parent,parent}.proof tree crefs'={},
},
before drawing tree={%
if={>{RR|}{line numbering}{justifications}}{%
proof tree proof line no/.option=!{parent,parent}.proof tree proof line no,
}{},
},
]%
},
},
},
]%
},
},
},
proof tree line no/.style={% creates the line numbers on the left; note that it *does* matter that these are part of the tree, even though they do not need to be packed or to have xy computed; moreover, it matters that each is the child of the previous line number... so it won't do for them to *remain* siblings, even though that's fine when they are created.
anchor=base west,
no edge,
proof tree line number,
text width/.register=line no width,
x'/.register=proof tree line no x,
process keylist register=line no format,
delay={%
proof tree lcount'+=1,
tempcounta/.process={RRw2+n}{proof tree lcount}{line no shift}{##1+##2},
content/.process={Rw1}{tempcounta}{\linenumberstyle{##1}},% content i.e. the line number
name/.expanded={line no \foresteregister{tempcounta}},% name them so they can be moved later
typeset node,
if proof tree lcount>=3{% the initial location of most line numbers is incorrect and they must be moved
for previous={% move the line number below the previous line number
append/.expanded={line no \foresteregister{tempcounta}}
},
}{},
},
},
proof tree line justification/.style={% creates the justifications on the right but does not yet specify any content
anchor=base west,
no edge,
proof tree justification,
x'/.register=proof tree justification x,
process keylist register=just format,
delay={%
proof tree jcount'+=1,
tempcounta/.process={RRw2+n}{proof tree jcount}{line no shift}{##1+##2},
name/.expanded={just \foresteregister{tempcounta}},% name them so they can be moved
typeset node,% angen i osgoi broblemau 'da highlight just/line etc.
if proof tree jcount>=3{% correct the location as for the line numbers (cf. line no style)
for previous={%
append/.expanded={just \foresteregister{tempcounta}},
},
}{},
},
},
zero start/.style={%
line no shift'+=-1,
},
to prove/.style={% sets a proof statement
for root={%
before typesetting nodes={%
content={#1},
phantom=false,
baseline,
if line numbering={anchor=base west}{anchor=base},
process keylist register=proof statement format
},
before computing xy={%
delay={%
for children={%
l=1.5*\baselineskip,
},
},
},
},
},
proof tree/.style={% this style should **NOT** be used directly in a forest environment - see notes at top of this file
for tree={%
parent anchor=children,% manual 64
child anchor=parent,% manual 64
math content,
delay={%
if just={}{}{% if we've got justifications, make sure nodes are created for them later and split out cross-references so we identify the correct nodes before anything gets moved, allowing the use of relative node names
justifications,
temptoksa={},
split option={just}{:}{just,temptoksa},
if temptoksa={}{}{%
split register={temptoksa}{,}{proof tree jref},
},
},
if content={}{% if there's no proof statement
if level=0{}{%
shape=coordinate,
},
}{},
},
},
where level=0{%
for children={% no edges from phantom root or proof statement to children
before typesetting nodes={%
no edge,
},
},
delay={%
if content={}{phantom}{},
if line numbering={% create the line numbers if appropriate
parent anchor=south west,
if line no width={0pt}{%
line no width/.pgfmath={width("\noexpand\linenumberstyle{99}")},
}{},
}{},
},
proof tree creu nodiadau={% this is processed after computing xy
if={>{RR|}{line numbering}{justifications}}{% count proof lines if necessary
proof tree rhif lefelau'/.register=line no shift,
for proof tree camau={%
if level>=1{%
if={%
>{OO<}{y}{!back.y}%
}{%
proof tree rhif lefelau'+=1,
proof tree proof line no'/.register=proof tree rhif lefelau,
}{%
proof tree proof line no'/.register=proof tree rhif lefelau
},
}{},
},
proof tree inner proof midpoint/.min={>{OOw2+d}{x}{min x}{##1+##2}}{fake=root,descendants},
proof tree inner proof width/.max={>{OOw2+d}{x}{max x}{##1+##2}}{fake=root,descendants},
proof tree inner proof width-/.register=proof tree inner proof midpoint,
proof tree inner proof midpoint+/.process={Rw+d{proof tree inner proof width}{##1/2}},
}{},
if line numbering={% get the x position of line numbers and adjust the location and alignment of the proof statement
proof tree line no x/.min={>{OOw2+d}{x}{min x}{##1+##2}}{fake=root,descendants},
if={%
> Rd= {line no dist}{0pt}%
}{%
proof tree line no x-/.register=line no sep,
}{%
tempdima/.register=proof tree inner proof width,
tempdima:=2,
if={%
> RR< {line no dist}{tempdima}%
}{}{%
proof tree line no x/.register=proof tree inner proof midpoint,
proof tree line no x-/.register=line no dist,
},
},
proof tree line no x-/.register=line no width,
for root={%
tempdimc/.option=x,
x'+/.register=proof tree line no x,
x'-/.option=min x,
},
prepend={% create line numbers on left
[,
proof tree line no,
% () to group are required here - otherwise, the -1 (or -2 or whatever) is silently ignored
repeat={((proof_tree_rhif_lefelau)-1)-(line_no_shift)}{% most are created in the wrong place but proof tree line no moves them later
delay n={proof_tree_lcount}{
append={[, proof tree line no]},
},
},
]%
},
}{},
if justifications={% get the x position of justifications and create the nodes which will hold the justification content, if required
proof tree justification x/.max={>{OOw2+d}{x}{max x}{##1+##2}}{fake=root,descendants},
if={%
> Rd= {just dist}{0pt}%
}{%
proof tree justification x+/.register=just sep,
}{%
tempdima/.register=proof tree inner proof width,
tempdima:=2,
if={%
> RR< {just dist}{tempdima}%
}{}{%
proof tree justification x/.register=proof tree inner proof midpoint,
proof tree justification x+/.register=just dist,
},
},
append={%
[,
proof tree line justification,
repeat={((proof_tree_rhif_lefelau)-1)-(line_no_shift)}{% most are created in the wrong place but proof tree line justification moves them later
delay n={proof_tree_jcount}{%
append={[, proof tree line justification]},
},
}%
]%
},
}{},
},
}{%
delay={%
if single branches={}{% automatically group lines if not using single branches
if n children=1{%
for children={%
grouped,
},
}{},
},
},
before typesetting nodes={% apply wff-specific highlighting and additional TikZ keys
process keylist register=wff format,
if highlight wff={node options/.register=highlight format}{},
node options/.option=wff options,
},
},
proof tree ffurf={% processed before proof tree symud auto: adjusts the alignment of lines when some levels of the tree are grouped together either whenever the number of children is only 1 or by applying the grouped style to particular nodes when specifying the tree
if auto move={%
if single branches={%
where={%
>{O! _O< O &&}{grouped}{2}{level}{proof tree rhifo}%
}{%
if={%
>{_O= _O< &}{1}{!parent.n children}{1}{!parent,parent.n children}%
}{%
not tempboola,
for root/.process={Ow1}{level}{%
for level={##1}{%
if={%
>{_O< _O= &}{1}{!parent.n children}{1}{n}%
}{%
tempboola,
}{},
},
},
if tempboola={%
proof tree toing,
}{},
}{},
}{},
}{},
where={%
>{O _O< O &&}{grouped}{1}{level}{proof tree rhifo}%
}{% this searches for certain kinds of structural asymmetry in the tree and attempts to move lines appropriately in such cases - the algorithm is intended to be relatively conservative (not in the sense of 'cautious' or 'safe' but in the sense of 'reflection of the overlapping consensus of reasonable users' / 'what would be rationally agreed behind the prooftrees veil of ignorance'; however, I should have realised I actually had 'the overlapping concensus of reasonable Beamer users' in mind rather than 'the overlapping consensus of reasonable users', so there is now an option to turn it off;apologies if this comment previously misclassified you as 'unreasonable'; apologies for the inconvenience if you are an unreasonable user)
not tempboola,
for root/.process={Ow1}{level}{%
for level={##1}{%
if={%
>{_O< _O= &}{1}{!parent.n children}{1}{n}%
}{%
tempboola,
}{},
},
},% Sašo: http://chat.stackexchange.com/transcript/message/27874731#27874731, see also http://chat.stackexchange.com/transcript/message/27874722#27874722
if tempboola={%
if n children=0{%
if={>{OO|}{!parent.proof tree toing}{!parent.proof tree toing with}}{% we're already moving the parent and the child will move with the parent, so we can just mark this and do nothing else
proof tree toing with,
}{%
for root/.process={Ow1}{level}{% don't move a terminal node even in case of asymmetry: instead, create a separate proof line for terminal nodes on this level which are only children, by moving children with siblings on this level down a proof line, without altering their physical location
% this makes the tree more compact and stops it looking silly
for level={##1}{%
if={%
>{_O< _O= &}{1}{!parent.n children}{1}{n}%
}{% this just serves to keep the levels nice for the sub-tree and ensure things align. We need this because we want to skip a level here to allow room for the terminal node in the other branch
for parent={%
if proof tree rhiant cymysg={}{% we mark the parent to avoid increasing the line number of its descendants more than once
proof tree rhiant cymysg,
for descendants={%
proof tree toing by'+=1,
},
},
},
}{},
},
},% Sašo: http://chat.stackexchange.com/transcript/message/27874731#27874731, see also http://chat.stackexchange.com/transcript/message/27874722#27874722
},
no edge,
}{%
if={%
>{_O= _O< &}{1}{!parent.n children}{1}{!parent,parent.n children}%
}{% don't try to move if the node has more than 1 child or the grandparent has no more than that; otherwise, mark the node as one to move - we figure out where to move it later
proof tree toing,
}{no edge},
},
}{no edge},
}{},
}{},
},
proof tree symud awto={% processed before typesetting nodes: if _this_ could be done during packing, that would be very nice, even if the previous stuff can't be
if auto move={%
proof tree aros'=0,
for proof tree symud={%
if proof tree toing={% this relies on an experimental feature of forest, which is anffodus
for nodewalk={fake=parent,fake=sibling,descendants}{do dynamics},
delay n={\foresteregister{proof tree aros}}{%
tempcounta/.max={>{OOOOw4+n}{level}{proof tree toing by}{proof tree toooing}{proof tree rhifo}{(##1+##2+##3)*##4}}{parent,sibling,descendants},
if tempcounta>=1{%
if={%
>{Rw1+n OOw2+n >}{tempcounta}{##1+1}{level}{proof tree toing by}{##1+##2}%
}{%
tempcounta-/.option=level,
tempcounta'+=1,
move by/.register=tempcounta,
}{no edge},
}{no edge},
},
proof tree aros'+=4,
}{},
},
}{},
},
proof tree nodiadau={% processed after proof tree creu nodiadau and before before drawing tree: creates annotation content which may include cross-references, applies highlighting and additional TikZ keys to line numbers, justifications and to wffs where specified for entire proof lines
where proof tree crefs={}{}{% resolve cross-refs in closures
split option={proof tree crefs}{,}{proof tree rhif llinell cau},
if content={}{%
content/.option=proof tree rhestr rhifau llinellau cau,
}{%
content+/.process={_O}{\ }{proof tree rhestr rhifau llinellau cau},
},
typeset node,
},
if line numbering={% apply highlighting and additional TikZ keys to line numbers; initial alignment of numbers with proof lines
for proof tree wffs={%
if highlight line no={%
for name/.process={Ow1OOOw3}{proof tree proof line no}{line no ##1}{proof tree proof line no}{line no options}{y}{% from Sašo's anti-pgfmath version - rhaid ddweud proof tree proof line no yn ddwywaith ?! dim yn bosibl i ailddefnyddio'r gyntaf ?!
node options/.register=highlight format,
##2,
y'=##3,
proof tree proof line no'=##1,
typeset node,
}%
}{%
if line no options={}{%
if proof tree phantom={}{%
for name/.process={Ow1OOw2}{proof tree proof line no}{line no ##1}{proof tree proof line no}{y}{%
y'=##2,
proof tree proof line no'=##1,
}%
},
}{%
for name/.process={Ow1OOOw3}{proof tree proof line no}{line no ##1}{proof tree proof line no}{line no options}{y}{%
##2,
y'=##3,
proof tree proof line no'=##1,
typeset node,
}%
},
},
},
}{},
if justifications={% initial alignment of justifications with proof lines, addition of content, resolution of cross-references and application of highlighting and additional TikZ keys
for proof tree wffs={%
if just={}{%
if proof tree phantom={}{%
for name/.process={Ow1OOw2}{proof tree proof line no}{just ##1}{proof tree proof line no}{y}{% from Sašo's anti-pgfmath version - rhaid ddweud proof tree proof line no yn ddwywaith ?! dim yn bosibl i ailddefnyddio'r gyntaf ?!
y'=##2,
proof tree proof line no'=##1,
}%
},
}{% puts the content of the justifications into the empty justification nodes on the right; because this is done late, the nodes need to be typeset again
if proof tree jrefs={}{}{% resolve cross-refs in justifications
split option={proof tree jrefs}{,}{proof tree rhif llinell},
if just refs left={%
+just/.process={O_}{proof tree rhestr rhifau llinellau}{\ },
}{%
just+/.process={_O}{\ }{proof tree rhestr rhifau llinellau},
},
},
if highlight just={% apply highlighting and additional TikZ keys to justifications, set content and merge any conflicting specifications, warning user if appropriate
for name/.process={Ow1OOOOw4}{proof tree proof line no}{just ##1}{proof tree proof line no}{just}{just options}{y}{% from Sašo's anti-pgfmath version - rhaid ddweud proof tree proof line no yn ddwywaith ?! dim yn bosibl i ailddefnyddio'r gyntaf ?!
if={%
>{O_= O_= |}{content}{}{content}{##2}%
}{% gweler isod - o gôd Sašo
content={##2},
}{%
content+'={\foresteregister{merge delimiter}##2},
TeX={\PackageWarning{prooftrees}{Merging conflicting justifications for line ##1! Please examine the output carefully and use "move by" to move lines later in the proof if required. Details of how to do this are included in the documentation.}},
},
node options/.register=highlight format,
##3,
y'=##4,
proof tree proof line no'=##1,
typeset node,
}% do NOT put a comma here!
}{%
for name/.process={Ow1OOOOw4}{proof tree proof line no}{just ##1}{proof tree proof line no}{just}{just options}{y}{% from Sašo's anti-pgfmath version - rhaid ddweud proof tree proof line no yn ddwywaith ?! dim yn bosibl i ailddefnyddio'r gyntaf ?!
if={% from Sašo's anti-pgfmath version - I appreciate this is faster, but why is it *required*?!
>{O_= O_= |}{content}{}{content}{##2}%
}{%
content={##2},
}{%
content+'={\foresteregister{merge delimiter}##2},
TeX={\PackageWarning{prooftrees}{Merging conflicting justifications for line ##1! Please examine the output carefully and use "move by" to move lines later in the proof if required. Details of how to do this are included in the documentation.}},
},
##3,
y'=##4,
proof tree proof line no'=##1,
typeset node,
}% do NOT put a comma here!
}
},
},
}{},
for proof tree wffs={% apply highlighting and TikZ keys which are specified for whole proof lines to all applicable wffs
if proof tree phantom={}{%
if highlight line={%
for proof tree wffs/.process={OOw2}{proof tree proof line no}{line options}{%
if proof tree proof line no={##1}{%
node options/.register=highlight format,
##2,
}{}%
},
}{%
for proof tree wffs/.process={OOw2}{proof tree proof line no}{line options}{%
if proof tree proof line no={##1}{##2}{},
},
},
delay={typeset node},
},
},
},
before packing={% initial alignment so we don't get proof line numbers incrementing due to varying height/depth of nodes, for example - when single branches is true and few nodes are grouped, this is also a reasonable first approximation
for tree={%
tier/.process={OOw2+nw1}{level}{proof tree toing by}{##1+##2}{tier ##1},
},
for root={% if there's no proof statement, adjust the alignment of the proof relative to the surrounding text
if content={}{%
!{n=1}.baseline,
}{},
},
},
before computing xy={% adjust distance between levels for grouped nodes after tree is packed
for tree={%
if={%
>{O _O< &}{grouped}{1}{level}%
}{% osgoi overlapping nodes, if posibl: cwestiwn https://tex.stackexchange.com/q/456254/
not tempboola,
tempcounta/.option=level,
tempcountb/.option=proof tree toing,
tempcountb+/.option=proof tree toooing,
for nodewalk={fake=root, descendants}{if={> RO= On> O! O! OOw2+nR= &&&&
{tempcounta}{level} {!u.n children}{1} {proof tree arweinydd} {proof tree phantom} {proof tree toing by} {proof tree toooing}{##1+##2} {tempcountb}
}{tempboola}{}},
if tempboola={}{l'=\baselineskip},
}{},
},
},
before drawing tree={% set final alignment for proof lines which have been moved by effectively grouping lead nodes and moving their subtrees accordingly - this requires that each line number and justification be the child of the previous one and that if justifications are used at all, then justifications exist for all proof lines, even if empty
if={>{RR|R!&}{line numbering}{justifications}{single branches}}{% correct the alignment of move by lines when single branches is false - o fersiwn anti-pgfmath Sašo
tempdimc'=0pt,% track cumulative adjustments to line numbers and justifications
for proof tree cywiro symud={%
if proof tree arweinydd={% only examine the lead nodes - their descendants need the same (cumulative) adjustments
tempdima'/.option=y,
if line numbering={% if there are line numbers, we use the previous line number's vertical position
for name/.process={Ow1+nw1}{proof tree proof line no}{##1-1}{line no ##1}{% arafach ?
tempdimb'/.option=y,
}%
}{% if not, we use the previous justification's vertical position
for name/.process={Ow1+nw1}{proof tree proof line no}{##1-1}{just ##1}{% arafach ?
tempdimb'/.option=y,
}%
},
for parent={% the parent (which will be a phantom) gets aligned with the previous line
y'/.register=tempdimb,
},
if tempdimb<={0pt}{% adjust so we align this line below the previous one (assuming we're going down)
tempdimb'-=\baselineskip,
}{%
tempdimb'+=\baselineskip,
},
tempdimb'-/.register=tempdima,% how far are we moving?
for tree={% adjust this node and all descendants
y'+/.register=tempdimb,
},
tempdimb'-/.register=tempdimc,% deduct any tracked cumulative adjustments to line numbers and justifications
if line numbering={% adjust the line numbers, if any
for name/.process={Ow1}{proof tree proof line no}{line no ##1}{%
for tree={%
y'+/.register=tempdimb,
},
}%
}{},
if justifications={% adjust the justifications, if any
for name/.process={Ow1}{proof tree proof line no}{just ##1}{% t. 60 manual 2.1 rc1
for tree={%
y'+/.register=tempdimb,
},
}%
}{},
tempdimc'/.register=tempdimb,% add the adjustment just implemented to the tracked cumulative adjustments for line numbers and/or justifications
}{},
},
}{},
if={%
> RR| {auto move}{single branches}%
}{}{%
where proof tree arweinydd={%
for nodewalk={%
save append={proof tree walk}{%
current,
do until={%
> O+t_+t=! {content}{}%
}{parent}%
}%
}{},
}{},
where level>=1{%
if grouped={%
if in saved nodewalk={current}{proof tree walk}{}{%
no edge,
},
}{},
}{},
},
},
},
move by/.style={% this implements both the automated moves prooftrees finds necessary and any additional moves requested by the user - more accurately, it implements initial moves, which may get corrected later (e.g. to avoid skipping numbers or creating empty proof lines, which we assume aren't wanted)
if={
>{_n<}{0}{#1}%
}{% only try to move the node if the target line number exceeds the one i.e. the line number is to be positively incremented
proof tree cadw toing by/.option=proof tree toing by,
proof tree arweinydd,
for tree={%
if={%
>{_n<}{1}{#1}%
}{% track skipped lines for which we won't be creating phantom nodes
proof tree toing by+=#1-2,
proof tree toooing'+=1,
}{},
},
delay={%
replace by={% insert our first phantom
[,
if={%
>{_n<}{1}{#1}%
}{%
child anchor=parent,
parent anchor=parent,
}{%
child anchor=children,
parent anchor=children,
},
proof tree phantom,
edge path/.option=!last dynamic node.edge path,% Sašo Živanović: http://chat.stackexchange.com/transcript/message/27990955#27990955
edge/.option=!last dynamic node.edge,
append,
before drawing tree={%
if={>{RR|}{line numbering}{justifications}}{%
proof tree proof line no/.process={Ow1+n}{!parent.proof tree proof line no}{##1+1},
}{},
},
if={%
>{_n<}{1}{#1}%
}{% if we are moving by more than 1, we insert a second phantom so that a node with siblings which is moved a long way will not get a unidirectional edge but an edge which looks similar to others in the tree (by default, sloping down a line or so and then plummeting straight down rather than a sharply-angled steep descent)
delay={%
append={%
[,
child anchor=parent,
parent anchor=parent,
proof tree toing by=#1-2+proof_tree_cadw_toing_by,
proof tree phantom,
edge path/.option=!u.edge path,
edge/.option=!u.edge,
before drawing tree={%
if={>{RR|}{line numbering}{justifications}}{%
proof tree proof line no/.process={Ow1+n}{!n=1.proof tree proof line no}{##1-1},
}{},
},
append=!sibling,
]%
},
},
}{%
if single branches={}{%
delay={%
for children={%
no edge,
},
},
},
},
]%
},
},
}{%
TeX/.process={Ow1}{name}{\PackageWarning{prooftrees}{Line not moved! I can only move things later in the proof. Please see the documentation for details. ##1}},
},
},
proof tree cref/.style={% get the names of nodes cross-referenced in closure annotations for use later
proof tree crefs+/.option=#1.name,
},
proof tree rhif llinell cau/.style={% get the proof line numbers of the cross-referenced nodes in closure annotations, using the list of names created earlier
if proof tree rhestr rhifau llinellau cau={}{}{%
proof tree rhestr rhifau llinellau cau+={,\,},
},
proof tree rhestr rhifau llinellau cau+/.option=#1.proof tree proof line no,
},
proof tree jref/.style={% get the names of nodes cross-referenced in justifications for use later
proof tree jrefs+/.option=#1.name,
},
proof tree rhif llinell/.style={% get the proof line numbers of the cross-referenced nodes in justifications, using the list of names created earlier
if proof tree rhestr rhifau llinellau={}{}{%
proof tree rhestr rhifau llinellau+={,\,},
},
proof tree rhestr rhifau llinellau+/.option=#1.proof tree proof line no,% works according to Sašo's anti-pgfmath version
},
line no override/.style={% 2018-02-19 ateb https://tex.stackexchange.com/a/416037/
before drawing tree={
for name/.process={Ow}{proof tree proof line no}{line no ##1}{
content=\linenumberstyle{#1},
typeset node,
},
},
},
no line no/.style={% 2018-02-19 gweler uchod
before drawing tree={
for name/.process={Ow}{proof tree proof line no}{line no ##1}{
content=,
typeset node,
},
},
},
proof tree dadfygio/.style={% style for use in debugging moves which displays information about nodes in the tree
before packing={%
for tree={%
label/.process={OOOw3}{level}{proof tree toing by}{id}{[red,font=\tiny,inner sep=0pt,outer sep=0pt, anchor=south]below:##1/##2/##3},
},
},
before drawing tree={%
for tree={%
delay={%
tikz+/.process={Ow1}{proof tree proof line no}{\node [anchor=west, font=\tiny, text=blue, inner sep=0pt] at (.east) {##1}; },
},
},
},
},
proof tree alino/.style={% debugging / dangos dimension stuff
before drawing tree={%
tikz+/.process={%
RRRRw4{proof tree inner proof midpoint}{line no width}{line no dist}{just dist}
{
\begin{scope}[densely dashed]
\draw [darkgray] (##1,0) coordinate (a) -- (a |- current bounding box.south);
\draw [green] (current bounding box.west) -- ++(##2,0) coordinate (b);
\draw [blue] (b) -- ++(##3,0) coordinate (c);
\draw [magenta] (c) -- ++(##4,0);
\end{scope}
}%
},
},
},
}
% \environbodyname\prooftreebody
\bracketset{action character=@}
\NewDocumentEnvironment{\prooftrees@enw}{ m +b }{% \forest/\endforest from egreg's answer at http://tex.stackexchange.com/a/229608/
\forest
(%
stages={% customised definition of stages - we don't use any custom stages, but we do use several custom keylists, where the processing order of these is critical
for root'={% nothing is removed from the standard forest definition - we only change it by adding to it
process keylist register=default preamble,
process keylist register=preamble,
},
process keylist=given options,
process keylist=before typesetting nodes,
% first two additions: process two custom keylists after before typesetting nodes and before typesetting nodes
process keylist=proof tree ffurf,
process keylist=proof tree symud awto,
typeset nodes stage,
process keylist=before packing,
pack stage,
process keylist=before computing xy,
compute xy stage,
% second two additions: process two custom keylists after computing xy and before before drawing tree
process keylist=proof tree creu nodiadau,
process keylist=proof tree nodiadau,
process keylist=before drawing tree,
draw tree stage,
},
)%
proof tree,% apply the proof tree style, which sets keylists from both forest's defaults and our custom additions
#1,% insert user's preamble, empty or otherwise - this allows the user both to override our defaults (e.g. by setting a non-empty proof statement or a custom format for line numbers) and to customise the tree using forest's facilities in the usual way - BUT customisations of the latter kind may or may not be effective, may or may not have undesirable - not to say chaotic - consequences, and may or may not cause compilation failures (structural changes, in particular, should be avoided completely)
[, name=proof statement @#2]%
\endforest
}{}
\ExplSyntaxOn
\cs_new_protected_nopar:Npn \__prooftrees_memoize:n #1
{
\mmzset{
auto = { #1 } { memoize },
}
}
\cs_generate_variant:Nn \__prooftrees_memoize:n { V }
\hook_gput_code:nnn { begindocument / before } { . }
{% paid â memoize bussproofs prooftree ...
\@ifpackageloaded{memoize}{
\__prooftrees_memoize:V \prooftrees@enw
}{}
}
\ExplSyntaxOff
\endinput
%% end prooftrees.sty