31#include "table/strings.h"
41static const uint INVALID_DATAPOINT_POS = UINT_MAX;
55 for (
CompanyID c = CompanyID::Begin(); c < MAX_COMPANIES; ++c) {
99 void OnInvalidateData([[maybe_unused]]
int data = 0, [[maybe_unused]]
bool gui_scope =
true)
override
101 if (!gui_scope)
return;
104 _legend_excluded_companies.
Set(
static_cast<CompanyID>(data));
99 void OnInvalidateData([[maybe_unused]]
int data = 0, [[maybe_unused]]
bool gui_scope =
true)
override {
…}
116 vert->SetPadding(2, 2, 2, 2);
120 auto panel = std::make_unique<NWidgetBackground>(
WWT_PANEL, COLOUR_BROWN, widnum);
123 panel->SetFill(1, 1);
124 panel->SetToolTip(STR_GRAPH_KEY_COMPANY_SELECTION_TOOLTIP);
125 vert->Add(std::move(panel));
130static constexpr NWidgetPart _nested_graph_legend_widgets[] = {
146 _nested_graph_legend_widgets
149static void ShowGraphLegend()
151 AllocateWindowDescFront<GraphLegendWindow>(_graph_legend_desc, 0);
166 static const int GRAPH_MAX_DATASETS = 64;
167 static const int GRAPH_BASE_COLOUR =
GREY_SCALE(2);
168 static const int GRAPH_GRID_COLOUR =
GREY_SCALE(3);
169 static const int GRAPH_AXIS_LINE_COLOUR =
GREY_SCALE(1);
170 static const int GRAPH_ZERO_LINE_COLOUR =
GREY_SCALE(8);
171 static const int GRAPH_YEAR_LINE_COLOUR =
GREY_SCALE(5);
186 uint8_t num_on_x_axis = 0;
198 bool x_values_reversed =
true;
204 std::array<OverflowSafeInt64, GRAPH_NUM_MONTHS> values;
210 std::vector<DataSet> data{};
212 std::span<const StringID> ranges = {};
225 return {std::begin(dataset.values), std::begin(dataset.values) + this->num_on_x_axis};
236 assert(num_hori_lines > 0);
239 current_interval.
highest = INT64_MIN;
240 current_interval.
lowest = INT64_MAX;
242 for (
const DataSet &dataset : this->data) {
243 if (
HasBit(this->excluded_data, dataset.exclude_bit))
continue;
244 if (
HasBit(this->excluded_range, dataset.range_bit))
continue;
247 if (datapoint != INVALID_DATAPOINT) {
248 current_interval.
highest = std::max(current_interval.
highest, datapoint);
249 current_interval.
lowest = std::min(current_interval.
lowest, datapoint);
255 double abs_lower = (current_interval.
lowest > 0) ? 0 : (double)
abs(current_interval.
lowest);
256 double abs_higher = (current_interval.
highest < 0) ? 0 : (double)current_interval.
highest;
259 abs_higher = (11.0 * abs_higher) / 10.0;
260 abs_lower = (11.0 * abs_lower) / 10.0;
265 if (abs_lower != 0 || abs_higher != 0) {
267 num_pos_grids = (int)floor(0.5 + num_hori_lines * abs_higher / (abs_higher + abs_lower));
270 if (num_pos_grids == 0 && abs_higher != 0) num_pos_grids++;
271 if (num_pos_grids == num_hori_lines && abs_lower != 0) num_pos_grids--;
276 if (abs_higher > 0) {
277 grid_size_higher = abs_higher >
INT64_MAX_IN_DOUBLE ? INT64_MAX :
static_cast<int64_t
>(abs_higher);
278 grid_size_higher = (grid_size_higher + num_pos_grids - 1) / num_pos_grids;
283 grid_size_lower = abs_lower >
INT64_MAX_IN_DOUBLE ? INT64_MAX :
static_cast<int64_t
>(abs_lower);
284 grid_size_lower = (grid_size_lower + num_hori_lines - num_pos_grids - 1) / (num_hori_lines - num_pos_grids);
287 grid_size = std::max(grid_size_higher, grid_size_lower);
290 num_pos_grids = num_hori_lines / 2;
294 current_interval.
highest = num_pos_grids * grid_size;
295 current_interval.
lowest = -(num_hori_lines - num_pos_grids) * grid_size;
296 return current_interval;
307 int64_t y_label = current_interval.
highest;
308 int64_t y_label_separation = (current_interval.
highest - current_interval.
lowest) / num_hori_lines;
312 for (
int i = 0; i < (num_hori_lines + 1); i++) {
314 if (d.width > max_width) max_width = d.width;
316 y_label -= y_label_separation;
334 static_assert(GRAPH_MAX_DATASETS >= (int)
NUM_CARGO && GRAPH_MAX_DATASETS >= (
int)MAX_COMPANIES);
335 assert(this->num_vert_lines > 0);
357 r.right -= label_width;
359 r.left += label_width;
362 int x_sep = (r.right - r.left) / this->num_vert_lines;
363 int y_sep = (r.bottom - r.top) / num_hori_lines;
368 r.left = r.right - x_sep * this->num_vert_lines;
370 r.right = r.left + x_sep * this->num_vert_lines;
372 r.bottom = r.top + y_sep * num_hori_lines;
376 x_axis_offset = (int)((r.bottom - r.top) * (double)interval.
highest / (
double)interval_size);
379 GfxFillRect(r.left, r.top, r.right, r.bottom, GRAPH_BASE_COLOUR);
391 int grid_colour = GRAPH_GRID_COLOUR;
392 for (
int i = 1; i < this->num_vert_lines + 1; i++) {
395 grid_colour = (i % 4 == 0) ? GRAPH_YEAR_LINE_COLOUR : GRAPH_GRID_COLOUR;
404 for (
int i = 0; i < (num_hori_lines + 1); i++) {
410 GfxFillRect(r.left, y, r.right, y, GRAPH_GRID_COLOUR);
415 GfxFillRect(r.left, r.top, r.left, r.bottom, GRAPH_AXIS_LINE_COLOUR);
418 y = x_axis_offset + r.top;
419 GfxFillRect(r.left, y, r.right, y, GRAPH_ZERO_LINE_COLOUR);
422 if (this->num_on_x_axis == 0)
return;
424 assert(this->num_on_x_axis > 0);
427 int64_t y_label = interval.
highest;
428 int64_t y_label_separation =
abs(interval.
highest - interval.
lowest) / num_hori_lines;
432 for (
int i = 0; i < (num_hori_lines + 1); i++) {
435 GetString(STR_GRAPH_Y_LABEL, this->format_str_y_axis, y_label),
439 GetString(STR_GRAPH_Y_LABEL, this->format_str_y_axis, y_label),
443 y_label -= y_label_separation;
447 x = rtl ? r.right : r.left;
451 x += (this->num_vert_lines - this->num_on_x_axis) * x_sep;
454 if (this->draw_dates) {
457 for (
int i = 0; i < this->num_on_x_axis; i++) {
460 GetString(month == 0 ? STR_GRAPH_X_LABEL_MONTH_YEAR : STR_GRAPH_X_LABEL_MONTH, STR_MONTH_ABBREV_JAN + month, year),
464 GetString(month == 0 ? STR_GRAPH_X_LABEL_MONTH_YEAR : STR_GRAPH_X_LABEL_MONTH, STR_MONTH_ABBREV_JAN + month, year),
474 GfxFillRect(x + x_sep, r.top + 1, x + x_sep, r.bottom - 1, GRAPH_YEAR_LINE_COLOUR);
482 if (this->x_values_reversed) {
483 label = this->x_values_increment * this->num_on_x_axis;
484 iterator = -this->x_values_increment;
486 label = this->x_values_increment;
487 iterator = this->x_values_increment;
490 for (
int i = 0; i < this->num_on_x_axis; i++) {
504 uint pointoffs1 = (linewidth + 1) / 2;
505 uint pointoffs2 = linewidth + 1 - pointoffs1;
507 auto draw_dataset = [&](
const DataSet &dataset, uint8_t colour) {
508 if (
HasBit(this->excluded_data, dataset.exclude_bit))
return;
509 if (
HasBit(this->excluded_range, dataset.range_bit))
return;
513 x = r.right + (x_sep / 2);
515 x = r.left + (x_sep / 2);
519 x += (this->num_vert_lines - this->num_on_x_axis) * x_sep;
521 uint prev_x = INVALID_DATAPOINT_POS;
522 uint prev_y = INVALID_DATAPOINT_POS;
526 if (datapoint != INVALID_DATAPOINT) {
538 int mult_range = FindLastBit<uint32_t>(x_axis_offset) + FindLastBit<uint64_t>(
abs(datapoint));
539 int reduce_range = std::max(mult_range - 31, 0);
543 datapoint = -(
abs(datapoint) >> reduce_range);
545 datapoint >>= reduce_range;
547 y = r.top + x_axis_offset - ((r.bottom - r.top) * datapoint) / (interval_size >> reduce_range);
550 GfxFillRect(x - pointoffs1, y - pointoffs1, x + pointoffs2, y + pointoffs2, colour);
553 if (prev_x != INVALID_DATAPOINT_POS) GfxDrawLine(prev_x, prev_y, x, y, colour, linewidth, dash);
558 prev_x = INVALID_DATAPOINT_POS;
559 prev_y = INVALID_DATAPOINT_POS;
567 for (
const DataSet &dataset : this->data) {
568 if (dataset.exclude_bit != this->highlight_data && dataset.range_bit != this->highlight_range) {
569 draw_dataset(dataset, dataset.colour);
575 if (this->highlight_state && (this->highlight_data != UINT8_MAX || this->highlight_range != UINT8_MAX)) {
576 for (
const DataSet &dataset : this->data) {
577 if (dataset.exclude_bit == this->highlight_data || dataset.range_bit == this->highlight_range) {
586 format_str_y_axis(format_str_y_axis)
593 if (this->highlight_data == UINT8_MAX && this->highlight_range == UINT8_MAX)
return;
605 for (
const StringID &str : this->ranges) {
613 size.height *=
static_cast<uint
>(std::size(this->ranges));
617 this->GetWidget<NWidgetCore>(
WID_GRAPH_RANGE_MATRIX)->SetMatrixDimension(1, ClampTo<uint32_t>(std::size(this->ranges)));
621 uint x_label_width = 0;
624 if (this->draw_dates) {
627 for (
int i = 0; i < this->num_on_x_axis; i++) {
628 x_label_width = std::max(x_label_width,
GetStringBoundingBox(
GetString(month == 0 ? STR_GRAPH_X_LABEL_MONTH_YEAR : STR_GRAPH_X_LABEL_MONTH, STR_MONTH_ABBREV_JAN + month, year)).width);
646 size.height = std::max<uint>(size.height, size.width / 3);
665 for (
const auto &str : this->ranges) {
666 bool lowered = !
HasBit(this->excluded_range, index);
707 uint8_t new_highlight_range = UINT8_MAX;
710 if (!
HasBit(this->excluded_range, row)) new_highlight_range =
static_cast<uint8_t
>(row);
714 uint8_t new_highlight_data = UINT8_MAX;
716 auto dataset_index = this->GetDatasetIndex(pt.y);
717 if (dataset_index.has_value() && !
HasBit(this->excluded_data, *dataset_index)) new_highlight_data = *dataset_index;
720 if (this->highlight_data == new_highlight_data && this->highlight_range == new_highlight_range)
return;
723 this->highlight_data = new_highlight_data;
724 this->highlight_range = new_highlight_range;
725 this->highlight_state =
true;
731 this->UpdateStatistics(
false);
739 void OnInvalidateData([[maybe_unused]]
int data = 0, [[maybe_unused]]
bool gui_scope =
true)
override
741 if (!gui_scope)
return;
742 this->UpdateStatistics(
true);
739 void OnInvalidateData([[maybe_unused]]
int data = 0, [[maybe_unused]]
bool gui_scope =
true)
override {
…}
745 virtual void UpdateStatistics(
bool initialize) = 0;
747 virtual std::optional<uint8_t> GetDatasetIndex(
int) {
return std::nullopt; }
773 CompanyMask excluded_companies = _legend_excluded_companies;
776 for (
CompanyID c = CompanyID::Begin(); c < MAX_COMPANIES; ++c) {
782 nums = std::min(this->num_vert_lines, std::max(nums, c->num_valid_stat_ent));
792 if (!initialize && this->excluded_data == excluded_companies.
base() && this->num_on_x_axis == nums &&
793 this->year == yr && this->month == mo) {
798 this->excluded_data = excluded_companies.
base();
799 this->num_on_x_axis = nums;
804 for (
CompanyID k = CompanyID::Begin(); k < MAX_COMPANIES; ++k) {
806 if (c ==
nullptr)
continue;
808 DataSet &dataset = this->data.emplace_back();
810 dataset.exclude_bit = k.base();
812 for (
int j = this->num_on_x_axis, i = 0; --j >= 0;) {
814 dataset.values[i] = INVALID_DATAPOINT;
818 dataset.values[i] = std::min(GetGraphData(c, j), INVALID_DATAPOINT - 1);
850static constexpr NWidgetPart _nested_operating_profit_widgets[] = {
862 NWidget(
WWT_TEXT, INVALID_COLOUR,
WID_GRAPH_FOOTER),
SetFill(1, 0),
SetResize(1, 0),
SetPadding(2, 0, 2, 0),
SetTextStyle(TC_BLACK,
FS_SMALL),
SetAlignment(
SA_CENTER),
869 WDP_AUTO,
"graph_operating_profit", 0, 0,
872 _nested_operating_profit_widgets
876void ShowOperatingProfitGraph()
878 AllocateWindowDescFront<OperatingProfitGraphWindow>(_operating_profit_desc, 0);
903static constexpr NWidgetPart _nested_income_graph_widgets[] = {
915 NWidget(
WWT_TEXT, INVALID_COLOUR,
WID_GRAPH_FOOTER),
SetFill(1, 0),
SetResize(1, 0),
SetPadding(2, 0, 2, 0),
SetTextStyle(TC_BLACK,
FS_SMALL),
SetAlignment(
SA_CENTER),
925 _nested_income_graph_widgets
928void ShowIncomeGraph()
930 AllocateWindowDescFront<IncomeGraphWindow>(_income_graph_desc, 0);
954static constexpr NWidgetPart _nested_delivered_cargo_graph_widgets[] = {
966 NWidget(
WWT_TEXT, INVALID_COLOUR,
WID_GRAPH_FOOTER),
SetFill(1, 0),
SetResize(1, 0),
SetPadding(2, 0, 2, 0),
SetTextStyle(TC_BLACK,
FS_SMALL),
SetAlignment(
SA_CENTER),
973 WDP_AUTO,
"graph_delivered_cargo", 0, 0,
976 _nested_delivered_cargo_graph_widgets
979void ShowDeliveredCargoGraph()
981 AllocateWindowDescFront<DeliveredCargoGraphWindow>(_delivered_cargo_graph_desc, 0);
1011static constexpr NWidgetPart _nested_performance_history_widgets[] = {
1024 NWidget(
WWT_TEXT, INVALID_COLOUR,
WID_GRAPH_FOOTER),
SetFill(1, 0),
SetResize(1, 0),
SetPadding(2, 0, 2, 0),
SetTextStyle(TC_BLACK,
FS_SMALL),
SetAlignment(
SA_CENTER),
1031 WDP_AUTO,
"graph_performance", 0, 0,
1034 _nested_performance_history_widgets
1037void ShowPerformanceHistoryGraph()
1039 AllocateWindowDescFront<PerformanceHistoryGraphWindow>(_performance_history_desc, 0);
1063static constexpr NWidgetPart _nested_company_value_graph_widgets[] = {
1075 NWidget(
WWT_TEXT, INVALID_COLOUR,
WID_GRAPH_FOOTER),
SetFill(1, 0),
SetResize(1, 0),
SetPadding(2, 0, 2, 0),
SetTextStyle(TC_BLACK,
FS_SMALL),
SetAlignment(
SA_CENTER),
1082 WDP_AUTO,
"graph_company_value", 0, 0,
1085 _nested_company_value_graph_widgets
1088void ShowCompanyValueGraph()
1090 AllocateWindowDescFront<CompanyValueGraphWindow>(_company_value_graph_desc, 0);
1120 virtual CargoTypes GetCargoTypes(
WindowNumber number)
const = 0;
1121 virtual CargoTypes &GetExcludedCargoTypes()
const = 0;
1123 std::optional<uint8_t> GetDatasetIndex(
int y)
override
1126 if (row >= this->vscroll->
GetCount())
return std::nullopt;
1130 if (row-- > 0)
continue;
1135 return std::nullopt;
1163 this->line_height = size.height;
1164 size.height = this->line_height * 11;
1186 if (pos-- > 0)
continue;
1187 if (--max < 0)
break;
1189 bool lowered = !
HasBit(this->excluded_data, cs->Index());
1199 uint8_t pc = cs->legend_colour;
1200 if (this->highlight_data == cs->Index()) pc = this->highlight_state ?
PC_WHITE :
PC_BLACK;
1206 line = line.
Translate(0, this->line_height);
1215 this->GetExcludedCargoTypes() = {};
1216 this->excluded_data = this->GetExcludedCargoTypes();
1222 this->GetExcludedCargoTypes() = this->
cargo_types;
1223 this->excluded_data = this->GetExcludedCargoTypes();
1230 if (row >= this->vscroll->
GetCount())
return;
1234 if (row-- > 0)
continue;
1236 ToggleBit(this->GetExcludedCargoTypes(), cs->Index());
1237 this->excluded_data = this->GetExcludedCargoTypes();
1261 static inline CargoTypes excluded_cargo_types{};
1267 this->draw_dates =
false;
1269 this->x_values_reversed =
false;
1273 this->InitializeWindow(
window_number, STR_GRAPH_CARGO_PAYMENT_RATES_SECONDS, STR_GRAPH_CARGO_PAYMENT_RATES_DAYS);
1281 CargoTypes &GetExcludedCargoTypes()
const override
1283 return PaymentRatesGraphWindow::excluded_cargo_types;
1291 void OnInvalidateData([[maybe_unused]]
int data = 0, [[maybe_unused]]
bool gui_scope =
true)
override
1293 if (!gui_scope)
return;
1291 void OnInvalidateData([[maybe_unused]]
int data = 0, [[maybe_unused]]
bool gui_scope =
true)
override {
…}
1302 void UpdateStatistics(
bool)
override {}
1309 this->excluded_data = this->GetExcludedCargoTypes();
1313 DataSet &dataset = this->data.emplace_back();
1314 dataset.colour = cs->legend_colour;
1315 dataset.exclude_bit = cs->Index();
1317 for (uint j = 0; j != this->num_on_x_axis; j++) {
1318 dataset.values[j] = GetTransportedGoodsIncome(10, 20, j * 4 + 4, cs->Index());
1324static constexpr NWidgetPart _nested_cargo_payment_rates_widgets[] = {
1333 NWidget(
WWT_TEXT, INVALID_COLOUR,
WID_GRAPH_HEADER),
SetFill(1, 0),
SetResize(1, 0),
SetPadding(2, 0, 2, 0),
SetStringTip(STR_GRAPH_CARGO_PAYMENT_RATES_TITLE),
SetTextStyle(TC_BLACK,
FS_SMALL),
SetAlignment(
SA_CENTER),
1342 NWidget(
WWT_MATRIX, COLOUR_BROWN,
WID_GRAPH_MATRIX),
SetFill(1, 0),
SetResize(0, 2),
SetMatrixDataTip(1, 0, STR_GRAPH_CARGO_PAYMENT_TOGGLE_CARGO),
SetScrollbar(
WID_GRAPH_MATRIX_SCROLLBAR),
1350 NWidget(
WWT_TEXT, INVALID_COLOUR,
WID_GRAPH_FOOTER),
SetFill(1, 0),
SetResize(1, 0),
SetPadding(2, 0, 2, 0),
SetTextStyle(TC_BLACK,
FS_SMALL),
SetAlignment(
SA_CENTER),
1357 WDP_AUTO,
"graph_cargo_payment_rates", 0, 0,
1360 _nested_cargo_payment_rates_widgets
1364void ShowCargoPaymentRates()
1366 AllocateWindowDescFront<PaymentRatesGraphWindow>(_cargo_payment_rates_desc, 0);
1376 uint score_info_left = 0;
1377 uint score_info_right = 0;
1381 uint bar_height = 0;
1382 uint score_detail_left = 0;
1383 uint score_detail_right = 0;
1387 this->UpdateCompanyStats();
1393 void UpdateCompanyStats()
1411 uint score_info_width = 0;
1412 for (uint i = SCORE_BEGIN; i <
SCORE_END; i++) {
1425 int max = -(999999999 - 500);
1446 this->score_info_left = rtl ? right - score_info_width :
left;
1447 this->score_info_right = rtl ? right :
left + score_info_width;
1449 this->score_detail_left = rtl ?
left : right - score_detail_width;
1450 this->score_detail_right = rtl ?
left + score_detail_width : right;
1453 this->bar_right = this->bar_left + this->bar_width - 1;
1461 if (this->company == CompanyID::Invalid())
return;
1480 int64_t val = _score_part[company][score_type];
1490 uint bar_top =
CentreBounds(r.top, r.bottom, this->bar_height);
1493 DrawString(this->score_info_left, this->score_info_right, text_top, STR_PERFORMANCE_DETAIL_VEHICLES + score_type);
1499 uint x = Clamp<int64_t>(val, 0, needed) * this->bar_width / needed;
1502 x = this->bar_right - x;
1504 x = this->bar_left + x;
1508 if (x != this->bar_left)
GfxFillRect(this->bar_left, bar_top, x, bar_top + this->bar_height - 1, rtl ? colour_notdone : colour_done);
1509 if (x != this->bar_right)
GfxFillRect(x, bar_top, this->bar_right, bar_top + this->bar_height - 1, rtl ? colour_done : colour_notdone);
1512 DrawString(this->bar_left, this->bar_right, text_top,
GetString(STR_PERFORMANCE_DETAIL_PERCENT, Clamp<int64_t>(val, 0, needed) * 100 / needed), TC_FROMSTRING,
SA_HOR_CENTER);
1515 if (score_type == SCORE_LOAN) val = needed - val;
1519 switch (score_type) {
1520 case SCORE_MIN_PROFIT:
1521 case SCORE_MIN_INCOME:
1522 case SCORE_MAX_INCOME:
1525 DrawString(this->score_detail_left, this->score_detail_right, text_top,
GetString(STR_PERFORMANCE_DETAIL_AMOUNT_CURRENCY, val, needed));
1528 DrawString(this->score_detail_left, this->score_detail_right, text_top,
GetString(STR_PERFORMANCE_DETAIL_AMOUNT_INT, val, needed));
1550 if (--this->timeout == 0) {
1551 this->UpdateCompanyStats();
1561 void OnInvalidateData([[maybe_unused]]
int data = 0, [[maybe_unused]]
bool gui_scope =
true)
override
1563 if (!gui_scope)
return;
1565 for (
CompanyID i = CompanyID::Begin(); i < MAX_COMPANIES; ++i) {
1573 this->company = CompanyID::Invalid();
1576 if (this->company == CompanyID::Invalid()) {
1578 this->company = c->index;
1584 if (this->company != CompanyID::Invalid()) {
1561 void OnInvalidateData([[maybe_unused]]
int data = 0, [[maybe_unused]]
bool gui_scope =
true)
override {
…}
1590CompanyID PerformanceRatingDetailWindow::company = CompanyID::Invalid();
1597 static inline constexpr StringID RANGE_LABELS[] = {
1598 STR_GRAPH_INDUSTRY_RANGE_PRODUCED,
1599 STR_GRAPH_INDUSTRY_RANGE_TRANSPORTED
1602 static inline CargoTypes excluded_cargo_types{};
1609 this->month_increment = 1;
1612 this->ranges = RANGE_LABELS;
1614 this->InitializeWindow(
window_number, STR_GRAPH_LAST_24_MINUTES_TIME_LABEL);
1621 for (
const auto &p : i->
produced) {
1627 CargoTypes &GetExcludedCargoTypes()
const override
1629 return IndustryProductionGraphWindow::excluded_cargo_types;
1639 void UpdateStatistics(
bool initialize)
override
1648 if (!initialize && this->excluded_data == this->GetExcludedCargoTypes() && this->num_on_x_axis == this->num_vert_lines && this->year == yr && this->month == mo) {
1653 this->excluded_data = this->GetExcludedCargoTypes();
1660 for (
const auto &p : i->produced) {
1664 DataSet &produced = this->data.emplace_back();
1665 produced.colour = cs->legend_colour;
1666 produced.exclude_bit = cs->
Index();
1667 produced.range_bit = 0;
1673 DataSet &transported = this->data.emplace_back();
1674 transported.colour = cs->legend_colour;
1675 transported.exclude_bit = cs->
Index();
1676 transported.range_bit = 1;
1677 transported.dash = 2;
1688static constexpr NWidgetPart _nested_industry_production_widgets[] = {
1701 NWidget(
WWT_MATRIX, COLOUR_BROWN,
WID_GRAPH_RANGE_MATRIX),
SetFill(1, 0),
SetResize(0, 0),
SetMatrixDataTip(1, 0, STR_GRAPH_CARGO_PAYMENT_TOGGLE_CARGO),
1707 NWidget(
WWT_MATRIX, COLOUR_BROWN,
WID_GRAPH_MATRIX),
SetFill(1, 0),
SetResize(0, 2),
SetMatrixDataTip(1, 0, STR_GRAPH_CARGO_PAYMENT_TOGGLE_CARGO),
SetScrollbar(
WID_GRAPH_MATRIX_SCROLLBAR),
1715 NWidget(
WWT_TEXT, INVALID_COLOUR,
WID_GRAPH_FOOTER),
SetFill(1, 0),
SetResize(1, 0),
SetPadding(2, 0, 2, 0),
SetTextStyle(TC_BLACK,
FS_SMALL),
SetAlignment(
SA_CENTER),
1722 WDP_AUTO,
"graph_industry_production", 0, 0,
1725 _nested_industry_production_widgets
1728void ShowIndustryProductionGraph(
WindowNumber window_number)
1730 AllocateWindowDescFront<IndustryProductionGraphWindow>(_industry_production_desc, window_number);
1740 const StringID performance_tips[] = {
1741 realtime ? STR_PERFORMANCE_DETAIL_VEHICLES_TOOLTIP_PERIODS : STR_PERFORMANCE_DETAIL_VEHICLES_TOOLTIP_YEARS,
1742 STR_PERFORMANCE_DETAIL_STATIONS_TOOLTIP,
1743 realtime ? STR_PERFORMANCE_DETAIL_MIN_PROFIT_TOOLTIP_PERIODS : STR_PERFORMANCE_DETAIL_MIN_PROFIT_TOOLTIP_YEARS,
1744 STR_PERFORMANCE_DETAIL_MIN_INCOME_TOOLTIP,
1745 STR_PERFORMANCE_DETAIL_MAX_INCOME_TOOLTIP,
1746 STR_PERFORMANCE_DETAIL_DELIVERED_TOOLTIP,
1747 STR_PERFORMANCE_DETAIL_CARGO_TOOLTIP,
1748 STR_PERFORMANCE_DETAIL_MONEY_TOOLTIP,
1749 STR_PERFORMANCE_DETAIL_LOAN_TOOLTIP,
1750 STR_PERFORMANCE_DETAIL_TOTAL_TOOLTIP,
1757 auto panel = std::make_unique<NWidgetBackground>(
WWT_PANEL, COLOUR_BROWN, widnum);
1758 panel->SetFill(1, 1);
1760 vert->Add(std::move(panel));
1771static constexpr NWidgetPart _nested_performance_rating_detail_widgets[] = {
1784static WindowDesc _performance_rating_detail_desc(
1788 _nested_performance_rating_detail_widgets
1791void ShowPerformanceRatingDetail()
1793 AllocateWindowDescFront<PerformanceRatingDetailWindow>(_performance_rating_detail_desc, 0);
1796void InitializeGraphGui()
1799 PaymentRatesGraphWindow::excluded_cargo_types = {};
1800 IndustryProductionGraphWindow::excluded_cargo_types = {};
debug_inline constexpr bool HasBit(const T x, const uint8_t y)
Checks if a bit in a value is set.
constexpr T SetBit(T &x, const uint8_t y)
Set a bit in a variable.
constexpr uint CountBits(T value)
Counts the number of set bits in a variable.
constexpr T ToggleBit(T &x, const uint8_t y)
Toggles a bit in a variable.
uint8_t CargoType
Cargo slots to indicate a cargo type within a game.
bool IsValidCargoType(CargoType cargo)
Test whether cargo type is not INVALID_CARGO.
static const CargoType NUM_CARGO
Maximum number of cargo types in a game.
std::span< const CargoSpec * > _sorted_standard_cargo_specs
Standard cargo specifications sorted alphabetically by name.
CargoTypes _standard_cargo_mask
Bitmask of real cargo types available.
std::vector< const CargoSpec * > _sorted_cargo_specs
Cargo specifications sorted alphabetically by name.
Types/functions related to cargoes.
constexpr bool Test(Tvalue_type value) const
Test if the value-th bit is set.
constexpr Tstorage base() const noexcept
Retrieve the raw value behind this bit set.
constexpr Timpl & Flip(Tvalue_type value)
Flip the value-th bit.
constexpr Timpl & Set()
Set all bits.
void UpdateStatistics(bool initialize) override
Update the statistics.
An interval timer will fire every interval, and will continue to fire until it is deleted.
static constexpr TimerGameTick::Ticks DAY_TICKS
1 day is 74 ticks; TimerGameCalendar::date_fract used to be uint16_t and incremented by 885.
static Year year
Current year, starting at 0.
static Month month
Current month (0..11).
static bool UsingWallclockUnits(bool newgame=false)
Check if we are using wallclock units.
uint8_t Month
Type for the month, note: 0 based, i.e.
Definition of stuff that is very close to a company, like the company struct itself.
void DrawCompanyIcon(CompanyID c, int x, int y)
Draw the icon of a company.
GUI Functions related to companies.
Functions to handle different currencies.
const CurrencySpec & GetCurrency()
Get the currently selected currency.
int UpdateCompanyRatingAndValue(Company *c, bool update)
if update is set to true, the economy is updated with this score (also the house is updated,...
const ScoreInfo _score_info[]
Score info, values used for computing the detailed performance rating.
Functions related to the economy.
ScoreID
Score categories in the detailed performance rating.
@ SCORE_END
How many scores are there..
@ SCORE_TOTAL
This must always be the last entry.
static constexpr int SCORE_MAX
The max score that can be in the performance history.
int GetCharacterHeight(FontSize size)
Get height of a character for a given font size.
Dimension maxdim(const Dimension &d1, const Dimension &d2)
Compute bounding box of both dimensions.
int CentreBounds(int min, int max, int size)
Determine where to position a centred object.
Dimension GetSpriteSize(SpriteID sprid, Point *offset, ZoomLevel zoom)
Get the size of a sprite.
Dimension GetStringBoundingBox(std::string_view str, FontSize start_fontsize)
Return the string dimension in pixels.
int DrawString(int left, int right, int top, std::string_view str, TextColour colour, StringAlignment align, bool underline, FontSize fontsize)
Draw string, possibly truncated to make it fit in its allocated space.
void GfxFillRect(int left, int top, int right, int bottom, int colour, FillRectMode mode)
Applies a certain FillRectMode-operation to a rectangle [left, right] x [top, bottom] on the screen.
bool DrawStringMultiLineWithClipping(int left, int right, int top, int bottom, std::string_view str, TextColour colour, StringAlignment align, bool underline, FontSize fontsize)
Draw a multiline string, possibly over multiple lines, if the region is within the current display cl...
Functions related to the gfx engine.
@ FS_SMALL
Index of the small font in the font tables.
@ FS_NORMAL
Index of the normal font in the font tables.
@ SA_LEFT
Left align the text.
@ SA_RIGHT
Right align the text (must be a single bit).
@ SA_HOR_CENTER
Horizontally center the text.
@ SA_FORCE
Force the alignment, i.e. don't swap for RTL languages.
@ SA_CENTER
Center both horizontally and vertically.
TextColour
Colour of the strings, see _string_colourmap in table/string_colours.h or docs/ottd-colourtext-palett...
constexpr double INT64_MAX_IN_DOUBLE
The biggest double that when cast to int64_t still fits in a int64_t.
static std::unique_ptr< NWidgetBase > MakeNWidgetCompanyLines()
Construct a vertical list of buttons, one for each company.
std::unique_ptr< NWidgetBase > MakeCompanyButtonRowsGraphGUI()
Make a number of rows with buttons for each company for the performance rating detail window.
static std::unique_ptr< NWidgetBase > MakePerformanceDetailPanels()
Make a vertical list of panels for outputting score details.
void SetDirty() const
Mark entire window as dirty (in need of re-paint)
constexpr bool IsInsideMM(const size_t x, const size_t min, const size_t max) noexcept
Checks if a value is in an interval.
constexpr T abs(const T a)
Returns the absolute value of (scalar) variable.
uint8_t GetColourGradient(Colours colour, ColourShade shade)
Get colour gradient palette index.
static const uint8_t PC_WHITE
White palette colour.
constexpr uint8_t GREY_SCALE(uint8_t level)
Return the colour for a particular greyscale level.
static const uint8_t PC_BLACK
Black palette colour.
A number of safeguards to prevent using unsafe methods.
ClientSettings _settings_client
The current settings for this game.
This file contains all sprite-related enums and defines.
Definition of base types and functions in a cross-platform compatible way.
#define lengthof(array)
Return the length of an fixed size array.
uint64_t GetParamMaxValue(uint64_t max_value, uint min_count, FontSize size)
Get some number that is suitable for string size computations.
std::string GetString(StringID string)
Resolve the given StringID into a std::string with formatting but no parameters.
TextDirection _current_text_dir
Text direction of the currently selected language.
Functions related to OTTD's strings.
uint32_t StringID
Numeric value that represents a string, independent of the selected language.
@ TD_RTL
Text is written right-to-left by default.
uint legend_width
Width of legend 'blob'.
void OnClick(Point pt, WidgetID widget, int click_count) override
A click with the left mouse button has been made on the window.
void OnInit() override
Notification that the nested widget tree gets initialized.
void UpdateWidgetSize(WidgetID widget, Dimension &size, const Dimension &padding, Dimension &fill, Dimension &resize) override
Update size and resize step of a widget in the window.
void DrawWidget(const Rect &r, WidgetID widget) const override
Draw the contents of a nested widget.
CargoTypes cargo_types
Cargo types that can be selected.
void OnResize() override
Called after the window got resized.
Scrollbar * vscroll
Cargo list scrollbar.
uint line_height
Pixel height of each cargo type row.
uint GetYLabelWidth(ValuesInterval current_interval, int num_hori_lines) const
Get width for Y labels.
void OnGameTick() override
Called once per (game) tick.
static const int MIN_GRID_PIXEL_SIZE
Minimum distance between graph lines.
static const int GRAPH_NUM_MONTHS
Number of months displayed in the graph.
void DrawWidget(const Rect &r, WidgetID widget) const override
Draw the contents of a nested widget.
static const int ECONOMY_QUARTER_MINUTES
Minutes per economic quarter.
static const int GRAPH_PAYMENT_RATE_STEPS
Number of steps on Payment rate graph.
void OnMouseOver(Point pt, WidgetID widget) override
The mouse is currently moving over the window or has just moved outside of the window.
std::span< const OverflowSafeInt64 > GetDataSetRange(const DataSet &dataset) const
Get appropriate part of dataset values for the current number of horizontal points.
void OnClick(Point pt, WidgetID widget, int click_count) override
A click with the left mouse button has been made on the window.
static const int ECONOMY_MONTH_MINUTES
Minutes per economic month.
uint64_t excluded_data
bitmask of the datasets that shouldn't be displayed.
bool draw_dates
Should we draw months and years on the time axis?
static const int MIN_GRAPH_NUM_LINES_Y
Minimal number of horizontal lines to draw.
uint64_t excluded_range
bitmask of ranges that should not be displayed.
uint8_t highlight_range
Data range that should be highlighted, or UINT8_MAX for none.
bool highlight_state
Current state of highlight, toggled every TIMER_BLINK_INTERVAL period.
uint8_t month_increment
month increment between vertical lines. must be divisor of 12.
static const int PAYMENT_GRAPH_X_STEP_DAYS
X-axis step label for cargo payment rates "Days in transit".
ValuesInterval GetValuesInterval(int num_hori_lines) const
Get the interval that contains the graph's data.
static const TextColour GRAPH_AXIS_LABEL_COLOUR
colour of the graph axis label.
void OnInvalidateData(int data=0, bool gui_scope=true) override
Some data on this window has become invalid.
void DrawGraph(Rect r) const
Actually draw the graph.
static const int PAYMENT_GRAPH_X_STEP_SECONDS
X-axis step label for cargo payment rates "Seconds in transit".
void UpdateWidgetSize(WidgetID widget, Dimension &size, const Dimension &padding, Dimension &fill, Dimension &resize) override
Update size and resize step of a widget in the window.
uint8_t highlight_data
Data set that should be highlighted, or UINT8_MAX for none.
Specification of a cargo type.
static CargoSpec * Get(size_t index)
Retrieve cargo details for the given cargo type.
CargoType Index() const
Determines index of this cargospec.
StringID name
Name of this type of cargo.
GUISettings gui
settings related to the GUI
Colours colour
Company colour.
std::array< CompanyEconomyEntry, MAX_HISTORY_QUARTERS > old_economy
Economic data of the company of the last MAX_HISTORY_QUARTERS quarters.
uint8_t num_valid_stat_ent
Number of valid statistical entries in old_economy.
uint16_t rate
The conversion rate compared to the base currency.
Dimensions (a width and height) of a rectangle in 2D.
uint8_t graph_line_thickness
the thickness of the lines in the various graph guis
void OnClick(Point pt, WidgetID widget, int click_count) override
A click with the left mouse button has been made on the window.
void DrawWidget(const Rect &r, WidgetID widget) const override
Draw the contents of a nested widget.
void OnInvalidateData(int data=0, bool gui_scope=true) override
Some data on this window has become invalid.
std::string GetWidgetString(WidgetID widget, StringID stringid) const override
Get the raw string for a widget.
Defines the internal data of a functional industry.
ProducedCargoes produced
produced cargo slots
void UpdatePaymentRates()
Update the payment rates according to the latest information.
void OnInvalidateData(int data=0, bool gui_scope=true) override
Some data on this window has become invalid.
const IntervalTimer< TimerWindow > update_payment_interval
Update the payment rates on a regular interval.
void OnClick(Point pt, WidgetID widget, int click_count) override
A click with the left mouse button has been made on the window.
Coordinates of a point in 2D.
static Pool::IterateWrapper< Titem > Iterate(size_t from=0)
Returns an iterable ensemble of all valid Titem.
static Titem * Get(auto index)
Returns Titem with given index.
static bool IsValidID(auto index)
Tests whether given index can be used to get valid (non-nullptr) Titem.
static Titem * GetIfValid(auto index)
Returns Titem with given index.
constexpr uint Horizontal() const
Get total horizontal padding of RectPadding.
constexpr uint Vertical() const
Get total vertical padding of RectPadding.
Specification of a rectangle with absolute coordinates of all edges.
Rect WithWidth(int width, bool end) const
Copy Rect and set its width.
Rect Shrink(int s) const
Copy and shrink Rect by s pixels.
Rect WithHeight(int height, bool end=false) const
Copy Rect and set its height.
Rect Indent(int indent, bool end) const
Copy Rect and indent it from its position.
Rect Translate(int x, int y) const
Copy and translate Rect by x,y pixels.
int needed
How much you need to get the perfect score.
int score
How much score it will give.
Iterable ensemble of each set bit in a value.
Templated helper to make a type-safe 'typedef' representing a single POD value.
Contains the interval of a graph's data.
OverflowSafeInt64 lowest
Lowest value of this interval. Must be zero or less.
OverflowSafeInt64 highest
Highest value of this interval. Must be zero or greater.
High level window description.
Number to differentiate different windows of the same class.
Data structure for an opened window.
void FinishInitNested(WindowNumber window_number=0)
Perform the second part of the initialization of a nested widget tree.
void InvalidateData(int data=0, bool gui_scope=true)
Mark this window's data as invalid (in need of re-computing)
void RaiseWidget(WidgetID widget_index)
Marks a widget as raised.
virtual std::string GetWidgetString(WidgetID widget, StringID stringid) const
Get the raw string for a widget.
ResizeInfo resize
Resize information.
void CreateNestedTree()
Perform the first part of the initialization of a nested widget tree.
bool IsWidgetDisabled(WidgetID widget_index) const
Gets the enabled/disabled status of a widget.
int left
x position of left edge of the window
int GetRowFromWidget(int clickpos, WidgetID widget, int padding, int line_height=-1) const
Compute the row of a widget that a user clicked in.
void LowerWidget(WidgetID widget_index)
Marks a widget as lowered.
void InitNested(WindowNumber number=0)
Perform complete initialization of the Window with nested widgets, to allow use.
const Scrollbar * GetScrollbar(WidgetID widnum) const
Return the Scrollbar to a widget index.
void SetWidgetDisabledState(WidgetID widget_index, bool disab_stat)
Sets the enabled/disabled status of a widget.
int height
Height of the window (number of pixels down in y direction)
int width
width of the window (number of pixels to the right in x direction)
void ToggleWidgetLoweredState(WidgetID widget_index)
Invert the lowered/raised status of a widget.
WindowNumber window_number
Window number within the window class.
Definition of Interval and OneShot timers.
Definition of the game-economy-timer.
Definition of the tick-based game-timer.
Definition of the Window system.
static constexpr std::chrono::milliseconds TIMER_BLINK_INTERVAL
Interval used by blinking interface elements.
void InvalidateWindowData(WindowClass cls, WindowNumber number, int data, bool gui_scope)
Mark window data of the window of a given class and specific window number as invalid (in need of re-...
void SetWindowDirty(WindowClass cls, WindowNumber number)
Mark window as dirty (in need of repainting)
Window functions not directly related to making/drawing windows.
Functions, definitions and such used only by the GUI.
@ Lowered
If set the frame is lowered and the background colour brighter (ie. buttons when pressed)
@ WDP_AUTO
Find a place automatically.
@ WC_PERFORMANCE_HISTORY
Performance history graph; Window numbers:
@ WC_PERFORMANCE_DETAIL
Performance detail window; Window numbers:
@ WC_PAYMENT_RATES
Payment rates graph; Window numbers:
@ WC_GRAPH_LEGEND
Legend for graphs; Window numbers:
@ WC_NONE
No window, redirects to WC_MAIN_WINDOW.
@ WC_OPERATING_PROFIT
Operating profit graph; Window numbers:
@ WC_INDUSTRY_PRODUCTION
Industry production history graph; Window numbers:
@ WC_INDUSTRY_VIEW
Industry view; Window numbers:
@ WC_INCOME_GRAPH
Income graph; Window numbers:
@ WC_DELIVERED_CARGO
Delivered cargo graph; Window numbers:
@ WC_COMPANY_VALUE
Company value graph; Window numbers:
Functions related to zooming.
@ Normal
The normal zoom level.