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 }