company_gui.cpp

Go to the documentation of this file.
00001 /* $Id$ */
00002 
00003 /*
00004  * This file is part of OpenTTD.
00005  * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
00006  * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
00007  * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
00008  */
00009 
00012 #include "stdafx.h"
00013 #include "gui.h"
00014 #include "window_gui.h"
00015 #include "textbuf_gui.h"
00016 #include "viewport_func.h"
00017 #include "company_func.h"
00018 #include "command_func.h"
00019 #include "network/network.h"
00020 #include "network/network_gui.h"
00021 #include "network/network_func.h"
00022 #include "vehicle_func.h"
00023 #include "newgrf.h"
00024 #include "company_manager_face.h"
00025 #include "strings_func.h"
00026 #include "date_func.h"
00027 #include "widgets/dropdown_type.h"
00028 #include "tilehighlight_func.h"
00029 #include "sprite.h"
00030 #include "company_base.h"
00031 #include "core/geometry_func.hpp"
00032 #include "economy_func.h"
00033 #include "object_type.h"
00034 #include "rail.h"
00035 #include "engine_base.h"
00036 #include "window_func.h"
00037 #include "road_func.h"
00038 #include "water.h"
00039 #include "station_func.h"
00040 
00041 #include "table/strings.h"
00042 
00044 static const uint EXP_LINESPACE  = 2;      
00045 static const uint EXP_BLOCKSPACE = 10;     
00046 
00047 static void DoSelectCompanyManagerFace(Window *parent);
00048 static void ShowCompanyInfrastructure(CompanyID company);
00049 
00051 static ExpensesType _expenses_list_1[] = {
00052   EXPENSES_CONSTRUCTION,
00053   EXPENSES_NEW_VEHICLES,
00054   EXPENSES_TRAIN_RUN,
00055   EXPENSES_ROADVEH_RUN,
00056   EXPENSES_AIRCRAFT_RUN,
00057   EXPENSES_SHIP_RUN,
00058   EXPENSES_PROPERTY,
00059   EXPENSES_TRAIN_INC,
00060   EXPENSES_ROADVEH_INC,
00061   EXPENSES_AIRCRAFT_INC,
00062   EXPENSES_SHIP_INC,
00063   EXPENSES_LOAN_INT,
00064   EXPENSES_OTHER,
00065 };
00066 
00068 static ExpensesType _expenses_list_2[] = {
00069   EXPENSES_TRAIN_INC,
00070   EXPENSES_ROADVEH_INC,
00071   EXPENSES_AIRCRAFT_INC,
00072   EXPENSES_SHIP_INC,
00073   INVALID_EXPENSES,
00074   EXPENSES_TRAIN_RUN,
00075   EXPENSES_ROADVEH_RUN,
00076   EXPENSES_AIRCRAFT_RUN,
00077   EXPENSES_SHIP_RUN,
00078   EXPENSES_PROPERTY,
00079   EXPENSES_LOAN_INT,
00080   INVALID_EXPENSES,
00081   EXPENSES_CONSTRUCTION,
00082   EXPENSES_NEW_VEHICLES,
00083   EXPENSES_OTHER,
00084   INVALID_EXPENSES,
00085 };
00086 
00088 struct ExpensesList {
00089   const ExpensesType *et;   
00090   const uint length;        
00091   const uint num_subtotals; 
00092 
00093   ExpensesList(ExpensesType *et, int length, int num_subtotals) : et(et), length(length), num_subtotals(num_subtotals)
00094   {
00095   }
00096 
00097   uint GetHeight() const
00098   {
00099     /* heading + line + texts of expenses + sub-totals + total line + total text */
00100     return FONT_HEIGHT_NORMAL + EXP_LINESPACE + this->length * FONT_HEIGHT_NORMAL + num_subtotals * (EXP_BLOCKSPACE + EXP_LINESPACE) + EXP_LINESPACE + FONT_HEIGHT_NORMAL;
00101   }
00102 
00104   uint GetCategoriesWidth() const
00105   {
00106     uint width = 0;
00107     bool invalid_expenses_measured = false; // Measure 'Total' width only once.
00108     for (uint i = 0; i < this->length; i++) {
00109       ExpensesType et = this->et[i];
00110       if (et == INVALID_EXPENSES) {
00111         if (!invalid_expenses_measured) {
00112           width = max(width, GetStringBoundingBox(STR_FINANCES_TOTAL_CAPTION).width);
00113           invalid_expenses_measured = true;
00114         }
00115       } else {
00116         width = max(width, GetStringBoundingBox(STR_FINANCES_SECTION_CONSTRUCTION + et).width);
00117       }
00118     }
00119     return width;
00120   }
00121 };
00122 
00123 static const ExpensesList _expenses_list_types[] = {
00124   ExpensesList(_expenses_list_1, lengthof(_expenses_list_1), 0),
00125   ExpensesList(_expenses_list_2, lengthof(_expenses_list_2), 3),
00126 };
00127 
00129 enum CompanyFinancesWindowWidgets {
00130   CFW_CAPTION,       
00131   CFW_TOGGLE_SIZE,   
00132   CFW_SEL_PANEL,     
00133   CFW_EXPS_CATEGORY, 
00134   CFW_EXPS_PRICE1,   
00135   CFW_EXPS_PRICE2,   
00136   CFW_EXPS_PRICE3,   
00137   CFW_TOTAL_PANEL,   
00138   CFW_SEL_MAXLOAN,   
00139   CFW_BALANCE_VALUE, 
00140   CFW_LOAN_VALUE,    
00141   CFW_LOAN_LINE,     
00142   CFW_TOTAL_VALUE,   
00143   CFW_MAXLOAN_GAP,   
00144   CFW_MAXLOAN_VALUE, 
00145   CFW_SEL_BUTTONS,   
00146   CFW_INCREASE_LOAN, 
00147   CFW_REPAY_LOAN,    
00148   CFW_INFRASTRUCTURE,
00149 };
00150 
00156 static void DrawCategories(const Rect &r)
00157 {
00158   int y = r.top;
00159 
00160   DrawString(r.left, r.right, y, STR_FINANCES_EXPENDITURE_INCOME_TITLE, TC_FROMSTRING, SA_HOR_CENTER, true);
00161   y += FONT_HEIGHT_NORMAL + EXP_LINESPACE;
00162 
00163   int type = _settings_client.gui.expenses_layout;
00164   for (uint i = 0; i < _expenses_list_types[type].length; i++) {
00165     const ExpensesType et = _expenses_list_types[type].et[i];
00166     if (et == INVALID_EXPENSES) {
00167       y += EXP_LINESPACE;
00168       DrawString(r.left, r.right, y, STR_FINANCES_TOTAL_CAPTION, TC_FROMSTRING, SA_RIGHT);
00169       y += FONT_HEIGHT_NORMAL + EXP_BLOCKSPACE;
00170     } else {
00171       DrawString(r.left, r.right, y, STR_FINANCES_SECTION_CONSTRUCTION + et);
00172       y += FONT_HEIGHT_NORMAL;
00173     }
00174   }
00175 
00176   DrawString(r.left, r.right, y + EXP_LINESPACE, STR_FINANCES_TOTAL_CAPTION, TC_FROMSTRING, SA_RIGHT);
00177 }
00178 
00186 static void DrawPrice(Money amount, int left, int right, int top)
00187 {
00188   StringID str = STR_FINANCES_NEGATIVE_INCOME;
00189   if (amount < 0) {
00190     amount = -amount;
00191     str++;
00192   }
00193   SetDParam(0, amount);
00194   DrawString(left, right, top, str, TC_FROMSTRING, SA_RIGHT);
00195 }
00196 
00204 static void DrawYearColumn(const Rect &r, int year, const Money (*tbl)[EXPENSES_END])
00205 {
00206   int y = r.top;
00207 
00208   SetDParam(0, year);
00209   DrawString(r.left, r.right, y, STR_FINANCES_YEAR, TC_FROMSTRING, SA_RIGHT, true);
00210   y += FONT_HEIGHT_NORMAL + EXP_LINESPACE;
00211 
00212   Money sum = 0;
00213   Money subtotal = 0;
00214   int type = _settings_client.gui.expenses_layout;
00215   for (uint i = 0; i < _expenses_list_types[type].length; i++) {
00216     const ExpensesType et = _expenses_list_types[type].et[i];
00217     if (et == INVALID_EXPENSES) {
00218       Money cost = subtotal;
00219       subtotal = 0;
00220       GfxFillRect(r.left, y, r.right, y, PC_BLACK);
00221       y += EXP_LINESPACE;
00222       DrawPrice(cost, r.left, r.right, y);
00223       y += FONT_HEIGHT_NORMAL + EXP_BLOCKSPACE;
00224     } else {
00225       Money cost = (*tbl)[et];
00226       subtotal += cost;
00227       sum += cost;
00228       if (cost != 0) DrawPrice(cost, r.left, r.right, y);
00229       y += FONT_HEIGHT_NORMAL;
00230     }
00231   }
00232 
00233   GfxFillRect(r.left, y, r.right, y, PC_BLACK);
00234   y += EXP_LINESPACE;
00235   DrawPrice(sum, r.left, r.right, y);
00236 }
00237 
00238 static const NWidgetPart _nested_company_finances_widgets[] = {
00239   NWidget(NWID_HORIZONTAL),
00240     NWidget(WWT_CLOSEBOX, COLOUR_GREY),
00241     NWidget(WWT_CAPTION, COLOUR_GREY, CFW_CAPTION), SetDataTip(STR_FINANCES_CAPTION, STR_TOOLTIP_WINDOW_TITLE_DRAG_THIS),
00242     NWidget(WWT_IMGBTN, COLOUR_GREY, CFW_TOGGLE_SIZE), SetDataTip(SPR_LARGE_SMALL_WINDOW, STR_TOOLTIP_TOGGLE_LARGE_SMALL_WINDOW),
00243     NWidget(WWT_SHADEBOX, COLOUR_GREY),
00244     NWidget(WWT_STICKYBOX, COLOUR_GREY),
00245   EndContainer(),
00246   NWidget(NWID_SELECTION, INVALID_COLOUR, CFW_SEL_PANEL),
00247     NWidget(WWT_PANEL, COLOUR_GREY),
00248       NWidget(NWID_HORIZONTAL), SetPadding(WD_FRAMERECT_TOP, WD_FRAMERECT_RIGHT, WD_FRAMERECT_BOTTOM, WD_FRAMERECT_LEFT), SetPIP(0, 9, 0),
00249         NWidget(WWT_EMPTY, COLOUR_GREY, CFW_EXPS_CATEGORY), SetMinimalSize(120, 0), SetFill(0, 0),
00250         NWidget(WWT_EMPTY, COLOUR_GREY, CFW_EXPS_PRICE1), SetMinimalSize(86, 0), SetFill(0, 0),
00251         NWidget(WWT_EMPTY, COLOUR_GREY, CFW_EXPS_PRICE2), SetMinimalSize(86, 0), SetFill(0, 0),
00252         NWidget(WWT_EMPTY, COLOUR_GREY, CFW_EXPS_PRICE3), SetMinimalSize(86, 0), SetFill(0, 0),
00253       EndContainer(),
00254     EndContainer(),
00255   EndContainer(),
00256   NWidget(WWT_PANEL, COLOUR_GREY),
00257     NWidget(NWID_HORIZONTAL), SetPadding(WD_FRAMERECT_TOP, WD_FRAMERECT_RIGHT, WD_FRAMERECT_BOTTOM, WD_FRAMERECT_LEFT),
00258       NWidget(NWID_VERTICAL), // Vertical column with 'bank balance', 'loan'
00259         NWidget(WWT_TEXT, COLOUR_GREY), SetDataTip(STR_FINANCES_BANK_BALANCE_TITLE, STR_NULL), SetFill(1, 0),
00260         NWidget(WWT_TEXT, COLOUR_GREY), SetDataTip(STR_FINANCES_LOAN_TITLE, STR_NULL), SetFill(1, 0),
00261         NWidget(NWID_SPACER), SetFill(0, 1),
00262       EndContainer(),
00263       NWidget(NWID_SPACER), SetFill(0, 0), SetMinimalSize(30, 0),
00264       NWidget(NWID_VERTICAL), // Vertical column with bank balance amount, loan amount, and total.
00265         NWidget(WWT_TEXT, COLOUR_GREY, CFW_BALANCE_VALUE), SetDataTip(STR_NULL, STR_NULL),
00266         NWidget(WWT_TEXT, COLOUR_GREY, CFW_LOAN_VALUE), SetDataTip(STR_NULL, STR_NULL),
00267         NWidget(WWT_EMPTY, COLOUR_GREY, CFW_LOAN_LINE), SetMinimalSize(0, 2), SetFill(1, 0),
00268         NWidget(WWT_TEXT, COLOUR_GREY, CFW_TOTAL_VALUE), SetDataTip(STR_NULL, STR_NULL),
00269       EndContainer(),
00270       NWidget(NWID_SELECTION, INVALID_COLOUR, CFW_SEL_MAXLOAN),
00271         NWidget(NWID_HORIZONTAL),
00272           NWidget(NWID_SPACER), SetFill(0, 1), SetMinimalSize(25, 0),
00273           NWidget(NWID_VERTICAL), // Max loan information
00274             NWidget(WWT_EMPTY, COLOUR_GREY, CFW_MAXLOAN_GAP), SetFill(0, 0),
00275             NWidget(WWT_TEXT, COLOUR_GREY, CFW_MAXLOAN_VALUE), SetDataTip(STR_FINANCES_MAX_LOAN, STR_NULL),
00276             NWidget(NWID_SPACER), SetFill(0, 1),
00277           EndContainer(),
00278         EndContainer(),
00279       EndContainer(),
00280       NWidget(NWID_SPACER), SetFill(1, 1),
00281     EndContainer(),
00282   EndContainer(),
00283   NWidget(NWID_SELECTION, INVALID_COLOUR, CFW_SEL_BUTTONS),
00284     NWidget(NWID_HORIZONTAL, NC_EQUALSIZE),
00285       NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CFW_INCREASE_LOAN), SetFill(1, 0), SetDataTip(STR_FINANCES_BORROW_BUTTON, STR_FINANCES_BORROW_TOOLTIP),
00286       NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CFW_REPAY_LOAN), SetFill(1, 0), SetDataTip(STR_FINANCES_REPAY_BUTTON, STR_FINANCES_REPAY_TOOLTIP),
00287       NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CFW_INFRASTRUCTURE), SetFill(1, 0), SetDataTip(STR_FINANCES_INFRASTRUCTURE_BUTTON, STR_COMPANY_VIEW_INFRASTRUCTURE_TOOLTIP),
00288     EndContainer(),
00289   EndContainer(),
00290 };
00291 
00296 struct CompanyFinancesWindow : Window {
00297   static Money max_money; 
00298   bool small;             
00299 
00300   CompanyFinancesWindow(const WindowDesc *desc, CompanyID company) : Window()
00301   {
00302     this->small = false;
00303     this->CreateNestedTree(desc);
00304     this->SetupWidgets();
00305     this->FinishInitNested(desc, company);
00306 
00307     this->owner = (Owner)this->window_number;
00308   }
00309 
00310   virtual void SetStringParameters(int widget) const
00311   {
00312     switch (widget) {
00313       case CFW_CAPTION:
00314         SetDParam(0, (CompanyID)this->window_number);
00315         SetDParam(1, (CompanyID)this->window_number);
00316         break;
00317 
00318       case CFW_MAXLOAN_VALUE:
00319         SetDParam(0, _economy.max_loan);
00320         break;
00321 
00322       case CFW_INCREASE_LOAN:
00323       case CFW_REPAY_LOAN:
00324         SetDParam(0, LOAN_INTERVAL);
00325         break;
00326     }
00327   }
00328 
00329   virtual void UpdateWidgetSize(int widget, Dimension *size, const Dimension &padding, Dimension *fill, Dimension *resize)
00330   {
00331     int type = _settings_client.gui.expenses_layout;
00332     switch (widget) {
00333       case CFW_EXPS_CATEGORY:
00334         size->width  = _expenses_list_types[type].GetCategoriesWidth();
00335         size->height = _expenses_list_types[type].GetHeight();
00336         break;
00337 
00338       case CFW_EXPS_PRICE1:
00339       case CFW_EXPS_PRICE2:
00340       case CFW_EXPS_PRICE3:
00341         size->height = _expenses_list_types[type].GetHeight();
00342         /* FALL THROUGH */
00343       case CFW_BALANCE_VALUE:
00344       case CFW_LOAN_VALUE:
00345       case CFW_TOTAL_VALUE:
00346         SetDParam(0, CompanyFinancesWindow::max_money);
00347         size->width = max(GetStringBoundingBox(STR_FINANCES_NEGATIVE_INCOME).width, GetStringBoundingBox(STR_FINANCES_POSITIVE_INCOME).width) + padding.width;
00348         break;
00349 
00350       case CFW_MAXLOAN_GAP:
00351         size->height = FONT_HEIGHT_NORMAL;
00352         break;
00353     }
00354   }
00355 
00356   virtual void DrawWidget(const Rect &r, int widget) const
00357   {
00358     switch (widget) {
00359       case CFW_EXPS_CATEGORY:
00360         DrawCategories(r);
00361         break;
00362 
00363       case CFW_EXPS_PRICE1:
00364       case CFW_EXPS_PRICE2:
00365       case CFW_EXPS_PRICE3: {
00366         const Company *c = Company::Get((CompanyID)this->window_number);
00367         int age = min(_cur_year - c->inaugurated_year, 2);
00368         int wid_offset = widget - CFW_EXPS_PRICE1;
00369         if (wid_offset <= age) {
00370           DrawYearColumn(r, _cur_year - (age - wid_offset), c->yearly_expenses + (age - wid_offset));
00371         }
00372         break;
00373       }
00374 
00375       case CFW_BALANCE_VALUE: {
00376         const Company *c = Company::Get((CompanyID)this->window_number);
00377         SetDParam(0, c->money);
00378         DrawString(r.left, r.right, r.top, STR_FINANCES_TOTAL_CURRENCY, TC_FROMSTRING, SA_RIGHT);
00379         break;
00380       }
00381 
00382       case CFW_LOAN_VALUE: {
00383         const Company *c = Company::Get((CompanyID)this->window_number);
00384         SetDParam(0, c->current_loan);
00385         DrawString(r.left, r.right, r.top, STR_FINANCES_TOTAL_CURRENCY, TC_FROMSTRING, SA_RIGHT);
00386         break;
00387       }
00388 
00389       case CFW_TOTAL_VALUE: {
00390         const Company *c = Company::Get((CompanyID)this->window_number);
00391         SetDParam(0, c->money - c->current_loan);
00392         DrawString(r.left, r.right, r.top, STR_FINANCES_TOTAL_CURRENCY, TC_FROMSTRING, SA_RIGHT);
00393         break;
00394       }
00395 
00396       case CFW_LOAN_LINE:
00397         GfxFillRect(r.left, r.top, r.right, r.top, PC_BLACK);
00398         break;
00399     }
00400   }
00401 
00406   void SetupWidgets()
00407   {
00408     int plane = this->small ? SZSP_NONE : 0;
00409     this->GetWidget<NWidgetStacked>(CFW_SEL_PANEL)->SetDisplayedPlane(plane);
00410     this->GetWidget<NWidgetStacked>(CFW_SEL_MAXLOAN)->SetDisplayedPlane(plane);
00411 
00412     CompanyID company = (CompanyID)this->window_number;
00413     plane = (company != _local_company) ? SZSP_NONE : 0;
00414     this->GetWidget<NWidgetStacked>(CFW_SEL_BUTTONS)->SetDisplayedPlane(plane);
00415   }
00416 
00417   virtual void OnPaint()
00418   {
00419     if (!this->IsShaded()) {
00420       if (!this->small) {
00421         /* Check that the expenses panel height matches the height needed for the layout. */
00422         int type = _settings_client.gui.expenses_layout;
00423         if (_expenses_list_types[type].GetHeight() != this->GetWidget<NWidgetBase>(CFW_EXPS_CATEGORY)->current_y) {
00424           this->SetupWidgets();
00425           this->ReInit();
00426           return;
00427         }
00428       }
00429 
00430       /* Check that the loan buttons are shown only when the user owns the company. */
00431       CompanyID company = (CompanyID)this->window_number;
00432       int req_plane = (company != _local_company) ? SZSP_NONE : 0;
00433       if (req_plane != this->GetWidget<NWidgetStacked>(CFW_SEL_BUTTONS)->shown_plane) {
00434         this->SetupWidgets();
00435         this->ReInit();
00436         return;
00437       }
00438 
00439       const Company *c = Company::Get(company);
00440       this->SetWidgetDisabledState(CFW_INCREASE_LOAN, c->current_loan == _economy.max_loan); // Borrow button only shows when there is any more money to loan.
00441       this->SetWidgetDisabledState(CFW_REPAY_LOAN, company != _local_company || c->current_loan == 0); // Repay button only shows when there is any more money to repay.
00442     }
00443 
00444     this->DrawWidgets();
00445   }
00446 
00447   virtual void OnClick(Point pt, int widget, int click_count)
00448   {
00449     switch (widget) {
00450       case CFW_TOGGLE_SIZE: // toggle size
00451         this->small = !this->small;
00452         this->SetupWidgets();
00453         if (this->IsShaded()) {
00454           /* Finances window is not resizable, so size hints given during unshading have no effect
00455            * on the changed appearance of the window. */
00456           this->SetShaded(false);
00457         } else {
00458           this->ReInit();
00459         }
00460         break;
00461 
00462       case CFW_INCREASE_LOAN: // increase loan
00463         DoCommandP(0, 0, _ctrl_pressed, CMD_INCREASE_LOAN | CMD_MSG(STR_ERROR_CAN_T_BORROW_ANY_MORE_MONEY));
00464         break;
00465 
00466       case CFW_REPAY_LOAN: // repay loan
00467         DoCommandP(0, 0, _ctrl_pressed, CMD_DECREASE_LOAN | CMD_MSG(STR_ERROR_CAN_T_REPAY_LOAN));
00468         break;
00469 
00470       case CFW_INFRASTRUCTURE: // show infrastructure details
00471         ShowCompanyInfrastructure((CompanyID)this->window_number);
00472         break;
00473     }
00474   }
00475 
00476   virtual void OnHundredthTick()
00477   {
00478     const Company *c = Company::Get((CompanyID)this->window_number);
00479     if (c->money > CompanyFinancesWindow::max_money) {
00480       CompanyFinancesWindow::max_money = max(c->money * 2, CompanyFinancesWindow::max_money * 4);
00481       this->SetupWidgets();
00482       this->ReInit();
00483     }
00484   }
00485 };
00486 
00488 Money CompanyFinancesWindow::max_money = INT32_MAX;
00489 
00490 static const WindowDesc _company_finances_desc(
00491   WDP_AUTO, 0, 0,
00492   WC_FINANCES, WC_NONE,
00493   WDF_UNCLICK_BUTTONS,
00494   _nested_company_finances_widgets, lengthof(_nested_company_finances_widgets)
00495 );
00496 
00502 void ShowCompanyFinances(CompanyID company)
00503 {
00504   if (!Company::IsValidID(company)) return;
00505   if (BringWindowToFrontById(WC_FINANCES, company)) return;
00506 
00507   new CompanyFinancesWindow(&_company_finances_desc, company);
00508 }
00509 
00510 /* List of colours for the livery window */
00511 static const StringID _colour_dropdown[] = {
00512   STR_COLOUR_DARK_BLUE,
00513   STR_COLOUR_PALE_GREEN,
00514   STR_COLOUR_PINK,
00515   STR_COLOUR_YELLOW,
00516   STR_COLOUR_RED,
00517   STR_COLOUR_LIGHT_BLUE,
00518   STR_COLOUR_GREEN,
00519   STR_COLOUR_DARK_GREEN,
00520   STR_COLOUR_BLUE,
00521   STR_COLOUR_CREAM,
00522   STR_COLOUR_MAUVE,
00523   STR_COLOUR_PURPLE,
00524   STR_COLOUR_ORANGE,
00525   STR_COLOUR_BROWN,
00526   STR_COLOUR_GREY,
00527   STR_COLOUR_WHITE,
00528 };
00529 
00530 /* Association of liveries to livery classes */
00531 static const LiveryClass _livery_class[LS_END] = {
00532   LC_OTHER,
00533   LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL, LC_RAIL,
00534   LC_ROAD, LC_ROAD,
00535   LC_SHIP, LC_SHIP,
00536   LC_AIRCRAFT, LC_AIRCRAFT, LC_AIRCRAFT,
00537   LC_ROAD, LC_ROAD,
00538 };
00539 
00540 class DropDownListColourItem : public DropDownListItem {
00541 public:
00542   DropDownListColourItem(int result, bool masked) : DropDownListItem(result, masked) {}
00543 
00544   virtual ~DropDownListColourItem() {}
00545 
00546   StringID String() const
00547   {
00548     return _colour_dropdown[this->result];
00549   }
00550 
00551   uint Height(uint width) const
00552   {
00553     return max(FONT_HEIGHT_NORMAL, (byte)14);
00554   }
00555 
00556   bool Selectable() const
00557   {
00558     return true;
00559   }
00560 
00561   void Draw(int left, int right, int top, int bottom, bool sel, int bg_colour) const
00562   {
00563     bool rtl = _current_text_dir == TD_RTL;
00564     DrawSprite(SPR_VEH_BUS_SIDE_VIEW, PALETTE_RECOLOUR_START + this->result, rtl ? right - 16 : left + 16, top + 7);
00565     DrawString(rtl ? left + 2 : left + 32, rtl ? right - 32 : right - 2, top + max(0, 13 - FONT_HEIGHT_NORMAL), this->String(), sel ? TC_WHITE : TC_BLACK);
00566   }
00567 };
00568 
00570 enum SelectCompanyLiveryWindowWidgets {
00571   SCLW_WIDGET_CAPTION,
00572   SCLW_WIDGET_CLASS_GENERAL,
00573   SCLW_WIDGET_CLASS_RAIL,
00574   SCLW_WIDGET_CLASS_ROAD,
00575   SCLW_WIDGET_CLASS_SHIP,
00576   SCLW_WIDGET_CLASS_AIRCRAFT,
00577   SCLW_WIDGET_SPACER_DROPDOWN,
00578   SCLW_WIDGET_PRI_COL_DROPDOWN,
00579   SCLW_WIDGET_SEC_COL_DROPDOWN,
00580   SCLW_WIDGET_MATRIX,
00581 };
00582 
00584 struct SelectCompanyLiveryWindow : public Window {
00585 private:
00586   uint32 sel;
00587   LiveryClass livery_class;
00588   Dimension square;
00589   Dimension box;
00590   uint line_height;
00591 
00592   void ShowColourDropDownMenu(uint32 widget)
00593   {
00594     uint32 used_colours = 0;
00595     const Livery *livery;
00596     LiveryScheme scheme;
00597 
00598     /* Disallow other company colours for the primary colour */
00599     if (HasBit(this->sel, LS_DEFAULT) && widget == SCLW_WIDGET_PRI_COL_DROPDOWN) {
00600       const Company *c;
00601       FOR_ALL_COMPANIES(c) {
00602         if (c->index != _local_company) SetBit(used_colours, c->colour);
00603       }
00604     }
00605 
00606     /* Get the first selected livery to use as the default dropdown item */
00607     for (scheme = LS_BEGIN; scheme < LS_END; scheme++) {
00608       if (HasBit(this->sel, scheme)) break;
00609     }
00610     if (scheme == LS_END) scheme = LS_DEFAULT;
00611     livery = &Company::Get((CompanyID)this->window_number)->livery[scheme];
00612 
00613     DropDownList *list = new DropDownList();
00614     for (uint i = 0; i < lengthof(_colour_dropdown); i++) {
00615       list->push_back(new DropDownListColourItem(i, HasBit(used_colours, i)));
00616     }
00617 
00618     ShowDropDownList(this, list, widget == SCLW_WIDGET_PRI_COL_DROPDOWN ? livery->colour1 : livery->colour2, widget);
00619   }
00620 
00621 public:
00622   SelectCompanyLiveryWindow(const WindowDesc *desc, CompanyID company) : Window()
00623   {
00624     this->livery_class = LC_OTHER;
00625     this->sel = 1;
00626 
00627     this->square = GetSpriteSize(SPR_SQUARE);
00628     this->box    = maxdim(GetSpriteSize(SPR_BOX_CHECKED), GetSpriteSize(SPR_BOX_EMPTY));
00629     this->line_height = max(max(this->square.height, this->box.height), (uint)FONT_HEIGHT_NORMAL) + 4;
00630 
00631     this->InitNested(desc, company);
00632     this->owner = company;
00633     this->LowerWidget(SCLW_WIDGET_CLASS_GENERAL);
00634     this->InvalidateData(1);
00635   }
00636 
00637   virtual void UpdateWidgetSize(int widget, Dimension *size, const Dimension &padding, Dimension *fill, Dimension *resize)
00638   {
00639     switch (widget) {
00640       case SCLW_WIDGET_SPACER_DROPDOWN: {
00641         /* The matrix widget below needs enough room to print all the schemes. */
00642         Dimension d = {0, 0};
00643         for (LiveryScheme scheme = LS_DEFAULT; scheme < LS_END; scheme++) {
00644           d = maxdim(d, GetStringBoundingBox(STR_LIVERY_DEFAULT + scheme));
00645         }
00646         size->width = max(size->width, 5 + this->box.width + d.width + WD_FRAMERECT_RIGHT);
00647         break;
00648       }
00649 
00650       case SCLW_WIDGET_MATRIX: {
00651         uint livery_height = 0;
00652         for (LiveryScheme scheme = LS_DEFAULT; scheme < LS_END; scheme++) {
00653           if (_livery_class[scheme] == this->livery_class && HasBit(_loaded_newgrf_features.used_liveries, scheme)) {
00654             livery_height++;
00655           }
00656         }
00657         size->height = livery_height * this->line_height;
00658         this->GetWidget<NWidgetCore>(SCLW_WIDGET_MATRIX)->widget_data = (livery_height << MAT_ROW_START) | (1 << MAT_COL_START);
00659         break;
00660       }
00661 
00662       case SCLW_WIDGET_SEC_COL_DROPDOWN:
00663         if (!_loaded_newgrf_features.has_2CC) {
00664           size->width = 0;
00665           break;
00666         }
00667         /* FALL THROUGH */
00668       case SCLW_WIDGET_PRI_COL_DROPDOWN: {
00669         int padding = this->square.width + NWidgetScrollbar::GetVerticalDimension().width + 10;
00670         for (const StringID *id = _colour_dropdown; id != endof(_colour_dropdown); id++) {
00671           size->width = max(size->width, GetStringBoundingBox(*id).width + padding);
00672         }
00673         break;
00674       }
00675     }
00676   }
00677 
00678   virtual void OnPaint()
00679   {
00680     /* Disable dropdown controls if no scheme is selected */
00681     this->SetWidgetDisabledState(SCLW_WIDGET_PRI_COL_DROPDOWN, this->sel == 0);
00682     this->SetWidgetDisabledState(SCLW_WIDGET_SEC_COL_DROPDOWN, this->sel == 0);
00683 
00684     this->DrawWidgets();
00685   }
00686 
00687   virtual void SetStringParameters(int widget) const
00688   {
00689     switch (widget) {
00690       case SCLW_WIDGET_PRI_COL_DROPDOWN:
00691       case SCLW_WIDGET_SEC_COL_DROPDOWN: {
00692         const Company *c = Company::Get((CompanyID)this->window_number);
00693         LiveryScheme scheme = LS_DEFAULT;
00694 
00695         if (this->sel != 0) {
00696           for (scheme = LS_BEGIN; scheme < LS_END; scheme++) {
00697             if (HasBit(this->sel, scheme)) break;
00698           }
00699           if (scheme == LS_END) scheme = LS_DEFAULT;
00700         }
00701         SetDParam(0, STR_COLOUR_DARK_BLUE + ((widget == SCLW_WIDGET_PRI_COL_DROPDOWN) ? c->livery[scheme].colour1 : c->livery[scheme].colour2));
00702         break;
00703       }
00704     }
00705   }
00706 
00707   virtual void DrawWidget(const Rect &r, int widget) const
00708   {
00709     if (widget != SCLW_WIDGET_MATRIX) return;
00710 
00711     bool rtl = _current_text_dir == TD_RTL;
00712 
00713     /* Horizontal coordinates of scheme name column. */
00714     const NWidgetBase *nwi = this->GetWidget<NWidgetBase>(SCLW_WIDGET_SPACER_DROPDOWN);
00715     int sch_left = nwi->pos_x;
00716     int sch_right = sch_left + nwi->current_x - 1;
00717     /* Horizontal coordinates of first dropdown. */
00718     nwi = this->GetWidget<NWidgetBase>(SCLW_WIDGET_PRI_COL_DROPDOWN);
00719     int pri_left = nwi->pos_x;
00720     int pri_right = pri_left + nwi->current_x - 1;
00721     /* Horizontal coordinates of second dropdown. */
00722     nwi = this->GetWidget<NWidgetBase>(SCLW_WIDGET_SEC_COL_DROPDOWN);
00723     int sec_left = nwi->pos_x;
00724     int sec_right = sec_left + nwi->current_x - 1;
00725 
00726     int text_left  = (rtl ? (uint)WD_FRAMERECT_LEFT : (this->box.width + 5));
00727     int text_right = (rtl ? (this->box.width + 5) : (uint)WD_FRAMERECT_RIGHT);
00728 
00729     int box_offs    = (this->line_height - this->box.height) / 2;
00730     int square_offs = (this->line_height - this->square.height) / 2 + 1;
00731     int text_offs   = (this->line_height - FONT_HEIGHT_NORMAL) / 2 + 1;
00732 
00733     int y = r.top;
00734     const Company *c = Company::Get((CompanyID)this->window_number);
00735     for (LiveryScheme scheme = LS_DEFAULT; scheme < LS_END; scheme++) {
00736       if (_livery_class[scheme] == this->livery_class && HasBit(_loaded_newgrf_features.used_liveries, scheme)) {
00737         bool sel = HasBit(this->sel, scheme) != 0;
00738 
00739         /* Optional check box + scheme name. */
00740         if (scheme != LS_DEFAULT) {
00741           DrawSprite(c->livery[scheme].in_use ? SPR_BOX_CHECKED : SPR_BOX_EMPTY, PAL_NONE, (rtl ? sch_right - (this->box.width + 5) + WD_FRAMERECT_RIGHT : sch_left) + WD_FRAMERECT_LEFT, y + box_offs);
00742         }
00743         DrawString(sch_left + text_left, sch_right - text_right, y + text_offs, STR_LIVERY_DEFAULT + scheme, sel ? TC_WHITE : TC_BLACK);
00744 
00745         /* Text below the first dropdown. */
00746         DrawSprite(SPR_SQUARE, GENERAL_SPRITE_COLOUR(c->livery[scheme].colour1), (rtl ? pri_right - (this->box.width + 5) + WD_FRAMERECT_RIGHT : pri_left) + WD_FRAMERECT_LEFT, y + square_offs);
00747         DrawString(pri_left + text_left, pri_right - text_right, y + text_offs, STR_COLOUR_DARK_BLUE + c->livery[scheme].colour1, sel ? TC_WHITE : TC_GOLD);
00748 
00749         /* Text below the second dropdown. */
00750         if (sec_right > sec_left) { // Second dropdown has non-zero size.
00751           DrawSprite(SPR_SQUARE, GENERAL_SPRITE_COLOUR(c->livery[scheme].colour2), (rtl ? sec_right - (this->box.width + 5) + WD_FRAMERECT_RIGHT : sec_left) + WD_FRAMERECT_LEFT, y + square_offs);
00752           DrawString(sec_left + text_left, sec_right - text_right, y + text_offs, STR_COLOUR_DARK_BLUE + c->livery[scheme].colour2, sel ? TC_WHITE : TC_GOLD);
00753         }
00754 
00755         y += this->line_height;
00756       }
00757     }
00758   }
00759 
00760   virtual void OnClick(Point pt, int widget, int click_count)
00761   {
00762     switch (widget) {
00763       /* Livery Class buttons */
00764       case SCLW_WIDGET_CLASS_GENERAL:
00765       case SCLW_WIDGET_CLASS_RAIL:
00766       case SCLW_WIDGET_CLASS_ROAD:
00767       case SCLW_WIDGET_CLASS_SHIP:
00768       case SCLW_WIDGET_CLASS_AIRCRAFT:
00769         this->RaiseWidget(this->livery_class + SCLW_WIDGET_CLASS_GENERAL);
00770         this->livery_class = (LiveryClass)(widget - SCLW_WIDGET_CLASS_GENERAL);
00771         this->LowerWidget(this->livery_class + SCLW_WIDGET_CLASS_GENERAL);
00772 
00773         /* Select the first item in the list */
00774         this->sel = 0;
00775         for (LiveryScheme scheme = LS_DEFAULT; scheme < LS_END; scheme++) {
00776           if (_livery_class[scheme] == this->livery_class && HasBit(_loaded_newgrf_features.used_liveries, scheme)) {
00777             this->sel = 1 << scheme;
00778             break;
00779           }
00780         }
00781 
00782         this->ReInit();
00783         break;
00784 
00785       case SCLW_WIDGET_PRI_COL_DROPDOWN: // First colour dropdown
00786         ShowColourDropDownMenu(SCLW_WIDGET_PRI_COL_DROPDOWN);
00787         break;
00788 
00789       case SCLW_WIDGET_SEC_COL_DROPDOWN: // Second colour dropdown
00790         ShowColourDropDownMenu(SCLW_WIDGET_SEC_COL_DROPDOWN);
00791         break;
00792 
00793       case SCLW_WIDGET_MATRIX: {
00794         const NWidgetBase *wid = this->GetWidget<NWidgetBase>(SCLW_WIDGET_MATRIX);
00795         LiveryScheme j = (LiveryScheme)((pt.y - wid->pos_y) / this->line_height);
00796 
00797         for (LiveryScheme scheme = LS_BEGIN; scheme <= j; scheme++) {
00798           if (_livery_class[scheme] != this->livery_class || !HasBit(_loaded_newgrf_features.used_liveries, scheme)) j++;
00799           if (scheme >= LS_END) return;
00800         }
00801         if (j >= LS_END) return;
00802 
00803         /* If clicking on the left edge, toggle using the livery */
00804         if (_current_text_dir == TD_RTL ? pt.x - wid->pos_x > wid->current_x - (this->box.width + 5) : pt.x - wid->pos_x < (this->box.width + 5)) {
00805           DoCommandP(0, j | (2 << 8), !Company::Get((CompanyID)this->window_number)->livery[j].in_use, CMD_SET_COMPANY_COLOUR);
00806         }
00807 
00808         if (_ctrl_pressed) {
00809           ToggleBit(this->sel, j);
00810         } else {
00811           this->sel = 1 << j;
00812         }
00813         this->SetDirty();
00814         break;
00815       }
00816     }
00817   }
00818 
00819   virtual void OnDropdownSelect(int widget, int index)
00820   {
00821     for (LiveryScheme scheme = LS_DEFAULT; scheme < LS_END; scheme++) {
00822       if (HasBit(this->sel, scheme)) {
00823         DoCommandP(0, scheme | (widget == SCLW_WIDGET_PRI_COL_DROPDOWN ? 0 : 256), index, CMD_SET_COMPANY_COLOUR);
00824       }
00825     }
00826   }
00827 
00833   virtual void OnInvalidateData(int data = 0, bool gui_scope = true)
00834   {
00835     if (!gui_scope) return;
00836     this->SetWidgetsDisabledState(true, SCLW_WIDGET_CLASS_RAIL, SCLW_WIDGET_CLASS_ROAD, SCLW_WIDGET_CLASS_SHIP, SCLW_WIDGET_CLASS_AIRCRAFT, WIDGET_LIST_END);
00837 
00838     bool current_class_valid = this->livery_class == LC_OTHER;
00839     if (_settings_client.gui.liveries == LIT_ALL || (_settings_client.gui.liveries == LIT_COMPANY && this->window_number == _local_company)) {
00840       for (LiveryScheme scheme = LS_DEFAULT; scheme < LS_END; scheme++) {
00841         if (HasBit(_loaded_newgrf_features.used_liveries, scheme)) {
00842           if (_livery_class[scheme] == this->livery_class) current_class_valid = true;
00843           this->EnableWidget(SCLW_WIDGET_CLASS_GENERAL + _livery_class[scheme]);
00844         } else {
00845           ClrBit(this->sel, scheme);
00846         }
00847       }
00848     }
00849 
00850     if (!current_class_valid) {
00851       Point pt = {0, 0};
00852       this->OnClick(pt, SCLW_WIDGET_CLASS_GENERAL, 1);
00853     } else if (data == 0) {
00854       this->ReInit();
00855     }
00856   }
00857 };
00858 
00859 static const NWidgetPart _nested_select_company_livery_widgets [] = {
00860   NWidget(NWID_HORIZONTAL),
00861     NWidget(WWT_CLOSEBOX, COLOUR_GREY),
00862     NWidget(WWT_CAPTION, COLOUR_GREY, SCLW_WIDGET_CAPTION), SetDataTip(STR_LIVERY_CAPTION, STR_TOOLTIP_WINDOW_TITLE_DRAG_THIS),
00863   EndContainer(),
00864   NWidget(NWID_HORIZONTAL),
00865     NWidget(WWT_IMGBTN, COLOUR_GREY, SCLW_WIDGET_CLASS_GENERAL), SetMinimalSize(22, 22), SetFill(0, 1), SetDataTip(SPR_IMG_COMPANY_GENERAL, STR_LIVERY_GENERAL_TOOLTIP),
00866     NWidget(WWT_IMGBTN, COLOUR_GREY, SCLW_WIDGET_CLASS_RAIL), SetMinimalSize(22, 22), SetFill(0, 1), SetDataTip(SPR_IMG_TRAINLIST, STR_LIVERY_TRAIN_TOOLTIP),
00867     NWidget(WWT_IMGBTN, COLOUR_GREY, SCLW_WIDGET_CLASS_ROAD), SetMinimalSize(22, 22), SetFill(0, 1), SetDataTip(SPR_IMG_TRUCKLIST, STR_LIVERY_ROAD_VEHICLE_TOOLTIP),
00868     NWidget(WWT_IMGBTN, COLOUR_GREY, SCLW_WIDGET_CLASS_SHIP), SetMinimalSize(22, 22), SetFill(0, 1), SetDataTip(SPR_IMG_SHIPLIST, STR_LIVERY_SHIP_TOOLTIP),
00869     NWidget(WWT_IMGBTN, COLOUR_GREY, SCLW_WIDGET_CLASS_AIRCRAFT), SetMinimalSize(22, 22), SetFill(0, 1), SetDataTip(SPR_IMG_AIRPLANESLIST, STR_LIVERY_AIRCRAFT_TOOLTIP),
00870     NWidget(WWT_PANEL, COLOUR_GREY), SetMinimalSize(90, 22), SetFill(1, 1), EndContainer(),
00871   EndContainer(),
00872   NWidget(NWID_HORIZONTAL),
00873     NWidget(WWT_PANEL, COLOUR_GREY, SCLW_WIDGET_SPACER_DROPDOWN), SetMinimalSize(150, 12), SetFill(1, 1), EndContainer(),
00874     NWidget(WWT_DROPDOWN, COLOUR_GREY, SCLW_WIDGET_PRI_COL_DROPDOWN), SetMinimalSize(125, 12), SetFill(0, 1), SetDataTip(STR_BLACK_STRING, STR_LIVERY_PRIMARY_TOOLTIP),
00875     NWidget(WWT_DROPDOWN, COLOUR_GREY, SCLW_WIDGET_SEC_COL_DROPDOWN), SetMinimalSize(125, 12), SetFill(0, 1),
00876         SetDataTip(STR_BLACK_STRING, STR_LIVERY_SECONDARY_TOOLTIP),
00877   EndContainer(),
00878   NWidget(WWT_MATRIX, COLOUR_GREY, SCLW_WIDGET_MATRIX), SetMinimalSize(275, 15), SetFill(1, 0), SetDataTip((1 << MAT_ROW_START) | (1 << MAT_COL_START), STR_LIVERY_PANEL_TOOLTIP),
00879 };
00880 
00881 static const WindowDesc _select_company_livery_desc(
00882   WDP_AUTO, 0, 0,
00883   WC_COMPANY_COLOUR, WC_NONE,
00884   0,
00885   _nested_select_company_livery_widgets, lengthof(_nested_select_company_livery_widgets)
00886 );
00887 
00895 void DrawCompanyManagerFace(CompanyManagerFace cmf, int colour, int x, int y)
00896 {
00897   GenderEthnicity ge = (GenderEthnicity)GetCompanyManagerFaceBits(cmf, CMFV_GEN_ETHN, GE_WM);
00898 
00899   bool has_moustache   = !HasBit(ge, GENDER_FEMALE) && GetCompanyManagerFaceBits(cmf, CMFV_HAS_MOUSTACHE,   ge) != 0;
00900   bool has_tie_earring = !HasBit(ge, GENDER_FEMALE) || GetCompanyManagerFaceBits(cmf, CMFV_HAS_TIE_EARRING, ge) != 0;
00901   bool has_glasses     = GetCompanyManagerFaceBits(cmf, CMFV_HAS_GLASSES, ge) != 0;
00902   PaletteID pal;
00903 
00904   /* Modify eye colour palette only if 2 or more valid values exist */
00905   if (_cmf_info[CMFV_EYE_COLOUR].valid_values[ge] < 2) {
00906     pal = PAL_NONE;
00907   } else {
00908     switch (GetCompanyManagerFaceBits(cmf, CMFV_EYE_COLOUR, ge)) {
00909       default: NOT_REACHED();
00910       case 0: pal = PALETTE_TO_BROWN; break;
00911       case 1: pal = PALETTE_TO_BLUE;  break;
00912       case 2: pal = PALETTE_TO_GREEN; break;
00913     }
00914   }
00915 
00916   /* Draw the gradient (background) */
00917   DrawSprite(SPR_GRADIENT, GENERAL_SPRITE_COLOUR(colour), x, y);
00918 
00919   for (CompanyManagerFaceVariable cmfv = CMFV_CHEEKS; cmfv < CMFV_END; cmfv++) {
00920     switch (cmfv) {
00921       case CMFV_MOUSTACHE:   if (!has_moustache)   continue; break;
00922       case CMFV_LIPS:        // FALL THROUGH
00923       case CMFV_NOSE:        if (has_moustache)    continue; break;
00924       case CMFV_TIE_EARRING: if (!has_tie_earring) continue; break;
00925       case CMFV_GLASSES:     if (!has_glasses)     continue; break;
00926       default: break;
00927     }
00928     DrawSprite(GetCompanyManagerFaceSprite(cmf, cmfv, ge), (cmfv == CMFV_EYEBROWS) ? pal : PAL_NONE, x, y);
00929   }
00930 }
00931 
00937 enum SelectCompanyManagerFaceWidgets {
00938   SCMFW_WIDGET_CAPTION,
00939   SCMFW_WIDGET_TOGGLE_LARGE_SMALL,
00940   SCMFW_WIDGET_SELECT_FACE,
00941   SCMFW_WIDGET_CANCEL,
00942   SCMFW_WIDGET_ACCEPT,
00943   SCMFW_WIDGET_MALE,           
00944   SCMFW_WIDGET_FEMALE,         
00945   SCMFW_WIDGET_MALE2,          
00946   SCMFW_WIDGET_FEMALE2,        
00947   SCMFW_WIDGET_SEL_LOADSAVE,   
00948   SCMFW_WIDGET_SEL_MALEFEMALE, 
00949   SCMFW_WIDGET_SEL_PARTS,      
00950   SCMFW_WIDGET_RANDOM_NEW_FACE,
00951   SCMFW_WIDGET_TOGGLE_LARGE_SMALL_BUTTON,
00952   SCMFM_WIDGET_FACE,
00953   SCMFW_WIDGET_LOAD,
00954   SCMFW_WIDGET_FACECODE,
00955   SCMFW_WIDGET_SAVE,
00956   SCMFW_WIDGET_HAS_MOUSTACHE_EARRING_TEXT,
00957   SCMFW_WIDGET_TIE_EARRING_TEXT,
00958   SCMFW_WIDGET_LIPS_MOUSTACHE_TEXT,
00959   SCMFW_WIDGET_HAS_GLASSES_TEXT,
00960   SCMFW_WIDGET_HAIR_TEXT,
00961   SCMFW_WIDGET_EYEBROWS_TEXT,
00962   SCMFW_WIDGET_EYECOLOUR_TEXT,
00963   SCMFW_WIDGET_GLASSES_TEXT,
00964   SCMFW_WIDGET_NOSE_TEXT,
00965   SCMFW_WIDGET_CHIN_TEXT,
00966   SCMFW_WIDGET_JACKET_TEXT,
00967   SCMFW_WIDGET_COLLAR_TEXT,
00968   SCMFW_WIDGET_ETHNICITY_EUR,
00969   SCMFW_WIDGET_ETHNICITY_AFR,
00970   SCMFW_WIDGET_HAS_MOUSTACHE_EARRING,
00971   SCMFW_WIDGET_HAS_GLASSES,
00972   SCMFW_WIDGET_EYECOLOUR_L,
00973   SCMFW_WIDGET_EYECOLOUR,
00974   SCMFW_WIDGET_EYECOLOUR_R,
00975   SCMFW_WIDGET_CHIN_L,
00976   SCMFW_WIDGET_CHIN,
00977   SCMFW_WIDGET_CHIN_R,
00978   SCMFW_WIDGET_EYEBROWS_L,
00979   SCMFW_WIDGET_EYEBROWS,
00980   SCMFW_WIDGET_EYEBROWS_R,
00981   SCMFW_WIDGET_LIPS_MOUSTACHE_L,
00982   SCMFW_WIDGET_LIPS_MOUSTACHE,
00983   SCMFW_WIDGET_LIPS_MOUSTACHE_R,
00984   SCMFW_WIDGET_NOSE_L,
00985   SCMFW_WIDGET_NOSE,
00986   SCMFW_WIDGET_NOSE_R,
00987   SCMFW_WIDGET_HAIR_L,
00988   SCMFW_WIDGET_HAIR,
00989   SCMFW_WIDGET_HAIR_R,
00990   SCMFW_WIDGET_JACKET_L,
00991   SCMFW_WIDGET_JACKET,
00992   SCMFW_WIDGET_JACKET_R,
00993   SCMFW_WIDGET_COLLAR_L,
00994   SCMFW_WIDGET_COLLAR,
00995   SCMFW_WIDGET_COLLAR_R,
00996   SCMFW_WIDGET_TIE_EARRING_L,
00997   SCMFW_WIDGET_TIE_EARRING,
00998   SCMFW_WIDGET_TIE_EARRING_R,
00999   SCMFW_WIDGET_GLASSES_L,
01000   SCMFW_WIDGET_GLASSES,
01001   SCMFW_WIDGET_GLASSES_R,
01002 };
01003 
01005 static const NWidgetPart _nested_select_company_manager_face_widgets[] = {
01006   NWidget(NWID_HORIZONTAL),
01007     NWidget(WWT_CLOSEBOX, COLOUR_GREY),
01008     NWidget(WWT_CAPTION, COLOUR_GREY, SCMFW_WIDGET_CAPTION), SetDataTip(STR_FACE_CAPTION, STR_TOOLTIP_WINDOW_TITLE_DRAG_THIS),
01009     NWidget(WWT_IMGBTN, COLOUR_GREY, SCMFW_WIDGET_TOGGLE_LARGE_SMALL), SetDataTip(SPR_LARGE_SMALL_WINDOW, STR_FACE_ADVANCED_TOOLTIP),
01010   EndContainer(),
01011   NWidget(WWT_PANEL, COLOUR_GREY, SCMFW_WIDGET_SELECT_FACE),
01012     NWidget(NWID_SPACER), SetMinimalSize(0, 2),
01013     NWidget(NWID_HORIZONTAL), SetPIP(2, 2, 2),
01014       NWidget(NWID_VERTICAL),
01015         NWidget(NWID_HORIZONTAL),
01016           NWidget(NWID_SPACER), SetFill(1, 0),
01017           NWidget(WWT_EMPTY, COLOUR_GREY, SCMFM_WIDGET_FACE), SetMinimalSize(92, 119),
01018           NWidget(NWID_SPACER), SetFill(1, 0),
01019         EndContainer(),
01020         NWidget(NWID_SPACER), SetMinimalSize(0, 2),
01021         NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_RANDOM_NEW_FACE), SetFill(1, 0), SetDataTip(STR_FACE_NEW_FACE_BUTTON, STR_FACE_NEW_FACE_TOOLTIP),
01022         NWidget(NWID_SELECTION, INVALID_COLOUR, SCMFW_WIDGET_SEL_LOADSAVE), // Load/number/save buttons under the portrait in the advanced view.
01023           NWidget(NWID_VERTICAL),
01024             NWidget(NWID_SPACER), SetMinimalSize(0, 5), SetFill(0, 1),
01025             NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_LOAD), SetFill(1, 0), SetDataTip(STR_FACE_LOAD, STR_FACE_LOAD_TOOLTIP),
01026             NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_FACECODE), SetFill(1, 0), SetDataTip(STR_FACE_FACECODE, STR_FACE_FACECODE_TOOLTIP),
01027             NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_SAVE), SetFill(1, 0), SetDataTip(STR_FACE_SAVE, STR_FACE_SAVE_TOOLTIP),
01028             NWidget(NWID_SPACER), SetMinimalSize(0, 5), SetFill(0, 1),
01029           EndContainer(),
01030         EndContainer(),
01031       EndContainer(),
01032       NWidget(NWID_VERTICAL),
01033         NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_TOGGLE_LARGE_SMALL_BUTTON), SetFill(1, 0), SetDataTip(STR_FACE_ADVANCED, STR_FACE_ADVANCED_TOOLTIP),
01034         NWidget(NWID_SPACER), SetMinimalSize(0, 2),
01035         NWidget(NWID_SELECTION, INVALID_COLOUR, SCMFW_WIDGET_SEL_MALEFEMALE), // Simple male/female face setting.
01036           NWidget(NWID_VERTICAL),
01037             NWidget(NWID_SPACER), SetFill(0, 1),
01038             NWidget(WWT_TEXTBTN, COLOUR_GREY, SCMFW_WIDGET_MALE), SetFill(1, 0), SetDataTip(STR_FACE_MALE_BUTTON, STR_FACE_MALE_TOOLTIP),
01039             NWidget(WWT_TEXTBTN, COLOUR_GREY, SCMFW_WIDGET_FEMALE), SetFill(1, 0), SetDataTip(STR_FACE_FEMALE_BUTTON, STR_FACE_FEMALE_TOOLTIP),
01040             NWidget(NWID_SPACER), SetFill(0, 1),
01041           EndContainer(),
01042         EndContainer(),
01043         NWidget(NWID_SELECTION, INVALID_COLOUR, SCMFW_WIDGET_SEL_PARTS), // Advanced face parts setting.
01044           NWidget(NWID_VERTICAL),
01045             NWidget(NWID_SPACER), SetMinimalSize(0, 2),
01046             NWidget(NWID_HORIZONTAL, NC_EQUALSIZE),
01047               NWidget(WWT_TEXTBTN, COLOUR_GREY, SCMFW_WIDGET_MALE2), SetFill(1, 0), SetDataTip(STR_FACE_MALE_BUTTON, STR_FACE_MALE_TOOLTIP),
01048               NWidget(WWT_TEXTBTN, COLOUR_GREY, SCMFW_WIDGET_FEMALE2), SetFill(1, 0), SetDataTip(STR_FACE_FEMALE_BUTTON, STR_FACE_FEMALE_TOOLTIP),
01049             EndContainer(),
01050             NWidget(NWID_SPACER), SetMinimalSize(0, 2),
01051             NWidget(NWID_HORIZONTAL, NC_EQUALSIZE),
01052               NWidget(WWT_TEXTBTN, COLOUR_GREY, SCMFW_WIDGET_ETHNICITY_EUR), SetFill(1, 0), SetDataTip(STR_FACE_EUROPEAN, STR_FACE_SELECT_EUROPEAN),
01053               NWidget(WWT_TEXTBTN, COLOUR_GREY, SCMFW_WIDGET_ETHNICITY_AFR), SetFill(1, 0), SetDataTip(STR_FACE_AFRICAN, STR_FACE_SELECT_AFRICAN),
01054             EndContainer(),
01055             NWidget(NWID_SPACER), SetMinimalSize(0, 4),
01056             NWidget(NWID_HORIZONTAL),
01057               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_HAS_MOUSTACHE_EARRING_TEXT), SetFill(1, 0),
01058               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_HAS_MOUSTACHE_EARRING), SetDataTip(STR_EMPTY, STR_FACE_MOUSTACHE_EARRING_TOOLTIP),
01059             EndContainer(),
01060             NWidget(NWID_HORIZONTAL),
01061               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_HAS_GLASSES_TEXT), SetFill(1, 0),
01062               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_HAS_GLASSES), SetDataTip(STR_EMPTY, STR_FACE_GLASSES_TOOLTIP),
01063             EndContainer(),
01064             NWidget(NWID_SPACER), SetMinimalSize(0, 2), SetFill(1, 0),
01065             NWidget(NWID_HORIZONTAL),
01066               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_HAIR_TEXT), SetFill(1, 0),
01067               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_HAIR_L), SetDataTip(AWV_DECREASE, STR_FACE_HAIR_TOOLTIP),
01068               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_HAIR), SetDataTip(STR_EMPTY, STR_FACE_HAIR_TOOLTIP),
01069               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_HAIR_R), SetDataTip(AWV_INCREASE, STR_FACE_HAIR_TOOLTIP),
01070             EndContainer(),
01071             NWidget(NWID_HORIZONTAL),
01072               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_EYEBROWS_TEXT), SetFill(1, 0),
01073               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_EYEBROWS_L), SetDataTip(AWV_DECREASE, STR_FACE_EYEBROWS_TOOLTIP),
01074               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_EYEBROWS), SetDataTip(STR_EMPTY, STR_FACE_EYEBROWS_TOOLTIP),
01075               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_EYEBROWS_R), SetDataTip(AWV_INCREASE, STR_FACE_EYEBROWS_TOOLTIP),
01076             EndContainer(),
01077             NWidget(NWID_HORIZONTAL),
01078               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_EYECOLOUR_TEXT), SetFill(1, 0),
01079               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_EYECOLOUR_L), SetDataTip(AWV_DECREASE, STR_FACE_EYECOLOUR_TOOLTIP),
01080               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_EYECOLOUR), SetDataTip(STR_EMPTY, STR_FACE_EYECOLOUR_TOOLTIP),
01081               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_EYECOLOUR_R), SetDataTip(AWV_INCREASE, STR_FACE_EYECOLOUR_TOOLTIP),
01082             EndContainer(),
01083             NWidget(NWID_HORIZONTAL),
01084               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_GLASSES_TEXT), SetFill(1, 0),
01085               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_GLASSES_L), SetDataTip(AWV_DECREASE, STR_FACE_GLASSES_TOOLTIP_2),
01086               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_GLASSES), SetDataTip(STR_EMPTY, STR_FACE_GLASSES_TOOLTIP_2),
01087               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_GLASSES_R), SetDataTip(AWV_INCREASE, STR_FACE_GLASSES_TOOLTIP_2),
01088             EndContainer(),
01089             NWidget(NWID_HORIZONTAL),
01090               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_NOSE_TEXT), SetFill(1, 0),
01091               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_NOSE_L), SetDataTip(AWV_DECREASE, STR_FACE_NOSE_TOOLTIP),
01092               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_NOSE), SetDataTip(STR_EMPTY, STR_FACE_NOSE_TOOLTIP),
01093               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_NOSE_R), SetDataTip(AWV_INCREASE, STR_FACE_NOSE_TOOLTIP),
01094             EndContainer(),
01095             NWidget(NWID_HORIZONTAL),
01096               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_LIPS_MOUSTACHE_TEXT), SetFill(1, 0),
01097               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_LIPS_MOUSTACHE_L), SetDataTip(AWV_DECREASE, STR_FACE_LIPS_MOUSTACHE_TOOLTIP),
01098               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_LIPS_MOUSTACHE), SetDataTip(STR_EMPTY, STR_FACE_LIPS_MOUSTACHE_TOOLTIP),
01099               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_LIPS_MOUSTACHE_R), SetDataTip(AWV_INCREASE, STR_FACE_LIPS_MOUSTACHE_TOOLTIP),
01100             EndContainer(),
01101             NWidget(NWID_HORIZONTAL),
01102               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_CHIN_TEXT), SetFill(1, 0),
01103               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_CHIN_L), SetDataTip(AWV_DECREASE, STR_FACE_CHIN_TOOLTIP),
01104               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_CHIN), SetDataTip(STR_EMPTY, STR_FACE_CHIN_TOOLTIP),
01105               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_CHIN_R), SetDataTip(AWV_INCREASE, STR_FACE_CHIN_TOOLTIP),
01106             EndContainer(),
01107             NWidget(NWID_HORIZONTAL),
01108               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_JACKET_TEXT), SetFill(1, 0),
01109               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_JACKET_L), SetDataTip(AWV_DECREASE, STR_FACE_JACKET_TOOLTIP),
01110               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_JACKET), SetDataTip(STR_EMPTY, STR_FACE_JACKET_TOOLTIP),
01111               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_JACKET_R), SetDataTip(AWV_INCREASE, STR_FACE_JACKET_TOOLTIP),
01112             EndContainer(),
01113             NWidget(NWID_HORIZONTAL),
01114               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_COLLAR_TEXT), SetFill(1, 0),
01115               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_COLLAR_L), SetDataTip(AWV_DECREASE, STR_FACE_COLLAR_TOOLTIP),
01116               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_COLLAR), SetDataTip(STR_EMPTY, STR_FACE_COLLAR_TOOLTIP),
01117               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_COLLAR_R), SetDataTip(AWV_INCREASE, STR_FACE_COLLAR_TOOLTIP),
01118             EndContainer(),
01119             NWidget(NWID_HORIZONTAL),
01120               NWidget(WWT_EMPTY, INVALID_COLOUR, SCMFW_WIDGET_TIE_EARRING_TEXT), SetFill(1, 0),
01121               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_TIE_EARRING_L), SetDataTip(AWV_DECREASE, STR_FACE_TIE_EARRING_TOOLTIP),
01122               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_TIE_EARRING), SetDataTip(STR_EMPTY, STR_FACE_TIE_EARRING_TOOLTIP),
01123               NWidget(WWT_PUSHARROWBTN, COLOUR_GREY, SCMFW_WIDGET_TIE_EARRING_R), SetDataTip(AWV_INCREASE, STR_FACE_TIE_EARRING_TOOLTIP),
01124             EndContainer(),
01125             NWidget(NWID_SPACER), SetFill(0, 1),
01126           EndContainer(),
01127         EndContainer(),
01128       EndContainer(),
01129     EndContainer(),
01130     NWidget(NWID_SPACER), SetMinimalSize(0, 2),
01131   EndContainer(),
01132   NWidget(NWID_HORIZONTAL, NC_EQUALSIZE),
01133     NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_CANCEL), SetFill(1, 0), SetDataTip(STR_BUTTON_CANCEL, STR_FACE_CANCEL_TOOLTIP),
01134     NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, SCMFW_WIDGET_ACCEPT), SetFill(1, 0), SetDataTip(STR_BUTTON_OK, STR_FACE_OK_TOOLTIP),
01135   EndContainer(),
01136 };
01137 
01139 class SelectCompanyManagerFaceWindow : public Window
01140 {
01141   CompanyManagerFace face; 
01142   bool advanced; 
01143 
01144   GenderEthnicity ge; 
01145   bool is_female;     
01146   bool is_moust_male; 
01147 
01148   Dimension yesno_dim;  
01149   Dimension number_dim; 
01150 
01151   static const StringID PART_TEXTS_IS_FEMALE[]; 
01152   static const StringID PART_TEXTS[];           
01153 
01161   void DrawFaceStringLabel(byte widget_index, uint8 val, bool is_bool_widget) const
01162   {
01163     StringID str;
01164     const NWidgetCore *nwi_widget = this->GetWidget<NWidgetCore>(widget_index);
01165     if (!nwi_widget->IsDisabled()) {
01166       if (is_bool_widget) {
01167         /* if it a bool button write yes or no */
01168         str = (val != 0) ? STR_FACE_YES : STR_FACE_NO;
01169       } else {
01170         /* else write the value + 1 */
01171         SetDParam(0, val + 1);
01172         str = STR_JUST_INT;
01173       }
01174 
01175       /* Draw the value/bool in white (0xC). If the button clicked adds 1px to x and y text coordinates (IsWindowWidgetLowered()). */
01176       DrawString(nwi_widget->pos_x + nwi_widget->IsLowered(), nwi_widget->pos_x + nwi_widget->current_x - 1 - nwi_widget->IsLowered(),
01177           nwi_widget->pos_y + 1 + nwi_widget->IsLowered(), str, TC_WHITE, SA_HOR_CENTER);
01178     }
01179   }
01180 
01181   void UpdateData()
01182   {
01183     this->ge = (GenderEthnicity)GB(this->face, _cmf_info[CMFV_GEN_ETHN].offset, _cmf_info[CMFV_GEN_ETHN].length); // get the gender and ethnicity
01184     this->is_female = HasBit(this->ge, GENDER_FEMALE); // get the gender: 0 == male and 1 == female
01185     this->is_moust_male = !is_female && GetCompanyManagerFaceBits(this->face, CMFV_HAS_MOUSTACHE, this->ge) != 0; // is a male face with moustache
01186   }
01187 
01188 public:
01189   SelectCompanyManagerFaceWindow(const WindowDesc *desc, Window *parent) : Window()
01190   {
01191     this->advanced = false;
01192     this->CreateNestedTree(desc);
01193     this->SelectDisplayPlanes(this->advanced);
01194     this->FinishInitNested(desc, parent->window_number);
01195     this->parent = parent;
01196     this->owner = (Owner)this->window_number;
01197     this->face = Company::Get((CompanyID)this->window_number)->face;
01198 
01199     this->UpdateData();
01200   }
01201 
01206   void SelectDisplayPlanes(bool advanced)
01207   {
01208     this->GetWidget<NWidgetStacked>(SCMFW_WIDGET_SEL_LOADSAVE)->SetDisplayedPlane(advanced ? 0 : SZSP_NONE);
01209     this->GetWidget<NWidgetStacked>(SCMFW_WIDGET_SEL_PARTS)->SetDisplayedPlane(advanced ? 0 : SZSP_NONE);
01210     this->GetWidget<NWidgetStacked>(SCMFW_WIDGET_SEL_MALEFEMALE)->SetDisplayedPlane(advanced ? SZSP_NONE : 0);
01211     this->GetWidget<NWidgetCore>(SCMFW_WIDGET_RANDOM_NEW_FACE)->widget_data = advanced ? STR_MAPGEN_RANDOM : STR_FACE_NEW_FACE_BUTTON;
01212 
01213     NWidgetCore *wi = this->GetWidget<NWidgetCore>(SCMFW_WIDGET_TOGGLE_LARGE_SMALL_BUTTON);
01214     if (advanced) {
01215       wi->SetDataTip(STR_FACE_SIMPLE, STR_FACE_SIMPLE_TOOLTIP);
01216     } else {
01217       wi->SetDataTip(STR_FACE_ADVANCED, STR_FACE_ADVANCED_TOOLTIP);
01218     }
01219   }
01220 
01221   virtual void OnInit()
01222   {
01223     /* Size of the boolean yes/no button. */
01224     Dimension yesno_dim = maxdim(GetStringBoundingBox(STR_FACE_YES), GetStringBoundingBox(STR_FACE_NO));
01225     yesno_dim.width  += WD_FRAMERECT_LEFT + WD_FRAMERECT_RIGHT;
01226     yesno_dim.height += WD_FRAMERECT_TOP + WD_FRAMERECT_BOTTOM;
01227     /* Size of the number button + arrows. */
01228     Dimension number_dim = {0, 0};
01229     for (int val = 1; val <= 12; val++) {
01230       SetDParam(0, val);
01231       number_dim = maxdim(number_dim, GetStringBoundingBox(STR_JUST_INT));
01232     }
01233     uint arrows_width = GetSpriteSize(SPR_ARROW_LEFT).width + GetSpriteSize(SPR_ARROW_RIGHT).width + 2 * (WD_IMGBTN_LEFT + WD_IMGBTN_RIGHT);
01234     number_dim.width += WD_FRAMERECT_LEFT + WD_FRAMERECT_RIGHT + arrows_width;
01235     number_dim.height += WD_FRAMERECT_TOP + WD_FRAMERECT_BOTTOM;
01236     /* Compute width of both buttons. */
01237     yesno_dim.width = max(yesno_dim.width, number_dim.width);
01238     number_dim.width = yesno_dim.width - arrows_width;
01239 
01240     this->yesno_dim = yesno_dim;
01241     this->number_dim = number_dim;
01242   }
01243 
01244   virtual void UpdateWidgetSize(int widget, Dimension *size, const Dimension &padding, Dimension *fill, Dimension *resize)
01245   {
01246     switch (widget) {
01247       case SCMFW_WIDGET_HAS_MOUSTACHE_EARRING_TEXT:
01248       case SCMFW_WIDGET_TIE_EARRING_TEXT: {
01249         int offset = (widget - SCMFW_WIDGET_HAS_MOUSTACHE_EARRING_TEXT) * 2;
01250         *size = maxdim(GetStringBoundingBox(PART_TEXTS_IS_FEMALE[offset]), GetStringBoundingBox(PART_TEXTS_IS_FEMALE[offset + 1]));
01251         size->width  += WD_FRAMERECT_LEFT + WD_FRAMERECT_RIGHT;
01252         size->height += WD_FRAMERECT_TOP + WD_FRAMERECT_BOTTOM;
01253         break;
01254       }
01255 
01256       case SCMFW_WIDGET_LIPS_MOUSTACHE_TEXT:
01257         *size = maxdim(GetStringBoundingBox(STR_FACE_LIPS), GetStringBoundingBox(STR_FACE_MOUSTACHE));
01258         size->width  += WD_FRAMERECT_LEFT + WD_FRAMERECT_RIGHT;
01259         size->height += WD_FRAMERECT_TOP + WD_FRAMERECT_BOTTOM;
01260         break;
01261 
01262       case SCMFW_WIDGET_HAS_GLASSES_TEXT:
01263       case SCMFW_WIDGET_HAIR_TEXT:
01264       case SCMFW_WIDGET_EYEBROWS_TEXT:
01265       case SCMFW_WIDGET_EYECOLOUR_TEXT:
01266       case SCMFW_WIDGET_GLASSES_TEXT:
01267       case SCMFW_WIDGET_NOSE_TEXT:
01268       case SCMFW_WIDGET_CHIN_TEXT:
01269       case SCMFW_WIDGET_JACKET_TEXT:
01270       case SCMFW_WIDGET_COLLAR_TEXT:
01271         *size = GetStringBoundingBox(PART_TEXTS[widget - SCMFW_WIDGET_HAS_GLASSES_TEXT]);
01272         size->width  += WD_FRAMERECT_LEFT + WD_FRAMERECT_RIGHT;
01273         size->height += WD_FRAMERECT_TOP + WD_FRAMERECT_BOTTOM;
01274         break;
01275 
01276       case SCMFW_WIDGET_HAS_MOUSTACHE_EARRING:
01277       case SCMFW_WIDGET_HAS_GLASSES:
01278         *size = this->yesno_dim;
01279         break;
01280 
01281       case SCMFW_WIDGET_EYECOLOUR:
01282       case SCMFW_WIDGET_CHIN:
01283       case SCMFW_WIDGET_EYEBROWS:
01284       case SCMFW_WIDGET_LIPS_MOUSTACHE:
01285       case SCMFW_WIDGET_NOSE:
01286       case SCMFW_WIDGET_HAIR:
01287       case SCMFW_WIDGET_JACKET:
01288       case SCMFW_WIDGET_COLLAR:
01289       case SCMFW_WIDGET_TIE_EARRING:
01290       case SCMFW_WIDGET_GLASSES:
01291         *size = this->number_dim;
01292         break;
01293     }
01294   }
01295 
01296   virtual void OnPaint()
01297   {
01298     /* lower the non-selected gender button */
01299     this->SetWidgetsLoweredState(!this->is_female, SCMFW_WIDGET_MALE, SCMFW_WIDGET_MALE2, WIDGET_LIST_END);
01300     this->SetWidgetsLoweredState( this->is_female, SCMFW_WIDGET_FEMALE, SCMFW_WIDGET_FEMALE2, WIDGET_LIST_END);
01301 
01302     /* advanced company manager face selection window */
01303 
01304     /* lower the non-selected ethnicity button */
01305     this->SetWidgetLoweredState(SCMFW_WIDGET_ETHNICITY_EUR, !HasBit(this->ge, ETHNICITY_BLACK));
01306     this->SetWidgetLoweredState(SCMFW_WIDGET_ETHNICITY_AFR,  HasBit(this->ge, ETHNICITY_BLACK));
01307 
01308 
01309     /* Disable dynamically the widgets which CompanyManagerFaceVariable has less than 2 options
01310      * (or in other words you haven't any choice).
01311      * If the widgets depend on a HAS-variable and this is false the widgets will be disabled, too. */
01312 
01313     /* Eye colour buttons */
01314     this->SetWidgetsDisabledState(_cmf_info[CMFV_EYE_COLOUR].valid_values[this->ge] < 2,
01315         SCMFW_WIDGET_EYECOLOUR, SCMFW_WIDGET_EYECOLOUR_L, SCMFW_WIDGET_EYECOLOUR_R, WIDGET_LIST_END);
01316 
01317     /* Chin buttons */
01318     this->SetWidgetsDisabledState(_cmf_info[CMFV_CHIN].valid_values[this->ge] < 2,
01319         SCMFW_WIDGET_CHIN, SCMFW_WIDGET_CHIN_L, SCMFW_WIDGET_CHIN_R, WIDGET_LIST_END);
01320 
01321     /* Eyebrows buttons */
01322     this->SetWidgetsDisabledState(_cmf_info[CMFV_EYEBROWS].valid_values[this->ge] < 2,
01323         SCMFW_WIDGET_EYEBROWS, SCMFW_WIDGET_EYEBROWS_L, SCMFW_WIDGET_EYEBROWS_R, WIDGET_LIST_END);
01324 
01325     /* Lips or (if it a male face with a moustache) moustache buttons */
01326     this->SetWidgetsDisabledState(_cmf_info[this->is_moust_male ? CMFV_MOUSTACHE : CMFV_LIPS].valid_values[this->ge] < 2,
01327         SCMFW_WIDGET_LIPS_MOUSTACHE, SCMFW_WIDGET_LIPS_MOUSTACHE_L, SCMFW_WIDGET_LIPS_MOUSTACHE_R, WIDGET_LIST_END);
01328 
01329     /* Nose buttons | male faces with moustache haven't any nose options */
01330     this->SetWidgetsDisabledState(_cmf_info[CMFV_NOSE].valid_values[this->ge] < 2 || this->is_moust_male,
01331         SCMFW_WIDGET_NOSE, SCMFW_WIDGET_NOSE_L, SCMFW_WIDGET_NOSE_R, WIDGET_LIST_END);
01332 
01333     /* Hair buttons */
01334     this->SetWidgetsDisabledState(_cmf_info[CMFV_HAIR].valid_values[this->ge] < 2,
01335         SCMFW_WIDGET_HAIR, SCMFW_WIDGET_HAIR_L, SCMFW_WIDGET_HAIR_R, WIDGET_LIST_END);
01336 
01337     /* Jacket buttons */
01338     this->SetWidgetsDisabledState(_cmf_info[CMFV_JACKET].valid_values[this->ge] < 2,
01339         SCMFW_WIDGET_JACKET, SCMFW_WIDGET_JACKET_L, SCMFW_WIDGET_JACKET_R, WIDGET_LIST_END);
01340 
01341     /* Collar buttons */
01342     this->SetWidgetsDisabledState(_cmf_info[CMFV_COLLAR].valid_values[this->ge] < 2,
01343         SCMFW_WIDGET_COLLAR, SCMFW_WIDGET_COLLAR_L, SCMFW_WIDGET_COLLAR_R, WIDGET_LIST_END);
01344 
01345     /* Tie/earring buttons | female faces without earring haven't any earring options */
01346     this->SetWidgetsDisabledState(_cmf_info[CMFV_TIE_EARRING].valid_values[this->ge] < 2 ||
01347           (this->is_female && GetCompanyManagerFaceBits(this->face, CMFV_HAS_TIE_EARRING, this->ge) == 0),
01348         SCMFW_WIDGET_TIE_EARRING, SCMFW_WIDGET_TIE_EARRING_L, SCMFW_WIDGET_TIE_EARRING_R, WIDGET_LIST_END);
01349 
01350     /* Glasses buttons | faces without glasses haven't any glasses options */
01351     this->SetWidgetsDisabledState(_cmf_info[CMFV_GLASSES].valid_values[this->ge] < 2 || GetCompanyManagerFaceBits(this->face, CMFV_HAS_GLASSES, this->ge) == 0,
01352         SCMFW_WIDGET_GLASSES, SCMFW_WIDGET_GLASSES_L, SCMFW_WIDGET_GLASSES_R, WIDGET_LIST_END);
01353 
01354     this->DrawWidgets();
01355   }
01356 
01357   virtual void DrawWidget(const Rect &r, int widget) const
01358   {
01359     switch (widget) {
01360       case SCMFW_WIDGET_HAS_MOUSTACHE_EARRING_TEXT:
01361       case SCMFW_WIDGET_TIE_EARRING_TEXT: {
01362         StringID str = PART_TEXTS_IS_FEMALE[(widget - SCMFW_WIDGET_HAS_MOUSTACHE_EARRING_TEXT) * 2 + this->is_female];
01363         DrawString(r.left + WD_FRAMERECT_LEFT, r.right - WD_FRAMERECT_RIGHT, r.top + WD_FRAMERECT_TOP, str, TC_GOLD, SA_RIGHT);
01364         break;
01365       }
01366 
01367       case SCMFW_WIDGET_LIPS_MOUSTACHE_TEXT:
01368         DrawString(r.left + WD_FRAMERECT_LEFT, r.right - WD_FRAMERECT_RIGHT, r.top + WD_FRAMERECT_TOP, (this->is_moust_male) ? STR_FACE_MOUSTACHE : STR_FACE_LIPS, TC_GOLD, SA_RIGHT);
01369         break;
01370 
01371       case SCMFW_WIDGET_HAS_GLASSES_TEXT:
01372       case SCMFW_WIDGET_HAIR_TEXT:
01373       case SCMFW_WIDGET_EYEBROWS_TEXT:
01374       case SCMFW_WIDGET_EYECOLOUR_TEXT:
01375       case SCMFW_WIDGET_GLASSES_TEXT:
01376       case SCMFW_WIDGET_NOSE_TEXT:
01377       case SCMFW_WIDGET_CHIN_TEXT:
01378       case SCMFW_WIDGET_JACKET_TEXT:
01379       case SCMFW_WIDGET_COLLAR_TEXT:
01380         DrawString(r.left + WD_FRAMERECT_LEFT, r.right - WD_FRAMERECT_RIGHT, r.top + WD_FRAMERECT_TOP, PART_TEXTS[widget - SCMFW_WIDGET_HAS_GLASSES_TEXT], TC_GOLD, SA_RIGHT);
01381         break;
01382 
01383 
01384       case SCMFW_WIDGET_HAS_MOUSTACHE_EARRING:
01385         if (this->is_female) { // Only for female faces
01386           this->DrawFaceStringLabel(SCMFW_WIDGET_HAS_MOUSTACHE_EARRING, GetCompanyManagerFaceBits(this->face, CMFV_HAS_TIE_EARRING, this->ge), true);
01387         } else { // Only for male faces
01388           this->DrawFaceStringLabel(SCMFW_WIDGET_HAS_MOUSTACHE_EARRING, GetCompanyManagerFaceBits(this->face, CMFV_HAS_MOUSTACHE,   this->ge), true);
01389         }
01390         break;
01391 
01392       case SCMFW_WIDGET_TIE_EARRING:
01393         if (this->is_female) { // Only for female faces
01394           this->DrawFaceStringLabel(SCMFW_WIDGET_TIE_EARRING, GetCompanyManagerFaceBits(this->face, CMFV_TIE_EARRING, this->ge), false);
01395         } else { // Only for male faces
01396           this->DrawFaceStringLabel(SCMFW_WIDGET_TIE_EARRING, GetCompanyManagerFaceBits(this->face, CMFV_TIE_EARRING, this->ge), false);
01397         }
01398         break;
01399 
01400       case SCMFW_WIDGET_LIPS_MOUSTACHE:
01401         if (this->is_moust_male) { // Only for male faces with moustache
01402           this->DrawFaceStringLabel(SCMFW_WIDGET_LIPS_MOUSTACHE, GetCompanyManagerFaceBits(this->face, CMFV_MOUSTACHE, this->ge), false);
01403         } else { // Only for female faces or male faces without moustache
01404           this->DrawFaceStringLabel(SCMFW_WIDGET_LIPS_MOUSTACHE, GetCompanyManagerFaceBits(this->face, CMFV_LIPS,      this->ge), false);
01405         }
01406         break;
01407 
01408       case SCMFW_WIDGET_HAS_GLASSES:
01409         this->DrawFaceStringLabel(SCMFW_WIDGET_HAS_GLASSES, GetCompanyManagerFaceBits(this->face, CMFV_HAS_GLASSES, this->ge), true );
01410         break;
01411 
01412       case SCMFW_WIDGET_HAIR:
01413         this->DrawFaceStringLabel(SCMFW_WIDGET_HAIR,        GetCompanyManagerFaceBits(this->face, CMFV_HAIR,        this->ge), false);
01414         break;
01415 
01416       case SCMFW_WIDGET_EYEBROWS:
01417         this->DrawFaceStringLabel(SCMFW_WIDGET_EYEBROWS,    GetCompanyManagerFaceBits(this->face, CMFV_EYEBROWS,    this->ge), false);
01418         break;
01419 
01420       case SCMFW_WIDGET_EYECOLOUR:
01421         this->DrawFaceStringLabel(SCMFW_WIDGET_EYECOLOUR,   GetCompanyManagerFaceBits(this->face, CMFV_EYE_COLOUR,  this->ge), false);
01422         break;
01423 
01424       case SCMFW_WIDGET_GLASSES:
01425         this->DrawFaceStringLabel(SCMFW_WIDGET_GLASSES,     GetCompanyManagerFaceBits(this->face, CMFV_GLASSES,     this->ge), false);
01426         break;
01427 
01428       case SCMFW_WIDGET_NOSE:
01429         this->DrawFaceStringLabel(SCMFW_WIDGET_NOSE,        GetCompanyManagerFaceBits(this->face, CMFV_NOSE,        this->ge), false);
01430         break;
01431 
01432       case SCMFW_WIDGET_CHIN:
01433         this->DrawFaceStringLabel(SCMFW_WIDGET_CHIN,        GetCompanyManagerFaceBits(this->face, CMFV_CHIN,        this->ge), false);
01434         break;
01435 
01436       case SCMFW_WIDGET_JACKET:
01437         this->DrawFaceStringLabel(SCMFW_WIDGET_JACKET,      GetCompanyManagerFaceBits(this->face, CMFV_JACKET,      this->ge), false);
01438         break;
01439 
01440       case SCMFW_WIDGET_COLLAR:
01441         this->DrawFaceStringLabel(SCMFW_WIDGET_COLLAR,      GetCompanyManagerFaceBits(this->face, CMFV_COLLAR,      this->ge), false);
01442         break;
01443 
01444       case SCMFM_WIDGET_FACE:
01445         DrawCompanyManagerFace(this->face, Company::Get((CompanyID)this->window_number)->colour, r.left, r.top);
01446         break;
01447     }
01448   }
01449 
01450   virtual void OnClick(Point pt, int widget, int click_count)
01451   {
01452     switch (widget) {
01453       /* Toggle size, advanced/simple face selection */
01454       case SCMFW_WIDGET_TOGGLE_LARGE_SMALL:
01455       case SCMFW_WIDGET_TOGGLE_LARGE_SMALL_BUTTON:
01456         this->advanced = !this->advanced;
01457         this->SelectDisplayPlanes(this->advanced);
01458         this->ReInit();
01459         break;
01460 
01461       /* OK button */
01462       case SCMFW_WIDGET_ACCEPT:
01463         DoCommandP(0, 0, this->face, CMD_SET_COMPANY_MANAGER_FACE);
01464         /* FALL THROUGH */
01465 
01466       /* Cancel button */
01467       case SCMFW_WIDGET_CANCEL:
01468         delete this;
01469         break;
01470 
01471       /* Load button */
01472       case SCMFW_WIDGET_LOAD:
01473         this->face = _company_manager_face;
01474         ScaleAllCompanyManagerFaceBits(this->face);
01475         ShowErrorMessage(STR_FACE_LOAD_DONE, INVALID_STRING_ID, WL_INFO);
01476         this->UpdateData();
01477         this->SetDirty();
01478         break;
01479 
01480       /* 'Company manager face number' button, view and/or set company manager face number */
01481       case SCMFW_WIDGET_FACECODE:
01482         SetDParam(0, this->face);
01483         ShowQueryString(STR_JUST_INT, STR_FACE_FACECODE_CAPTION, 10 + 1, this, CS_NUMERAL, QSF_NONE);
01484         break;
01485 
01486       /* Save button */
01487       case SCMFW_WIDGET_SAVE:
01488         _company_manager_face = this->face;
01489         ShowErrorMessage(STR_FACE_SAVE_DONE, INVALID_STRING_ID, WL_INFO);
01490         break;
01491 
01492       /* Toggle gender (male/female) button */
01493       case SCMFW_WIDGET_MALE:
01494       case SCMFW_WIDGET_FEMALE:
01495       case SCMFW_WIDGET_MALE2:
01496       case SCMFW_WIDGET_FEMALE2:
01497         SetCompanyManagerFaceBits(this->face, CMFV_GENDER, this->ge, (widget == SCMFW_WIDGET_FEMALE || widget == SCMFW_WIDGET_FEMALE2));
01498         ScaleAllCompanyManagerFaceBits(this->face);
01499         this->UpdateData();
01500         this->SetDirty();
01501         break;
01502 
01503       /* Randomize face button */
01504       case SCMFW_WIDGET_RANDOM_NEW_FACE:
01505         RandomCompanyManagerFaceBits(this->face, this->ge, this->advanced);
01506         this->UpdateData();
01507         this->SetDirty();
01508         break;
01509 
01510       /* Toggle ethnicity (european/african) button */
01511       case SCMFW_WIDGET_ETHNICITY_EUR:
01512       case SCMFW_WIDGET_ETHNICITY_AFR:
01513         SetCompanyManagerFaceBits(this->face, CMFV_ETHNICITY, this->ge, widget - SCMFW_WIDGET_ETHNICITY_EUR);
01514         ScaleAllCompanyManagerFaceBits(this->face);
01515         this->UpdateData();
01516         this->SetDirty();
01517         break;
01518 
01519       default:
01520         /* Here all buttons from SCMFW_WIDGET_HAS_MOUSTACHE_EARRING to SCMFW_WIDGET_GLASSES_R are handled.
01521          * First it checks which CompanyManagerFaceVariable is being changed, and then either
01522          * a: invert the value for boolean variables, or
01523          * b: it checks inside of IncreaseCompanyManagerFaceBits() if a left (_L) butten is pressed and then decrease else increase the variable */
01524         if (widget >= SCMFW_WIDGET_HAS_MOUSTACHE_EARRING && widget <= SCMFW_WIDGET_GLASSES_R) {
01525           CompanyManagerFaceVariable cmfv; // which CompanyManagerFaceVariable shall be edited
01526 
01527           if (widget < SCMFW_WIDGET_EYECOLOUR_L) { // Bool buttons
01528             switch (widget - SCMFW_WIDGET_HAS_MOUSTACHE_EARRING) {
01529               default: NOT_REACHED();
01530               case 0: cmfv = this->is_female ? CMFV_HAS_TIE_EARRING : CMFV_HAS_MOUSTACHE; break; // Has earring/moustache button
01531               case 1: cmfv = CMFV_HAS_GLASSES; break; // Has glasses button
01532             }
01533             SetCompanyManagerFaceBits(this->face, cmfv, this->ge, !GetCompanyManagerFaceBits(this->face, cmfv, this->ge));
01534             ScaleAllCompanyManagerFaceBits(this->face);
01535           } else { // Value buttons
01536             switch ((widget - SCMFW_WIDGET_EYECOLOUR_L) / 3) {
01537               default: NOT_REACHED();
01538               case 0: cmfv = CMFV_EYE_COLOUR; break;  // Eye colour buttons
01539               case 1: cmfv = CMFV_CHIN; break;        // Chin buttons
01540               case 2: cmfv = CMFV_EYEBROWS; break;    // Eyebrows buttons
01541               case 3: cmfv = this->is_moust_male ? CMFV_MOUSTACHE : CMFV_LIPS; break; // Moustache or lips buttons
01542               case 4: cmfv = CMFV_NOSE; break;        // Nose buttons
01543               case 5: cmfv = CMFV_HAIR; break;        // Hair buttons
01544               case 6: cmfv = CMFV_JACKET; break;      // Jacket buttons
01545               case 7: cmfv = CMFV_COLLAR; break;      // Collar buttons
01546               case 8: cmfv = CMFV_TIE_EARRING; break; // Tie/earring buttons
01547               case 9: cmfv = CMFV_GLASSES; break;     // Glasses buttons
01548             }
01549             /* 0 == left (_L), 1 == middle or 2 == right (_R) - button click */
01550             IncreaseCompanyManagerFaceBits(this->face, cmfv, this->ge, (((widget - SCMFW_WIDGET_EYECOLOUR_L) % 3) != 0) ? 1 : -1);
01551           }
01552           this->UpdateData();
01553           this->SetDirty();
01554         }
01555         break;
01556     }
01557   }
01558 
01559   virtual void OnQueryTextFinished(char *str)
01560   {
01561     if (str == NULL) return;
01562     /* Set a new company manager face number */
01563     if (!StrEmpty(str)) {
01564       this->face = strtoul(str, NULL, 10);
01565       ScaleAllCompanyManagerFaceBits(this->face);
01566       ShowErrorMessage(STR_FACE_FACECODE_SET, INVALID_STRING_ID, WL_INFO);
01567       this->UpdateData();
01568       this->SetDirty();
01569     } else {
01570       ShowErrorMessage(STR_FACE_FACECODE_ERR, INVALID_STRING_ID, WL_INFO);
01571     }
01572   }
01573 };
01574 
01576 const StringID SelectCompanyManagerFaceWindow::PART_TEXTS_IS_FEMALE[] = {
01577   STR_FACE_MOUSTACHE, STR_FACE_EARRING, // SCMFW_WIDGET_HAS_MOUSTACHE_EARRING_TEXT
01578   STR_FACE_TIE,       STR_FACE_EARRING, // SCMFW_WIDGET_TIE_EARRING_TEXT
01579 };
01580 
01582 const StringID SelectCompanyManagerFaceWindow::PART_TEXTS[] = {
01583   STR_FACE_GLASSES,   // SCMFW_WIDGET_HAS_GLASSES_TEXT
01584   STR_FACE_HAIR,      // SCMFW_WIDGET_HAIR_TEXT
01585   STR_FACE_EYEBROWS,  // SCMFW_WIDGET_EYEBROWS_TEXT
01586   STR_FACE_EYECOLOUR, // SCMFW_WIDGET_EYECOLOUR_TEXT
01587   STR_FACE_GLASSES,   // SCMFW_WIDGET_GLASSES_TEXT
01588   STR_FACE_NOSE,      // SCMFW_WIDGET_NOSE_TEXT
01589   STR_FACE_CHIN,      // SCMFW_WIDGET_CHIN_TEXT
01590   STR_FACE_JACKET,    // SCMFW_WIDGET_JACKET_TEXT
01591   STR_FACE_COLLAR,    // SCMFW_WIDGET_COLLAR_TEXT
01592 };
01593 
01595 static const WindowDesc _select_company_manager_face_desc(
01596   WDP_AUTO, 0, 0,
01597   WC_COMPANY_MANAGER_FACE, WC_NONE,
01598   WDF_UNCLICK_BUTTONS | WDF_CONSTRUCTION,
01599   _nested_select_company_manager_face_widgets, lengthof(_nested_select_company_manager_face_widgets)
01600 );
01601 
01610 static void DoSelectCompanyManagerFace(Window *parent)
01611 {
01612   if (!Company::IsValidID((CompanyID)parent->window_number)) return;
01613 
01614   if (BringWindowToFrontById(WC_COMPANY_MANAGER_FACE, parent->window_number)) return;
01615   new SelectCompanyManagerFaceWindow(&_select_company_manager_face_desc, parent);
01616 }
01617 
01618 
01620 enum CompanyInfrastructureWindowWidgets {
01621   CIW_WIDGET_CAPTION,
01622   CIW_WIDGET_RAIL_DESC,
01623   CIW_WIDGET_RAIL_COUNT,
01624   CIW_WIDGET_ROAD_DESC,
01625   CIW_WIDGET_ROAD_COUNT,
01626   CIW_WIDGET_WATER_DESC,
01627   CIW_WIDGET_WATER_COUNT,
01628   CIW_WIDGET_STATION_DESC,
01629   CIW_WIDGET_STATION_COUNT,
01630   CIW_WIDGET_TOTAL_DESC,
01631   CIW_WIDGET_TOTAL,
01632 };
01633 
01634 static const NWidgetPart _nested_company_infrastructure_widgets[] = {
01635   NWidget(NWID_HORIZONTAL),
01636     NWidget(WWT_CLOSEBOX, COLOUR_GREY),
01637     NWidget(WWT_CAPTION, COLOUR_GREY, CIW_WIDGET_CAPTION), SetDataTip(STR_COMPANY_INFRASTRUCTURE_VIEW_CAPTION, STR_TOOLTIP_WINDOW_TITLE_DRAG_THIS),
01638     NWidget(WWT_SHADEBOX, COLOUR_GREY),
01639     NWidget(WWT_STICKYBOX, COLOUR_GREY),
01640   EndContainer(),
01641   NWidget(WWT_PANEL, COLOUR_GREY),
01642     NWidget(NWID_VERTICAL), SetPIP(WD_FRAMERECT_TOP, 4, WD_FRAMETEXT_BOTTOM),
01643       NWidget(NWID_HORIZONTAL), SetPIP(2, 4, 2),
01644         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_RAIL_DESC), SetMinimalTextLines(2, 0), SetFill(1, 0),
01645         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_RAIL_COUNT), SetMinimalTextLines(2, 0), SetFill(0, 1),
01646       EndContainer(),
01647       NWidget(NWID_HORIZONTAL), SetPIP(2, 4, 2),
01648         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_ROAD_DESC), SetMinimalTextLines(2, 0), SetFill(1, 0),
01649         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_ROAD_COUNT), SetMinimalTextLines(2, 0), SetFill(0, 1),
01650       EndContainer(),
01651       NWidget(NWID_HORIZONTAL), SetPIP(2, 4, 2),
01652         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_WATER_DESC), SetMinimalTextLines(2, 0), SetFill(1, 0),
01653         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_WATER_COUNT), SetMinimalTextLines(2, 0), SetFill(0, 1),
01654       EndContainer(),
01655       NWidget(NWID_HORIZONTAL), SetPIP(2, 4, 2),
01656         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_STATION_DESC), SetMinimalTextLines(3, 0), SetFill(1, 0),
01657         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_STATION_COUNT), SetMinimalTextLines(3, 0), SetFill(0, 1),
01658       EndContainer(),
01659       NWidget(NWID_HORIZONTAL), SetPIP(2, 4, 2),
01660         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_TOTAL_DESC), SetFill(1, 0),
01661         NWidget(WWT_EMPTY, COLOUR_GREY, CIW_WIDGET_TOTAL), SetFill(0, 1),
01662       EndContainer(),
01663     EndContainer(),
01664   EndContainer(),
01665 };
01666 
01670 struct CompanyInfrastructureWindow : Window
01671 {
01672   RailTypes railtypes; 
01673   RoadTypes roadtypes; 
01674 
01675   uint total_width; 
01676 
01677   CompanyInfrastructureWindow(const WindowDesc *desc, WindowNumber window_number) : Window()
01678   {
01679     this->UpdateRailRoadTypes();
01680 
01681     this->InitNested(desc, window_number);
01682     this->owner = (Owner)this->window_number;
01683   }
01684 
01685   void UpdateRailRoadTypes()
01686   {
01687     this->railtypes = RAILTYPES_NONE;
01688     this->roadtypes = ROADTYPES_ROAD; // Road is always available.
01689 
01690     /* Find the used railtypes. */
01691     Engine *e;
01692     FOR_ALL_ENGINES_OF_TYPE(e, VEH_TRAIN) {
01693       if (!HasBit(e->info.climates, _settings_game.game_creation.landscape)) continue;
01694 
01695       this->railtypes |= GetRailTypeInfo(e->u.rail.railtype)->introduces_railtypes;
01696     }
01697 
01698     /* Get the date introduced railtypes as well. */
01699     this->railtypes = AddDateIntroducedRailTypes(this->railtypes, MAX_DAY);
01700 
01701     /* Tram is only visible when there will be a tram. */
01702     FOR_ALL_ENGINES_OF_TYPE(e, VEH_ROAD) {
01703       if (!HasBit(e->info.climates, _settings_game.game_creation.landscape)) continue;
01704       if (!HasBit(e->info.misc_flags, EF_ROAD_TRAM)) continue;
01705 
01706       this->roadtypes |= ROADTYPES_TRAM;
01707       break;
01708     }
01709   }
01710 
01712   Money GetTotalMaintenanceCost() const
01713   {
01714     const Company *c = Company::Get((CompanyID)this->window_number);
01715     Money total;
01716 
01717     for (RailType rt = RAILTYPE_BEGIN; rt != RAILTYPE_END; rt++) {
01718       if (HasBit(this->railtypes, rt)) total += RailMaintenanceCost(rt, c->infrastructure.rail[rt]);
01719     }
01720     total += SignalMaintenanceCost(c->infrastructure.signal);
01721 
01722     if (HasBit(this->roadtypes, ROADTYPE_ROAD)) total += RoadMaintenanceCost(ROADTYPE_ROAD, c->infrastructure.road[ROADTYPE_ROAD]);
01723     if (HasBit(this->roadtypes, ROADTYPE_TRAM)) total += RoadMaintenanceCost(ROADTYPE_TRAM, c->infrastructure.road[ROADTYPE_TRAM]);
01724 
01725     total += CanalMaintenanceCost(c->infrastructure.water);
01726     total += StationMaintenanceCost(c->infrastructure.station);
01727     total += AirportMaintenanceCost(c->index);
01728 
01729     return total;
01730   }
01731 
01732   virtual void SetStringParameters(int widget) const
01733   {
01734     switch (widget) {
01735       case CIW_WIDGET_CAPTION:
01736         SetDParam(0, (CompanyID)this->window_number);
01737         break;
01738     }
01739   }
01740 
01741   virtual void UpdateWidgetSize(int widget, Dimension *size, const Dimension &padding, Dimension *fill, Dimension *resize)
01742   {
01743     const Company *c = Company::Get((CompanyID)this->window_number);
01744 
01745     switch (widget) {
01746       case CIW_WIDGET_RAIL_DESC: {
01747         uint lines = 1;
01748 
01749         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_RAIL_SECT).width);
01750 
01751         for (RailType rt = RAILTYPE_BEGIN; rt < RAILTYPE_END; rt++) {
01752           if (HasBit(this->railtypes, rt)) {
01753             lines++;
01754             SetDParam(0, GetRailTypeInfo(rt)->strings.name);
01755             size->width = max(size->width, GetStringBoundingBox(STR_WHITE_STRING).width + WD_FRAMERECT_LEFT);
01756           }
01757         }
01758         if (this->railtypes != RAILTYPES_NONE) {
01759           lines++;
01760           size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_SIGNALS).width + WD_FRAMERECT_LEFT);
01761         }
01762 
01763         size->height = max(size->height, lines * FONT_HEIGHT_NORMAL);
01764         break;
01765       }
01766 
01767       case CIW_WIDGET_ROAD_DESC: {
01768         uint lines = 1;
01769 
01770         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_ROAD_SECT).width);
01771 
01772         if (HasBit(this->roadtypes, ROADTYPE_ROAD)) {
01773           lines++;
01774           size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_ROAD).width + WD_FRAMERECT_LEFT);
01775         }
01776         if (HasBit(this->roadtypes, ROADTYPE_TRAM)) {
01777           lines++;
01778           size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_TRAMWAY).width + WD_FRAMERECT_LEFT);
01779         }
01780 
01781         size->height = max(size->height, lines * FONT_HEIGHT_NORMAL);
01782         break;
01783       }
01784 
01785       case CIW_WIDGET_WATER_DESC:
01786         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_WATER_SECT).width);
01787         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_CANALS).width + WD_FRAMERECT_LEFT);
01788         break;
01789 
01790       case CIW_WIDGET_STATION_DESC:
01791         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_STATION_SECT).width);
01792         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_STATIONS).width + WD_FRAMERECT_LEFT);
01793         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_AIRPORTS).width + WD_FRAMERECT_LEFT);
01794         break;
01795 
01796       case CIW_WIDGET_RAIL_COUNT:
01797       case CIW_WIDGET_ROAD_COUNT:
01798       case CIW_WIDGET_WATER_COUNT:
01799       case CIW_WIDGET_STATION_COUNT:
01800       case CIW_WIDGET_TOTAL: {
01801         /* Find the maximum count that is displayed. */
01802         uint32 max_val = 1000;  // Some random number to reserve enough space.
01803         Money max_cost = 10000; // Some random number to reserve enough space.
01804         for (RailType rt = RAILTYPE_BEGIN; rt < RAILTYPE_END; rt++) {
01805           max_val = max(max_val, c->infrastructure.rail[rt]);
01806           max_cost = max(max_cost, RailMaintenanceCost(rt, c->infrastructure.rail[rt]));
01807         }
01808         max_val = max(max_val, c->infrastructure.signal);
01809         max_cost = max(max_cost, SignalMaintenanceCost(c->infrastructure.signal));
01810         for (RoadType rt = ROADTYPE_BEGIN; rt < ROADTYPE_END; rt++) {
01811           max_val = max(max_val, c->infrastructure.road[rt]);
01812           max_cost = max(max_cost, RoadMaintenanceCost(rt, c->infrastructure.road[rt]));
01813         }
01814         max_val = max(max_val, c->infrastructure.water);
01815         max_cost = max(max_cost, CanalMaintenanceCost(c->infrastructure.water));
01816         max_val = max(max_val, c->infrastructure.station);
01817         max_cost = max(max_cost, StationMaintenanceCost(c->infrastructure.station));
01818         max_val = max(max_val, c->infrastructure.airport);
01819         max_cost = max(max_cost, AirportMaintenanceCost(c->index));
01820 
01821         SetDParam(0, max_val);
01822         SetDParam(1, max_cost * 12); // Convert to per year
01823         size->width = max(size->width, GetStringBoundingBox(_settings_game.economy.infrastructure_maintenance ? STR_COMPANY_INFRASTRUCTURE_VIEW_COST : STR_WHITE_COMMA).width + 20); // Reserve some wiggle room.
01824 
01825         if (_settings_game.economy.infrastructure_maintenance) {
01826           SetDParam(0, this->GetTotalMaintenanceCost() * 12); // Convert to per year
01827           this->total_width = GetStringBoundingBox(STR_COMPANY_INFRASTRUCTURE_VIEW_TOTAL).width + 20;
01828           size->width = max(size->width, this->total_width);
01829         }
01830 
01831         /* Set height of the total line. */
01832         if (widget == CIW_WIDGET_TOTAL) {
01833           size->height = _settings_game.economy.infrastructure_maintenance ? max(size->height, EXP_LINESPACE + FONT_HEIGHT_NORMAL) : 0;
01834         }
01835         break;
01836       }
01837     }
01838   }
01839 
01840   virtual void DrawWidget(const Rect &r, int widget) const
01841   {
01842     const Company *c = Company::Get((CompanyID)this->window_number);
01843     int y = r.top;
01844 
01845     int offs_left = _current_text_dir == TD_LTR ? WD_FRAMERECT_LEFT : 0;
01846     int offs_right = _current_text_dir == TD_LTR ? 0 : WD_FRAMERECT_LEFT;
01847 
01848     switch (widget) {
01849       case CIW_WIDGET_RAIL_DESC:
01850         DrawString(r.left, r.right, y, STR_COMPANY_INFRASTRUCTURE_VIEW_RAIL_SECT);
01851 
01852         if (this->railtypes != RAILTYPES_NONE) {
01853           /* Draw name of each valid railtype. */
01854           for (RailType rt = RAILTYPE_BEGIN; rt != RAILTYPE_END; rt++) {
01855             if (HasBit(this->railtypes, rt)) {
01856               SetDParam(0, GetRailTypeInfo(rt)->strings.name);
01857               DrawString(r.left + offs_left, r.right - offs_right, y += FONT_HEIGHT_NORMAL, STR_WHITE_STRING);
01858             }
01859           }
01860           DrawString(r.left + offs_left, r.right - offs_right, y += FONT_HEIGHT_NORMAL, STR_COMPANY_INFRASTRUCTURE_VIEW_SIGNALS);
01861         } else {
01862           /* No valid railtype. */
01863           DrawString(r.left + offs_left, r.right - offs_right, y += FONT_HEIGHT_NORMAL, STR_COMPANY_VIEW_INFRASTRUCTURE_NONE);
01864         }
01865 
01866         break;
01867 
01868       case CIW_WIDGET_RAIL_COUNT:
01869         /* Draw infrastructure count for each valid railtype. */
01870         for (RailType rt = RAILTYPE_BEGIN; rt != RAILTYPE_END; rt++) {
01871           if (HasBit(this->railtypes, rt)) {
01872             SetDParam(0, c->infrastructure.rail[rt]);
01873             SetDParam(1, RailMaintenanceCost(rt, c->infrastructure.rail[rt]) * 12); // Convert to per year
01874             DrawString(r.left, r.right, y += FONT_HEIGHT_NORMAL, _settings_game.economy.infrastructure_maintenance ? STR_COMPANY_INFRASTRUCTURE_VIEW_COST : STR_WHITE_COMMA);
01875           }
01876         }
01877         if (this->railtypes != RAILTYPES_NONE) {
01878           SetDParam(0, c->infrastructure.signal);
01879           SetDParam(1, SignalMaintenanceCost(c->infrastructure.signal) * 12); // Convert to per year
01880           DrawString(r.left, r.right, y += FONT_HEIGHT_NORMAL, _settings_game.economy.infrastructure_maintenance ? STR_COMPANY_INFRASTRUCTURE_VIEW_COST : STR_WHITE_COMMA);
01881         }
01882         break;
01883 
01884       case CIW_WIDGET_ROAD_DESC:
01885         DrawString(r.left, r.right, y, STR_COMPANY_INFRASTRUCTURE_VIEW_ROAD_SECT);
01886 
01887         if (this->roadtypes != ROADTYPES_NONE) {
01888           if (HasBit(this->roadtypes, ROADTYPE_ROAD)) DrawString(r.left + offs_left, r.right - offs_right, y += FONT_HEIGHT_NORMAL, STR_COMPANY_INFRASTRUCTURE_VIEW_ROAD);
01889           if (HasBit(this->roadtypes, ROADTYPE_TRAM)) DrawString(r.left + offs_left, r.right - offs_right, y += FONT_HEIGHT_NORMAL, STR_COMPANY_INFRASTRUCTURE_VIEW_TRAMWAY);
01890         } else {
01891           /* No valid roadtypes. */
01892           DrawString(r.left + offs_left, r.right - offs_right, y += FONT_HEIGHT_NORMAL, STR_COMPANY_VIEW_INFRASTRUCTURE_NONE);
01893         }
01894 
01895         break;
01896 
01897       case CIW_WIDGET_ROAD_COUNT:
01898         if (HasBit(this->roadtypes, ROADTYPE_ROAD)) {
01899           SetDParam(0, c->infrastructure.road[ROADTYPE_ROAD]);
01900           SetDParam(1, RoadMaintenanceCost(ROADTYPE_ROAD, c->infrastructure.road[ROADTYPE_ROAD]) * 12); // Convert to per year
01901           DrawString(r.left, r.right, y += FONT_HEIGHT_NORMAL, _settings_game.economy.infrastructure_maintenance ? STR_COMPANY_INFRASTRUCTURE_VIEW_COST : STR_WHITE_COMMA);
01902         }
01903         if (HasBit(this->roadtypes, ROADTYPE_TRAM)) {
01904           SetDParam(0, c->infrastructure.road[ROADTYPE_TRAM]);
01905           SetDParam(1, RoadMaintenanceCost(ROADTYPE_TRAM, c->infrastructure.road[ROADTYPE_TRAM]) * 12); // Convert to per year
01906           DrawString(r.left, r.right, y += FONT_HEIGHT_NORMAL, _settings_game.economy.infrastructure_maintenance ? STR_COMPANY_INFRASTRUCTURE_VIEW_COST : STR_WHITE_COMMA);
01907         }
01908         break;
01909 
01910       case CIW_WIDGET_WATER_DESC:
01911         DrawString(r.left, r.right, y, STR_COMPANY_INFRASTRUCTURE_VIEW_WATER_SECT);
01912         DrawString(r.left + offs_left, r.right - offs_right, y += FONT_HEIGHT_NORMAL, STR_COMPANY_INFRASTRUCTURE_VIEW_CANALS);
01913         break;
01914 
01915       case CIW_WIDGET_WATER_COUNT:
01916         SetDParam(0, c->infrastructure.water);
01917         SetDParam(1, CanalMaintenanceCost(c->infrastructure.water) * 12); // Convert to per year
01918         DrawString(r.left, r.right, y += FONT_HEIGHT_NORMAL, _settings_game.economy.infrastructure_maintenance ? STR_COMPANY_INFRASTRUCTURE_VIEW_COST : STR_WHITE_COMMA);
01919         break;
01920 
01921       case CIW_WIDGET_TOTAL:
01922         if (_settings_game.economy.infrastructure_maintenance) {
01923           GfxFillRect(r.left, y, r.left + this->total_width, y, PC_WHITE);
01924           y += EXP_LINESPACE;
01925           SetDParam(0, this->GetTotalMaintenanceCost() * 12); // Convert to per year
01926           DrawString(r.left, r.right, y, STR_COMPANY_INFRASTRUCTURE_VIEW_TOTAL);
01927         }
01928         break;
01929 
01930       case CIW_WIDGET_STATION_DESC:
01931         DrawString(r.left, r.right, y, STR_COMPANY_INFRASTRUCTURE_VIEW_STATION_SECT);
01932         DrawString(r.left + offs_left, r.right - offs_right, y += FONT_HEIGHT_NORMAL, STR_COMPANY_INFRASTRUCTURE_VIEW_STATIONS);
01933         DrawString(r.left + offs_left, r.right - offs_right, y += FONT_HEIGHT_NORMAL, STR_COMPANY_INFRASTRUCTURE_VIEW_AIRPORTS);
01934         break;
01935 
01936       case CIW_WIDGET_STATION_COUNT:
01937         SetDParam(0, c->infrastructure.station);
01938         SetDParam(1, StationMaintenanceCost(c->infrastructure.station) * 12); // Convert to per year
01939         DrawString(r.left, r.right, y += FONT_HEIGHT_NORMAL, _settings_game.economy.infrastructure_maintenance ? STR_COMPANY_INFRASTRUCTURE_VIEW_COST : STR_WHITE_COMMA);
01940         SetDParam(0, c->infrastructure.airport);
01941         SetDParam(1, AirportMaintenanceCost(c->index) * 12); // Convert to per year
01942         DrawString(r.left, r.right, y += FONT_HEIGHT_NORMAL, _settings_game.economy.infrastructure_maintenance ? STR_COMPANY_INFRASTRUCTURE_VIEW_COST : STR_WHITE_COMMA);
01943         break;
01944     }
01945   }
01946 
01952   virtual void OnInvalidateData(int data = 0, bool gui_scope = true)
01953   {
01954     if (!gui_scope) return;
01955 
01956     this->UpdateRailRoadTypes();
01957     this->ReInit();
01958   }
01959 };
01960 
01961 static const WindowDesc _company_infrastructure_desc(
01962   WDP_AUTO, 0, 0,
01963   WC_COMPANY_INFRASTRUCTURE, WC_NONE,
01964   WDF_UNCLICK_BUTTONS,
01965   _nested_company_infrastructure_widgets, lengthof(_nested_company_infrastructure_widgets)
01966 );
01967 
01972 static void ShowCompanyInfrastructure(CompanyID company)
01973 {
01974   if (!Company::IsValidID(company)) return;
01975   AllocateWindowDescFront<CompanyInfrastructureWindow>(&_company_infrastructure_desc, company);
01976 }
01977 
01978 
01980 enum CompanyWindowWidgets {
01981   CW_WIDGET_CAPTION,
01982 
01983   CW_WIDGET_FACE,
01984   CW_WIDGET_FACE_TITLE,
01985 
01986   CW_WIDGET_DESC_INAUGURATION,
01987   CW_WIDGET_DESC_COLOUR_SCHEME,
01988   CW_WIDGET_DESC_COLOUR_SCHEME_EXAMPLE,
01989   CW_WIDGET_DESC_VEHICLE,
01990   CW_WIDGET_DESC_VEHICLE_COUNTS,
01991   CW_WIDGET_DESC_COMPANY_VALUE,
01992   CW_WIDGET_DESC_INFRASTRUCTURE,
01993   CW_WIDGET_DESC_INFRASTRUCTURE_COUNTS,
01994 
01995   CW_WIDGET_SELECT_DESC_OWNERS,
01996   CW_WIDGET_DESC_OWNERS,
01997 
01998   CW_WIDGET_SELECT_BUTTONS,     
01999   CW_WIDGET_NEW_FACE,
02000   CW_WIDGET_COLOUR_SCHEME,
02001   CW_WIDGET_PRESIDENT_NAME,
02002   CW_WIDGET_COMPANY_NAME,
02003   CW_WIDGET_BUY_SHARE,
02004   CW_WIDGET_SELL_SHARE,
02005 
02006   CW_WIDGET_SELECT_VIEW_BUILD_HQ,
02007   CW_WIDGET_VIEW_HQ,
02008   CW_WIDGET_BUILD_HQ,
02009 
02010   CW_WIDGET_SELECT_RELOCATE,    
02011   CW_WIDGET_RELOCATE_HQ,
02012 
02013   CW_WIDGET_VIEW_INFRASTRUCTURE,
02014 
02015   CW_WIDGET_HAS_PASSWORD,       
02016   CW_WIDGET_SELECT_MULTIPLAYER, 
02017   CW_WIDGET_COMPANY_PASSWORD,
02018   CW_WIDGET_COMPANY_JOIN,
02019 };
02020 
02021 static const NWidgetPart _nested_company_widgets[] = {
02022   NWidget(NWID_HORIZONTAL),
02023     NWidget(WWT_CLOSEBOX, COLOUR_GREY),
02024     NWidget(WWT_CAPTION, COLOUR_GREY, CW_WIDGET_CAPTION), SetDataTip(STR_COMPANY_VIEW_CAPTION, STR_TOOLTIP_WINDOW_TITLE_DRAG_THIS),
02025     NWidget(WWT_SHADEBOX, COLOUR_GREY),
02026     NWidget(WWT_STICKYBOX, COLOUR_GREY),
02027   EndContainer(),
02028   NWidget(WWT_PANEL, COLOUR_GREY),
02029     NWidget(NWID_HORIZONTAL), SetPIP(4, 6, 4),
02030       NWidget(NWID_VERTICAL), SetPIP(4, 2, 4),
02031         NWidget(WWT_EMPTY, INVALID_COLOUR, CW_WIDGET_FACE), SetMinimalSize(92, 119), SetFill(1, 0),
02032         NWidget(WWT_EMPTY, INVALID_COLOUR, CW_WIDGET_FACE_TITLE), SetFill(1, 1), SetMinimalTextLines(2, 0),
02033       EndContainer(),
02034       NWidget(NWID_VERTICAL),
02035         NWidget(NWID_HORIZONTAL),
02036           NWidget(NWID_VERTICAL), SetPIP(4, 5, 5),
02037             NWidget(WWT_TEXT, COLOUR_GREY, CW_WIDGET_DESC_INAUGURATION), SetDataTip(STR_COMPANY_VIEW_INAUGURATED_TITLE, STR_NULL), SetFill(1, 0),
02038             NWidget(NWID_HORIZONTAL), SetPIP(0, 5, 0),
02039               NWidget(WWT_LABEL, COLOUR_GREY, CW_WIDGET_DESC_COLOUR_SCHEME), SetDataTip(STR_COMPANY_VIEW_COLOUR_SCHEME_TITLE, STR_NULL),
02040               NWidget(WWT_EMPTY, INVALID_COLOUR, CW_WIDGET_DESC_COLOUR_SCHEME_EXAMPLE), SetMinimalSize(30, 0), SetFill(0, 1),
02041               NWidget(NWID_SPACER), SetFill(1, 0),
02042             EndContainer(),
02043             NWidget(NWID_HORIZONTAL), SetPIP(0, 4, 0),
02044               NWidget(NWID_VERTICAL),
02045                 NWidget(WWT_TEXT, COLOUR_GREY, CW_WIDGET_DESC_VEHICLE), SetDataTip(STR_COMPANY_VIEW_VEHICLES_TITLE, STR_NULL),
02046                 NWidget(NWID_SPACER), SetFill(0, 1),
02047               EndContainer(),
02048               NWidget(WWT_EMPTY, INVALID_COLOUR, CW_WIDGET_DESC_VEHICLE_COUNTS), SetMinimalTextLines(4, 0),
02049               NWidget(NWID_SPACER), SetFill(1, 0),
02050             EndContainer(),
02051           EndContainer(),
02052           NWidget(NWID_VERTICAL), SetPIP(4, 2, 4),
02053             NWidget(NWID_SELECTION, INVALID_COLOUR, CW_WIDGET_SELECT_VIEW_BUILD_HQ),
02054               NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_VIEW_HQ), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_VIEW_HQ_BUTTON, STR_COMPANY_VIEW_VIEW_HQ_TOOLTIP),
02055               NWidget(WWT_TEXTBTN, COLOUR_GREY, CW_WIDGET_BUILD_HQ), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_BUILD_HQ_BUTTON, STR_COMPANY_VIEW_BUILD_HQ_TOOLTIP),
02056             EndContainer(),
02057             NWidget(NWID_SELECTION, INVALID_COLOUR, CW_WIDGET_SELECT_RELOCATE),
02058               NWidget(WWT_TEXTBTN, COLOUR_GREY, CW_WIDGET_RELOCATE_HQ), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_RELOCATE_HQ, STR_COMPANY_VIEW_RELOCATE_COMPANY_HEADQUARTERS),
02059               NWidget(NWID_SPACER), SetMinimalSize(90, 0),
02060             EndContainer(),
02061             NWidget(NWID_SPACER), SetFill(0, 1),
02062           EndContainer(),
02063         EndContainer(),
02064         NWidget(WWT_TEXT, COLOUR_GREY, CW_WIDGET_DESC_COMPANY_VALUE), SetDataTip(STR_COMPANY_VIEW_COMPANY_VALUE, STR_NULL), SetFill(1, 0),
02065           NWidget(NWID_VERTICAL), SetPIP(4, 2, 4),
02066             NWidget(NWID_HORIZONTAL), SetPIP(0, 4, 0),
02067               NWidget(NWID_VERTICAL),
02068                 NWidget(WWT_TEXT, COLOUR_GREY, CW_WIDGET_DESC_INFRASTRUCTURE), SetDataTip(STR_COMPANY_VIEW_INFRASTRUCTURE, STR_NULL),
02069                 NWidget(NWID_SPACER), SetFill(0, 1),
02070               EndContainer(),
02071               NWidget(WWT_EMPTY, INVALID_COLOUR, CW_WIDGET_DESC_INFRASTRUCTURE_COUNTS), SetMinimalTextLines(5, 0), SetFill(1, 0),
02072               NWidget(NWID_VERTICAL),
02073                 NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_VIEW_INFRASTRUCTURE), SetDataTip(STR_COMPANY_VIEW_INFRASTRUCTURE_BUTTON, STR_COMPANY_VIEW_INFRASTRUCTURE_TOOLTIP),
02074                 NWidget(NWID_SPACER), SetFill(0, 1), SetMinimalSize(90, 0),
02075               EndContainer(),
02076             EndContainer(),
02077           EndContainer(),
02078         NWidget(NWID_HORIZONTAL),
02079           NWidget(NWID_SELECTION, INVALID_COLOUR, CW_WIDGET_SELECT_DESC_OWNERS),
02080             NWidget(NWID_VERTICAL), SetPIP(5, 5, 4),
02081               NWidget(WWT_EMPTY, INVALID_COLOUR, CW_WIDGET_DESC_OWNERS), SetMinimalTextLines(3, 0),
02082               NWidget(NWID_SPACER), SetFill(0, 1),
02083             EndContainer(),
02084           EndContainer(),
02085           NWidget(NWID_VERTICAL), SetPIP(4, 2, 4),
02086             NWidget(NWID_SPACER), SetMinimalSize(90, 0), SetFill(0, 1),
02087             /* Multi player buttons. */
02088             NWidget(NWID_HORIZONTAL),
02089               NWidget(WWT_EMPTY, COLOUR_GREY, CW_WIDGET_HAS_PASSWORD),
02090               NWidget(NWID_SELECTION, INVALID_COLOUR, CW_WIDGET_SELECT_MULTIPLAYER),
02091                 NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_COMPANY_PASSWORD), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_PASSWORD, STR_COMPANY_VIEW_PASSWORD_TOOLTIP),
02092                 NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_COMPANY_JOIN), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_JOIN, STR_COMPANY_VIEW_JOIN_TOOLTIP),
02093               EndContainer(),
02094             EndContainer(),
02095           EndContainer(),
02096         EndContainer(),
02097       EndContainer(),
02098     EndContainer(),
02099   EndContainer(),
02100   /* Button bars at the bottom. */
02101   NWidget(NWID_SELECTION, INVALID_COLOUR, CW_WIDGET_SELECT_BUTTONS),
02102     NWidget(NWID_HORIZONTAL, NC_EQUALSIZE),
02103       NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_NEW_FACE), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_NEW_FACE_BUTTON, STR_COMPANY_VIEW_NEW_FACE_TOOLTIP),
02104       NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_COLOUR_SCHEME), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_COLOUR_SCHEME_BUTTON, STR_COMPANY_VIEW_COLOUR_SCHEME_TOOLTIP),
02105       NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_PRESIDENT_NAME), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_PRESIDENT_NAME_BUTTON, STR_COMPANY_VIEW_PRESIDENT_NAME_TOOLTIP),
02106       NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_COMPANY_NAME), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_COMPANY_NAME_BUTTON, STR_COMPANY_VIEW_COMPANY_NAME_TOOLTIP),
02107     EndContainer(),
02108     NWidget(NWID_HORIZONTAL, NC_EQUALSIZE),
02109       NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_BUY_SHARE), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_BUY_SHARE_BUTTON, STR_COMPANY_VIEW_BUY_SHARE_TOOLTIP),
02110       NWidget(WWT_PUSHTXTBTN, COLOUR_GREY, CW_WIDGET_SELL_SHARE), SetFill(1, 0), SetDataTip(STR_COMPANY_VIEW_SELL_SHARE_BUTTON, STR_COMPANY_VIEW_SELL_SHARE_TOOLTIP),
02111     EndContainer(),
02112   EndContainer(),
02113 };
02114 
02115 int GetAmountOwnedBy(const Company *c, Owner owner)
02116 {
02117   return (c->share_owners[0] == owner) +
02118          (c->share_owners[1] == owner) +
02119          (c->share_owners[2] == owner) +
02120          (c->share_owners[3] == owner);
02121 }
02122 
02124 static const StringID _company_view_vehicle_count_strings[] = {
02125   STR_COMPANY_VIEW_TRAINS, STR_COMPANY_VIEW_ROAD_VEHICLES, STR_COMPANY_VIEW_SHIPS, STR_COMPANY_VIEW_AIRCRAFT
02126 };
02127 
02131 struct CompanyWindow : Window
02132 {
02133   CompanyWindowWidgets query_widget;
02134 
02136   enum CompanyWindowPlanes {
02137     /* Display planes of the #CW_WIDGET_SELECT_MULTIPLAYER selection widget. */
02138     CWP_MP_C_PWD = 0, 
02139     CWP_MP_C_JOIN,    
02140 
02141     /* Display planes of the #CW_WIDGET_SELECT_VIEW_BUILD_HQ selection widget. */
02142     CWP_VB_VIEW = 0,  
02143     CWP_VB_BUILD,     
02144 
02145     /* Display planes of the #CW_WIDGET_SELECT_RELOCATE selection widget. */
02146     CWP_RELOCATE_SHOW = 0, 
02147     CWP_RELOCATE_HIDE,     
02148 
02149     /* Display planes of the #CW_WIDGET_SELECT_BUTTONS selection widget. */
02150     CWP_BUTTONS_LOCAL = 0, 
02151     CWP_BUTTONS_OTHER,     
02152   };
02153 
02154   CompanyWindow(const WindowDesc *desc, WindowNumber window_number) : Window()
02155   {
02156     this->InitNested(desc, window_number);
02157     this->owner = (Owner)this->window_number;
02158   }
02159 
02160   virtual void OnPaint()
02161   {
02162     const Company *c = Company::Get((CompanyID)this->window_number);
02163     bool local = this->window_number == _local_company;
02164 
02165     if (!this->IsShaded()) {
02166       bool reinit = false;
02167 
02168       /* Button bar selection. */
02169       int plane = local ? CWP_BUTTONS_LOCAL : CWP_BUTTONS_OTHER;
02170       NWidgetStacked *wi = this->GetWidget<NWidgetStacked>(CW_WIDGET_SELECT_BUTTONS);
02171       if (plane != wi->shown_plane) {
02172         wi->SetDisplayedPlane(plane);
02173         this->SetDirty();
02174         return;
02175       }
02176 
02177       /* Build HQ button handling. */
02178       plane = (local && c->location_of_HQ == INVALID_TILE) ? CWP_VB_BUILD : CWP_VB_VIEW;
02179       wi = this->GetWidget<NWidgetStacked>(CW_WIDGET_SELECT_VIEW_BUILD_HQ);
02180       if (plane != wi->shown_plane) {
02181         wi->SetDisplayedPlane(plane);
02182         this->SetDirty();
02183         return;
02184       }
02185 
02186       this->SetWidgetDisabledState(CW_WIDGET_VIEW_HQ, c->location_of_HQ == INVALID_TILE);
02187 
02188       /* Enable/disable 'Relocate HQ' button. */
02189       plane = (!local || c->location_of_HQ == INVALID_TILE) ? CWP_RELOCATE_HIDE : CWP_RELOCATE_SHOW;
02190       wi = this->GetWidget<NWidgetStacked>(CW_WIDGET_SELECT_RELOCATE);
02191       if (plane != wi->shown_plane) {
02192         wi->SetDisplayedPlane(plane);
02193         this->SetDirty();
02194         return;
02195       }
02196 
02197       /* Owners of company */
02198       plane = SZSP_HORIZONTAL;
02199       for (uint i = 0; i < lengthof(c->share_owners); i++) {
02200         if (c->share_owners[i] != INVALID_COMPANY) {
02201           plane = 0;
02202           break;
02203         }
02204       }
02205       wi = this->GetWidget<NWidgetStacked>(CW_WIDGET_SELECT_DESC_OWNERS);
02206       if (plane != wi->shown_plane) {
02207         wi->SetDisplayedPlane(plane);
02208         reinit = true;
02209       }
02210 
02211       /* Multiplayer buttons. */
02212       plane = ((!_networking) ? (int)SZSP_NONE : (int)(local ? CWP_MP_C_PWD : CWP_MP_C_JOIN));
02213       wi = this->GetWidget<NWidgetStacked>(CW_WIDGET_SELECT_MULTIPLAYER);
02214       if (plane != wi->shown_plane) {
02215         wi->SetDisplayedPlane(plane);
02216         reinit = true;
02217       }
02218       this->SetWidgetDisabledState(CW_WIDGET_COMPANY_JOIN,   c->is_ai);
02219 
02220       if (reinit) {
02221         this->ReInit();
02222         return;
02223       }
02224     }
02225 
02226     if (!local) {
02227       if (_settings_game.economy.allow_shares) { // Shares are allowed
02228         /* If all shares are owned by someone (none by nobody), disable buy button */
02229         this->SetWidgetDisabledState(CW_WIDGET_BUY_SHARE, GetAmountOwnedBy(c, INVALID_OWNER) == 0 ||
02230             /* Only 25% left to buy. If the company is human, disable buying it up.. TODO issues! */
02231             (GetAmountOwnedBy(c, INVALID_OWNER) == 1 && !c->is_ai) ||
02232             /* Spectators cannot do anything of course */
02233             _local_company == COMPANY_SPECTATOR);
02234 
02235         /* If the company doesn't own any shares, disable sell button */
02236         this->SetWidgetDisabledState(CW_WIDGET_SELL_SHARE, (GetAmountOwnedBy(c, _local_company) == 0) ||
02237             /* Spectators cannot do anything of course */
02238             _local_company == COMPANY_SPECTATOR);
02239       } else { // Shares are not allowed, disable buy/sell buttons
02240         this->DisableWidget(CW_WIDGET_BUY_SHARE);
02241         this->DisableWidget(CW_WIDGET_SELL_SHARE);
02242       }
02243     }
02244 
02245     this->DrawWidgets();
02246   }
02247 
02248   virtual void UpdateWidgetSize(int widget, Dimension *size, const Dimension &padding, Dimension *fill, Dimension *resize)
02249   {
02250     switch (widget) {
02251       case CW_WIDGET_DESC_COLOUR_SCHEME_EXAMPLE: {
02252         Point offset;
02253         Dimension d = GetSpriteSize(SPR_VEH_BUS_SW_VIEW, &offset);
02254         d.width -= offset.x;
02255         d.height -= offset.y;
02256         *size = maxdim(*size, d);
02257         break;
02258       }
02259 
02260       case CW_WIDGET_DESC_COMPANY_VALUE:
02261         SetDParam(0, INT64_MAX); // Arguably the maximum company value
02262         size->width = GetStringBoundingBox(STR_COMPANY_VIEW_COMPANY_VALUE).width;
02263         break;
02264 
02265       case CW_WIDGET_DESC_VEHICLE_COUNTS:
02266         SetDParam(0, 5000); // Maximum number of vehicles
02267         for (uint i = 0; i < lengthof(_company_view_vehicle_count_strings); i++) {
02268           size->width = max(size->width, GetStringBoundingBox(_company_view_vehicle_count_strings[i]).width);
02269         }
02270         break;
02271 
02272       case CW_WIDGET_DESC_INFRASTRUCTURE_COUNTS:
02273         SetDParam(0, UINT_MAX);
02274         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_VIEW_INFRASTRUCTURE_RAIL).width);
02275         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_VIEW_INFRASTRUCTURE_ROAD).width);
02276         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_VIEW_INFRASTRUCTURE_WATER).width);
02277         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_VIEW_INFRASTRUCTURE_STATION).width);
02278         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_VIEW_INFRASTRUCTURE_AIRPORT).width);
02279         size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_VIEW_INFRASTRUCTURE_NONE).width);
02280         break;
02281 
02282       case CW_WIDGET_DESC_OWNERS: {
02283         const Company *c2;
02284 
02285         FOR_ALL_COMPANIES(c2) {
02286           SetDParam(0, 25);
02287           SetDParam(1, c2->index);
02288 
02289           size->width = max(size->width, GetStringBoundingBox(STR_COMPANY_VIEW_SHARES_OWNED_BY).width);
02290         }
02291         break;
02292       }
02293 
02294 #ifdef ENABLE_NETWORK
02295       case CW_WIDGET_HAS_PASSWORD:
02296         *size = maxdim(*size, GetSpriteSize(SPR_LOCK));
02297         break;
02298 #endif /* ENABLE_NETWORK */
02299     }
02300   }
02301 
02302   virtual void DrawWidget(const Rect &r, int widget) const
02303   {
02304     const Company *c = Company::Get((CompanyID)this->window_number);
02305     switch (widget) {
02306       case CW_WIDGET_FACE:
02307         DrawCompanyManagerFace(c->face, c->colour, r.left, r.top);
02308         break;
02309 
02310       case CW_WIDGET_FACE_TITLE:
02311         SetDParam(0, c->index);
02312         DrawStringMultiLine(r.left, r.right, r.top, r.bottom, STR_COMPANY_VIEW_PRESIDENT_MANAGER_TITLE, TC_FROMSTRING, SA_HOR_CENTER);
02313         break;
02314 
02315       case CW_WIDGET_DESC_COLOUR_SCHEME_EXAMPLE: {
02316         Point offset;
02317         Dimension d = GetSpriteSize(SPR_VEH_BUS_SW_VIEW, &offset);
02318         d.height -= offset.y;
02319         DrawSprite(SPR_VEH_BUS_SW_VIEW, COMPANY_SPRITE_COLOUR(c->index), r.left - offset.x, (r.top + r.bottom - d.height) / 2 - offset.y);
02320         break;
02321       }
02322 
02323       case CW_WIDGET_DESC_VEHICLE_COUNTS: {
02324         uint amounts[4];
02325         amounts[0] = c->group_all[VEH_TRAIN].num_vehicle;
02326         amounts[1] = c->group_all[VEH_ROAD].num_vehicle;
02327         amounts[2] = c->group_all[VEH_SHIP].num_vehicle;
02328         amounts[3] = c->group_all[VEH_AIRCRAFT].num_vehicle;
02329 
02330         int y = r.top;
02331         if (amounts[0] + amounts[1] + amounts[2] + amounts[3] == 0) {
02332           DrawString(r.left, r.right, y, STR_COMPANY_VIEW_VEHICLES_NONE);
02333         } else {
02334           assert_compile(lengthof(amounts) == lengthof(_company_view_vehicle_count_strings));
02335 
02336           for (uint i = 0; i < lengthof(amounts); i++) {
02337             if (amounts[i] != 0) {
02338               SetDParam(0, amounts[i]);
02339               DrawString(r.left, r.right, y, _company_view_vehicle_count_strings[i]);
02340               y += FONT_HEIGHT_NORMAL;
02341             }
02342           }
02343         }
02344         break;
02345       }
02346 
02347       case CW_WIDGET_DESC_INFRASTRUCTURE_COUNTS: {
02348         uint y = r.top;
02349 
02350         /* Collect rail and road counts. */
02351         uint rail_pices = c->infrastructure.signal;
02352         uint road_pieces = 0;
02353         for (uint i = 0; i < lengthof(c->infrastructure.rail); i++) rail_pices += c->infrastructure.rail[i];
02354         for (uint i = 0; i < lengthof(c->infrastructure.road); i++) road_pieces += c->infrastructure.road[i];
02355 
02356         if (rail_pices == 0 && road_pieces == 0 && c->infrastructure.water == 0 && c->infrastructure.station == 0 && c->infrastructure.airport == 0) {
02357           DrawString(r.left, r.right, y, STR_COMPANY_VIEW_INFRASTRUCTURE_NONE);
02358         } else {
02359           if (rail_pices != 0) {
02360             SetDParam(0, rail_pices);
02361             DrawString(r.left, r.right, y, STR_COMPANY_VIEW_INFRASTRUCTURE_RAIL);
02362             y += FONT_HEIGHT_NORMAL;
02363           }
02364           if (road_pieces != 0) {
02365             SetDParam(0, road_pieces);
02366             DrawString(r.left, r.right, y, STR_COMPANY_VIEW_INFRASTRUCTURE_ROAD);
02367             y += FONT_HEIGHT_NORMAL;
02368           }
02369           if (c->infrastructure.water != 0) {
02370             SetDParam(0, c->infrastructure.water);
02371             DrawString(r.left, r.right, y, STR_COMPANY_VIEW_INFRASTRUCTURE_WATER);
02372             y += FONT_HEIGHT_NORMAL;
02373           }
02374           if (c->infrastructure.station != 0) {
02375             SetDParam(0, c->infrastructure.station);
02376             DrawString(r.left, r.right, y, STR_COMPANY_VIEW_INFRASTRUCTURE_STATION);
02377             y += FONT_HEIGHT_NORMAL;
02378           }
02379           if (c->infrastructure.airport != 0) {
02380             SetDParam(0, c->infrastructure.airport);
02381             DrawString(r.left, r.right, y, STR_COMPANY_VIEW_INFRASTRUCTURE_AIRPORT);
02382           }
02383         }
02384 
02385         break;
02386       }
02387 
02388       case CW_WIDGET_DESC_OWNERS: {
02389         const Company *c2;
02390         uint y = r.top;
02391 
02392         FOR_ALL_COMPANIES(c2) {
02393           uint amt = GetAmountOwnedBy(c, c2->index);
02394           if (amt != 0) {
02395             SetDParam(0, amt * 25);
02396             SetDParam(1, c2->index);
02397 
02398             DrawString(r.left, r.right, y, STR_COMPANY_VIEW_SHARES_OWNED_BY);
02399             y += FONT_HEIGHT_NORMAL;
02400           }
02401         }
02402         break;
02403       }
02404 
02405 #ifdef ENABLE_NETWORK
02406       case CW_WIDGET_HAS_PASSWORD:
02407         if (_networking && NetworkCompanyIsPassworded(c->index)) {
02408           DrawSprite(SPR_LOCK, PAL_NONE, r.left, r.top);
02409         }
02410         break;
02411 #endif /* ENABLE_NETWORK */
02412     }
02413   }
02414 
02415   virtual void SetStringParameters(int widget) const
02416   {
02417     switch (widget) {
02418       case CW_WIDGET_CAPTION:
02419         SetDParam(0, (CompanyID)this->window_number);
02420         SetDParam(1, (CompanyID)this->window_number);
02421         break;
02422 
02423       case CW_WIDGET_DESC_INAUGURATION:
02424         SetDParam(0, Company::Get((CompanyID)this->window_number)->inaugurated_year);
02425         break;
02426 
02427       case CW_WIDGET_DESC_COMPANY_VALUE:
02428         SetDParam(0, CalculateCompanyValue(Company::Get((CompanyID)this->window_number)));
02429         break;
02430     }
02431   }
02432 
02433   virtual void OnClick(Point pt, int widget, int click_count)
02434   {
02435     switch (widget) {
02436       case CW_WIDGET_NEW_FACE: DoSelectCompanyManagerFace(this); break;
02437 
02438       case CW_WIDGET_COLOUR_SCHEME:
02439         if (BringWindowToFrontById(WC_COMPANY_COLOUR, this->window_number)) break;
02440         new SelectCompanyLiveryWindow(&_select_company_livery_desc, (CompanyID)this->window_number);
02441         break;
02442 
02443       case CW_WIDGET_PRESIDENT_NAME:
02444         this->query_widget = CW_WIDGET_PRESIDENT_NAME;
02445         SetDParam(0, this->window_number);
02446         ShowQueryString(STR_PRESIDENT_NAME, STR_COMPANY_VIEW_PRESIDENT_S_NAME_QUERY_CAPTION, MAX_LENGTH_PRESIDENT_NAME_CHARS, this, CS_ALPHANUMERAL, QSF_ENABLE_DEFAULT | QSF_LEN_IN_CHARS);
02447         break;
02448 
02449       case CW_WIDGET_COMPANY_NAME:
02450         this->query_widget = CW_WIDGET_COMPANY_NAME;
02451         SetDParam(0, this->window_number);
02452         ShowQueryString(STR_COMPANY_NAME, STR_COMPANY_VIEW_COMPANY_NAME_QUERY_CAPTION, MAX_LENGTH_COMPANY_NAME_CHARS, this, CS_ALPHANUMERAL, QSF_ENABLE_DEFAULT | QSF_LEN_IN_CHARS);
02453         break;
02454 
02455       case CW_WIDGET_VIEW_HQ: {
02456         TileIndex tile = Company::Get((CompanyID)this->window_number)->location_of_HQ;
02457         if (_ctrl_pressed) {
02458           ShowExtraViewPortWindow(tile);
02459         } else {
02460           ScrollMainWindowToTile(tile);
02461         }
02462         break;
02463       }
02464 
02465       case CW_WIDGET_BUILD_HQ:
02466         if ((byte)this->window_number != _local_company) return;
02467         if (this->IsWidgetLowered(CW_WIDGET_BUILD_HQ)) {
02468           ResetObjectToPlace();
02469           this->RaiseButtons();
02470           break;
02471         }
02472         SetObjectToPlaceWnd(SPR_CURSOR_HQ, PAL_NONE, HT_RECT, this);
02473         SetTileSelectSize(2, 2);
02474         this->LowerWidget(CW_WIDGET_BUILD_HQ);
02475         this->SetWidgetDirty(CW_WIDGET_BUILD_HQ);
02476         break;
02477 
02478       case CW_WIDGET_RELOCATE_HQ:
02479         if (this->IsWidgetLowered(CW_WIDGET_RELOCATE_HQ)) {
02480           ResetObjectToPlace();
02481           this->RaiseButtons();
02482           break;
02483         }
02484         SetObjectToPlaceWnd(SPR_CURSOR_HQ, PAL_NONE, HT_RECT, this);
02485         SetTileSelectSize(2, 2);
02486         this->LowerWidget(CW_WIDGET_RELOCATE_HQ);
02487         this->SetWidgetDirty(CW_WIDGET_RELOCATE_HQ);
02488         break;
02489 
02490       case CW_WIDGET_VIEW_INFRASTRUCTURE:
02491         ShowCompanyInfrastructure((CompanyID)this->window_number);
02492         break;
02493 
02494       case CW_WIDGET_BUY_SHARE:
02495         DoCommandP(0, this->window_number, 0, CMD_BUY_SHARE_IN_COMPANY | CMD_MSG(STR_ERROR_CAN_T_BUY_25_SHARE_IN_THIS));
02496         break;
02497 
02498       case CW_WIDGET_SELL_SHARE:
02499         DoCommandP(0, this->window_number, 0, CMD_SELL_SHARE_IN_COMPANY | CMD_MSG(STR_ERROR_CAN_T_SELL_25_SHARE_IN));
02500         break;
02501 
02502 #ifdef ENABLE_NETWORK
02503       case CW_WIDGET_COMPANY_PASSWORD:
02504         if (this->window_number == _local_company) ShowNetworkCompanyPasswordWindow(this);
02505         break;
02506 
02507       case CW_WIDGET_COMPANY_JOIN: {
02508         this->query_widget = CW_WIDGET_COMPANY_JOIN;
02509         CompanyID company = (CompanyID)this->window_number;
02510         if (_network_server) {
02511           NetworkServerDoMove(CLIENT_ID_SERVER, company);
02512           MarkWholeScreenDirty();
02513         } else if (NetworkCompanyIsPassworded(company)) {
02514           /* ask for the password */
02515           ShowQueryString(STR_EMPTY, STR_NETWORK_NEED_COMPANY_PASSWORD_CAPTION, NETWORK_PASSWORD_LENGTH, this, CS_ALPHANUMERAL, QSF_NONE);
02516         } else {
02517           /* just send the join command */
02518           NetworkClientRequestMove(company);
02519         }
02520         break;
02521       }
02522 #endif /* ENABLE_NETWORK */
02523     }
02524   }
02525 
02526   virtual void OnHundredthTick()
02527   {
02528     /* redraw the window every now and then */
02529     this->SetDirty();
02530   }
02531 
02532   virtual void OnPlaceObject(Point pt, TileIndex tile)
02533   {
02534     if (DoCommandP(tile, OBJECT_HQ, 0, CMD_BUILD_OBJECT | CMD_MSG(STR_ERROR_CAN_T_BUILD_COMPANY_HEADQUARTERS))) {
02535       ResetObjectToPlace();
02536       this->RaiseButtons();
02537     }
02538   }
02539 
02540   virtual void OnPlaceObjectAbort()
02541   {
02542     this->RaiseButtons();
02543   }
02544 
02545   virtual void OnQueryTextFinished(char *str)
02546   {
02547     if (str == NULL) return;
02548 
02549     switch (this->query_widget) {
02550       default: NOT_REACHED();
02551 
02552       case CW_WIDGET_PRESIDENT_NAME:
02553         DoCommandP(0, 0, 0, CMD_RENAME_PRESIDENT | CMD_MSG(STR_ERROR_CAN_T_CHANGE_PRESIDENT), NULL, str);
02554         break;
02555 
02556       case CW_WIDGET_COMPANY_NAME:
02557         DoCommandP(0, 0, 0, CMD_RENAME_COMPANY | CMD_MSG(STR_ERROR_CAN_T_CHANGE_COMPANY_NAME), NULL, str);
02558         break;
02559 
02560 #ifdef ENABLE_NETWORK
02561       case CW_WIDGET_COMPANY_JOIN:
02562         NetworkClientRequestMove((CompanyID)this->window_number, str);
02563         break;
02564 #endif /* ENABLE_NETWORK */
02565     }
02566   }
02567 };
02568 
02569 static const WindowDesc _company_desc(
02570   WDP_AUTO, 0, 0,
02571   WC_COMPANY, WC_NONE,
02572   WDF_UNCLICK_BUTTONS,
02573   _nested_company_widgets, lengthof(_nested_company_widgets)
02574 );
02575 
02580 void ShowCompany(CompanyID company)
02581 {
02582   if (!Company::IsValidID(company)) return;
02583 
02584   AllocateWindowDescFront<CompanyWindow>(&_company_desc, company);
02585 }
02586 
02591 void DirtyCompanyInfrastructureWindows(CompanyID company)
02592 {
02593   SetWindowDirty(WC_COMPANY, company);
02594   SetWindowDirty(WC_COMPANY_INFRASTRUCTURE, company);
02595 }
02596 
02597 
02599 enum BuyCompanyWidgets {
02600   BCW_CAPTION,
02601   BCW_FACE,
02602   BCW_QUESTION,
02603   BCW_NO,
02604   BCW_YES,
02605 };
02606 
02607 struct BuyCompanyWindow : Window {
02608   BuyCompanyWindow(const WindowDesc *desc, WindowNumber window_number) : Window()
02609   {
02610     this->InitNested(desc, window_number);
02611   }
02612 
02613   virtual void UpdateWidgetSize(int widget, Dimension *size, const Dimension &padding, Dimension *fill, Dimension *resize)
02614   {
02615     switch (widget) {
02616       case BCW_FACE:
02617         *size = GetSpriteSize(SPR_GRADIENT);
02618         break;
02619 
02620       case BCW_QUESTION:
02621         const Company *c = Company::Get((CompanyID)this->window_number);
02622         SetDParam(0, c->index);
02623         SetDParam(1, c->bankrupt_value);
02624         size->height = GetStringHeight(STR_BUY_COMPANY_MESSAGE, size->width);
02625         break;
02626     }
02627   }
02628 
02629   virtual void SetStringParameters(int widget) const
02630   {
02631     switch (widget) {
02632       case BCW_CAPTION:
02633         SetDParam(0, STR_COMPANY_NAME);
02634         SetDParam(1, Company::Get((CompanyID)this->window_number)->index);
02635         break;
02636     }
02637   }
02638 
02639   virtual void DrawWidget(const Rect &r, int widget) const
02640   {
02641     switch (widget) {
02642       case BCW_FACE: {
02643         const Company *c = Company::Get((CompanyID)this->window_number);
02644         DrawCompanyManagerFace(c->face, c->colour, r.left, r.top);
02645         break;
02646       }
02647 
02648       case BCW_QUESTION: {
02649         const Company *c = Company::Get((CompanyID)this->window_number);
02650         SetDParam(0, c->index);
02651         SetDParam(1, c->bankrupt_value);
02652         DrawStringMultiLine(r.left, r.right, r.top, r.bottom, STR_BUY_COMPANY_MESSAGE, TC_FROMSTRING, SA_CENTER);
02653         break;
02654       }
02655     }
02656   }
02657 
02658   virtual void OnClick(Point pt, int widget, int click_count)
02659   {
02660     switch (widget) {
02661       case BCW_NO:
02662         delete this;
02663         break;
02664 
02665       case BCW_YES:
02666         DoCommandP(0, this->window_number, 0, CMD_BUY_COMPANY | CMD_MSG(STR_ERROR_CAN_T_BUY_COMPANY));
02667         break;
02668     }
02669   }
02670 };
02671 
02672 static const NWidgetPart _nested_buy_company_widgets[] = {
02673   NWidget(NWID_HORIZONTAL),
02674     NWidget(WWT_CLOSEBOX, COLOUR_LIGHT_BLUE),
02675     NWidget(WWT_CAPTION, COLOUR_LIGHT_BLUE, BCW_CAPTION), SetDataTip(STR_ERROR_MESSAGE_CAPTION_OTHER_COMPANY, STR_TOOLTIP_WINDOW_TITLE_DRAG_THIS),
02676   EndContainer(),
02677   NWidget(WWT_PANEL, COLOUR_LIGHT_BLUE),
02678     NWidget(NWID_VERTICAL), SetPIP(8, 8, 8),
02679       NWidget(NWID_HORIZONTAL), SetPIP(8, 10, 8),
02680         NWidget(WWT_EMPTY, INVALID_COLOUR, BCW_FACE), SetFill(0, 1),
02681         NWidget(WWT_EMPTY, INVALID_COLOUR, BCW_QUESTION), SetMinimalSize(240, 0), SetFill(1, 1),
02682       EndContainer(),
02683       NWidget(NWID_HORIZONTAL, NC_EQUALSIZE), SetPIP(100, 10, 100),
02684         NWidget(WWT_TEXTBTN, COLOUR_LIGHT_BLUE, BCW_NO), SetMinimalSize(60, 12), SetDataTip(STR_QUIT_NO, STR_NULL), SetFill(1, 0),
02685         NWidget(WWT_TEXTBTN, COLOUR_LIGHT_BLUE, BCW_YES), SetMinimalSize(60, 12), SetDataTip(STR_QUIT_YES, STR_NULL), SetFill(1, 0),
02686       EndContainer(),
02687     EndContainer(),
02688   EndContainer(),
02689 };
02690 
02691 static const WindowDesc _buy_company_desc(
02692   WDP_AUTO, 0, 0,
02693   WC_BUY_COMPANY, WC_NONE,
02694   WDF_CONSTRUCTION,
02695   _nested_buy_company_widgets, lengthof(_nested_buy_company_widgets)
02696 );
02697 
02702 void ShowBuyCompanyDialog(CompanyID company)
02703 {
02704   AllocateWindowDescFront<BuyCompanyWindow>(&_buy_company_desc, company);
02705 }