]> git.localhorst.tv Git - blobs.git/blob - src/ui/ui.cpp
color mutation
[blobs.git] / src / ui / ui.cpp
1 #include "CreaturePanel.hpp"
2 #include "RecordsPanel.hpp"
3 #include "string.hpp"
4 #include "TimePanel.hpp"
5
6 #include "Label.hpp"
7 #include "Meter.hpp"
8 #include "../app/Assets.hpp"
9 #include "../creature/Creature.hpp"
10 #include "../graphics/Viewport.hpp"
11 #include "../world/Body.hpp"
12 #include "../world/Simulation.hpp"
13
14 #include <iomanip>
15 #include <iostream>
16 #include <sstream>
17 #include <glm/gtx/io.hpp>
18 #include <glm/gtx/transform.hpp>
19
20
21 namespace blobs {
22 namespace ui {
23
24 CreaturePanel::CreaturePanel(app::Assets &assets)
25 : assets(assets)
26 , c(nullptr)
27 , name(new Label(assets.fonts.large))
28 , parents(new Label(assets.fonts.medium))
29 , born(new Label(assets.fonts.medium))
30 , age(new Label(assets.fonts.medium))
31 , mass(new Label(assets.fonts.medium))
32 , goal(new Label(assets.fonts.medium))
33 , composition(new Panel)
34 , stats{nullptr}
35 , props{nullptr}
36 , panel() {
37         Label *parents_label = new Label(assets.fonts.medium);
38         parents_label->Text("Parents");
39         Label *born_label = new Label(assets.fonts.medium);
40         born_label->Text("Born");
41         Label *age_label = new Label(assets.fonts.medium);
42         age_label->Text("Age");
43         Label *mass_label = new Label(assets.fonts.medium);
44         mass_label->Text("Mass");
45         Label *goal_label = new Label(assets.fonts.medium);
46         goal_label->Text("Goal");
47
48         Panel *info_label_panel = new Panel;
49         info_label_panel
50                 ->Direction(Panel::VERTICAL)
51                 ->Add(parents_label)
52                 ->Add(born_label)
53                 ->Add(age_label)
54                 ->Add(mass_label)
55                 ->Add(goal_label);
56         Panel *info_value_panel = new Panel;
57         info_value_panel
58                 ->Direction(Panel::VERTICAL)
59                 ->Add(parents)
60                 ->Add(born)
61                 ->Add(age)
62                 ->Add(mass)
63                 ->Add(goal);
64         Panel *info_panel = new Panel;
65         info_panel
66                 ->Direction(Panel::HORIZONTAL)
67                 ->Spacing(10.0f)
68                 ->Add(info_label_panel)
69                 ->Add(info_value_panel);
70
71         Label *stat_label[7];
72         for (int i = 0; i < 7; ++i) {
73                 stat_label[i] = new Label(assets.fonts.medium);
74                 stats[i] = new Meter;
75                 stats[i]
76                         ->Size(glm::vec2(100.0f, assets.fonts.medium.Height() + assets.fonts.medium.Descent()))
77                         ->Padding(glm::vec2(1.0f))
78                         ->Border(1.0f)
79                         ->BorderColor(glm::vec4(0.0f, 0.0f, 0.0f, 1.0f));
80         }
81         stat_label[0]->Text("Damage");
82         stat_label[1]->Text("Breath");
83         stat_label[2]->Text("Thirst");
84         stat_label[3]->Text("Hunger");
85         stat_label[4]->Text("Exhaustion");
86         stat_label[5]->Text("Fatigue");
87         stat_label[6]->Text("Boredom");
88
89         Panel *stat_label_panel = new Panel;
90         stat_label_panel
91                 ->Spacing(2)
92                 ->Direction(Panel::VERTICAL);
93         Panel *stat_meter_panel = new Panel;
94         stat_meter_panel
95                 ->Spacing(stat_label[0]->Size().y - stats[0]->Size().y + 2)
96                 ->Direction(Panel::VERTICAL);
97         for (int i = 0; i < 7; ++i) {
98                 stat_label_panel->Add(stat_label[i]);
99                 stat_meter_panel->Add(stats[i]);
100         }
101         Panel *stat_panel = new Panel;
102         stat_panel
103                 ->Direction(Panel::HORIZONTAL)
104                 ->Spacing(10)
105                 ->Add(stat_label_panel)
106                 ->Add(stat_meter_panel);
107
108         Label *prop_label[9];
109         for (int i = 0; i < 9; ++i) {
110                 prop_label[i] = new Label(assets.fonts.medium);
111                 props[i] = new Label(assets.fonts.medium);
112         }
113         prop_label[0]->Text("Strength");
114         prop_label[1]->Text("Stamina");
115         prop_label[2]->Text("Dexerty");
116         prop_label[3]->Text("Intelligence");
117         prop_label[4]->Text("Lifetime");
118         prop_label[5]->Text("Fertility");
119         prop_label[6]->Text("Mutability");
120         prop_label[7]->Text("Adaptability");
121         prop_label[8]->Text("Offspring mass");
122
123         Panel *prop_label_panel = new Panel;
124         prop_label_panel
125                 ->Spacing(2)
126                 ->Direction(Panel::VERTICAL);
127         Panel *prop_meter_panel = new Panel;
128         prop_meter_panel
129                 ->Spacing(2)
130                 ->Direction(Panel::VERTICAL);
131         for (int i = 0; i < 9; ++i) {
132                 prop_label_panel->Add(prop_label[i]);
133                 prop_meter_panel->Add(props[i]);
134         }
135         Panel *prop_panel = new Panel;
136         prop_panel
137                 ->Direction(Panel::HORIZONTAL)
138                 ->Spacing(10)
139                 ->Add(prop_label_panel)
140                 ->Add(prop_meter_panel);
141
142         panel
143                 .Add(name)
144                 ->Add(info_panel)
145                 ->Add(composition)
146                 ->Add(stat_panel)
147                 ->Add(prop_panel)
148                 ->Padding(glm::vec2(10.0f))
149                 ->Spacing(10.0f)
150                 ->Direction(Panel::VERTICAL)
151                 ->Background(glm::vec4(0.7f, 0.7f, 0.7f, 1.0f));
152 }
153
154 CreaturePanel::~CreaturePanel() {
155 }
156
157
158 void CreaturePanel::Show(creature::Creature &cr) {
159         c = &cr;
160         name->Text(c->Name());
161         born->Text(TimeString(c->Born()));
162
163         if (c->Parents().empty()) {
164                 parents->Text("none");
165         } else {
166                 std::string parent_string;
167                 bool first = true;
168                 for (auto p : c->Parents()) {
169                         if (first) {
170                                 first = false;
171                         } else {
172                                 parent_string += " and ";
173                         }
174                         parent_string += p->Name();
175                 }
176                 parents->Text(parent_string);
177         }
178 }
179
180 void CreaturePanel::Hide() noexcept {
181         c = nullptr;
182 }
183
184 void CreaturePanel::Draw(graphics::Viewport &viewport) noexcept {
185         if (!c) return;
186
187         age->Text(TimeString(c->Age()));
188         mass->Text(MassString(c->Mass()));
189         if (c->Goals().empty()) {
190                 goal->Text("none");
191         } else {
192                 goal->Text(c->Goals()[0]->Describe());
193         }
194
195         const creature::Composition &comp = c->GetComposition();
196         if (comp.size() < components.size()) {
197                 composition->Clear();
198                 while (comp.size() < components.size()) {
199                         delete components.back();
200                         components.pop_back();
201                 }
202                 for (auto l : components) {
203                         composition->Add(l);
204                 }
205         } else {
206                 while (comp.size() > components.size()) {
207                         components.emplace_back(new Label(assets.fonts.medium));
208                         composition->Add(components.back());
209                 }
210         }
211         {
212                 int i = 0;
213                 for (auto &cmp : comp) {
214                         components[i]->Text(
215                                 PercentageString(cmp.value / comp.TotalMass())
216                                 + " " + assets.data.resources[cmp.resource].label);
217                         ++i;
218                 }
219         }
220
221         for (int i = 0; i < 7; ++i) {
222                 stats[i]->Value(c->GetStats().stat[i].value);
223                 if (c->GetStats().stat[i].Okay()) {
224                         stats[i]->FillColor(glm::vec4(0.0f, 0.7f, 0.0f, 1.0f));
225                 } else if (c->GetStats().stat[i].Critical()) {
226                         stats[i]->FillColor(glm::vec4(0.7f, 0.0f, 0.0f, 1.0f));
227                 } else {
228                         stats[i]->FillColor(glm::vec4(0.9f, 0.4f, 0.0f, 1.0f));
229                 }
230         }
231
232         props[0]->Text(DecimalString(c->Strength(), 2));
233         props[1]->Text(DecimalString(c->Stamina(), 2));
234         props[2]->Text(DecimalString(c->Dexerty(), 2));
235         props[3]->Text(DecimalString(c->Intelligence(), 2));
236         props[4]->Text(TimeString(c->Lifetime()));
237         props[5]->Text(PercentageString(c->Fertility()));
238         props[6]->Text(PercentageString(c->Mutability()));
239         props[7]->Text(PercentageString(c->Adaptability()));
240         props[8]->Text(MassString(c->OffspringMass()));
241
242         const glm::vec2 margin(20.0f);
243         panel.Position(glm::vec2(viewport.Width() - margin.x - panel.Size().x, margin.y));
244         panel.Draw(assets, viewport);
245 }
246
247
248 RecordsPanel::RecordsPanel(world::Simulation &sim)
249 : sim(sim)
250 , live(new Label(sim.Assets().fonts.medium))
251 , records()
252 , holders()
253 , panel() {
254         Label *live_label = new Label(sim.Assets().fonts.medium);
255         live_label->Text("Creatures alive");
256
257         Panel *label_panel = new Panel;
258         label_panel
259                 ->Direction(Panel::VERTICAL)
260                 ->Add(live_label);
261
262         Panel *value_panel = new Panel;
263         value_panel
264                 ->Direction(Panel::VERTICAL)
265                 ->Add(live);
266
267         Label *holder_label = new Label(sim.Assets().fonts.medium);
268         holder_label->Text("Holder");
269         Panel *holder_panel = new Panel;
270         holder_panel
271                 ->Direction(Panel::VERTICAL)
272                 ->Add(holder_label);
273
274         records.reserve(sim.Records().size());
275         for (const auto &r : sim.Records()) {
276                 Label *label = new Label(sim.Assets().fonts.medium);
277                 label->Text(r.name + " record");
278                 label_panel->Add(label);
279                 Label *value = new Label(sim.Assets().fonts.medium);
280                 value->Text("none");
281                 value_panel->Add(value);
282                 records.push_back(value);
283                 Label *holder = new Label(sim.Assets().fonts.medium);
284                 holder->Text("nobody");
285                 holder_panel->Add(holder);
286                 holders.push_back(holder);
287         }
288
289         panel
290                 .Direction(Panel::HORIZONTAL)
291                 ->Padding(glm::vec2(10.0f))
292                 ->Spacing(10.0f)
293                 ->Background(glm::vec4(0.7f, 0.7f, 0.7f, 1.0f))
294                 ->Add(label_panel)
295                 ->Add(value_panel)
296                 ->Add(holder_panel);
297 }
298
299 RecordsPanel::~RecordsPanel() {
300 }
301
302 void RecordsPanel::Draw(graphics::Viewport &viewport) noexcept {
303         live->Text(NumberString(sim.LiveCreatures().size()));
304         int i = 0;
305         for (const auto &r : sim.Records()) {
306                 if (!r) continue;
307                 switch (r.type) {
308                         default:
309                         case world::Record::VALUE:
310                                 records[i]->Text(DecimalString(r.value, 2));
311                                 break;
312                         case world::Record::LENGTH:
313                                 records[i]->Text(LengthString(r.value));
314                                 break;
315                         case world::Record::MASS:
316                                 records[i]->Text(MassString(r.value));
317                                 break;
318                         case world::Record::PERCENTAGE:
319                                 records[i]->Text(PercentageString(r.value));
320                                 break;
321                         case world::Record::TIME:
322                                 records[i]->Text(TimeString(r.value));
323                                 break;
324                 }
325                 std::string str(r.holder->Name());
326                 bool first = true;
327                 for (auto p : r.holder->Parents()) {
328                         if (first) {
329                                 first = false;
330                                 str += " of ";
331                         } else {
332                                 str += " and ";
333                         }
334                         str += p->Name();
335                 }
336                 holders[i]->Text(str);
337                 ++i;
338         }
339
340         const glm::vec2 margin(20.0f);
341         panel.Position(glm::vec2(margin.x, margin.y));
342         panel.Draw(sim.Assets(), viewport);
343 }
344
345
346 TimePanel::TimePanel(world::Simulation &sim)
347 : sim(sim)
348 , body(nullptr)
349 , time(new Label(sim.Assets().fonts.medium))
350 , clock(new Label(sim.Assets().fonts.medium))
351 , panel() {
352         Label *time_label = new Label(sim.Assets().fonts.medium);
353         time_label->Text("Time");
354         Label *clock_label = new Label(sim.Assets().fonts.medium);
355         clock_label->Text("Clock");
356
357         Panel *label_panel = new Panel;
358         label_panel
359                 ->Direction(Panel::VERTICAL)
360                 ->Add(time_label)
361                 ->Add(clock_label);
362
363         Panel *value_panel = new Panel;
364         value_panel
365                 ->Direction(Panel::VERTICAL)
366                 ->Add(time)
367                 ->Add(clock);
368
369         panel
370                 .Direction(Panel::HORIZONTAL)
371                 ->Padding(glm::vec2(10.0f))
372                 ->Spacing(10.0f)
373                 ->Background(glm::vec4(0.7f, 0.7f, 0.7f, 1.0f))
374                 ->Add(label_panel)
375                 ->Add(value_panel);
376 }
377
378 TimePanel::~TimePanel() {
379 }
380
381 void TimePanel::Draw(graphics::Viewport &viewport) noexcept {
382         time->Text(TimeString(sim.Time()));
383         if (body) {
384                 clock->Text(TimeString(std::fmod(sim.Time(), body->RotationalPeriod())));
385         } else {
386                 clock->Text("no reference");
387         }
388
389         const glm::vec2 margin(20.0f);
390         panel.Position(glm::vec2(margin.x, viewport.Height() - margin.y - panel.Size().y));
391         panel.Draw(sim.Assets(), viewport);
392 }
393
394
395 std::string DecimalString(double n, int p) {
396         std::stringstream s;
397         s << std::fixed << std::setprecision(p) << n;
398         return s.str();
399 }
400
401 std::string LengthString(double m) {
402         std::stringstream s;
403         s << std::fixed << std::setprecision(3);
404         if (m > 1500.0) {
405                 s << (m * 0.001) << "km";
406         } else if (m < 0.1) {
407                 s << (m * 1000.0) << "mm";
408         } else {
409                 s << m << "m";
410         }
411         return s.str();
412 }
413
414 std::string MassString(double kg) {
415         std::stringstream s;
416         s << std::fixed << std::setprecision(3);
417         if (kg > 1500.0) {
418                 s << (kg * 0.001) << "t";
419         } else if (kg < 1.0) {
420                 s << (kg * 1000.0) << "g";
421         } else if (kg < 0.001) {
422                 s << (kg * 1.0e6) << "mg";
423         } else {
424                 s << kg << "kg";
425         }
426         return s.str();
427 }
428
429 std::string NumberString(int n) {
430         return std::to_string(n);
431 }
432
433 std::string PercentageString(double n) {
434         std::stringstream s;
435         s << std::fixed << std::setprecision(1) << (n * 100.0) << '%';
436         return s.str();
437 }
438
439 std::string TimeString(double s) {
440         int is = int(s);
441         std::stringstream ss;
442         if (is >= 3600) {
443                 ss << (is / 3600) << "h ";
444                 is %= 3600;
445         }
446         if (is >= 60) {
447                 ss << (is / 60) << "m ";
448                 is %= 60;
449         }
450         ss << is << 's';
451         return ss.str();
452 }
453
454 std::string VectorString(const glm::dvec3 &v, int p) {
455         std::stringstream ss;
456         ss << std::fixed << std::setprecision(p)
457                 << "<" << v.x << ", " << v.y << ", " << v.z << ">";
458         return ss.str();
459 }
460
461 std::string VectorString(const glm::ivec2 &v) {
462         std::stringstream ss;
463         ss << "<" << v.x << ", " << v.y << ">";
464         return ss.str();
465 }
466
467 }
468 }