1 module nodes.switch_slot;
2 
3 import fluid;
4 
5 @safe:
6 
7 unittest {
8 
9     Node bigNode, smallNode;
10 
11     auto slot = switchSlot(
12         bigNode = new class Frame {
13 
14             override void resizeImpl(Vector2 space) {
15                 super.resizeImpl(space);
16                 minSize = Vector2(300, 300);
17             }
18 
19         },
20         smallNode = new class Frame {
21 
22             override void resizeImpl(Vector2 space) {
23                 super.resizeImpl(space);
24                 minSize = Vector2(100, 100);
25             }
26 
27         },
28     );
29     auto root = sizeLock!testSpace(
30         .sizeLimit(800, 600),
31         .nullTheme,
32         slot
33     );
34 
35     // At start, there should be enough space to draw the big frame
36     root.drawAndAssert(
37         slot.drawsChild(bigNode),
38         slot.doesNotDrawChildren(),
39     );
40     assert(slot.node is bigNode);
41 
42     // Reduce the viewport, this time the small frame should be drawn
43     root.limit = sizeLimit(200, 200);
44     root.updateSize();
45     root.drawAndAssert(
46         slot.drawsChild(smallNode),
47         slot.doesNotDrawChildren(),
48     );
49     assert(slot.node is smallNode);
50 
51     // Do it again, but make it so neither fit; the small one should prevail anyway
52     root.limit = sizeLimit(50, 50);
53     root.updateSize();
54     root.drawAndAssert(
55         slot.drawsChild(smallNode),
56         slot.doesNotDrawChildren(),
57     );
58     assert(slot.node is smallNode);
59 
60     // Unless a null node is added
61     slot.availableNodes ~= null;
62     root.updateSize();
63     root.drawAndAssertFailure(
64         slot.isDrawn(),
65     );
66     assert(slot.node is null);
67 
68     // Resize to fit the big node
69     root.limit = sizeLimit(400, 400);
70     root.updateSize();
71     root.drawAndAssert(
72         slot.drawsChild(bigNode),
73         slot.doesNotDrawChildren(),
74     );
75     assert(slot.node is bigNode);
76 
77 }
78 
79 @("Nodes can be moved between SwitchSlots")
80 unittest {
81 
82     int principalDrawn, deputyDrawn;
83 
84     auto principal = switchSlot(
85         layout!(1, "fill"),
86         new class Frame {
87             override void resizeImpl(Vector2) {
88                 minSize = Vector2(200, 200);
89             }
90             override void drawImpl(Rectangle outer, Rectangle) {
91                 io.drawRectangle(outer, color!"f00");
92                 principalDrawn++;
93             }
94         },
95         null
96     );
97     auto deputy = principal.retry(
98         layout!(1, "fill"),
99         new class Frame {
100             override void resizeImpl(Vector2 space) {
101                 minSize = Vector2(50, 200);
102             }
103             override void drawImpl(Rectangle outer, Rectangle) {
104                 io.drawRectangle(outer, color!"f00");
105                 deputyDrawn++;
106             }
107         }
108     );
109     auto root = sizeLock!testSpace(
110         .layout!(1, "fill"),
111         .sizeLimit(600, 600),
112         hframe(
113             .layout!(1, "fill"),
114             deputy,
115         ),
116         hframe(
117             .layout!(1, "fill"),
118             principal,
119         ),
120     );
121 
122     // At the initial size, the principal should be preferred
123     root.draw();
124 
125     assert(principalDrawn == 1);
126     assert(deputyDrawn == 0);
127 
128     // Resize the window so that the principal can't fit
129     root.limit = sizeLimit(300, 300);
130     root.updateSize();
131     root.draw();
132 
133     assert(principalDrawn == 1);
134     assert(deputyDrawn == 1);
135 
136 }
137 
138 @("Deputy SwitchSlot can be placed before its principal")
139 unittest {
140 
141     import std.algorithm;
142 
143     import fluid.space;
144     import fluid.structs;
145 
146     SwitchSlot slot;
147 
148     auto checker = new class Node {
149 
150         Vector2 size;
151         Vector2[] spacesGiven;
152 
153         override void resizeImpl(Vector2 space) {
154 
155             spacesGiven ~= space;
156             size = minSize = Vector2(500, 200);
157 
158         }
159 
160         override void drawImpl(Rectangle, Rectangle) {
161 
162         }
163 
164     };
165 
166     auto parentSlot = switchSlot(checker, null);
167     auto childSlot = parentSlot.retry(checker);
168 
169     auto root = vspace(
170         layout!"fill",
171         nullTheme,
172 
173         // Two slots: child slot that gets resized earlier
174         hspace(
175             layout!"fill",
176             childSlot,
177         ),
178 
179         // Parent slot that doesn't give enough space for the child to fit
180         hspace(
181             layout!"fill",
182             vspace(
183                 layout!(1, "fill"),
184                 parentSlot,
185             ),
186             vspace(
187                 layout!(3, "fill"),
188             ),
189         ),
190     );
191 
192     root.draw();
193 
194     // The principal slot gives the least space, namely the width of the window divided by 4
195     assert(checker.spacesGiven.map!"a.x".minElement == HeadlessBackend.defaultWindowSize.x / 4);
196 
197     // The window size that is accepted is equal to its size, as it was assigned by the fallback slot
198     assert(checker.spacesGiven[$-1] == checker.size);
199 
200     // A total of three resizes were performed: one by the fallback, one by the parent and one, final, by the parent
201     // using previous parameters
202     assert(checker.spacesGiven.length == 3);
203 
204     // The first one (which should be the child's) has the largest width given, equal to the window width
205     assert(checker.spacesGiven[0].x == HeadlessBackend.defaultWindowSize.x);
206 
207 }