source: tags/1.3.7/LocalPlayer/Term.pas

Last change on this file was 648, checked in by chronos, 6 weeks ago
  • Fixed: Tribes were not propertly detected in usr/share/c-evo directory.
File size: 279.3 KB
Line 
1{$INCLUDE Switches.inc}
2unit Term;
3
4interface
5
6uses
7{$IFDEF WINDOWS}
8 Windows,
9{$ENDIF}
10{$IFDEF UNIX}
11 LMessages, Messages,
12{$ENDIF}
13 Protocol, Tribes, PVSB, ClientTools, ScreenTools, BaseWin, Messg, ButtonBase,
14 LCLIntf, LCLType, SysUtils, Classes, DrawDlg, Types, Math,
15 DateUtils, Platform, ButtonB, ButtonC, EOTButton, Area, Help,
16 GraphicSet, MiniMap, IsoEngine, Wonders, TechTree, Enhance, Nego, CityType,
17 Diagram, CityScreen, Rates, Battle, NatStat, UnitStat, Draft, Select, MessgEx,
18 {$IFDEF DPI}Dpi.Graphics, Dpi.Controls, Dpi.Forms, Dpi.Menus, Dpi.ExtCtrls,
19 Dpi.PixelPointer, Dpi.Common, System.UITypes{$ELSE}
20 Graphics, Controls, Forms, Menus, ExtCtrls, PixelPointer{$ENDIF};
21
22const
23 WM_EOT = WM_USER;
24
25type
26 TPaintLocTempStyle = (pltsNormal, pltsBlink);
27
28 TSoundBlock = (sbStart, sbWonder, sbScience, sbContact, sbTurn);
29 TSoundBlocks = set of TSoundBlock;
30
31 { TMainScreen }
32
33 TMainScreen = class(TDrawDlg)
34 mBigTiles: TMenuItem;
35 mFillMap: TMenuItem;
36 mNextUnit: TMenuItem;
37 N13: TMenuItem;
38 mPrevUnit: TMenuItem;
39 Timer1: TTimer;
40 GamePopup: TPopupMenu;
41 UnitPopup: TPopupMenu;
42 mIrrigation: TMenuItem;
43 mCity: TMenuItem;
44 mRoad: TMenuItem;
45 mMine: TMenuItem;
46 mPollution: TMenuItem;
47 mHome: TMenuItem;
48 mStay: TMenuItem;
49 mDisband: TMenuItem;
50 mWait: TMenuItem;
51 mNoOrders: TMenuItem;
52 MTrans: TMenuItem;
53 UnitBtn: TButtonB;
54 mResign: TMenuItem;
55 mOptions: TMenuItem;
56 mEnMoves: TMenuItem;
57 mWaitTurn: TMenuItem;
58 mRep: TMenuItem;
59 mFort: TMenuItem;
60 mCentre: TMenuItem;
61 N1: TMenuItem;
62 mAirBase: TMenuItem;
63 N5: TMenuItem;
64 mCityTypes: TMenuItem;
65 mHelp: TMenuItem;
66 mCanal: TMenuItem;
67 mTest: TMenuItem;
68 mLocCodes: TMenuItem;
69 mLoad: TMenuItem;
70 StatPopup: TPopupMenu;
71 mCityStat: TMenuItem;
72 mUnitStat: TMenuItem;
73 mWonders: TMenuItem;
74 mScienceStat: TMenuItem;
75 mRailRoad: TMenuItem;
76 mClear: TMenuItem;
77 mFarm: TMenuItem;
78 mAfforest: TMenuItem;
79 mRep0: TMenuItem;
80 mRep1: TMenuItem;
81 mRep2: TMenuItem;
82 mRep3: TMenuItem;
83 mRep4: TMenuItem;
84 mRep5: TMenuItem;
85 mRep7: TMenuItem;
86 mRep8: TMenuItem;
87 mRep9: TMenuItem;
88 mRep15: TMenuItem;
89 mCancel: TMenuItem;
90 mLog: TMenuItem;
91 mEUnitStat: TMenuItem;
92 mRep10: TMenuItem;
93 mEnAttacks: TMenuItem;
94 mEnNoMoves: TMenuItem;
95 mDiagram: TMenuItem;
96 mJump: TMenuItem;
97 mNations: TMenuItem;
98 mManip: TMenuItem;
99 mManip0: TMenuItem;
100 mManip1: TMenuItem;
101 mManip2: TMenuItem;
102 mManip3: TMenuItem;
103 mManip4: TMenuItem;
104 mManip5: TMenuItem;
105 mEnhanceDef: TMenuItem;
106 mEnhance: TMenuItem;
107 mShips: TMenuItem;
108 mMacro: TMenuItem;
109 mRun: TMenuItem;
110 N10: TMenuItem;
111 mRepList: TMenuItem;
112 mRepScreens: TMenuItem;
113 mRep11: TMenuItem;
114 mNames: TMenuItem;
115 mManip6: TMenuItem;
116 mRep12: TMenuItem;
117 mRandomMap: TMenuItem;
118 mUnload: TMenuItem;
119 mRecover: TMenuItem;
120 MapBtn0: TButtonC;
121 MapBtn1: TButtonC;
122 MapBtn4: TButtonC;
123 MapBtn5: TButtonC;
124 EditPopup: TPopupMenu;
125 mCreateUnit: TMenuItem;
126 MapBtn6: TButtonC;
127 mDebugMap: TMenuItem;
128 mUtilize: TMenuItem;
129 mRep6: TMenuItem;
130 mEnemyMovement: TMenuItem;
131 mEnFastMoves: TMenuItem;
132 mOwnMovement: TMenuItem;
133 mSlowMoves: TMenuItem;
134 mFastMoves: TMenuItem;
135 mVeryFastMoves: TMenuItem;
136 mGoOn: TMenuItem;
137 mSound: TMenuItem;
138 mSoundOn: TMenuItem;
139 mSoundOnAlt: TMenuItem;
140 mSoundOff: TMenuItem;
141 N6: TMenuItem;
142 TerrainBtn: TButtonB;
143 TerrainPopup: TPopupMenu;
144 mScrolling: TMenuItem;
145 mScrollSlow: TMenuItem;
146 mScrollFast: TMenuItem;
147 mScrollOff: TMenuItem;
148 mPillage: TMenuItem;
149 mSelectTransport: TMenuItem;
150 mEmpire: TMenuItem;
151 N4: TMenuItem;
152 N2: TMenuItem;
153 mWebsite: TMenuItem;
154 N3: TMenuItem;
155 mRevolution: TMenuItem;
156 mRep13: TMenuItem;
157 UnitInfoBtn: TButtonB;
158 EOT: TEOTButton;
159 mAllyMovement: TMenuItem;
160 mAlSlowMoves: TMenuItem;
161 mAlFastMoves: TMenuItem;
162 N7: TMenuItem;
163 mEffectiveMovesOnly: TMenuItem;
164 N8: TMenuItem;
165 mAlEffectiveMovesOnly: TMenuItem;
166 mAlNoMoves: TMenuItem;
167 N9: TMenuItem;
168 mViewpoint: TMenuItem;
169 mTileSize: TMenuItem;
170 mNormalTiles: TMenuItem;
171 mSmallTiles: TMenuItem;
172 N11: TMenuItem;
173 MenuArea: TArea;
174 TreasuryArea: TArea;
175 ResearchArea: TArea;
176 ManagementArea: TArea;
177 mTechTree: TMenuItem;
178 MovieSpeed1Btn: TButtonB;
179 MovieSpeed2Btn: TButtonB;
180 MovieSpeed3Btn: TButtonB;
181 MovieSpeed4Btn: TButtonB;
182 N12: TMenuItem;
183 mRep14: TMenuItem;
184 procedure FormCreate(Sender: TObject);
185 procedure FormDestroy(Sender: TObject);
186 procedure FormMouseWheel(Sender: TObject; Shift: TShiftState;
187 WheelDelta: Integer; MousePos: TPoint; var Handled: Boolean);
188 procedure mAfforestClick(Sender: TObject);
189 procedure mAirBaseClick(Sender: TObject);
190 procedure mCanalClick(Sender: TObject);
191 procedure mCancelClick(Sender: TObject);
192 procedure mCentreClick(Sender: TObject);
193 procedure mcityClick(Sender: TObject);
194 procedure mCityStatClick(Sender: TObject);
195 procedure mCityTypesClick(Sender: TObject);
196 procedure mClearClick(Sender: TObject);
197 procedure mDiagramClick(Sender: TObject);
198 procedure mEmpireClick(Sender: TObject);
199 procedure mFillMapClick(Sender: TObject);
200 procedure mEnhanceClick(Sender: TObject);
201 procedure mEnhanceDefClick(Sender: TObject);
202 procedure mEUnitStatClick(Sender: TObject);
203 procedure mFarmClick(Sender: TObject);
204 procedure mfortClick(Sender: TObject);
205 procedure mGoOnClick(Sender: TObject);
206 procedure mHelpClick(Sender: TObject);
207 procedure mhomeClick(Sender: TObject);
208 procedure mirrigationClick(Sender: TObject);
209 procedure mirrigationDrawItem(Sender: TObject; ACanvas: TCanvas;
210 ARect: TRect; AState: TOwnerDrawState);
211 procedure mJumpClick(Sender: TObject);
212 procedure mLoadClick(Sender: TObject);
213 procedure mmineClick(Sender: TObject);
214 procedure mNationsClick(Sender: TObject);
215 procedure mNextUnitClick(Sender: TObject);
216 procedure mnoordersClick(Sender: TObject);
217 procedure mPillageClick(Sender: TObject);
218 procedure mpollutionClick(Sender: TObject);
219 procedure mPrevUnitClick(Sender: TObject);
220 procedure mRandomMapClick(Sender: TObject);
221 procedure mRecoverClick(Sender: TObject);
222 procedure mResignClick(Sender: TObject);
223 procedure mRevolutionClick(Sender: TObject);
224 procedure mroadClick(Sender: TObject);
225 procedure mRailRoadClick(Sender: TObject);
226 procedure mRunClick(Sender: TObject);
227 procedure mScienceStatClick(Sender: TObject);
228 procedure mSelectTransportClick(Sender: TObject);
229 procedure mShipsClick(Sender: TObject);
230 procedure mstayClick(Sender: TObject);
231 procedure mTechTreeClick(Sender: TObject);
232 procedure mtransClick(Sender: TObject);
233 procedure mUnitStatClick(Sender: TObject);
234 procedure mUnloadClick(Sender: TObject);
235 procedure mwaitClick(Sender: TObject);
236 procedure mWebsiteClick(Sender: TObject);
237 procedure mWondersClick(Sender: TObject);
238 procedure Timer1Timer(Sender: TObject);
239 procedure MapBoxMouseDown(Sender: TObject; Button: TMouseButton;
240 Shift: TShiftState; X, Y: Integer);
241 procedure EOTClick(Sender: TObject);
242 procedure PanelBoxMouseDown(Sender: TObject; Button: TMouseButton;
243 Shift: TShiftState; X, Y: Integer);
244 procedure FormKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);
245 procedure FormKeyUp(Sender: TObject; var Key: Word; Shift: TShiftState);
246 procedure mDisbandOrUtilizeClick(Sender: TObject);
247 procedure FormResize(Sender: TObject);
248 procedure PanelBtnClick(Sender: TObject);
249 procedure Toggle(Sender: TObject);
250 procedure PanelBoxMouseMove(Sender: TObject; Shift: TShiftState;
251 X, Y: Integer);
252 procedure PanelBoxMouseUp(Sender: TObject; Button: TMouseButton;
253 Shift: TShiftState; X, Y: Integer);
254 procedure MapBoxMouseMove(Sender: TObject; Shift: TShiftState;
255 X, Y: Integer);
256 procedure mShowClick(Sender: TObject);
257 procedure FormMouseDown(Sender: TObject; Button: TMouseButton;
258 Shift: TShiftState; X, Y: Integer);
259 procedure FormMouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer);
260 procedure FormMouseUp(Sender: TObject; Button: TMouseButton;
261 Shift: TShiftState; X, Y: Integer);
262 procedure FormPaint(Sender: TObject);
263 procedure FormClose(Sender: TObject; var Action: TCloseAction);
264 procedure FormCloseQuery(Sender: TObject; var CanClose: Boolean);
265 procedure FormShow(Sender: TObject);
266 procedure mRepClicked(Sender: TObject);
267 procedure mLogClick(Sender: TObject);
268 procedure Radio(Sender: TObject);
269 procedure mManipClick(Sender: TObject);
270 procedure mNamesClick(Sender: TObject);
271 procedure MapBtnClick(Sender: TObject);
272 procedure CreateUnitClick(Sender: TObject);
273 procedure mSoundOffClick(Sender: TObject);
274 procedure mSoundOnClick(Sender: TObject);
275 procedure mSoundOnAltClick(Sender: TObject);
276 procedure UnitInfoBtnClick(Sender: TObject);
277 procedure ViewpointClick(Sender: TObject);
278 procedure DebugMapClick(Sender: TObject);
279 procedure mSmallTilesClick(Sender: TObject);
280 procedure mNormalTilesClick(Sender: TObject);
281 procedure mBigTilesClick(Sender: TObject);
282 procedure GrWallBtnDownChanged(Sender: TObject);
283 procedure BareBtnDownChanged(Sender: TObject);
284 procedure MovieSpeedBtnClick(Sender: TObject);
285 private
286 xw: Integer; // Base map x
287 yw: Integer; // Base map y
288 xwd: Integer;
289 ywd: Integer;
290 xwMini: Integer;
291 ywMini: Integer;
292 xMidPanel: Integer;
293 xRightPanel: Integer;
294 xTroop: Integer;
295 xTerrain: Integer;
296 xMini: Integer;
297 yMini: Integer;
298 ywmax: Integer;
299 ywcenter: Integer;
300 TroopLoc: Integer;
301 TrCnt: Integer;
302 TrRow: Integer;
303 TrPitch: Integer;
304 MapWidth: Integer;
305 MapOffset: Integer;
306 MapHeight: Integer;
307 BlinkTime: Integer;
308 BrushLoc: Integer;
309 EditLoc: Integer;
310 xMouse: Integer;
311 yMouse: Integer;
312 BrushType: Cardinal;
313 trix: array [0 .. 63] of Integer;
314 AILogo: array [0 .. nPl - 1] of TBitmap;
315 MiniMap: TMiniMap;
316 Panel: TBitmap;
317 TopBar: TBitmap;
318 VerticalScrollBar: TPVScrollBar;
319 Closable: Boolean;
320 RepaintOnResize: Boolean;
321 Tracking: Boolean;
322 TurnComplete: Boolean;
323 Edited: Boolean;
324 GoOnPhase: Boolean;
325 HaveStrategyAdvice: Boolean;
326 FirstMovieTurn: Boolean;
327 FormFullScreen: Boolean;
328 FormRestoredSize: TRect;
329 FormWindowState: TWindowState;
330 MainMap: TIsoMap;
331 NoMap: TIsoMap;
332 NoMapPanel: TIsoMap;
333 LastResizeWidth: Integer;
334 LastResizeHeight: Integer;
335 LastWindowState: TWindowState;
336 KeyboardDisabled: Boolean;
337 // Forms
338 FWondersDlg: TWondersDlg;
339 FTechTreeDlg: TTechTreeDlg;
340 FEnhanceDlg: TEnhanceDlg;
341 FNegoDlg: TNegoDlg;
342 FCityTypeDlg: TCityTypeDlg;
343 FDiaDlg: TDiaDlg;
344 FCityDlg: TCityDlg;
345 FRatesDlg: TRatesDlg;
346 FBattleDlg: TBattleDlg;
347 FNatStatDlg: TNatStatDlg;
348 FUnitStatDlg: TUnitStatDlg;
349 FDraftDlg: TDraftDlg;
350 FModalSelectDlg: TModalSelectDlg;
351 FListDlg: TListDlg;
352 FMessgExDlg: TMessgExDlg;
353 FHelpDlg: THelpDlg;
354 procedure ResizeControls;
355 procedure ArrangeDialogs;
356 procedure ArrangeDialog(Form: TBufferedDrawDlg);
357 function ChooseUnusedTribe: Integer;
358 function DoJob(j0: Integer): Integer;
359 function GetBattleDlg: TBattleDlg;
360 function GetCityDlg: TCityDlg;
361 function GetCityTypeDlg: TCityTypeDlg;
362 function GetDiaDlg: TDiaDlg;
363 function GetDraftDlg: TDraftDlg;
364 function GetEnhanceDlg: TEnhanceDlg;
365 function GetHelpDlg: THelpDlg;
366 function GetListDlg: TListDlg;
367 function GetMessgExDlg: TMessgExDlg;
368 function GetModalSelectDlg: TModalSelectDlg;
369 function GetNatStatDlg: TNatStatDlg;
370 function GetNegoDlg: TNegoDlg;
371 function GetRatesDlg: TRatesDlg;
372 function GetTechTreeDlg: TTechTreeDlg;
373 procedure GetTribeList;
374 function GetUnitStatDlg: TUnitStatDlg;
375 function GetWondersDlg: TWondersDlg;
376 procedure InitModule;
377 procedure DoneModule;
378 procedure InitTurn(NewPlayer: Integer);
379 procedure SaveMenuItemsState;
380 procedure ScrollBarUpdate(Sender: TObject);
381 procedure ArrangeMidPanel;
382 procedure MainOffscreenPaint;
383 procedure MiniMapPaint;
384 procedure PaintAll;
385 procedure PaintAllMaps;
386 procedure CopyMiniToPanel;
387 procedure PanelPaint;
388 procedure FocusNextUnit(Dir: Integer = 1);
389 procedure NextUnit(NearLoc: Integer; AutoTurn: Boolean);
390 procedure Scroll(dx, dy: Integer);
391 procedure SetMapPos(Loc: Integer; MapPos: TPoint);
392 procedure Centre(Loc: Integer);
393 procedure SetTroopLoc(Loc: Integer);
394 procedure ProcessRect(x0, y0, nx, ny, Options: Integer);
395 procedure PaintLoc(Loc: Integer; Radius: Integer = 0);
396 procedure PaintLoc_BeforeMove(FromLoc: Integer);
397 procedure PaintLocTemp(Loc: Integer; Style: TPaintLocTempStyle = pltsNormal);
398 procedure PaintBufferToScreen(xMap, yMap, Width, Height: Integer);
399 procedure PaintDestination;
400 procedure SetUnFocus(uix: Integer);
401 function MoveUnit(dx, dy: Integer; Options: Integer = 0): Integer;
402 procedure MoveToLoc(Loc: Integer; CheckSuicide: Boolean);
403 procedure MoveOnScreen(ShowMove: TShowMove; Step0, Step1, nStep: Integer;
404 Restore: Boolean = True);
405 procedure FocusOnLoc(Loc: Integer; Options: Integer = 0);
406 function EndTurn(WasSkipped: Boolean = False): Boolean;
407 procedure EndNego;
408 function IsPanelPixel(X, Y: Integer): Boolean;
409 procedure InitPopup(Popup: TPopupMenu);
410 procedure SetMapOptions;
411 procedure CheckMovieSpeedBtnState;
412 procedure CheckTerrainBtnVisible;
413 procedure RememberPeaceViolation;
414 procedure SetDebugMap(P: Integer);
415 procedure SetViewpoint(P: Integer);
416 function LocationOfScreenPixel(X, Y: Integer): Integer;
417 function GetCenterLoc: Integer;
418 procedure SetTileSizeCenter(TileSize: TTileSize);
419 procedure SetTileSize(TileSize: TTileSize; Loc: Integer; MapPos: TPoint);
420 procedure RectInvalidate(Left, Top, Rigth, Bottom: Integer);
421 procedure ShowEnemyShipChange(ShowShipChange: TShowShipChange);
422 procedure SmartRectInvalidate(Left, Top, Rigth, Bottom: Integer);
423 procedure LoadSettings;
424 procedure SaveSettings;
425 procedure OnScroll(var Msg: TMessage); message WM_VSCROLL;
426 procedure OnEOT(var Msg: TMessage); message WM_EOT;
427 procedure SoundPreload(Check: TSoundBlocks);
428 procedure UpdateKeyShortcuts;
429 procedure SetFullScreen(Active: Boolean);
430 procedure PaintZoomedTile(Dst: TBitmap; X, Y, Loc: Integer);
431 procedure UpdateInterface;
432 public
433 UsedOffscreenWidth: Integer;
434 UsedOffscreenHeight: Integer;
435 Offscreen: TBitmap;
436 OffscreenUser: TForm;
437 procedure Client(Command, NewPlayer: Integer; var Data);
438 procedure SetAIName(P: Integer; Name: string);
439 function ZoomToCity(Loc: Integer; NextUnitOnClose: Boolean = False;
440 ShowEvent: Integer = 0): Boolean;
441 procedure CityClosed(Activateuix: Integer; StepFocus: Boolean = False;
442 SelectFocus: Boolean = False);
443 function DipCall(Command: Integer): Integer;
444 function OfferCall(var Offer: TOffer): Integer;
445 procedure UpdateViews(UpdateCityScreen: Boolean = False);
446 function ContactRefused(P: Integer; Item: String): Boolean;
447 procedure RepaintAll;
448 // Forms
449 property WondersDlg: TWondersDlg read GetWondersDlg;
450 property TechTreeDlg: TTechTreeDlg read GetTechTreeDlg;
451 property EnhanceDlg: TEnhanceDlg read GetEnhanceDlg;
452 property NegoDlg: TNegoDlg read GetNegoDlg;
453 property CityTypeDlg: TCityTypeDlg read GetCityTypeDlg;
454 property DiaDlg: TDiaDlg read GetDiaDlg;
455 property CityDlg: TCityDlg read GetCityDlg;
456 property RatesDlg: TRatesDlg read GetRatesDlg;
457 property BattleDlg: TBattleDlg read GetBattleDlg;
458 property NatStatDlg: TNatStatDlg read GetNatStatDlg;
459 property UnitStatDlg: TUnitStatDlg read GetUnitStatDlg;
460 property DraftDlg: TDraftDlg read GetDraftDlg;
461 property ModalSelectDlg: TModalSelectDlg read GetModalSelectDlg;
462 property ListDlg: TListDlg read GetListDlg;
463 property MessgExDlg: TMessgExDlg read GetMessgExDlg;
464 property HelpDlg: THelpDlg read GetHelpDlg;
465 end;
466
467var
468 MainScreen: TMainScreen;
469
470type
471
472 { TTribeInfo }
473
474 TTribeInfo = record
475 trix: Integer;
476 FileName: ShortString;
477 function GetCommandDataSize: Byte;
478 end;
479
480 { TCityNameInfo }
481
482 TCityNameInfo = record
483 ID: Integer;
484 NewName: ShortString;
485 function GetCommandDataSize: Byte;
486 end;
487
488 { TModelNameInfo }
489
490 TModelNameInfo = record
491 mix: Integer;
492 NewName: ShortString;
493 function GetCommandDataSize: Byte;
494 end;
495
496 TFormAction = (faClose, faEnable, faDisable, faUpdate, faSmartUpdateContent);
497
498const
499 crImpDrag = 2;
500 crFlatHand = 3;
501
502 xxu = 32;
503 yyu = 24; // half of unit slot size x/y
504 yyu_anchor = 32;
505 xxc = 32;
506 yyc = 16; // 1/2 of city slot size in x, 1/2 of ground tile size in y (=1/3 of slot)
507
508 // layout
509 TopBarHeight = 41;
510 PanelHeight = 168;
511 MidPanelHeight = 120;
512 // TopBarHeight+MidPanelHeight should be same as BaseWin.yUnused
513 MapCenterUp = (MidPanelHeight - TopBarHeight) div 2;
514
515 nCityType = 4;
516
517 { client exclusive commands: }
518 cSetTribe = $9000;
519 cSetNewModelPicture = $9100;
520 cSetModelName = $9110;
521 cSetModelPicture = $9120;
522 cSetSlaveIndex = $9131;
523 cSetCityName = $9200;
524
525 // city status flags
526 csTypeMask = $0007;
527 csToldDelay = $0008;
528 csResourceWeightsMask = $00F0;
529 csToldBombard = $0100;
530
531 { unit status flags }
532 usStay = $01;
533 usWaiting = $02;
534 usGoto = $04;
535 usEnhance = $08;
536 usRecover = $10;
537 usToldNoReturn = $100;
538 usPersistent = usStay or usGoto or usEnhance or usRecover or
539 Integer($FFFF0000);
540
541 { model status flags }
542 msObsolete = $1;
543 msAllowConscripts = $2;
544
545 { additional city happened flags }
546 chTypeDel = $8000;
547 chAllImpsMade = $4000;
548
549 adNone = $801;
550 adFar = $802;
551 adNexus = $803;
552
553 SpecialModelPictureCode: array [0 .. nSpecialModel - 1] of Integer = (10,
554 11, 40, 41, 21, 30, { 50,51, } 64, 74, { 71, } 73);
555
556 pixSlaves = 0;
557 pixNoSlaves = 1; // index of slaves in StdUnits
558
559 // icons.bmp properties
560 xSizeSmall = 36;
561 ySizeSmall = 20;
562 SystemIconLines = 2;
563 // lines of system icons in icons.bmp before improvements
564
565 nCityEventPriority = 16;
566 CityEventPriority: array [0 .. nCityEventPriority - 1] of Integer =
567 (chDisorder, chImprovementLost, chUnitLost, chAllImpsMade, chProduction,
568 chOldWonder, chNoSettlerProd, chPopDecrease, chProductionSabotaged,
569 chNoGrowthWarning, chPollution, chTypeDel, chFounded, chSiege,
570 chAfterCapture, chPopIncrease);
571
572 CityEventSoundItem: array [0 .. 15] of string = ('CITY_DISORDER', '',
573 'CITY_POPPLUS', 'CITY_POPMINUS', 'CITY_UNITLOST', 'CITY_IMPLOST',
574 'CITY_SABOTAGE', 'CITY_GROWTHNEEDSIMP', 'CITY_POLLUTION', 'CITY_SIEGE',
575 'CITY_WONDEREX', 'CITY_EMDELAY', 'CITY_FOUNDED', 'CITY_FOUNDED', '',
576 'CITY_INVALIDTYPE');
577
578type
579 TPersistentData = record
580 FarTech: Integer;
581 ToldAge: Integer;
582 ToldModels: Integer;
583 ToldAlive: Integer;
584 ToldContact: Integer;
585 ToldOwnCredibility: Integer;
586 ColdWarStart: Integer;
587 PeaceEvaHappened: Integer;
588 EnhancementJobs: TEnhancementJobs;
589 ImpOrder: array [0 .. nCityType - 1] of TImpOrder;
590 ToldWonders: array [0 .. nWonder - 1] of TWonderInfo;
591 ToldTech: array [0 .. nAdv - 1] of ShortInt;
592 end;
593
594 TDipMem = record
595 pContact: Integer;
596 SentCommand: Integer;
597 FormerTreaty: Integer;
598 SentOffer: TOffer;
599 DeliveredPrices: TPriceSet;
600 ReceivedPrices: TPriceSet;
601 end;
602
603 TCurrentMoveInfo = record
604 AfterMovePaintRadius: Integer;
605 AfterAttackExpeller: Integer;
606 DoShow: Boolean;
607 IsAlly: Boolean;
608 end;
609
610var
611 MyData: ^TPersistentData;
612 AdvIcon: array [0 .. nAdv - 1] of Integer;
613 { icons displayed with the technologies }
614 GameMode: Integer;
615 ClientMode: Integer;
616 Age: Integer;
617 UnFocus: Integer;
618 OptionChecked: TSaveOptions;
619 MapOptionChecked: TMapOptions;
620 nLostArmy: Integer;
621 ScienceSum: Integer;
622 TaxSum: Integer;
623 SoundPreloadDone: TSoundBlocks;
624 MarkCityLoc: Integer;
625 MovieSpeed: Integer;
626 CityRepMask: Cardinal;
627 ReceivedOffer: TOffer;
628 Buffer: TBitmap;
629 SmallImp: TBitmap;
630 BlinkON: Boolean;
631 DestinationMarkON: Boolean;
632 StartRunning: Boolean;
633 Supervising: Boolean;
634 UnusedTribeFiles: TStringList;
635 TribeNames: TStringList;
636 TribeOriginal: array [0 .. nPl - 1] of Boolean;
637 LostArmy: array [0 .. nPl * nMmax - 1] of Integer;
638 DipMem: array [0 .. nPl - 1] of TDipMem;
639
640function CityEventName(I: Integer): string;
641function RoughCredibility(Credibility: Integer): Integer;
642
643function InitEnemyModel(emix: Integer): Boolean;
644procedure InitAllEnemyModels;
645procedure InitMyModel(Mix: Integer; Final: Boolean);
646
647procedure ImpImage(ca: TCanvas; X, Y, iix: Integer; Government: Integer = -1;
648 IsControl: Boolean = False);
649procedure HelpOnTerrain(Loc: Integer; NewMode: TWindowMode);
650function AlignUp(Value, Alignment: Integer): Integer;
651
652
653implementation
654
655uses
656 Directories, Log, Sound, Registry, Global, KeyBindings, CmdList;
657
658{$R *.lfm}
659
660const
661 lxmax_xxx = 130;
662 LeftPanelWidth = 70;
663 Overlap = PanelHeight - MidPanelHeight;
664 yTroop = PanelHeight - 83;
665 xPalace = 66;
666 yPalace = 24; // 120;
667{ xAdvisor = 108;
668 yAdvisor = 48;}
669 xUnitText = 80;
670 BlinkOnTime = 12;
671 BlinkOffTime = 6;
672 MoveTime = 300; // {time for moving a unit in ms}
673 WaitAfterShowMove = 32;
674 FastScrolling = False; // causes problems with overlapping windows
675
676 BrushTypes: array[0..25] of Cardinal = (fPrefStartPos,
677 fStartPos, fShore, fGrass, fTundra, fPrairie, fDesert, fSwamp, fForest,
678 fHills, fMountains, fArctic, fDeadLands, fDeadLands or fCobalt,
679 fDeadLands or fUranium, fDeadLands or fMercury, fRiver, fRoad, fRR, fCanal,
680 tiIrrigation, tiFarm, tiMine, fPoll, tiFort, tiBase);
681
682 // MoveUnit options:
683 muAutoNoWait = $0001;
684 muAutoNext = $0002;
685 muNoSuicideCheck = $0004;
686
687 // ProcessRect options:
688 prPaint = $0001;
689 prAutoBounds = $0002;
690 prInvalidate = $0004;
691
692 // FocusOnLoc options:
693 flRepaintPanel = $0001;
694 flImmUpdate = $0002;
695
696var
697 Jump: array [0 .. nPl - 1] of Integer;
698 pTurn: Integer;
699 pLogo: Integer;
700 UnStartLoc: Integer;
701 ToldSlavery: Integer;
702 SmallScreen: Boolean;
703 GameOK: Boolean;
704 MapValid: Boolean;
705 Skipped: Boolean;
706 Idle: Boolean;
707
708 SaveOption: array of Integer;
709 CurrentMoveInfo: TCurrentMoveInfo;
710
711function CityEventName(I: Integer): string;
712begin
713 if I = 14 then // chAllImpsMade
714 if not Phrases2FallenBackToEnglish then
715 Result := Phrases2.Lookup('CITYEVENT_ALLIMPSMADE')
716 else
717 Result := Phrases.Lookup('CITYEVENTS', 1)
718 else
719 Result := Phrases.Lookup('CITYEVENTS', I);
720end;
721
722procedure InitSmallImp;
723begin
724 SmallImp.SetSize(BigImp.Width * xSizeSmall div xSizeBig,
725 BigImp.Height * ySizeSmall div ySizeBig);
726 UnshareBitmap(SmallImp);
727 SmallImp.Canvas.StretchDraw(Rect(0, 0, SmallImp.Width, SmallImp.Height), BigImp);
728end;
729
730procedure ImpImage(ca: TCanvas; X, Y, iix: Integer; Government: Integer;
731 IsControl: Boolean);
732begin
733 if Government < 0 then
734 Government := MyRO.Government;
735 if (iix = imPalace) and (Government <> gAnarchy) then
736 iix := Government - 8;
737 FrameImage(ca, BigImp, X, Y, xSizeBig, ySizeBig, (iix + SystemIconLines * 7)
738 mod 7 * xSizeBig, (iix + SystemIconLines * 7) div 7 * ySizeBig, IsControl);
739end;
740
741procedure HelpOnTerrain(Loc: Integer; NewMode: TWindowMode);
742begin
743 if MyMap[Loc] and fDeadLands <> 0 then
744 MainScreen.HelpDlg.ShowNewContent(NewMode, hkTer, 3 * 12)
745 else if (MyMap[Loc] and fTerrain = fForest) and IsJungle(Loc div G.lx) then
746 MainScreen.HelpDlg.ShowNewContent(NewMode, hkTer,
747 fJungle + (MyMap[Loc] shr 5 and 3) * 12)
748 else
749 MainScreen.HelpDlg.ShowNewContent(NewMode, hkTer, MyMap[Loc] and fTerrain +
750 (MyMap[Loc] shr 5 and 3) * 12);
751end;
752
753function AlignUp(Value, Alignment: Integer): Integer;
754begin
755 Result := Value or (Alignment - 1);
756end;
757
758{ *** tribe management procedures *** }
759
760function RoughCredibility(Credibility: Integer): Integer;
761begin
762 case Credibility of
763 0 .. 69:
764 Result := 0;
765 70 .. 89:
766 Result := 1;
767 90 .. 99:
768 Result := 2;
769 100:
770 Result := 3;
771 end;
772end;
773
774procedure ChooseModelPicture(P, mix, Code, Hash, Turn: Integer;
775 ForceNew, Final: Boolean);
776var
777 I: Integer;
778 Picture: TModelPictureInfo;
779 IsNew: Boolean;
780begin
781 Picture.trix := P;
782 Picture.mix := mix;
783 if Code = 74 then
784 begin // use correct pictures for slaves
785 if Tribe[P].mixSlaves < 0 then
786 if not TribeOriginal[P] then
787 Tribe[P].mixSlaves := mix
788 else
789 begin
790 I := mix + P shl 16;
791 Server(cSetSlaveIndex, 0, 0, I);
792 end;
793 if ToldSlavery = 1 then
794 Picture.pix := pixSlaves
795 else
796 Picture.pix := pixNoSlaves;
797 Picture.Hash := 0;
798 Picture.GrName := 'StdUnits.png';
799 IsNew := True;
800 end
801 else
802 begin
803 Picture.Hash := Hash;
804 IsNew := Tribe[P].ChooseModelPicture(Picture, Code, Turn, ForceNew);
805 end;
806 if Final then
807 if not TribeOriginal[P] then
808 Tribe[P].SetModelPicture(Picture, IsNew)
809 else if IsNew then
810 Server(CommandWithData(cSetNewModelPicture, Picture.GetCommandDataSize),
811 0, 0, Picture)
812 else
813 Server(CommandWithData(cSetModelPicture, Picture.GetCommandDataSize),
814 0, 0, Picture)
815 else
816 with Tribe[P].ModelPicture[mix] do
817 begin
818 HGr := LoadGraphicSet(Picture.GrName);
819 pix := Picture.pix;
820 end;
821end;
822
823function InitEnemyModel(emix: Integer): Boolean;
824begin
825 if GameMode = cMovie then
826 begin
827 Result := False;
828 Exit;
829 end;
830 with MyRO.EnemyModel[emix] do
831 ChooseModelPicture(Owner, mix, ModelCode(MyRO.EnemyModel[emix]),
832 ModelHash(MyRO.EnemyModel[emix]), MyRO.Turn, False, True);
833 Result := True;
834end;
835
836procedure InitAllEnemyModels;
837var
838 emix: Integer;
839begin
840 for emix := 0 to MyRO.nEnemyModel - 1 do
841 with MyRO.EnemyModel[emix] do
842 if not Assigned(Tribe[Owner].ModelPicture[mix].HGr) then
843 InitEnemyModel(emix);
844end;
845
846procedure InitMyModel(Mix: Integer; Final: Boolean);
847var
848 mi: TModelInfo;
849begin
850 if (GameMode = cMovie) and (MyModel[mix].Kind < $08) then
851 Exit;
852 // don't exit for special units because cSetModelPicture comes after TellNewModels
853 MakeModelInfo(Me, Mix, MyModel[mix], mi);
854 ChooseModelPicture(Me, Mix, ModelCode(mi), ModelHash(mi), MyRO.Turn,
855 False, Final);
856end;
857
858function AttackSound(Code: Integer): string;
859begin
860 Result := 'ATTACK_' + Char(48 + Code div 100 mod 10) +
861 Char(48 + Code div 10 mod 10) + Char(48 + Code mod 10);
862end;
863
864procedure CheckToldNoReturn(uix: Integer);
865// check whether aircraft survived low-fuel warning
866begin
867 Assert(not Supervising);
868 with MyUn[uix] do
869 if (Status and usToldNoReturn <> 0) and
870 ((MyMap[Loc] and fCity <> 0) or (MyMap[Loc] and fTerImp = tiBase) or
871 (Master >= 0)) then
872 Status := Status and not usToldNoReturn;
873end;
874
875function CreateTribe(P: Integer; FileName: string; Original: Boolean): Boolean;
876begin
877 FileName := LocalizedFilePath('Tribes' + DirectorySeparator + FileName +
878 CevoTribeExt);
879 if not FileExists(FileName) then
880 begin
881 Result := False;
882 Exit;
883 end;
884
885 TribeOriginal[P] := Original;
886 Tribe[P] := TTribe.Create(FileName);
887 with Tribe[P] do
888 begin
889 if (GameMode = cNewGame) or not Original then
890 begin
891 Term.ChooseModelPicture(P, 0, 010, 1, 0, True, True);
892 Term.ChooseModelPicture(P, 1, 040, 1, 0, True, True);
893 Term.ChooseModelPicture(P, 2, 041, 1, 0, True, True);
894 Term.ChooseModelPicture(P, -1, 017, 1, 0, True, True);
895 end;
896 DipMem[P].pContact := -1;
897 end;
898 Result := True;
899end;
900
901procedure TellNewContacts;
902var
903 p1: Integer;
904begin
905 if not Supervising then
906 for p1 := 0 to nPl - 1 do
907 if (p1 <> Me) and (1 shl p1 and MyData.ToldContact = 0) and
908 (1 shl p1 and MyRO.Alive <> 0) and (MyRO.Treaty[p1] > trNoContact) then
909 begin
910 TribeMessage(p1, Tribe[p1].TPhrase('FRNEWNATION'), '');
911 MyData.ToldContact := MyData.ToldContact or (1 shl p1);
912 end;
913end;
914
915procedure TellNewModels;
916var
917 mix: Integer;
918 ModelNameInfo: TModelNameInfo;
919begin
920 if Supervising then
921 Exit;
922 with Tribe[Me] do
923 while MyData.ToldModels < MyRO.nModel do
924 begin { new Unit class available }
925 if Assigned(ModelPicture[MyData.ToldModels].HGr) and
926 (MyModel[MyData.ToldModels].Kind <> mkSelfDeveloped) then
927 begin // save picture of DevModel
928 ModelPicture[MyData.ToldModels + 1] := ModelPicture[MyData.ToldModels];
929 ModelName[MyData.ToldModels + 1] := ModelName[MyData.ToldModels];
930 ModelPicture[MyData.ToldModels].HGr := nil;
931 end;
932 if not Assigned(ModelPicture[MyData.ToldModels].HGr) then
933 InitMyModel(MyData.ToldModels, True);
934 { only run if no researched model }
935 with MainScreen.MessgExDlg do
936 begin
937 { MakeModelInfo(me,MyData.ToldModels,MyModel[MyData.ToldModels],mi);
938 if mi.Attack=0 then OpenSound:='MSG_DEFAULT'
939 else OpenSound:=AttackSound(ModelCode(mi)); }
940 if MyModel[MyData.ToldModels].Kind = mkSelfDeveloped then
941 OpenSound := 'NEWMODEL_' + Char(48 + Age);
942 MessgText := Phrases.Lookup('MODELAVAILABLE');
943 if GameMode = cMovie then
944 begin
945 Kind := mkOkHelp; // doesn't matter
946 MessgText := MessgText + '\' + ModelName[MyData.ToldModels];
947 end
948 else
949 begin
950 Kind := mkModel;
951 EInput.Text := ModelName[MyData.ToldModels];
952 end;
953 IconKind := mikModel;
954 IconIndex := MyData.ToldModels;
955 ShowModal;
956 if (EInput.Text <> '') and (EInput.Text <> ModelName[MyData.ToldModels])
957 then
958 begin // user renamed model
959 ModelNameInfo.mix := MyData.ToldModels;
960 ModelNameInfo.NewName := EInput.Text;
961 if ModelNameInfo.GetCommandDataSize > CommandDataMaxSize then
962 Delete(ModelNameInfo.NewName, Length(ModelNameInfo.NewName) -
963 (ModelNameInfo.GetCommandDataSize - 1 - CommandDataMaxSize), MaxInt);
964 Server(CommandWithData(cSetModelName, ModelNameInfo.GetCommandDataSize),
965 Me, 0, ModelNameInfo);
966 end;
967 end;
968 if MyModel[MyData.ToldModels].Kind = mkSettler then
969 begin // engineers make settlers obsolete
970 for mix := 0 to MyData.ToldModels - 1 do
971 if MyModel[mix].Kind = mkSettler then
972 MyModel[mix].Status := MyModel[mix].Status or msObsolete;
973 end;
974 Inc(MyData.ToldModels);
975 end;
976end;
977
978{ TTribeInfo }
979
980function TTribeInfo.GetCommandDataSize: Byte;
981begin
982 Result := SizeOf(trix) + 1 + Length(FileName)
983end;
984
985{ TModelNameInfo }
986
987function TModelNameInfo.GetCommandDataSize: Byte;
988begin
989 Result := SizeOf(mix) + 1 + Length(NewName);
990end;
991
992{ TCityNameInfo }
993
994function TCityNameInfo.GetCommandDataSize: Byte;
995begin
996 Result := SizeOf(ID) + 1 + Length(NewName);
997end;
998
999procedure TMainScreen.PaintZoomedTile(Dst: TBitmap; X, Y, Loc: Integer);
1000
1001 procedure TerrainSprite(xDst, yDst, xSrc, ySrc: Integer);
1002 begin
1003 with NoMapPanel do
1004 Sprite(Dst, HGrTerrain, X + xDst, Y + yDst, xxt * 2, yyt * 3,
1005 1 + xSrc * (xxt * 2 + 1), 1 + ySrc * (yyt * 3 + 1));
1006 end;
1007
1008 procedure TerrainSprite4(xSrc, ySrc: Integer);
1009 begin
1010 with NoMapPanel do begin
1011 Sprite(Dst, HGrTerrain, X + xxt, Y + yyt + 2, xxt * 2, yyt * 2 - 2,
1012 1 + xSrc * (xxt * 2 + 1), 3 + yyt + ySrc * (yyt * 3 + 1));
1013 Sprite(Dst, HGrTerrain, X + 4, Y + 2 * yyt, xxt * 2 - 4, yyt * 2,
1014 5 + xSrc * (xxt * 2 + 1), 1 + yyt + ySrc * (yyt * 3 + 1));
1015 Sprite(Dst, HGrTerrain, X + xxt * 2, Y + 2 * yyt, xxt * 2 - 4, yyt * 2,
1016 1 + xSrc * (xxt * 2 + 1), 1 + yyt + ySrc * (yyt * 3 + 1));
1017 Sprite(Dst, HGrTerrain, X + xxt, Y + yyt * 3, xxt * 2, yyt * 2 - 2,
1018 1 + xSrc * (xxt * 2 + 1), 1 + yyt + ySrc * (yyt * 3 + 1));
1019 end;
1020 end;
1021
1022var
1023 cix, ySrc, Tile: Integer;
1024begin
1025 with NoMapPanel do begin
1026 Tile := MyMap[Loc];
1027 if Tile and fCity <> 0 then
1028 begin
1029 if MyRO.Tech[adRailroad] >= tsApplicable then
1030 Tile := Tile or fRR
1031 else
1032 Tile := Tile or fRoad;
1033 if Tile and fOwned <> 0 then
1034 begin
1035 cix := MyRO.nCity - 1;
1036 while (cix >= 0) and (MyCity[cix].Loc <> Loc) do
1037 Dec(cix);
1038 Assert(cix >= 0);
1039 if MyCity[cix].Built[imSupermarket] > 0 then
1040 Tile := Tile or tiFarm
1041 else
1042 Tile := Tile or tiIrrigation;
1043 end
1044 else Tile := Tile or tiIrrigation;
1045 end;
1046
1047 if Tile and fTerrain >= fForest then
1048 TerrainSprite4(2, 2)
1049 else
1050 TerrainSprite4(Tile and fTerrain, 0);
1051 if Tile and fTerrain >= fForest then
1052 begin
1053 if (Tile and fTerrain = fForest) and IsJungle(Loc div G.lx) then
1054 ySrc := 18
1055 else
1056 ySrc := 3 + 2 * (Tile and fTerrain - fForest);
1057 TerrainSprite(xxt, 0, 6, ySrc);
1058 TerrainSprite(0, yyt, 3, ySrc);
1059 TerrainSprite((xxt * 2), yyt, 4, ySrc + 1);
1060 TerrainSprite(xxt, (yyt * 2), 1, ySrc + 1);
1061 end;
1062
1063 // irrigation
1064 case Tile and fTerImp of
1065 tiIrrigation: begin
1066 TerrainSprite(xxt, 0, 0, 12);
1067 TerrainSprite(xxt * 2, yyt, 0, 12);
1068 end;
1069 tiFarm: begin
1070 TerrainSprite(xxt, 0, 1, 12);
1071 TerrainSprite(xxt * 2, yyt, 1, 12);
1072 end;
1073 end;
1074
1075 // river/canal/road/railroad
1076 if Tile and fRiver <> 0 then begin
1077 TerrainSprite(0, yyt, 2, 14);
1078 TerrainSprite(xxt, (yyt * 2), 2, 14);
1079 end;
1080 if Tile and fCanal <> 0 then begin
1081 TerrainSprite(xxt, 0, 7, 11);
1082 TerrainSprite(xxt, 0, 3, 11);
1083 TerrainSprite(xxt * 2, yyt, 7, 11);
1084 TerrainSprite(xxt * 2, yyt, 3, 11);
1085 end;
1086 if Tile and fRR <> 0 then begin
1087 TerrainSprite((xxt * 2), yyt, 1, 10);
1088 TerrainSprite((xxt * 2), yyt, 5, 10);
1089 TerrainSprite(xxt, (yyt * 2), 1, 10);
1090 TerrainSprite(xxt, (yyt * 2), 5, 10);
1091 end
1092 else if Tile and fRoad <> 0 then begin
1093 TerrainSprite((xxt * 2), yyt, 8, 9);
1094 TerrainSprite((xxt * 2), yyt, 5, 9);
1095 TerrainSprite(xxt, (yyt * 2), 1, 9);
1096 TerrainSprite(xxt, (yyt * 2), 5, 9);
1097 end;
1098
1099 if Tile and fPoll <> 0 then
1100 TerrainSprite(xxt, (yyt * 2), 6, 12);
1101
1102 // special
1103 if Tile and (fTerrain or fSpecial) = fGrass or fSpecial1 then TerrainSprite4(2, 1)
1104 else if Tile and fSpecial <> 0 then
1105 if Tile and fTerrain < fForest then
1106 TerrainSprite(0, yyt, Tile and fTerrain, Tile and fSpecial shr 5)
1107 else if (Tile and fTerrain = fForest) and IsJungle(Loc div G.lx) then
1108 TerrainSprite(0, yyt, 8, 17 + Tile and fSpecial shr 5)
1109 else
1110 TerrainSprite(0, yyt, 8, 2 + (Tile and fTerrain - fForest) * 2 + Tile and
1111 fSpecial shr 5)
1112 else if Tile and fDeadLands <> 0 then begin
1113 TerrainSprite4(6, 2);
1114 TerrainSprite(xxt, yyt, 8, 12 + Tile shr 25 and 3);
1115 end;
1116
1117 // other improvements
1118 case Tile and fTerImp of
1119 tiMine: TerrainSprite(xxt, 0, 2, 12);
1120 tiFort: begin
1121 TerrainSprite(xxt, 0, 7, 12);
1122 TerrainSprite(xxt, 0, 3, 12);
1123 end;
1124 tiBase: TerrainSprite(xxt, 0, 4, 12);
1125 end;
1126 end;
1127end;
1128
1129procedure TMainScreen.UpdateInterface;
1130begin
1131 mFillMap.Enabled := BrushType in [fShore, fGrass, fTundra, fPrairie, fDesert, fSwamp, fForest,
1132 fHills, fMountains, fArctic];
1133end;
1134
1135function ChooseResearch: Boolean;
1136var
1137 ChosenResearch: Integer;
1138begin
1139 if (MyData.FarTech <> adNone) and (MyRO.Tech[MyData.FarTech] >= tsApplicable)
1140 then
1141 MyData.FarTech := adNone;
1142 repeat
1143 { research complete -- select new }
1144 repeat
1145 MainScreen.ModalSelectDlg.ShowNewContent(wmModal, kAdvance);
1146 if MainScreen.ModalSelectDlg.Result < 0 then
1147 begin
1148 Result := False;
1149 Exit;
1150 end;
1151 ChosenResearch := MainScreen.ModalSelectDlg.Result;
1152 if ChosenResearch = adMilitary then
1153 begin
1154 MainScreen.DraftDlg.ShowNewContent(wmModal);
1155 if MainScreen.DraftDlg.ModalResult <> mrOK then
1156 Tribe[Me].ModelPicture[MyRO.nModel].HGr := nil;
1157 end;
1158 until (ChosenResearch <> adMilitary) or (MainScreen.DraftDlg.ModalResult = mrOK);
1159
1160 if ChosenResearch = adMilitary then
1161 InitMyModel(MyRO.nModel, True)
1162 else if ChosenResearch = adFar then
1163 begin
1164 MainScreen.ModalSelectDlg.ShowNewContent(wmModal, kFarAdvance);
1165 if MainScreen.ModalSelectDlg.Result >= 0 then
1166 if (MainScreen.ModalSelectDlg.Result = adNone) or
1167 (Server(sSetResearch - sExecute, Me, MainScreen.ModalSelectDlg.Result, nil^) <
1168 rExecuted) then
1169 MyData.FarTech := MainScreen.ModalSelectDlg.Result
1170 else
1171 begin
1172 ChosenResearch := MainScreen.ModalSelectDlg.Result;
1173 // can be researched immediately
1174 MyData.FarTech := adNone;
1175 end;
1176 end;
1177 until ChosenResearch <> adFar;
1178 if ChosenResearch = adNexus then
1179 MyData.FarTech := adNexus
1180 else
1181 Server(sSetResearch, Me, ChosenResearch, nil^);
1182 MainScreen.ListDlg.TechChange;
1183 Result := True;
1184end;
1185
1186procedure ApplyToVisibleForms(FormAction: TFormAction);
1187var
1188 I: Integer;
1189 Form: TForm;
1190begin
1191 for I := 0 to Screen.FormCount - 1 do begin
1192 Form := Screen.Forms[I];
1193 if Form.Visible and (Form is TBufferedDrawDlg) then begin
1194 case FormAction of
1195 faClose: Form.Close;
1196 faEnable: Form.Enabled := True;
1197 faDisable: Form.Enabled := False;
1198 faUpdate: begin
1199 if @Form.OnShow <> nil then Form.OnShow(nil);
1200 Form.Invalidate;
1201 Form.Update;
1202 end;
1203 faSmartUpdateContent: TBufferedDrawDlg(Form).SmartUpdateContent(False);
1204 end;
1205 end;
1206 end;
1207end;
1208
1209(* ** client function handling ** *)
1210
1211function TMainScreen.DipCall(Command: Integer): Integer;
1212var
1213 I: Integer;
1214 IsTreatyDeal: Boolean;
1215begin
1216 Result := Server(Command, Me, 0, nil^);
1217 if Result >= rExecuted then
1218 begin
1219 if Command and $FF0F = scContact then
1220 begin
1221 DipMem[Me].pContact := Command shr 4 and $F;
1222 NegoDlg.Initiate;
1223 DipMem[Me].DeliveredPrices := [];
1224 DipMem[Me].ReceivedPrices := [];
1225 end;
1226
1227 DipMem[Me].SentCommand := Command;
1228 DipMem[Me].FormerTreaty := MyRO.Treaty[DipMem[Me].pContact];
1229 if Command = scDipCancelTreaty then
1230 Play('CANCELTREATY')
1231 else if Command = scDipAccept then
1232 begin // remember delivered and received prices
1233 for I := 0 to ReceivedOffer.nDeliver - 1 do
1234 Include(DipMem[Me].ReceivedPrices, ReceivedOffer.Price[I] shr 24);
1235 for I := 0 to ReceivedOffer.nCost - 1 do
1236 Include(DipMem[Me].DeliveredPrices,
1237 ReceivedOffer.Price[ReceivedOffer.nDeliver + I] shr 24);
1238 IsTreatyDeal := False;
1239 for I := 0 to ReceivedOffer.nDeliver + ReceivedOffer.nCost - 1 do
1240 if ReceivedOffer.Price[I] and opMask = opTreaty then begin
1241 IsTreatyDeal := True;
1242 Break;
1243 end;
1244 if IsTreatyDeal then
1245 Play('NEWTREATY')
1246 else
1247 Play('ACCEPTOFFER');
1248 end;
1249 CityDlg.CloseAction := None;
1250 if G.RO[DipMem[Me].pContact] <> nil then
1251 begin // close windows for next player
1252 ApplyToVisibleForms(faClose);
1253 end
1254 else
1255 begin
1256 if CityDlg.Visible then
1257 CityDlg.Close;
1258 if UnitStatDlg.Visible then
1259 UnitStatDlg.Close;
1260 end;
1261 end;
1262end;
1263
1264function TMainScreen.OfferCall(var Offer: TOffer): Integer;
1265begin
1266 Result := Server(scDipOffer, Me, 0, Offer);
1267 if Result >= rExecuted then
1268 begin
1269 DipMem[Me].SentCommand := scDipOffer;
1270 DipMem[Me].FormerTreaty := MyRO.Treaty[DipMem[Me].pContact];
1271 DipMem[Me].SentOffer := Offer;
1272 CityDlg.CloseAction := None;
1273 if G.RO[DipMem[Me].pContact] <> nil then
1274 begin // close windows for next player
1275 ApplyToVisibleForms(faClose);
1276 end
1277 else
1278 begin
1279 if CityDlg.Visible then
1280 CityDlg.Close;
1281 if UnitStatDlg.Visible then
1282 UnitStatDlg.Close;
1283 end;
1284 end;
1285end;
1286
1287procedure TMainScreen.SetUnFocus(uix: Integer);
1288var
1289 Loc0: Integer;
1290begin
1291 Assert(not ((uix >= 0) and Supervising));
1292 if uix <> UnFocus then
1293 begin
1294 DestinationMarkON := False;
1295 PaintDestination;
1296 if uix >= 0 then
1297 UnStartLoc := MyUn[uix].Loc;
1298 BlinkON := False;
1299 BlinkTime := -1;
1300 if UnFocus >= 0 then
1301 begin
1302 Loc0 := MyUn[UnFocus].Loc;
1303 if (uix < 0) or (Loc0 <> MyUn[uix].Loc) then
1304 begin
1305 UnFocus := -1;
1306 PaintLoc(Loc0);
1307 end;
1308 end;
1309 UnFocus := uix;
1310 end;
1311 UnitInfoBtn.Visible := UnFocus >= 0;
1312 UnitBtn.Visible := UnFocus >= 0;
1313 CheckTerrainBtnVisible;
1314end;
1315
1316procedure TMainScreen.CheckTerrainBtnVisible;
1317var
1318 Tile: Integer;
1319 mox: ^TModel;
1320begin
1321 if UnFocus >= 0 then
1322 begin
1323 mox := @MyModel[MyUn[UnFocus].mix];
1324 Tile := MyMap[MyUn[UnFocus].Loc];
1325 TerrainBtn.Visible := (Tile and fCity = 0) and (MyUn[UnFocus].Master < 0)
1326 and ((mox.Kind = mkSettler) or (mox.Kind = mkSlaves) and
1327 (MyRO.Wonder[woPyramids].EffectiveOwner >= 0));
1328 end
1329 else
1330 TerrainBtn.Visible := False;
1331end;
1332
1333procedure TMainScreen.CheckMovieSpeedBtnState;
1334begin
1335 if GameMode = cMovie then
1336 begin
1337 MovieSpeed1Btn.Down := MovieSpeed = 1;
1338 MovieSpeed1Btn.Visible := True;
1339 MovieSpeed2Btn.Down := MovieSpeed = 2;
1340 MovieSpeed2Btn.Visible := True;
1341 MovieSpeed3Btn.Down := MovieSpeed = 3;
1342 MovieSpeed3Btn.Visible := True;
1343 MovieSpeed4Btn.Down := MovieSpeed = 4;
1344 MovieSpeed4Btn.Visible := True;
1345 end
1346 else
1347 begin
1348 MovieSpeed1Btn.Visible := False;
1349 MovieSpeed2Btn.Visible := False;
1350 MovieSpeed3Btn.Visible := False;
1351 MovieSpeed4Btn.Visible := False;
1352 end;
1353end;
1354
1355procedure TMainScreen.SetMapOptions;
1356begin
1357 MiniMap.MapOptions := MapOptionChecked;
1358 MapOptions := MapOptionChecked;
1359 if ClientMode = cEditMap then
1360 MapOptions := MapOptions + [moEditMode];
1361 if mLocCodes.Checked then
1362 MapOptions := MapOptions + [moLocCodes];
1363end;
1364
1365procedure TMainScreen.UpdateViews(UpdateCityScreen: Boolean);
1366begin
1367 SumCities(TaxSum, ScienceSum);
1368 PanelPaint; // TopBar was enough!!!
1369 ListDlg.EcoChange;
1370 NatStatDlg.EcoChange;
1371 if UpdateCityScreen then
1372 CityDlg.SmartUpdateContent;
1373end;
1374
1375procedure TMainScreen.SetAIName(P: Integer; Name: string);
1376begin
1377 if Name = '' then
1378 begin
1379 if AILogo[P] <> nil then
1380 begin
1381 FreeAndNil(AILogo[P]);
1382 end;
1383 end
1384 else
1385 begin
1386 if AILogo[P] = nil then
1387 AILogo[P] := TBitmap.Create;
1388 if not LoadGraphicFile(AILogo[P], GetAppSharePath(Name + '.png'), [gfNoError]) then
1389 begin
1390 FreeAndNil(AILogo[P]);
1391 end;
1392 end;
1393end;
1394
1395function TMainScreen.ContactRefused(P: Integer; Item: String): Boolean;
1396// return whether treaty was cancelled
1397var
1398 S: string;
1399begin
1400 Assert(MyRO.Treaty[P] >= trPeace);
1401 S := Tribe[P].TPhrase(Item);
1402 if MyRO.Turn < MyRO.LastCancelTreaty[P] + CancelTreatyTurns then
1403 begin
1404 SimpleMessage(S);
1405 Result := False;
1406 end
1407 else
1408 begin
1409 case MyRO.Treaty[P] of
1410 trPeace:
1411 S := S + ' ' + Phrases.Lookup('FRCANCELQUERY_PEACE');
1412 trFriendlyContact:
1413 S := S + ' ' + Phrases.Lookup('FRCANCELQUERY_FRIENDLY');
1414 trAlliance:
1415 S := S + ' ' + Phrases.Lookup('FRCANCELQUERY_ALLIANCE');
1416 end;
1417 Result := SimpleQuery(mkYesNo, S, 'NEGO_REJECTED') = mrOK;
1418 if Result then
1419 begin
1420 Play('CANCELTREATY');
1421 Server(sCancelTreaty, Me, 0, nil^);
1422 if MyRO.Treaty[P] = trNone then
1423 CityOptimizer_BeginOfTurn;
1424 // peace treaty was cancelled -- use formerly forbidden tiles
1425 MapValid := False;
1426 PaintAllMaps;
1427 end;
1428 end;
1429end;
1430
1431procedure TMainScreen.RememberPeaceViolation;
1432var
1433 uix, p1: Integer;
1434begin
1435 MyData.PeaceEvaHappened := 0;
1436 for uix := 0 to MyRO.nUn - 1 do
1437 with MyUn[uix] do
1438 if Loc >= 0 then
1439 begin
1440 p1 := MyRO.Territory[Loc];
1441 if (p1 <> Me) and (p1 >= 0) and
1442 (MyRO.Turn = MyRO.EvaStart[p1] + (PeaceEvaTurns - 1)) then
1443 MyData.PeaceEvaHappened := MyData.PeaceEvaHappened or (1 shl p1);
1444 end;
1445end;
1446
1447procedure TMainScreen.SoundPreload(Check: TSoundBlocks);
1448const
1449 nStartBlock = 27;
1450 StartBlock: array [0 .. nStartBlock - 1] of string = ('INVALID', 'TURNEND',
1451 'DISBAND', 'CHEAT', 'MSG_DEFAULT', 'WARNING_DISORDER', 'WARNING_FAMINE',
1452 'WARNING_LOWSUPPORT', 'WARNING_LOWFUNDS', 'MOVE_MOUNTAIN', 'MOVE_LOAD',
1453 'MOVE_UNLOAD', 'MOVE_DIE', 'NOMOVE_TIME', 'NOMOVE_DOMAIN',
1454 'NOMOVE_DEFAULT', 'CITY_SELLIMP', 'CITY_REBUILDIMP', 'CITY_BUYPROJECT',
1455 'CITY_UTILIZE', 'NEWMODEL_0', 'NEWADVANCE_0', 'AGE_0', 'REVOLUTION',
1456 'NEWGOV', 'CITY_INVALIDTYPE', 'MSG_GAMEOVER');
1457
1458 nWonderBlock = 6;
1459 WonderBlock: array [0 .. nWonderBlock - 1] of string = ('WONDER_BUILT',
1460 'WONDER_CAPTURED', 'WONDER_EXPIRED', 'WONDER_DESTROYED', 'MSG_COLDWAR',
1461 'NEWADVANCE_GRLIB');
1462
1463 nScienceBlock = 17;
1464 ScienceBlock: array [0 .. nScienceBlock - 1] of string = ('MOVE_PARACHUTE',
1465 'MOVE_PLANESTART', 'MOVE_PLANELANDING', 'MOVE_COVERT', 'NEWMODEL_1',
1466 'NEWMODEL_2', 'NEWMODEL_3', 'NEWADVANCE_1', 'NEWADVANCE_2',
1467 'NEWADVANCE_3', 'AGE_1', 'AGE_2', 'AGE_3', 'SHIP_BUILT', 'SHIP_TRADED',
1468 'SHIP_CAPTURED', 'SHIP_DESTROYED');
1469
1470 nContactBlock = 20;
1471 ContactBlock: array [0 .. nContactBlock - 1] of string = ('NEWTREATY',
1472 'CANCELTREATY', 'ACCEPTOFFER', 'MSG_WITHDRAW', 'MSG_BANKRUPT',
1473 'CONTACT_0', 'CONTACT_1', 'CONTACT_2', 'CONTACT_3', 'CONTACT_4',
1474 'CONTACT_5', 'CONTACT_5', 'CONTACT_6', 'NEGO_REJECTED', 'MOVE_CAPTURE',
1475 'MOVE_EXPEL', 'NOMOVE_TREATY', 'NOMOVE_ZOC', 'NOMOVE_SUBMARINE',
1476 'NOMOVE_STEALTH');
1477
1478var
1479 I, cix, mix: Integer;
1480 Need: Boolean;
1481 mi: TModelInfo;
1482begin
1483 if (sbStart in Check) and not (sbStart in SoundPreloadDone) then begin
1484 for I := 0 to nStartBlock - 1 do
1485 PreparePlay(StartBlock[I]);
1486 SoundPreloadDone := SoundPreloadDone + [sbStart];
1487 end;
1488 if (sbWonder in Check) and not (sbWonder in SoundPreloadDone) then begin
1489 Need := False;
1490 for I := 0 to nWonder - 1 do
1491 if MyRO.Wonder[I].CityID <> WonderNotBuiltYet then begin
1492 Need := True;
1493 Break;
1494 end;
1495 if Need then begin
1496 for I := 0 to nWonderBlock - 1 do
1497 PreparePlay(WonderBlock[I]);
1498 SoundPreloadDone := SoundPreloadDone + [sbWonder];
1499 end;
1500 end;
1501 if ((sbScience in Check) and not (sbScience in SoundPreloadDone)) and
1502 (MyRO.Tech[adScience] >= tsApplicable) then begin
1503 for I := 0 to nScienceBlock - 1 do
1504 PreparePlay(ScienceBlock[I]);
1505 SoundPreloadDone := SoundPreloadDone + [sbScience];
1506 end;
1507 if ((sbContact in Check) and not (sbContact in SoundPreloadDone)) and
1508 (MyRO.nEnemyModel + MyRO.nEnemyCity > 0) then begin
1509 for I := 0 to nContactBlock - 1 do
1510 PreparePlay(ContactBlock[I]);
1511 SoundPreloadDone := SoundPreloadDone + [sbContact];
1512 end;
1513 if sbTurn in Check then begin
1514 if MyRO.Happened and phShipComplete <> 0 then
1515 PreparePlay('MSG_YOUWIN');
1516 if MyData.ToldAlive <> MyRO.Alive then
1517 PreparePlay('MSG_EXTINCT');
1518 for cix := 0 to MyRO.nCity - 1 do
1519 with MyCity[cix] do
1520 if (Loc >= 0) and (Flags and CityRepMask <> 0) then
1521 for I := 0 to 12 do
1522 if 1 shl I and Flags and CityRepMask <> 0 then
1523 PreparePlay(CityEventSoundItem[I]);
1524 for mix := 0 to MyRO.nModel - 1 do
1525 with MyModel[mix] do
1526 if Attack > 0 then
1527 begin
1528 MakeModelInfo(Me, mix, MyModel[mix], mi);
1529 PreparePlay(AttackSound(ModelCode(mi)));
1530 end;
1531 end;
1532end;
1533
1534procedure TMainScreen.GetTribeList;
1535var
1536 SearchRec: TSearchRec;
1537 Color: TColor;
1538 Name: string;
1539 Ok: Boolean;
1540begin
1541 UnusedTribeFiles.Clear;
1542 Ok := FindFirst(LocalizedFilePath('Tribes') + DirectorySeparator + '*' + CevoTribeExt,
1543 faArchive + faReadOnly, SearchRec) = 0;
1544 if not Ok then
1545 begin
1546 FindClose(SearchRec);
1547 Ok := FindFirst(LocalizedFilePath('Tribes' + DirectorySeparator + '*' + CevoTribeExt),
1548 faArchive + faReadOnly, SearchRec) = 0;
1549 end;
1550 if Ok then
1551 repeat
1552 SearchRec.Name := Copy(SearchRec.Name, 1, Length(SearchRec.Name) - Length(CevoTribeExt));
1553 if GetTribeInfo(SearchRec.Name, Name, Color) then
1554 UnusedTribeFiles.AddObject(SearchRec.Name, TObject(Color));
1555 until FindNext(SearchRec) <> 0;
1556 FindClose(SearchRec);
1557end;
1558
1559function TMainScreen.GetUnitStatDlg: TUnitStatDlg;
1560begin
1561 if not Assigned(FUnitStatDlg) then begin
1562 FUnitStatDlg := TUnitStatDlg.Create(nil);
1563 ArrangeDialog(FUnitStatDlg);
1564 FUnitStatDlg.CheckAge;
1565 end;
1566 Result := FUnitStatDlg;
1567end;
1568
1569function TMainScreen.GetWondersDlg: TWondersDlg;
1570begin
1571 if not Assigned(FWondersDlg) then FWondersDlg := TWondersDlg.Create(nil);
1572 Result := FWondersDlg;
1573end;
1574
1575function TMainScreen.ChooseUnusedTribe: Integer;
1576var
1577 I: Integer;
1578 J: Integer;
1579 ColorDistance: Integer;
1580 BestColorDistance: Integer;
1581 TestColorDistance: Integer;
1582 CountBest: Integer;
1583begin
1584 Assert(UnusedTribeFiles.Count > 0);
1585 Result := -1;
1586 BestColorDistance := -1;
1587 for J := 0 to UnusedTribeFiles.Count - 1 do
1588 begin
1589 ColorDistance := 250; // consider differences more than this infinite
1590 for I := 0 to nPl - 1 do
1591 if Tribe[I] <> nil then
1592 begin
1593 TestColorDistance := Abs(Integer(UnusedTribeFiles.Objects[J])
1594 shr 16 and $FF - Tribe[I].Color shr 16 and $FF) +
1595 Abs(Integer(UnusedTribeFiles.Objects[J]) shr 8 and
1596 $FF - Tribe[I].Color shr 8 and $FF) * 3 +
1597 Abs(Integer(UnusedTribeFiles.Objects[J]) and
1598 $FF - Tribe[I].Color and $FF) * 2;
1599 if TestColorDistance < ColorDistance then
1600 ColorDistance := TestColorDistance;
1601 end;
1602 if ColorDistance > BestColorDistance then
1603 begin
1604 CountBest := 0;
1605 BestColorDistance := ColorDistance;
1606 end;
1607 if ColorDistance = BestColorDistance then
1608 begin
1609 Inc(CountBest);
1610 if DelphiRandom(CountBest) = 0 then
1611 Result := J;
1612 end;
1613 end;
1614end;
1615
1616procedure TMainScreen.ShowEnemyShipChange(ShowShipChange: TShowShipChange);
1617var
1618 I, TestCost, MostCost: Integer;
1619 Ship1Plus, Ship2Plus: Boolean;
1620begin
1621 with ShowShipChange, MessgExDlg do
1622 begin
1623 case Reason of
1624 scrProduction:
1625 begin
1626 OpenSound := 'SHIP_BUILT';
1627 MessgText := Tribe[Ship1Owner].TPhrase('SHIPBUILT');
1628 IconKind := mikShip;
1629 IconIndex := Ship1Owner;
1630 end;
1631
1632 scrDestruction:
1633 begin
1634 OpenSound := 'SHIP_DESTROYED';
1635 MessgText := Tribe[Ship1Owner].TPhrase('SHIPDESTROYED');
1636 IconKind := mikImp;
1637 end;
1638
1639 scrTrade:
1640 begin
1641 OpenSound := 'SHIP_TRADED';
1642 Ship1Plus := False;
1643 Ship2Plus := False;
1644 for I := 0 to nShipPart - 1 do
1645 begin
1646 if Ship1Change[I] > 0 then Ship1Plus := True;
1647 if Ship2Change[I] > 0 then Ship2Plus := True;
1648 end;
1649 if Ship1Plus and Ship2Plus then
1650 MessgText := Tribe[Ship1Owner].TPhrase('SHIPBITRADE1') + ' ' +
1651 Tribe[Ship2Owner].TPhrase('SHIPBITRADE2')
1652 else if Ship1Plus then
1653 MessgText := Tribe[Ship1Owner].TPhrase('SHIPUNITRADE1') + ' ' +
1654 Tribe[Ship2Owner].TPhrase('SHIPUNITRADE2')
1655 else // if Ship2Plus then
1656 MessgText := Tribe[Ship2Owner].TPhrase('SHIPUNITRADE1') + ' ' +
1657 Tribe[Ship1Owner].TPhrase('SHIPUNITRADE2');
1658 IconKind := mikImp;
1659 end;
1660
1661 scrCapture:
1662 begin
1663 OpenSound := 'SHIP_CAPTURED';
1664 MessgText := Tribe[Ship2Owner].TPhrase('SHIPCAPTURE1') + ' ' +
1665 Tribe[Ship1Owner].TPhrase('SHIPCAPTURE2');
1666 IconKind := mikShip;
1667 IconIndex := Ship2Owner;
1668 end;
1669 end;
1670
1671 if IconKind = mikImp then
1672 begin
1673 MostCost := 0;
1674 for I := 0 to nShipPart - 1 do
1675 begin
1676 TestCost := Abs(Ship1Change[I]) * Imp[imShipComp + I].Cost;
1677 if TestCost > MostCost then
1678 begin
1679 MostCost := TestCost;
1680 IconIndex := imShipComp + I;
1681 end;
1682 end;
1683 end;
1684
1685 Kind := mkOk;
1686 ShowModal;
1687 end;
1688end;
1689
1690procedure TMainScreen.InitModule;
1691var
1692 I, J, Domain: Integer;
1693begin
1694 { search icons for advances: }
1695 for I := 0 to nAdv - 1 do
1696 if I in FutureTech then
1697 AdvIcon[I] := 96 + I - futResearchTechnology
1698 else
1699 begin
1700 AdvIcon[I] := -1;
1701 for Domain := 0 to nDomains - 1 do
1702 for J := 0 to nUpgrade - 1 do
1703 if upgrade[Domain, J].Preq = I then
1704 if AdvIcon[I] >= 0 then
1705 AdvIcon[I] := 85
1706 else
1707 AdvIcon[I] := 86 + Domain;
1708 for J := 0 to nFeature - 1 do
1709 if Feature[J].Preq = I then
1710 for Domain := 0 to nDomains - 1 do
1711 if 1 shl Domain and Feature[J].Domains <> 0 then
1712 if (AdvIcon[I] >= 0) and (AdvIcon[I] <> 86 + Domain) then
1713 AdvIcon[I] := 85
1714 else
1715 AdvIcon[I] := 86 + Domain;
1716 for J := nWonder to nImp - 1 do
1717 if Imp[J].Preq = I then
1718 AdvIcon[I] := J;
1719 for J := nWonder to nImp - 1 do
1720 if (Imp[J].Preq = I) and (Imp[J].Kind <> ikCommon) then
1721 AdvIcon[I] := J;
1722 for J := 0 to nJob - 1 do
1723 if I = JobPreq[J] then
1724 AdvIcon[I] := 84;
1725 for J := 0 to nWonder - 1 do
1726 if Imp[J].Preq = I then
1727 AdvIcon[I] := J;
1728 if AdvIcon[I] < 0 then
1729 if AdvValue[I] < 1000 then
1730 AdvIcon[I] := -7
1731 else
1732 AdvIcon[I] := 24 + AdvValue[I] div 1000;
1733 for J := 2 to nGov - 1 do
1734 if GovPreq[J] = I then
1735 AdvIcon[I] := J - 8;
1736 end;
1737 AdvIcon[adConscription] := 86 + dGround;
1738
1739 UnusedTribeFiles := TStringList.Create;
1740 UnusedTribeFiles.Sorted := True;
1741 TribeNames := TStringList.Create;
1742
1743 IsoEngine.Init(InitEnemyModel);
1744 // non-default tile size is missing a file, switch to default
1745 MainMap.SetOutput(Offscreen);
1746
1747 HGrStdUnits := LoadGraphicSet('StdUnits.png');
1748 SmallImp := TBitmap.Create;
1749 SmallImp.PixelFormat := TPixelFormat.pf24bit;
1750 InitSmallImp;
1751 SoundPreloadDone := [];
1752 StartRunning := False;
1753
1754 VerticalScrollBar := TPVScrollbar.Create(Self);
1755 VerticalScrollBar.WheelSteps := 1;
1756 VerticalScrollBar.OnUpdate := ScrollBarUpdate;
1757end;
1758
1759procedure TMainScreen.DoneModule;
1760begin
1761 FreeAndNil(SmallImp);
1762 FreeAndNil(UnusedTribeFiles);
1763 FreeAndNil(TribeNames);
1764end;
1765
1766procedure TMainScreen.InitTurn(NewPlayer: Integer);
1767const
1768 nAdvBookIcon = 16;
1769 AdvBookIcon: array [0 .. nAdvBookIcon - 1] of record Adv,
1770 Icon: Integer end = ((Adv: adPolyTheism; Icon: woZeus),
1771 (Adv: adBronzeWorking; Icon: woColossus), (Adv: adMapMaking;
1772 Icon: woLighthouse), (Adv: adPoetry; Icon: imTheater), (Adv: adMonotheism;
1773 Icon: woMich), (Adv: adPhilosophy; Icon: woLeo), (Adv: adTheoryOfGravity;
1774 Icon: woNewton), (Adv: adSteel; Icon: woEiffel), (Adv: adDemocracy;
1775 Icon: woLiberty), (Adv: adAutomobile; Icon: imHighways),
1776 (Adv: adSanitation; Icon: imSewer), (Adv: adElectronics; Icon: woHoover),
1777 (Adv: adNuclearFission; Icon: woManhattan), (Adv: adRecycling;
1778 Icon: imRecycling), (Adv: adComputers; Icon: imResLab),
1779 (Adv: adSpaceFlight; Icon: woMIR));
1780 sbAll = [sbStart, sbWonder, sbScience, sbContact, sbTurn];
1781var
1782 p1, I, ad, uix, cix, MoveOptions, MoveResult, Loc1,
1783 NewAgeCenterTo, Winners, NewGovAvailable, dx,
1784 dy: Integer;
1785 MoveAdviceData: TMoveAdviceData;
1786 Picture: TModelPictureInfo;
1787 S, Item, Item2: string;
1788 UpdatePanel, OwnWonder, ok, Stop, ShowCityList, WondersOnly,
1789 AllowCityScreen: Boolean;
1790begin
1791 if IsMultiPlayerGame and (NewPlayer <> Me) then
1792 begin
1793 UnitInfoBtn.Visible := False;
1794 UnitBtn.Visible := False;
1795 TerrainBtn.Visible := False;
1796 EOT.Visible := False;
1797 end;
1798 if IsMultiPlayerGame and (NewPlayer <> Me) and
1799 (G.RO[0].Happened and phShipComplete = 0) then
1800 begin // inter player screen
1801 for I := 0 to ControlCount - 1 do
1802 if Controls[I] is TButtonC then
1803 Controls[I].Visible := False;
1804 Me := -1;
1805 MainTexture.Age := -1;
1806 with Panel.Canvas do
1807 begin
1808 Brush.Color := $000000;
1809 FillRect(Rect(0, 0, Panel.Width, Panel.Height));
1810 Brush.Style := TBrushStyle.bsClear;
1811 end;
1812 with TopBar.Canvas do
1813 begin
1814 Brush.Color := $000000;
1815 FillRect(Rect(0, 0, TopBar.Width, TopBar.Height));
1816 Brush.Style := TBrushStyle.bsClear;
1817 end;
1818 Invalidate;
1819
1820 S := TurnToString(G.RO[0].Turn);
1821 if Supervising then
1822 SimpleMessage(Format(Phrases.Lookup('SUPERTURN'), [S]))
1823 else
1824 SimpleMessage(Format(Tribe[NewPlayer].TPhrase('TURN'), [S]));
1825 end;
1826 for I := 0 to ControlCount - 1 do
1827 if Controls[I] is TButtonC then
1828 Controls[I].Visible := True;
1829
1830 ItsMeAgain(NewPlayer);
1831 MyData := G.RO[NewPlayer].Data;
1832 if not Supervising then
1833 SoundPreload(sbAll);
1834 if (Me = 0) and ((MyRO.Turn = 0) or (ClientMode = cResume)) then
1835 Invalidate; // colorize empty space
1836
1837 if not Supervising then
1838 begin
1839
1840 { if MyRO.Happened and phGameEnd<>0 then
1841 begin
1842 Age := 3;
1843 MainTexture.Age := -1;
1844 end
1845 else }
1846 begin
1847 Age := GetAge(Me);
1848 if MainTexture.Age <> Age then begin
1849 MainTexture.Age := Age;
1850 EOT.Invalidate; // has visible background parts in its bounds
1851 end;
1852 end;
1853 // age:=MyRO.Turn mod 4; //!!!
1854 if ClientMode = cMovieTurn then
1855 EOT.ButtonIndex := eotCancel
1856 else if ClientMode < scContact then
1857 EOT.ButtonIndex := eotGray
1858 else
1859 EOT.ButtonIndex := eotBackToNego;
1860 end
1861 else
1862 begin
1863 Age := 0;
1864 MainTexture.Age := -1;
1865 if ClientMode = cMovieTurn then
1866 EOT.ButtonIndex := eotCancel
1867 else
1868 EOT.ButtonIndex := eotBlinkOn;
1869 end;
1870 InitCityMark(MainTexture);
1871 if Assigned(FCityDlg) then FCityDlg.CheckAge;
1872 if Assigned(FNatStatDlg) then FNatStatDlg.CheckAge;
1873 if Assigned(FUnitStatDlg) then FUnitStatDlg.CheckAge;
1874 if Assigned(FHelpDlg) then FHelpDlg.Difficulty := G.Difficulty[Me];
1875
1876 UnFocus := -1;
1877 MarkCityLoc := -1;
1878 BlinkON := False;
1879 BlinkTime := -1;
1880 Tracking := False;
1881 TurnComplete := False;
1882
1883 if (ToldSlavery < 0) or
1884 ((ToldSlavery = 1) <> (MyRO.Wonder[woPyramids].EffectiveOwner >= 0)) then
1885 begin
1886 if MyRO.Wonder[woPyramids].EffectiveOwner >= 0 then
1887 ToldSlavery := 1
1888 else
1889 ToldSlavery := 0;
1890 for p1 := 0 to nPl - 1 do
1891 if (Tribe[p1] <> nil) and (Tribe[p1].mixSlaves >= 0) then
1892 with Picture do
1893 begin // replace unit picture
1894 mix := Tribe[p1].mixSlaves;
1895 if ToldSlavery = 1 then
1896 pix := pixSlaves
1897 else
1898 pix := pixNoSlaves;
1899 Hash := 0;
1900 GrName := 'StdUnits.png';
1901 Tribe[p1].SetModelPicture(Picture, True);
1902 end;
1903 end;
1904
1905 if not Supervising and (ClientMode = cTurn) then
1906 begin
1907 for cix := 0 to MyRO.nCity - 1 do
1908 if (MyCity[cix].Loc >= 0) and
1909 ((MyRO.Turn = 0) or (MyCity[cix].Flags and chFounded <> 0)) then
1910 MyCity[cix].Status := MyCity[cix].Status and
1911 not csResourceWeightsMask or (3 shl 4);
1912 // new city, set to maximum growth
1913 end;
1914 if (ClientMode = cTurn) or (ClientMode = cContinue) then
1915 CityOptimizer_BeginOfTurn; // maybe peace was made or has ended
1916 SumCities(TaxSum, ScienceSum);
1917
1918 if ClientMode = cMovieTurn then
1919 begin
1920 UnitInfoBtn.Visible := False;
1921 UnitBtn.Visible := False;
1922 TerrainBtn.Visible := False;
1923 EOT.Hint := Phrases.Lookup('BTN_STOP');
1924 EOT.Visible := True;
1925 end
1926 else if ClientMode < scContact then
1927 begin
1928 UnitInfoBtn.Visible := UnFocus >= 0;
1929 UnitBtn.Visible := UnFocus >= 0;
1930 CheckTerrainBtnVisible;
1931 TurnComplete := Supervising;
1932 EOT.Hint := Phrases.Lookup('BTN_ENDTURN');
1933 EOT.Visible := Server(sTurn - sExecute, Me, 0, nil^) >= rExecuted;
1934 end
1935 else
1936 begin
1937 UnitInfoBtn.Visible := False;
1938 UnitBtn.Visible := False;
1939 TerrainBtn.Visible := False;
1940 EOT.Hint := Phrases.Lookup('BTN_NEGO');
1941 EOT.Visible := True;
1942 end;
1943 SetTroopLoc(-1);
1944 MapValid := False;
1945 NewAgeCenterTo := 0;
1946 if ((MyRO.Turn = 0) and not Supervising or IsMultiPlayerGame or
1947 (ClientMode = cResume)) and (MyRO.nCity > 0) then
1948 begin
1949 Loc1 := MyCity[0].Loc;
1950 if (ClientMode = cTurn) and (MyRO.Turn = 0) then
1951 with MainMap do begin // move city out of center to not be covered by welcome screen
1952 dx := MapWidth div (xxt * 5);
1953 if dx > 5 then
1954 dx := 5;
1955 dy := MapHeight div (yyt * 5);
1956 if dy > 5 then
1957 dy := 5;
1958 if Loc1 >= G.lx * G.ly div 2 then
1959 begin
1960 NewAgeCenterTo := -1;
1961 Loc1 := dLoc(Loc1, -dx, -dy);
1962 end
1963 else
1964 begin
1965 NewAgeCenterTo := 1;
1966 Loc1 := dLoc(Loc1, -dx, dy);
1967 end
1968 end;
1969 Centre(Loc1);
1970 end;
1971
1972 ApplyToVisibleForms(faEnable);
1973
1974 if ClientMode <> cResume then
1975 begin
1976 PaintAll;
1977 if (MyRO.Happened and phChangeGov <> 0) and (MyRO.NatBuilt[imPalace] > 0)
1978 then
1979 ImpImage(Panel.Canvas, ClientWidth - xPalace, yPalace, imPalace,
1980 gAnarchy { , GameMode<>cMovie } );
1981 // first turn after anarchy -- don't show despotism palace!
1982 Update;
1983 ApplyToVisibleForms(faUpdate);
1984
1985 if MyRO.Happened and phGameEnd <> 0 then
1986 with MessgExDlg do
1987 begin // game ended
1988 if MyRO.Happened and phExtinct <> 0 then
1989 begin
1990 OpenSound := 'MSG_GAMEOVER';
1991 MessgText := Tribe[Me].TPhrase('GAMEOVER');
1992 IconKind := mikBigIcon;
1993 IconIndex := 8;
1994 end
1995 else if MyRO.Happened and phShipComplete <> 0 then
1996 begin
1997 Winners := 0;
1998 for p1 := 0 to nPl - 1 do
1999 if 1 shl p1 and MyRO.Alive <> 0 then
2000 begin
2001 Winners := Winners or 1 shl p1;
2002 for I := 0 to nShipPart - 1 do
2003 if MyRO.Ship[p1].Parts[I] < ShipNeed[I] then
2004 Winners := Winners and not (1 shl p1);
2005 end;
2006 Assert(Winners <> 0);
2007 if Winners and (1 shl Me) <> 0 then
2008 begin
2009 S := '';
2010 for p1 := 0 to nPl - 1 do
2011 if (p1 <> Me) and (1 shl p1 and Winners <> 0) then
2012 if S = '' then
2013 S := Tribe[p1].TPhrase('SHORTNAME')
2014 else
2015 S := Format(Phrases.Lookup('SHAREDWIN_CONCAT'),
2016 [S, Tribe[p1].TPhrase('SHORTNAME')]);
2017
2018 OpenSound := 'MSG_YOUWIN';
2019 MessgText := Tribe[Me].TPhrase('MYSPACESHIP');
2020 if S <> '' then
2021 MessgText := MessgText + '\' +
2022 Format(Phrases.Lookup('SHAREDWIN'), [S]);
2023 IconKind := mikBigIcon;
2024 IconIndex := 9;
2025 end
2026 else
2027 begin
2028 Assert(Me = 0);
2029 OpenSound := 'MSG_GAMEOVER';
2030 MessgText := '';
2031 for p1 := 0 to nPl - 1 do
2032 if Winners and (1 shl p1) <> 0 then
2033 MessgText := MessgText + Tribe[p1].TPhrase('SPACESHIP1');
2034 MessgText := MessgText + '\' + Phrases.Lookup('SPACESHIP2');
2035 IconKind := mikEnemyShipComplete;
2036 end
2037 end
2038 else { if MyRO.Happened and fTimeUp<>0 then }
2039 begin
2040 Assert(Me = 0);
2041 OpenSound := 'MSG_GAMEOVER';
2042 if not Supervising then
2043 MessgText := Tribe[Me].TPhrase('TIMEUP')
2044 else
2045 MessgText := Phrases.Lookup('TIMEUPSUPER');
2046 IconKind := mikImp;
2047 IconIndex := 22;
2048 end;
2049 Kind := mkOk;
2050 ShowModal;
2051 if MyRO.Happened and phExtinct = 0 then
2052 begin
2053 p1 := 0;
2054 while (p1 < nPl - 1) and (Winners and (1 shl p1) = 0) do
2055 Inc(p1);
2056 if MyRO.Happened and phShipComplete = 0 then
2057 DiaDlg.ShowNewContent_Charts(wmModal);
2058 end;
2059 TurnComplete := True;
2060 Exit;
2061 end;
2062 if not Supervising and (1 shl Me and MyRO.Alive = 0) then
2063 begin
2064 TurnComplete := True;
2065 Exit;
2066 end;
2067
2068 if (ClientMode = cContinue) and
2069 (DipMem[Me].SentCommand and $FF0F = scContact) then
2070 // contact was refused
2071 if MyRO.Treaty[DipMem[Me].pContact] >= trPeace then
2072 ContactRefused(DipMem[Me].pContact, 'FRREJECTED')
2073 else
2074 SoundMessage(Tribe[DipMem[Me].pContact].TPhrase('FRREJECTED'),
2075 'NEGO_REJECTED');
2076
2077 if not Supervising and (Age > MyData.ToldAge) and
2078 ((Age > 0) or (ClientMode <> cMovieTurn)) then
2079 with MessgExDlg do
2080 begin
2081 if Age = 0 then
2082 begin
2083 if Phrases2FallenBackToEnglish then
2084 begin
2085 S := Tribe[Me].TPhrase('AGE0');
2086 MessgText :=
2087 Format(S, [TurnToString(MyRO.Turn), CityName(MyCity[0].ID)]);
2088 end
2089 else
2090 begin
2091 S := Tribe[Me].TString(Phrases2.Lookup('AGE0'));
2092 MessgText := Format(S, [TurnToString(MyRO.Turn)]);
2093 end;
2094 end
2095 else
2096 begin
2097 S := Tribe[Me].TPhrase('AGE' + Char(48 + Age));
2098 MessgText := Format(S, [TurnToString(MyRO.Turn)]);
2099 end;
2100 IconKind := mikAge;
2101 IconIndex := Age;
2102 { if age = 0 then } Kind := mkOk
2103 { else begin Kind := mkOkHelp; HelpKind := hkAdv; HelpNo := AgePreq[age]; end; };
2104 CenterTo := NewAgeCenterTo;
2105 OpenSound := 'AGE_' + Char(48 + Age);
2106 Application.ProcessMessages;
2107 ShowModal;
2108 MyData.ToldAge := Age;
2109 if Age > 0 then
2110 MyData.ToldTech[AgePreq[Age]] := MyRO.Tech[AgePreq[Age]];
2111 end;
2112
2113 if MyData.ToldAlive <> MyRO.Alive then
2114 begin
2115 for p1 := 0 to nPl - 1 do
2116 if (MyData.ToldAlive - MyRO.Alive) and (1 shl p1) <> 0 then
2117 with MessgExDlg do
2118 begin
2119 OpenSound := 'MSG_EXTINCT';
2120 S := Tribe[p1].TPhrase('EXTINCT');
2121 MessgText := Format(S, [TurnToString(MyRO.Turn)]);
2122 if MyRO.Alive = 1 shl Me then
2123 MessgText := MessgText + Phrases.Lookup('EXTINCTALL');
2124 Kind := mkOk;
2125 IconKind := mikImp;
2126 IconIndex := 21;
2127 ShowModal;
2128 end;
2129 if (ClientMode <> cMovieTurn) and not Supervising then
2130 DiaDlg.ShowNewContent_Charts(wmModal);
2131 end;
2132
2133 // tell changes of own credibility
2134 if not Supervising then
2135 begin
2136 if RoughCredibility(MyRO.Credibility) <>
2137 RoughCredibility(MyData.ToldOwnCredibility) then
2138 begin
2139 if RoughCredibility(MyRO.Credibility) >
2140 RoughCredibility(MyData.ToldOwnCredibility) then
2141 S := Phrases.Lookup('CREDUP')
2142 else
2143 S := Phrases.Lookup('CREDDOWN');
2144 TribeMessage(Me, Format(S, [Phrases.Lookup('CREDIBILITY',
2145 RoughCredibility(MyRO.Credibility))]), '');
2146 end;
2147 MyData.ToldOwnCredibility := MyRO.Credibility;
2148 end;
2149
2150 for I := 0 to nWonder - 1 do
2151 begin
2152 OwnWonder := False;
2153 for cix := 0 to MyRO.nCity - 1 do
2154 if (MyCity[cix].Loc >= 0) and (MyCity[cix].ID = MyRO.Wonder[I].CityID)
2155 then begin
2156 OwnWonder := True;
2157 Break;
2158 end;
2159 if MyRO.Wonder[I].CityID <> MyData.ToldWonders[I].CityID then
2160 begin
2161 if MyRO.Wonder[I].CityID = WonderDestroyed then
2162 with MessgExDlg do
2163 begin { tell about destroyed wonders }
2164 OpenSound := 'WONDER_DESTROYED';
2165 MessgText := Format(Phrases.Lookup('WONDERDEST'),
2166 [Phrases.Lookup('IMPROVEMENTS', I)]);
2167 Kind := mkOkHelp;
2168 HelpKind := hkImp;
2169 HelpNo := I;
2170 IconKind := mikImp;
2171 IconIndex := I;
2172 ShowModal;
2173 end
2174 else
2175 begin
2176 if I = woManhattan then
2177 if MyRO.Wonder[I].EffectiveOwner > Me then
2178 MyData.ColdWarStart := MyRO.Turn - 1
2179 else
2180 MyData.ColdWarStart := MyRO.Turn;
2181 if not OwnWonder then
2182 with MessgExDlg do
2183 begin { tell about newly built wonders }
2184 if I = woManhattan then
2185 begin
2186 OpenSound := 'MSG_COLDWAR';
2187 S := Tribe[MyRO.Wonder[I].EffectiveOwner].TPhrase('COLDWAR');
2188 end
2189 else if MyRO.Wonder[I].EffectiveOwner >= 0 then
2190 begin
2191 OpenSound := 'WONDER_BUILT';
2192 S := Tribe[MyRO.Wonder[I].EffectiveOwner]
2193 .TPhrase('WONDERBUILT');
2194 end
2195 else
2196 begin
2197 OpenSound := 'MSG_DEFAULT';
2198 S := Phrases.Lookup('WONDERBUILTEXP');
2199 // already expired when built
2200 end;
2201 MessgText := Format(S, [Phrases.Lookup('IMPROVEMENTS', I),
2202 CityName(MyRO.Wonder[I].CityID)]);
2203 Kind := mkOkHelp;
2204 HelpKind := hkImp;
2205 HelpNo := I;
2206 IconKind := mikImp;
2207 IconIndex := I;
2208 ShowModal;
2209 end;
2210 end;
2211 end
2212 else if (MyRO.Wonder[I].EffectiveOwner <> MyData.ToldWonders[I]
2213 .EffectiveOwner) and (MyRO.Wonder[I].CityID > WonderDestroyed) then
2214 if MyRO.Wonder[I].EffectiveOwner < 0 then
2215 begin
2216 if I <> woMIR then
2217 with MessgExDlg do
2218 begin { tell about expired wonders }
2219 OpenSound := 'WONDER_EXPIRED';
2220 MessgText := Format(Phrases.Lookup('WONDEREXP'),
2221 [Phrases.Lookup('IMPROVEMENTS', I),
2222 CityName(MyRO.Wonder[I].CityID)]);
2223 Kind := mkOkHelp;
2224 HelpKind := hkImp;
2225 HelpNo := I;
2226 IconKind := mikImp;
2227 IconIndex := I;
2228 ShowModal;
2229 end;
2230 end
2231 else if (MyData.ToldWonders[I].EffectiveOwner >= 0) and not OwnWonder
2232 then
2233 with MessgExDlg do
2234 begin { tell about capture of wonders }
2235 OpenSound := 'WONDER_CAPTURED';
2236 S := Tribe[MyRO.Wonder[I].EffectiveOwner].TPhrase('WONDERCAPT');
2237 MessgText := Format(S, [Phrases.Lookup('IMPROVEMENTS', I),
2238 CityName(MyRO.Wonder[I].CityID)]);
2239 Kind := mkOkHelp;
2240 HelpKind := hkImp;
2241 HelpNo := I;
2242 IconKind := mikImp;
2243 IconIndex := I;
2244 ShowModal;
2245 end;
2246 end;
2247
2248 if MyRO.Turn = MyData.ColdWarStart + ColdWarTurns then
2249 begin
2250 SoundMessageEx(Phrases.Lookup('COLDWAREND'), 'MSG_DEFAULT');
2251 MyData.ColdWarStart := -ColdWarTurns - 1;
2252 end;
2253
2254 TellNewModels;
2255 end; // ClientMode<>cResume
2256 MyData.ToldAlive := MyRO.Alive;
2257 Move(MyRO.Wonder, MyData.ToldWonders, SizeOf(MyData.ToldWonders));
2258
2259 NewGovAvailable := -1;
2260 if ClientMode <> cResume then
2261 begin // tell about new techs
2262 for ad := 0 to nAdv - 1 do
2263 if (MyRO.TestFlags and tfAllTechs = 0) and
2264 ((MyRO.Tech[ad] >= tsApplicable) <> (MyData.ToldTech[ad] >=
2265 tsApplicable)) or (ad in FutureTech) and
2266 (MyRO.Tech[ad] <> MyData.ToldTech[ad]) then
2267 with MessgExDlg do
2268 begin
2269 Item := 'RESEARCH_GENERAL';
2270 if GameMode <> cMovie then
2271 OpenSound := 'NEWADVANCE_' + Char(48 + Age);
2272 Item2 := Phrases.Lookup('ADVANCES', ad);
2273 if ad in FutureTech then
2274 Item2 := Item2 + ' ' + IntToStr(MyRO.Tech[ad]);
2275 MessgText := Format(Phrases.Lookup(Item), [Item2]);
2276 Kind := mkOkHelp;
2277 HelpKind := hkAdv;
2278 HelpNo := ad;
2279 IconKind := mikBook;
2280 IconIndex := -1;
2281 for I := 0 to nAdvBookIcon - 1 do
2282 if AdvBookIcon[I].Adv = ad then
2283 IconIndex := AdvBookIcon[I].Icon;
2284 ShowModal;
2285 MyData.ToldTech[ad] := MyRO.Tech[ad];
2286 for I := gMonarchy to nGov - 1 do
2287 if GovPreq[I] = ad then
2288 NewGovAvailable := I;
2289 end;
2290 end;
2291
2292 ShowCityList := False;
2293 if ClientMode = cTurn then
2294 begin
2295 if (MyRO.Happened and phTech <> 0) and (MyData.FarTech <> adNexus) then
2296 ChooseResearch;
2297
2298 UpdatePanel := False;
2299 if MyRO.Happened and phChangeGov <> 0 then
2300 begin
2301 ModalSelectDlg.ShowNewContent(wmModal, kGov);
2302 Play('NEWGOV');
2303 Server(sSetGovernment, Me, ModalSelectDlg.Result, nil^);
2304 CityOptimizer_BeginOfTurn;
2305 UpdatePanel := True;
2306 end;
2307 end; // ClientMode=cTurn
2308
2309 if not Supervising and ((ClientMode = cTurn) or (ClientMode = cMovieTurn))
2310 then
2311 for cix := 0 to MyRO.nCity - 1 do
2312 with MyCity[cix] do
2313 Status := Status and not csToldBombard;
2314
2315 if ((ClientMode = cTurn) or (ClientMode = cMovieTurn)) and
2316 (MyRO.Government <> gAnarchy) then
2317 begin
2318 // tell what happened in cities
2319 for WondersOnly := True downto False do
2320 for cix := 0 to MyRO.nCity - 1 do
2321 with MyCity[cix] do
2322 if (MyRO.Turn > 0) and (Loc >= 0) and (Flags and chCaptured = 0) and
2323 (WondersOnly = (Flags and chProduction <> 0) and
2324 (Project0 and cpImp <> 0) and (Project0 and cpIndex < nWonder)) then
2325 begin
2326 if WondersOnly then
2327 with MessgExDlg do
2328 begin { tell about newly built wonder }
2329 OpenSound := 'WONDER_BUILT';
2330 S := Tribe[Me].TPhrase('WONDERBUILTOWN');
2331 MessgText :=
2332 Format(S, [Phrases.Lookup('IMPROVEMENTS',
2333 Project0 and cpIndex), CityName(ID)]);
2334 Kind := mkOkHelp;
2335 HelpKind := hkImp;
2336 HelpNo := Project0 and cpIndex;
2337 IconKind := mikImp;
2338 IconIndex := Project0 and cpIndex;
2339 ShowModal;
2340 end;
2341 if not Supervising and (ClientMode = cTurn) then
2342 begin
2343 AllowCityScreen := True;
2344 if (Status and 7 <> 0) and
2345 (Project and (cpImp + cpIndex) = cpImp + imTrGoods) then
2346 if (MyData.ImpOrder[Status and 7 - 1, 0] >= 0) then
2347 begin
2348 if AutoBuild(cix, MyData.ImpOrder[Status and 7 - 1]) then
2349 AllowCityScreen := False
2350 else if Flags and chProduction <> 0 then
2351 Flags := (Flags and not chProduction) or chAllImpsMade
2352 end
2353 else
2354 Flags := Flags or chTypeDel;
2355 if (Size >= NeedAqueductSize) and
2356 (MyRO.Tech[Imp[imAqueduct].Preq] < tsApplicable) or
2357 (Size >= NeedSewerSize) and
2358 (MyRO.Tech[Imp[imSewer].Preq] < tsApplicable) then
2359 Flags := Flags and not chNoGrowthWarning;
2360 // don't remind of unknown building
2361 if Flags and chNoSettlerProd = 0 then
2362 Status := Status and not csToldDelay
2363 else if Status and csToldDelay = 0 then
2364 Status := Status or csToldDelay
2365 else
2366 Flags := Flags and not chNoSettlerProd;
2367 if mRepScreens.Checked then
2368 begin
2369 if (Flags and CityRepMask <> 0) and AllowCityScreen then
2370 begin { show what happened in cities }
2371 SetTroopLoc(MyCity[cix].Loc);
2372 MarkCityLoc := MyCity[cix].Loc;
2373 PanelPaint;
2374 CityDlg.CloseAction := None;
2375 CityDlg.ShowNewContent(wmModal, MyCity[cix].Loc,
2376 Flags and CityRepMask);
2377 UpdatePanel := True;
2378 end;
2379 end
2380 else { if mRepList.Checked then }
2381 begin
2382 if Flags and CityRepMask <> 0 then
2383 ShowCityList := True;
2384 end;
2385 end;
2386 end; { city loop }
2387 end; // ClientMode=cTurn
2388
2389 if ClientMode = cTurn then
2390 begin
2391 if NewGovAvailable >= 0 then
2392 with MessgExDlg do
2393 begin
2394 MessgText := Format(Phrases.Lookup('AUTOREVOLUTION'),
2395 [Phrases.Lookup('GOVERNMENT', NewGovAvailable)]);
2396 Kind := mkYesNo;
2397 IconKind := mikPureIcon;
2398 IconIndex := 6 + NewGovAvailable;
2399 ShowModal;
2400 if ModalResult = mrOK then
2401 begin
2402 Play('REVOLUTION');
2403 Server(sRevolution, Me, 0, nil^);
2404 end;
2405 end;
2406 end; // ClientMode=cTurn
2407
2408 if (ClientMode = cTurn) or (ClientMode = cMovieTurn) then
2409 begin
2410 if MyRO.Happened and phGliderLost <> 0 then
2411 ContextMessage(Phrases.Lookup('GLIDERLOST'), 'MSG_DEFAULT',
2412 hkModel, 200);
2413 if MyRO.Happened and phPlaneLost <> 0 then
2414 ContextMessage(Phrases.Lookup('PLANELOST'), 'MSG_DEFAULT',
2415 hkFeature, mcFuel);
2416 if MyRO.Happened and phPeaceEvacuation <> 0 then
2417 for p1 := 0 to nPl - 1 do
2418 if 1 shl p1 and MyData.PeaceEvaHappened <> 0 then
2419 SoundMessageEx(Tribe[p1].TPhrase('WITHDRAW'), 'MSG_DEFAULT');
2420 if MyRO.Happened and phPeaceViolation <> 0 then
2421 for p1 := 0 to nPl - 1 do
2422 if (1 shl p1 and MyRO.Alive <> 0) and (MyRO.EvaStart[p1] = MyRO.Turn)
2423 then
2424 SoundMessageEx(Format(Tribe[p1].TPhrase('VIOLATION'),
2425 [TurnToString(MyRO.Turn + PeaceEvaTurns - 1)]), 'MSG_WITHDRAW');
2426 TellNewContacts;
2427 end;
2428
2429 if ClientMode = cMovieTurn then
2430 Update
2431 else if ClientMode = cTurn then
2432 begin
2433 if UpdatePanel then
2434 UpdateViews;
2435 Application.ProcessMessages;
2436
2437 if not Supervising then
2438 for uix := 0 to MyRO.nUn - 1 do
2439 with MyUn[uix] do
2440 if Loc >= 0 then
2441 begin
2442 if Flags and unWithdrawn <> 0 then
2443 Status := 0;
2444 if Health = 100 then
2445 Status := Status and not usRecover;
2446 if (Master >= 0) or UnitExhausted(uix) then
2447 Status := Status and not usWaiting
2448 else
2449 Status := Status or usWaiting;
2450 CheckToldNoReturn(uix);
2451 if Status and usGoto <> 0 then
2452 begin { continue multi-turn goto }
2453 SetUnFocus(uix);
2454 SetTroopLoc(Loc);
2455 FocusOnLoc(TroopLoc, flRepaintPanel or flImmUpdate);
2456 if Status shr 16 = $7FFF then
2457 MoveResult := GetMoveAdvice(UnFocus, maNextCity,
2458 MoveAdviceData)
2459 else
2460 MoveResult := GetMoveAdvice(UnFocus, Status shr 16,
2461 MoveAdviceData);
2462 if MoveResult >= rExecuted then
2463 begin // !!! Shinkansen
2464 MoveResult := eOK;
2465 Ok := True;
2466 for I := 0 to MoveAdviceData.nStep - 1 do
2467 begin
2468 Loc1 := dLoc(Loc, MoveAdviceData.dx[I],
2469 MoveAdviceData.dy[I]);
2470 if (MyMap[Loc1] and (fCity or fOwned) = fCity)
2471 // don't capture cities during auto move
2472 or (MyMap[Loc1] and (fUnit or fOwned) = fUnit) then
2473 // don't attack during auto move
2474 begin
2475 Ok := False;
2476 Break;
2477 end
2478 else
2479 begin
2480 if (Loc1 = MoveAdviceData.ToLoc) or
2481 (MoveAdviceData.ToLoc = maNextCity) and
2482 (MyMap[dLoc(Loc, MoveAdviceData.dx[I],
2483 MoveAdviceData.dy[I])] and fCity <> 0) then
2484 MoveOptions := muAutoNoWait
2485 else
2486 MoveOptions := 0;
2487 MoveResult := MoveUnit(MoveAdviceData.dx[I],
2488 MoveAdviceData.dy[I], MoveOptions);
2489 if (MoveResult < rExecuted) or (MoveResult = eEnemySpotted)
2490 then
2491 begin
2492 Ok := False;
2493 Break;
2494 end;
2495 end
2496 end;
2497 Stop := not ok or (Loc = MoveAdviceData.ToLoc) or
2498 (MoveAdviceData.ToLoc = maNextCity) and
2499 (MyMap[Loc] and fCity <> 0);
2500 end
2501 else
2502 begin
2503 MoveResult := eOK;
2504 Stop := True;
2505 end;
2506
2507 if MoveResult <> eDied then
2508 if Stop then
2509 Status := Status and ($FFFF - usGoto)
2510 else
2511 Status := Status and not usWaiting;
2512 end;
2513
2514 if Status and (usEnhance or usGoto) = usEnhance then
2515 // continue terrain enhancement
2516 begin
2517 MoveResult := ProcessEnhancement(uix, MyData.EnhancementJobs);
2518 if MoveResult <> eDied then
2519 if MoveResult = eJobDone then
2520 Status := Status and not usEnhance
2521 else
2522 Status := Status and not usWaiting;
2523 end;
2524 end;
2525 end; // ClientMode=cTurn
2526
2527 HaveStrategyAdvice := False;
2528 GoOnPhase := True;
2529 if Supervising or (GameMode = cMovie) then
2530 begin
2531 SetTroopLoc(-1);
2532 PaintAll;
2533 end { supervisor }
2534 { else if (ClientMode=cTurn) and (MyRO.Turn=0) then
2535 begin
2536 SetUnFocus(0);
2537 ZoomToCity(MyCity[0].Loc)
2538 end }
2539 else
2540 begin
2541 if ClientMode >= scContact then
2542 SetUnFocus(-1)
2543 else
2544 NextUnit(-1, False);
2545 if UnFocus < 0 then
2546 begin
2547 UnStartLoc := -1;
2548 if IsMultiPlayerGame or (ClientMode = cResume) then
2549 if MyRO.nCity > 0 then
2550 FocusOnLoc(MyCity[0].Loc)
2551 else
2552 FocusOnLoc(G.lx * G.ly div 2);
2553 SetTroopLoc(-1);
2554 PanelPaint;
2555 end;
2556 if ShowCityList then
2557 ListDlg.ShowNewContent(wmPersistent, kCityEvents);
2558 end;
2559end;
2560
2561procedure TMainScreen.Client(Command, NewPlayer: Integer; var Data);
2562var
2563 I, J, p1, mix, ToLoc, AnimationSpeed, ShowMoveDomain, cix, ecix: Integer;
2564 Color: TColor;
2565 Name, S: string;
2566 TribeInfo: TTribeInfo;
2567 mi: TModelInfo;
2568 SkipTurn, IsAlpine, IsTreatyDeal: Boolean;
2569begin
2570 case Command of
2571 cTurn, cResume, cContinue, cMovieTurn, scContact, scDipStart .. scDipBreak:
2572 begin
2573 Supervising := G.Difficulty[NewPlayer] = 0;
2574 ArrangeMidPanel;
2575 end
2576 end;
2577 case Command of
2578 cDebugMessage:
2579 LogDlg.Add(NewPlayer, G.RO[0].Turn, PChar(@Data));
2580
2581 cShowNego:
2582 with TShowNegoData(Data) do
2583 begin
2584 S := Format('P%d to P%d: ', [pSender, pTarget]);
2585 if (Action = scDipOffer) and (Offer.nDeliver + Offer.nCost > 0) then
2586 begin
2587 S := S + 'Offer ';
2588 for I := 0 to Offer.nDeliver + Offer.nCost - 1 do
2589 begin
2590 if I = Offer.nDeliver then
2591 S := S + ' for '
2592 else if I > 0 then
2593 S := S + '+';
2594 case Offer.Price[I] and opMask of
2595 opChoose:
2596 S := S + 'Price of choice';
2597 opCivilReport:
2598 S := S + 'State report';
2599 opMilReport:
2600 S := S + 'Military report';
2601 opMap:
2602 S := S + 'Map';
2603 opTreaty:
2604 S := S + 'Treaty';
2605 opShipParts:
2606 S := S + 'Ship part';
2607 opMoney:
2608 S := S + IntToStr(Offer.Price[I] and $FFFFFF) + 'o';
2609 opTribute:
2610 S := S + IntToStr(Offer.Price[I] and $FFFFFF) + 'o tribute';
2611 opTech:
2612 S := S + Phrases.Lookup('ADVANCES', Offer.Price[I] and $FFFFFF);
2613 opAllTech:
2614 S := S + 'All advances';
2615 opModel:
2616 S := S + Tribe[pSender].ModelName[Offer.Price[I] and $FFFFFF];
2617 opAllModel:
2618 S := S + 'All models';
2619 end;
2620 end;
2621 LogDlg.Add(NewPlayer, G.RO[0].Turn, PChar(S));
2622 end
2623 else if Action = scDipAccept then
2624 begin
2625 S := S + '--- ACCEPTED! ---';
2626 LogDlg.Add(NewPlayer, G.RO[0].Turn, PChar(S));
2627 end;
2628 end;
2629
2630 cInitModule:
2631 begin
2632 Server := TInitModuleData(Data).Server;
2633 InitModule;
2634 TInitModuleData(Data).DataSize := SizeOf(TPersistentData);
2635 TInitModuleData(Data).Flags := aiThreaded;
2636 end;
2637
2638 cReleaseModule: DoneModule;
2639
2640 cHelpOnly, cStartHelp, cStartCredits:
2641 begin
2642 Age := 0;
2643 if Command = cHelpOnly then
2644 MainTexture.Age := -1;
2645 Tribes.Init;
2646 HelpDlg.UserLeft := Screen.PrimaryMonitor.Left + (Screen.PrimaryMonitor.Width - HelpDlg.Width) div 2;
2647 HelpDlg.UserTop := Screen.PrimaryMonitor.Top + (Screen.PrimaryMonitor.Height - HelpDlg.Height) div 2;
2648 HelpDlg.Difficulty := 0;
2649 if Command = cStartCredits then
2650 HelpDlg.ShowNewContent(wmModal, hkMisc, Integer(miscCredits))
2651 else
2652 HelpDlg.ShowNewContent(wmModal, hkMisc, Integer(miscMain));
2653 Tribes.Done;
2654 end;
2655
2656 cNewGame, cLoadGame, cMovie, cNewMap:
2657 begin
2658 GenerateNames := mNames.Checked;
2659 GameOK := True;
2660 G := TNewGameData(Data);
2661 Me := -1;
2662 pLogo := -1;
2663 ClientMode := -1;
2664 SetMapOptions;
2665 MainMap.pDebugMap := -1;
2666 Idle := False;
2667 FillChar(Jump, SizeOf(Jump), 0);
2668 if StartRunning then
2669 Jump[0] := 999999;
2670 GameMode := Command;
2671 GrExt.ResetPixUsed;
2672 MainMap.Reset;
2673 NoMap.Reset;
2674 Tribes.Init;
2675 GetTribeList;
2676 for p1 := 0 to nPl - 1 do
2677 if (G.RO[p1] <> nil) and (G.RO[p1].Data <> nil) then
2678 with TPersistentData(G.RO[p1].Data^) do
2679 begin
2680 FarTech := adNone;
2681 FillChar(EnhancementJobs, SizeOf(EnhancementJobs), jNone);
2682 FillChar(ImpOrder, SizeOf(ImpOrder), Byte(-1));
2683 ColdWarStart := -ColdWarTurns - 1;
2684 ToldAge := -1;
2685 ToldModels := 3;
2686 ToldAlive := 0;
2687 ToldContact := 0;
2688 ToldOwnCredibility := InitialCredibility;
2689 for I := 0 to nPl - 1 do
2690 if G.Difficulty[I] > 0 then
2691 Inc(ToldAlive, 1 shl I);
2692 PeaceEvaHappened := 0;
2693 for I := 0 to nWonder - 1 do
2694 with ToldWonders[I] do
2695 begin
2696 CityID := -1;
2697 EffectiveOwner := -1;
2698 end;
2699 FillChar(ToldTech, SizeOf(ToldTech), Byte(tsNA));
2700 if G.Difficulty[p1] > 0 then
2701 SoundPreload([sbStart]);
2702 end;
2703
2704 ArrangeDialogs;
2705
2706 Age := 0;
2707 MovieSpeed := 1;
2708 LogDlg.mSlot.Visible := True;
2709 LogDlg.Host := Self;
2710 if Assigned(FHelpDlg) then FHelpDlg.ClearHistory;
2711 if Assigned(FCityDlg) then FCityDlg.Reset;
2712
2713 MiniMap.Size := Point(G.lx, G.ly);
2714 MiniMap.Bitmap.SetSize(MiniMap.Size.X * 2, MiniMap.Size.Y);
2715
2716 for I := 0 to nPl - 1 do begin
2717 Tribe[I] := nil;
2718 TribeOriginal[I] := False;
2719 end;
2720 ToldSlavery := -1;
2721 RepaintOnResize := False;
2722 Closable := False;
2723 FirstMovieTurn := True;
2724
2725 MenuArea.Visible := GameMode <> cMovie;
2726 TreasuryArea.Visible := GameMode < cMovie;
2727 ResearchArea.Visible := GameMode < cMovie;
2728 ManagementArea.Visible := GameMode < cMovie;
2729 end;
2730
2731 cGetReady, cReplay:
2732 if NewPlayer = 0 then
2733 begin
2734 I := 0;
2735 for p1 := 0 to nPl - 1 do
2736 if (G.Difficulty[p1] > 0) and (Tribe[p1] = nil) then
2737 Inc(I);
2738 if I > UnusedTribeFiles.Count then
2739 begin
2740 GameOK := False;
2741 SimpleMessage(Phrases.Lookup('TOOFEWTRIBES'));
2742 end
2743 else
2744 begin
2745 for p1 := 0 to nPl - 1 do
2746 if (G.Difficulty[p1] > 0) and (Tribe[p1] = nil) and (G.RO[p1] <> nil)
2747 then
2748 begin // let player select own tribes
2749 TribeInfo.trix := p1;
2750 TribeNames.Clear;
2751 for J := 0 to UnusedTribeFiles.Count - 1 do
2752 begin
2753 GetTribeInfo(UnusedTribeFiles[J], Name, Color);
2754 TribeNames.AddObject(Name, TObject(Color));
2755 end;
2756 Assert(TribeNames.Count > 0);
2757 ModalSelectDlg.ShowNewContent(wmModal, kTribe);
2758 Application.ProcessMessages;
2759 TribeInfo.FileName := UnusedTribeFiles[ModalSelectDlg.Result];
2760 UnusedTribeFiles.Delete(ModalSelectDlg.Result);
2761
2762 if GameMode = cLoadGame then
2763 CreateTribe(TribeInfo.trix, TribeInfo.FileName, False)
2764 else
2765 Server(CommandWithData(cSetTribe, TribeInfo.GetCommandDataSize),
2766 0, 0, TribeInfo);
2767 end;
2768
2769 for p1 := 0 to nPl - 1 do
2770 if (G.Difficulty[p1] > 0) and (Tribe[p1] = nil) and (G.RO[p1] = nil)
2771 then
2772 begin // autoselect enemy tribes
2773 J := ChooseUnusedTribe;
2774 TribeInfo.FileName := UnusedTribeFiles[J];
2775 UnusedTribeFiles.Delete(J);
2776 TribeInfo.trix := p1;
2777 if GameMode = cLoadGame then
2778 CreateTribe(TribeInfo.trix, TribeInfo.FileName, False)
2779 else
2780 Server(CommandWithData(cSetTribe, TribeInfo.GetCommandDataSize),
2781 0, 0, TribeInfo);
2782 end;
2783 end;
2784 if not mNames.Checked then
2785 for p1 := 0 to nPl - 1 do
2786 if Tribe[p1] <> nil then
2787 Tribe[p1].NumberName := p1;
2788 end;
2789
2790 cBreakGame:
2791 begin
2792 SaveSettings;
2793 if Assigned(FCityDlg) then CityDlg.CloseAction := None;
2794 ApplyToVisibleForms(faClose);
2795 if LogDlg.Visible then
2796 LogDlg.Close;
2797 LogDlg.List.Clear;
2798 StartRunning := not Idle and (Jump[0] > 0); // AI called Reload
2799 Me := -1;
2800 Idle := False;
2801 ClientMode := -1;
2802 UnitInfoBtn.Visible := False;
2803 UnitBtn.Visible := False;
2804 TerrainBtn.Visible := False;
2805 MovieSpeed1Btn.Visible := False;
2806 MovieSpeed2Btn.Visible := False;
2807 MovieSpeed3Btn.Visible := False;
2808 MovieSpeed4Btn.Visible := False;
2809 EOT.Visible := False;
2810 for I := 0 to ControlCount - 1 do
2811 if Controls[I] is TButtonC then
2812 Controls[I].Visible := False;
2813 VerticalScrollBar.Init(0, 1);
2814 for p1 := 0 to nPl - 1 do
2815 if Tribe[p1] <> nil then
2816 FreeAndNil(Tribe[p1]);
2817 Tribes.Done;
2818 RepaintOnResize := False;
2819 Closable := True;
2820 Close;
2821 end;
2822
2823 cShowGame:
2824 begin
2825 with Panel.Canvas do
2826 begin
2827 Brush.Color := $000000;
2828 FillRect(Rect(0, 0, Panel.Width, Panel.Height));
2829 Brush.Style := TBrushStyle.bsClear;
2830 end;
2831 with TopBar.Canvas do
2832 begin
2833 Brush.Color := $000000;
2834 FillRect(Rect(0, 0, TopBar.Width, TopBar.Height));
2835 Brush.Style := TBrushStyle.bsClear;
2836 end;
2837 Show;
2838 Update;
2839 ResizeControls;
2840 RepaintOnResize := True;
2841 xw := 0;
2842 yw := ywcenter;
2843 end;
2844
2845 cShowTurnChange:
2846 begin
2847 if Integer(Data) >= 0 then
2848 begin
2849 pLogo := Integer(Data);
2850 if G.RO[pLogo] = nil then
2851 begin
2852 if AILogo[pLogo] <> nil then
2853 BitBltCanvas(Canvas, (xRightPanel + 10) - (16 + 64),
2854 ClientHeight - PanelHeight, 64, 64, AILogo[pLogo].Canvas,
2855 0, 0);
2856 end;
2857 end;
2858 end;
2859
2860 cTurn, cResume, cContinue:
2861 if not GameOK then
2862 Server(sResign, NewPlayer, 0, nil^)
2863 else
2864 begin
2865 ClientMode := Command;
2866 pTurn := NewPlayer;
2867 pLogo := NewPlayer;
2868
2869 if Command = cResume then
2870 begin // init non-original model pictures (maybe tribes not found)
2871 for p1 := 0 to nPl - 1 do
2872 if G.RO[p1] <> nil then
2873 begin
2874 ItsMeAgain(p1);
2875 for mix := 0 to MyRO.nModel - 1 do
2876 if not Assigned(Tribe[Me].ModelPicture[mix].HGr) then
2877 InitMyModel(mix, True);
2878 end;
2879 Me := -1;
2880 end;
2881
2882 if Jump[pTurn] > 0 then
2883 Application.ProcessMessages;
2884 if Jump[pTurn] > 0 then
2885 if G.RO[NewPlayer].Happened and phGameEnd <> 0 then
2886 Jump[pTurn] := 0
2887 else
2888 Dec(Jump[pTurn]);
2889 SkipTurn := Jump[pTurn] > 0;
2890 if SkipTurn then
2891 begin
2892 ItsMeAgain(NewPlayer);
2893 MyData := G.RO[NewPlayer].Data;
2894 SetTroopLoc(-1);
2895 MiniMapPaint;
2896 InitAllEnemyModels; // necessary for correct replay
2897 if not EndTurn(True) then
2898 SkipTurn := False;
2899 end;
2900 if not SkipTurn then
2901 begin
2902 if ((ClientMode < scDipStart) or (ClientMode > scDipBreak)) and
2903 Assigned(FNegoDlg) and NegoDlg.Visible then
2904 NegoDlg.Close;
2905 Skipped := False; // always show my moves during my turn
2906 Idle := True;
2907 InitTurn(NewPlayer);
2908 DipMem[Me].pContact := -1;
2909 (* if (Me=0) and (MyRO.Alive and (1 shl Me)=0)} then
2910 begin
2911 if SimpleQuery(Phrases.Lookup('RESIGN'))=mrIgnore then
2912 Server(sResign,Me,0,nil^)
2913 else Server(sBreak,Me,0,nil^)
2914 end
2915 else Play('TURNSTART'); *)
2916 end;
2917 end;
2918
2919 cMovieTurn:
2920 begin
2921 ClientMode := Command;
2922 pTurn := NewPlayer;
2923 pLogo := -1;
2924 Skipped := False; // always show my moves during my turn
2925 Idle := True;
2926 if FirstMovieTurn then
2927 begin
2928 CheckMovieSpeedBtnState;
2929 FirstMovieTurn := False;
2930 end;
2931 InitTurn(NewPlayer);
2932 Application.ProcessMessages;
2933 if MovieSpeed = 4 then
2934 begin
2935 Sleep(75);
2936 // this break will ensure speed of fast forward does not depend on cpu speed
2937 Application.ProcessMessages;
2938 end;
2939 end;
2940
2941 cMovieEndTurn:
2942 begin
2943 RememberPeaceViolation;
2944 pTurn := -1;
2945 pLogo := -1;
2946 MapValid := False;
2947 ClientMode := -1;
2948 Idle := False;
2949 Skipped := False;
2950 end;
2951
2952 cEditMap:
2953 begin
2954 ClientMode := cEditMap;
2955 SetMapOptions;
2956 MainMap.pDebugMap := -1;
2957 ItsMeAgain(0);
2958 MyData := nil;
2959 UnitInfoBtn.Visible := False;
2960 UnitBtn.Visible := False;
2961 TerrainBtn.Visible := False;
2962 MovieSpeed1Btn.Visible := False;
2963 MovieSpeed2Btn.Visible := False;
2964 MovieSpeed3Btn.Visible := False;
2965 MovieSpeed4Btn.Visible := False;
2966 EOT.Visible := False;
2967 HelpDlg.Difficulty := 0;
2968 BrushType := fGrass;
2969 UpdateInterface;
2970 BrushLoc := -1;
2971 Edited := False;
2972 UnFocus := -1;
2973 MarkCityLoc := -1;
2974 Tracking := False;
2975 TurnComplete := False;
2976 MapValid := False;
2977 ResizeControls;
2978 SetTroopLoc(-1);
2979 Idle := True;
2980 end;
2981
2982 (* cNewContact:
2983 begin
2984 end;
2985 *)
2986
2987 scContact:
2988 begin
2989 DipMem[NewPlayer].pContact := Integer(Data);
2990 if Jump[NewPlayer] > 0 then
2991 DipCall(scReject)
2992 else
2993 begin
2994 ClientMode := Command;
2995 InitTurn(NewPlayer);
2996 MyData.ToldContact := MyData.ToldContact or (1 shl Integer(Data));
2997 // don't tell about new nation when already contacted by them
2998 with MessgExDlg do
2999 begin
3000 OpenSound := 'CONTACT_' + Char(48 + MyRO.EnemyReport[Integer(Data)
3001 ].Attitude);
3002 MessgText := Tribe[Integer(Data)].TPhrase('FRCONTACT');
3003 Kind := mkYesNo;
3004 IconKind := mikTribe;
3005 IconIndex := Integer(Data);
3006 ShowModal;
3007 if ModalResult = mrOK then
3008 begin
3009 NegoDlg.Respond;
3010 DipMem[Me].DeliveredPrices := [];
3011 DipMem[Me].ReceivedPrices := [];
3012 DipCall(scDipStart);
3013 end
3014 else
3015 begin
3016 DipCall(scReject);
3017 EndNego;
3018 end;
3019 end;
3020 end;
3021 end;
3022
3023 scDipStart .. scDipBreak:
3024 begin
3025 ClientMode := Command;
3026 InitTurn(NewPlayer);
3027 if Command = scDipStart then
3028 Play('CONTACT_' + Char(48 + MyRO.Attitude[DipMem[NewPlayer]
3029 .pContact]))
3030 else if Command = scDipCancelTreaty then
3031 Play('CANCELTREATY')
3032 else if Command = scDipOffer then
3033 begin
3034 ReceivedOffer := TOffer(Data);
3035 InitAllEnemyModels;
3036 end
3037 else if Command = scDipAccept then
3038 begin // remember delivered and received prices
3039 for I := 0 to DipMem[Me].SentOffer.nDeliver - 1 do
3040 Include(DipMem[Me].DeliveredPrices,
3041 DipMem[Me].SentOffer.Price[I] shr 24);
3042 for I := 0 to DipMem[Me].SentOffer.nCost - 1 do
3043 Include(DipMem[Me].ReceivedPrices,
3044 DipMem[Me].SentOffer.Price[DipMem[Me].SentOffer.nDeliver +
3045 I] shr 24);
3046 IsTreatyDeal := False;
3047 for I := 0 to ReceivedOffer.nDeliver + ReceivedOffer.nCost - 1 do
3048 if DipMem[Me].SentOffer.Price[I] and opMask = opTreaty then begin
3049 IsTreatyDeal := True;
3050 Break;
3051 end;
3052 if IsTreatyDeal then
3053 Play('NEWTREATY')
3054 else
3055 Play('ACCEPTOFFER');
3056 end;
3057 NegoDlg.Start;
3058 Idle := True;
3059 end;
3060
3061 cShowCancelTreaty:
3062 if not IsMultiPlayerGame then
3063 begin
3064 case G.RO[NewPlayer].Treaty[Integer(Data)] of
3065 trPeace:
3066 S := Tribe[Integer(Data)].TPhrase('FRCANCELBYREJECT_PEACE');
3067 trFriendlyContact:
3068 S := Tribe[Integer(Data)].TPhrase('FRCANCELBYREJECT_FRIENDLY');
3069 trAlliance:
3070 S := Tribe[Integer(Data)].TPhrase('FRCANCELBYREJECT_ALLIANCE');
3071 end;
3072 TribeMessage(Integer(Data), S, 'CANCELTREATY');
3073 end;
3074
3075 cShowCancelTreatyByAlliance:
3076 if Idle and (NewPlayer = Me) then
3077 TribeMessage(Integer(Data), Tribe[Integer(Data)
3078 ].TPhrase('FRENEMYALLIANCE'), 'CANCELTREATY');
3079
3080 cShowSupportAllianceAgainst:
3081 if not IsMultiPlayerGame and (Jump[0] = 0) then
3082 TribeMessage(Integer(Data) and $F, Tribe[Integer(Data) and $F]
3083 .TPhrase('FRMYALLIANCE1') + ' ' + Tribe[Integer(Data) shr 4]
3084 .TPhrase('FRMYALLIANCE2'), 'CANCELTREATY');
3085
3086 cShowPeaceViolation:
3087 if not IsMultiPlayerGame and (Jump[0] = 0) then
3088 TribeMessage(Integer(Data),
3089 Format(Tribe[Integer(Data)].TPhrase('EVIOLATION'),
3090 [TurnToString(MyRO.Turn + PeaceEvaTurns - 1)]), 'MSG_WITHDRAW');
3091
3092 cShowEndContact:
3093 EndNego;
3094
3095 cShowUnitChanged, cShowCityChanged, cShowAfterMove, cShowAfterAttack:
3096 if (Idle and (NewPlayer = Me) or not Idle and not skipped) and
3097 not ((GameMode = cMovie) and (MovieSpeed = 4)) then
3098 begin
3099 Assert(NewPlayer = Me);
3100 if not Idle or (GameMode = cMovie) then
3101 Application.ProcessMessages;
3102 if Command = cShowCityChanged then
3103 begin
3104 CurrentMoveInfo.DoShow := False;
3105 if Idle then
3106 CurrentMoveInfo.DoShow := True
3107 else if CurrentMoveInfo.IsAlly then
3108 CurrentMoveInfo.DoShow := not mAlNoMoves.Checked
3109 else
3110 CurrentMoveInfo.DoShow := not mEnNoMoves.Checked;
3111 end
3112 else if Command = cShowUnitChanged then
3113 begin
3114 CurrentMoveInfo.DoShow := False;
3115 if Idle then
3116 CurrentMoveInfo.DoShow := not mEffectiveMovesOnly.Checked
3117 else if CurrentMoveInfo.IsAlly then
3118 CurrentMoveInfo.DoShow :=
3119 not (mAlNoMoves.Checked or mAlEffectiveMovesOnly.Checked)
3120 else
3121 CurrentMoveInfo.DoShow :=
3122 not (mEnNoMoves.Checked or mEnAttacks.Checked);
3123 end;
3124 // else keep DoShow from cShowMove/cShowAttack
3125
3126 if CurrentMoveInfo.DoShow then
3127 begin
3128 if Command = cShowCityChanged then
3129 MapValid := False;
3130 FocusOnLoc(Integer(Data), flImmUpdate);
3131 // OldUnFocus:=UnFocus;
3132 // UnFocus:=-1;
3133 if Command = cShowAfterMove then
3134 PaintLoc(Integer(Data), CurrentMoveInfo.AfterMovePaintRadius)
3135 // show discovered areas
3136 else
3137 PaintLoc(Integer(Data), 1);
3138 // UnFocus:=OldUnFocus;
3139 if (Command = cShowAfterAttack) and
3140 (CurrentMoveInfo.AfterAttackExpeller >= 0) then
3141 begin
3142 SoundMessageEx(Tribe[CurrentMoveInfo.AfterAttackExpeller]
3143 .TPhrase('EXPEL'), '');
3144 CurrentMoveInfo.AfterAttackExpeller := -1;
3145 Update; // remove message box from screen
3146 end
3147 else if not Idle then
3148 if Command = cShowCityChanged then
3149 Sleep(MoveTime * WaitAfterShowMove div 16)
3150 else if (Command = cShowUnitChanged) and
3151 (MyMap[Integer(Data)] and fUnit <> 0) then
3152 Sleep(MoveTime * WaitAfterShowMove div 32)
3153 end // if CurrentMoveInfo.DoShow
3154 else
3155 MapValid := False;
3156 end;
3157
3158 cShowMoving, cShowCapturing:
3159 if (Idle and (NewPlayer = Me) or not Idle and not skipped and
3160 (TShowMove(Data).emix <> $FFFF)) and
3161 not ((GameMode = cMovie) and (MovieSpeed = 4)) then
3162 begin
3163 Assert(NewPlayer = Me);
3164 if not Idle or (GameMode = cMovie) then
3165 Application.ProcessMessages;
3166 with TShowMove(Data) do
3167 begin
3168 CurrentMoveInfo.DoShow := False;
3169 if not Idle and (not Assigned(Tribe[Owner].ModelPicture[mix].HGr)) then
3170 InitEnemyModel(emix);
3171
3172 ToLoc := dLoc(FromLoc, dx, dy);
3173 if Idle then
3174 begin // own unit -- make discovered land visible
3175 Assert(Owner = Me); // no foreign moves during my turn!
3176 CurrentMoveInfo.DoShow := not mEffectiveMovesOnly.Checked or
3177 (Command = cShowCapturing);
3178 if CurrentMoveInfo.DoShow then
3179 begin
3180 if GameMode = cMovie then
3181 begin
3182 if MovieSpeed = 3 then
3183 AnimationSpeed := 4
3184 else if MovieSpeed = 2 then
3185 AnimationSpeed := 8
3186 else
3187 AnimationSpeed := 16;
3188 end
3189 else
3190 begin
3191 if mVeryFastMoves.Checked then
3192 AnimationSpeed := 4
3193 else if mFastMoves.Checked then
3194 AnimationSpeed := 8
3195 else
3196 AnimationSpeed := 16;
3197 end;
3198 with MyModel[mix] do
3199 begin
3200 if (Kind = mkDiplomat) or (Domain = dAir) or
3201 (Cap[mcRadar] + Cap[mcCarrier] + Cap[mcAcademy] > 0) or
3202 (MyMap[ToLoc] and fTerrain = fMountains) or
3203 (MyMap[ToLoc] and fTerImp = tiFort) or
3204 (MyMap[ToLoc] and fTerImp = tiBase) then
3205 CurrentMoveInfo.AfterMovePaintRadius := 2
3206 else
3207 CurrentMoveInfo.AfterMovePaintRadius := 1;
3208 if (MyRO.Wonder[woShinkansen].EffectiveOwner = Me) and
3209 (Domain = dGround) and
3210 (MyMap[FromLoc] and (fRR or fCity) <> 0) and
3211 (MyMap[ToLoc] and (fRR or fCity) <> 0) and
3212 (Flags and umPlaneUnloading = 0) then
3213 AnimationSpeed := 4;
3214 ShowMoveDomain := Domain;
3215 IsAlpine := Cap[mcAlpine] > 0;
3216 end;
3217 end;
3218 end
3219 else
3220 begin
3221 CurrentMoveInfo.IsAlly := MyRO.Treaty[Owner] = trAlliance;
3222 if GameMode = cMovie then
3223 CurrentMoveInfo.DoShow := True
3224 else if CurrentMoveInfo.IsAlly then
3225 CurrentMoveInfo.DoShow := not mAlNoMoves.Checked and
3226 not (mAlEffectiveMovesOnly.Checked and
3227 (Command <> cShowCapturing))
3228 else
3229 CurrentMoveInfo.DoShow := not mEnNoMoves.Checked and
3230 not (mEnAttacks.Checked and (Command <> cShowCapturing));
3231 if CurrentMoveInfo.DoShow then
3232 begin
3233 if Command = cShowCapturing then
3234 begin // show capture message
3235 if MyMap[ToLoc] and fOwned <> 0 then
3236 begin // own city, search
3237 cix := MyRO.nCity - 1;
3238 while (cix >= 0) and (MyCity[cix].Loc <> ToLoc) do
3239 Dec(cix);
3240 S := CityName(MyCity[cix].ID);
3241 end
3242 else
3243 begin // foreign city, search
3244 ecix := MyRO.nEnemyCity - 1;
3245 while (ecix >= 0) and (MyRO.EnemyCity[ecix].Loc <> ToLoc) do
3246 Dec(ecix);
3247 S := CityName(MyRO.EnemyCity[ecix].ID);
3248 end;
3249 TribeMessage(Owner, Format(Tribe[Owner].TPhrase('CAPTURE'),
3250 [S]), '');
3251 Update; // remove message box from screen
3252 end;
3253
3254 if CurrentMoveInfo.IsAlly then
3255 begin // allied unit -- make discovered land visible
3256 if mAlFastMoves.Checked then
3257 AnimationSpeed := 8
3258 else
3259 AnimationSpeed := 16;
3260 with MyRO.EnemyModel[emix] do
3261 if (Kind = mkDiplomat) or (Domain = dAir) or (ATrans_Fuel > 0)
3262 or (Cap and (1 shl (mcRadar - mcFirstNonCap) or
3263 1 shl (mcAcademy - mcFirstNonCap)) <> 0) or
3264 (MyMap[ToLoc] and fTerrain = fMountains) or
3265 (MyMap[ToLoc] and fTerImp = tiFort) or
3266 (MyMap[ToLoc] and fTerImp = tiBase) then
3267 CurrentMoveInfo.AfterMovePaintRadius := 2
3268 else
3269 CurrentMoveInfo.AfterMovePaintRadius := 1;
3270 end
3271 else
3272 begin
3273 if mEnFastMoves.Checked then
3274 AnimationSpeed := 8
3275 else
3276 AnimationSpeed := 16;
3277 CurrentMoveInfo.AfterMovePaintRadius := 0;
3278 // enemy unit, nothing discovered
3279 end;
3280 if GameMode = cMovie then
3281 begin
3282 if MovieSpeed = 3 then
3283 AnimationSpeed := 4
3284 else if MovieSpeed = 2 then
3285 AnimationSpeed := 8
3286 else
3287 AnimationSpeed := 16;
3288 end;
3289 ShowMoveDomain := MyRO.EnemyModel[emix].Domain;
3290 IsAlpine := MyRO.EnemyModel[emix].Cap and
3291 (1 shl (mcAlpine - mcFirstNonCap)) <> 0;
3292 end
3293 end;
3294
3295 if CurrentMoveInfo.DoShow then
3296 begin
3297 if Command = cShowCapturing then
3298 Play('MOVE_CAPTURE')
3299 else if EndHealth <= 0 then
3300 Play('MOVE_DIE')
3301 else if Flags and umSpyMission <> 0 then
3302 Play('MOVE_COVERT')
3303 else if Flags and umShipLoading <> 0 then
3304 if ShowMoveDomain = dAir then
3305 Play('MOVE_PLANELANDING')
3306 else
3307 Play('MOVE_LOAD')
3308 else if Flags and umPlaneLoading <> 0 then
3309 Play('MOVE_LOAD')
3310 else if Flags and umShipUnloading <> 0 then
3311 if ShowMoveDomain = dAir then
3312 Play('MOVE_PLANESTART')
3313 else
3314 Play('MOVE_UNLOAD')
3315 else if Flags and umPlaneUnloading <> 0 then
3316 if (MyMap[FromLoc] and fCity = 0) and
3317 (MyMap[FromLoc] and fTerImp <> tiBase) then
3318 Play('MOVE_PARACHUTE')
3319 else
3320 Play('MOVE_UNLOAD')
3321 else if (ShowMoveDomain = dGround) and not IsAlpine and
3322 (MyMap[ToLoc] and fTerrain = fMountains) and
3323 ((MyMap[FromLoc] and (fRoad or fRR or fCity) = 0) or
3324 (MyMap[ToLoc] and (fRoad or fRR or fCity) = 0)) then
3325 Play('MOVE_MOUNTAIN');
3326
3327 FocusOnLoc(FromLoc, flImmUpdate);
3328 PaintLoc_BeforeMove(FromLoc);
3329 if Command = cShowCapturing then
3330 MoveOnScreen(TShowMove(Data), 1, 32, 32)
3331 else
3332 MoveOnScreen(TShowMove(Data), 1, AnimationSpeed, AnimationSpeed)
3333 end // if CurrentMoveInfo.DoShow
3334 else
3335 MapValid := False;
3336 end;
3337 end;
3338
3339 cShowAttacking:
3340 if (Idle and (NewPlayer = Me) or not Idle and not skipped and
3341 (TShowMove(Data).emix <> $FFFF)) and
3342 not ((GameMode = cMovie) and (MovieSpeed = 4)) then
3343 begin
3344 Assert(NewPlayer = Me);
3345 if not Idle or (GameMode = cMovie) then
3346 Application.ProcessMessages;
3347 with TShowMove(Data) do
3348 begin
3349 CurrentMoveInfo.AfterAttackExpeller := -1;
3350 CurrentMoveInfo.DoShow := False;
3351 if Idle then
3352 CurrentMoveInfo.DoShow := True // own unit -- always show attacks
3353 else
3354 begin
3355 CurrentMoveInfo.IsAlly := MyRO.Treaty[Owner] = trAlliance;
3356 if CurrentMoveInfo.IsAlly then
3357 CurrentMoveInfo.DoShow := not mAlNoMoves.Checked
3358 else
3359 CurrentMoveInfo.DoShow := not mEnNoMoves.Checked;
3360 end;
3361 if CurrentMoveInfo.DoShow then
3362 begin
3363 ToLoc := dLoc(FromLoc, dx, dy);
3364 if not Assigned(Tribe[Owner].ModelPicture[mix].HGr) then
3365 InitEnemyModel(emix);
3366
3367 if (MyMap[ToLoc] and (fCity or fUnit or fOwned) = fCity or fOwned)
3368 then
3369 begin // tell about bombardment
3370 cix := MyRO.nCity - 1;
3371 while (cix >= 0) and (MyCity[cix].Loc <> ToLoc) do
3372 Dec(cix);
3373 if MyCity[cix].Status and csToldBombard = 0 then
3374 begin
3375 if not Supervising then
3376 MyCity[cix].Status := MyCity[cix].Status or csToldBombard;
3377 S := CityName(MyCity[cix].ID);
3378 SoundMessageEx(Format(Tribe[Owner].TPhrase('BOMBARD'),
3379 [S]), '');
3380 Update; // remove message box from screen
3381 end;
3382 end
3383 else if Flags and umExpelling <> 0 then
3384 CurrentMoveInfo.AfterAttackExpeller := Owner;
3385
3386 if Flags and umExpelling <> 0 then
3387 Play('MOVE_EXPEL')
3388 else if Owner = Me then
3389 begin
3390 MakeModelInfo(Me, mix, MyModel[mix], mi);
3391 Play(AttackSound(ModelCode(mi)));
3392 end
3393 else
3394 Play(AttackSound(ModelCode(MyRO.EnemyModel[emix])));
3395
3396 FocusOnLoc(FromLoc, flImmUpdate);
3397
3398 // before combat
3399 MainMap.AttackBegin(TShowMove(Data));
3400 if MyMap[ToLoc] and fCity <> 0 then
3401 PaintLoc(ToLoc);
3402 PaintLoc(FromLoc);
3403 MoveOnScreen(TShowMove(Data), 1, 9, 16);
3404 MoveOnScreen(TShowMove(Data), 17, 12, 32);
3405 MoveOnScreen(TShowMove(Data), 7, 11, 16);
3406
3407 // after combat
3408 MainMap.AttackEffect(TShowMove(Data));
3409 PaintLoc(ToLoc);
3410 if EndHealth > 0 then
3411 begin
3412 Health := EndHealth;
3413 MoveOnScreen(TShowMove(Data), 10, 0, 16);
3414 end
3415 else if not Idle then
3416 Sleep(MoveTime div 2);
3417 MainMap.AttackEnd;
3418 end // if CurrentMoveInfo.DoShow
3419 else
3420 MapValid := False;
3421 end;
3422 end;
3423
3424 cShowMissionResult:
3425 if Cardinal(Data) = 0 then
3426 SoundMessageEx(Phrases.Lookup('NOFOREIGNINFO'), '')
3427 else
3428 begin
3429 S := Phrases.Lookup('FOREIGNINFO');
3430 for p1 := 0 to nPl - 1 do
3431 if 3 shl (p1 * 2) and Cardinal(Data) <> 0 then
3432 S := S + '\' + Tribe[p1].TPhrase('SHORTNAME');
3433 SoundMessageEx(S, '');
3434 end;
3435
3436 cShowShipChange:
3437 if not IsMultiPlayerGame and (Jump[0] = 0) then
3438 ShowEnemyShipChange(TShowShipChange(Data));
3439
3440 cShowGreatLibTech:
3441 if not IsMultiPlayerGame and (Jump[0] = 0) then
3442 with MessgExDlg do
3443 begin
3444 MessgText := Format(Phrases.Lookup('GRLIB_GENERAL'),
3445 [Phrases.Lookup('ADVANCES', Integer(Data))]);
3446 OpenSound := 'NEWADVANCE_GRLIB';
3447 Kind := mkOk;
3448 IconKind := mikImp;
3449 IconIndex := woGrLibrary;
3450 ShowModal;
3451 end;
3452
3453 cRefreshDebugMap:
3454 begin
3455 if Integer(Data) = MainMap.pDebugMap then
3456 begin
3457 MapValid := False;
3458 MainOffscreenPaint;
3459 Update;
3460 end;
3461 end;
3462
3463 else
3464 if Command >= cClientEx then
3465 case Command and (not Integer(CommandDataElementCountMask)) of
3466 cSetTribe:
3467 with TTribeInfo(Data) do begin
3468 I := UnusedTribeFiles.Count - 1;
3469 while (I >= 0) and
3470 (AnsiCompareFileName(UnusedTribeFiles[I], FileName) <> 0) do
3471 Dec(I);
3472 if I >= 0 then
3473 UnusedTribeFiles.Delete(I);
3474 CreateTribe(trix, FileName, True);
3475 end;
3476 cSetNewModelPicture:
3477 if TribeOriginal[TModelPictureInfo(Data).trix] then
3478 Tribe[TModelPictureInfo(Data).trix].SetModelPicture
3479 (TModelPictureInfo(Data), True);
3480 cSetModelPicture:
3481 if TribeOriginal[TModelPictureInfo(Data).trix] then
3482 Tribe[TModelPictureInfo(Data).trix].SetModelPicture
3483 (TModelPictureInfo(Data), False);
3484 cSetSlaveIndex:
3485 Tribe[Integer(Data) shr 16].mixSlaves := Integer(Data) and $FFFF;
3486 cSetCityName:
3487 with TCityNameInfo(Data) do
3488 if TribeOriginal[ID shr 12] then
3489 Tribe[ID shr 12].SetCityName(ID and $FFF, NewName);
3490 cSetModelName:
3491 with TModelNameInfo(Data) do
3492 if TribeOriginal[NewPlayer] then
3493 Tribe[NewPlayer].ModelName[mix] := NewName;
3494 end;
3495 end;
3496end;
3497
3498{ *** main part *** }
3499
3500procedure TMainScreen.FormCreate(Sender: TObject);
3501var
3502 I, J: Integer;
3503begin
3504 TroopLoc := -1;
3505 NoMap := TIsoMap.Create;
3506 MainMap := TIsoMap.Create;
3507 NoMapPanel := TIsoMap.Create;
3508
3509 UpdateKeyShortcuts;
3510
3511 MainFormKeyDown := FormKeyDown;
3512 BaseWin.CreateOffscreen(Offscreen, Width, Height);
3513
3514 // define which menu settings to save
3515 SetLength(SaveOption, 22);
3516 SaveOption[0] := mAlEffectiveMovesOnly.Tag;
3517 SaveOption[1] := mEnMoves.Tag;
3518 SaveOption[2] := mEnAttacks.Tag;
3519 SaveOption[3] := mEnNoMoves.Tag;
3520 SaveOption[4] := mWaitTurn.Tag;
3521 SaveOption[5] := mEffectiveMovesOnly.Tag;
3522 SaveOption[6] := mEnFastMoves.Tag;
3523 SaveOption[7] := mSlowMoves.Tag;
3524 SaveOption[8] := mFastMoves.Tag;
3525 SaveOption[9] := mVeryFastMoves.Tag;
3526 SaveOption[10] := mNames.Tag;
3527 SaveOption[11] := mRepList.Tag;
3528 SaveOption[12] := mRepScreens.Tag;
3529 SaveOption[13] := mSoundOff.Tag;
3530 SaveOption[14] := mSoundOn.Tag;
3531 SaveOption[15] := mSoundOnAlt.Tag;
3532 SaveOption[16] := mScrollSlow.Tag;
3533 SaveOption[17] := mScrollFast.Tag;
3534 SaveOption[18] := mScrollOff.Tag;
3535 SaveOption[19] := mAlSlowMoves.Tag;
3536 SaveOption[20] := mAlFastMoves.Tag;
3537 SaveOption[21] := mAlNoMoves.Tag;
3538
3539 LoadSettings;
3540
3541 Screen.Cursors[crImpDrag] := LoadCursor(HInstance, 'DRAG');
3542 Screen.Cursors[crFlatHand] := LoadCursor(HInstance, 'FLATHAND');
3543
3544 // tag-controlled language
3545 for I := 0 to ComponentCount - 1 do
3546 if Components[I].Tag and $FF <> 0 then
3547 if Components[I] is TMenuItem then begin
3548 TMenuItem(Components[I]).Caption := Phrases.Lookup('CONTROLS',
3549 -1 + Components[I].Tag and $FF);
3550 for J := 0 to Length(SaveOption) - 1 do
3551 if Components[I].Tag and $FF = SaveOption[J] then
3552 TMenuItem(Components[I]).Checked := TSaveOption(J) in OptionChecked;
3553 end else
3554 if Components[I] is TButtonBase then begin
3555 TButtonBase(Components[I]).Hint := Phrases.Lookup('CONTROLS',
3556 -1 + Components[I].Tag and $FF);
3557 if (Components[I] is TButtonC) and
3558 (TButtonC(Components[I]).ButtonIndex <> 1) then
3559 TButtonC(Components[I]).ButtonIndex :=
3560 Integer(MapOptionChecked) shr (Components[I].Tag shr 8) and 1 + 2;
3561 end;
3562
3563 // non-tag-controlled language
3564 mTechTree.Caption := Phrases2.Lookup('MENU_ADVTREE');
3565 mViewpoint.Caption := Phrases2.Lookup('MENU_VIEWPOINT');
3566 if not Phrases2FallenBackToEnglish then
3567 begin
3568 MenuArea.Hint := Phrases2.Lookup('BTN_MENU');
3569 TreasuryArea.Hint := Phrases2.Lookup('TIP_TREASURY');
3570 ResearchArea.Hint := Phrases.Lookup('SCIENCE');
3571 ManagementArea.Hint := Phrases2.Lookup('BTN_MANAGE');
3572 end;
3573 for I := 0 to mRep.Count - 1 do
3574 begin
3575 J := mRep[I].Tag shr 8;
3576 mRep[I].Caption := CityEventName(J);
3577 mRep[I].Checked := CityRepMask and (1 shl J) <> 0;
3578 end;
3579
3580 MiniMap := TMiniMap.Create;
3581 Panel := TBitmap.Create;
3582 Panel.PixelFormat := TPixelFormat.pf24bit;
3583 Panel.Canvas.Font.Assign(UniFont[ftSmall]);
3584 Panel.Canvas.Brush.Style := TBrushStyle.bsClear;
3585 TopBar := TBitmap.Create;
3586 TopBar.PixelFormat := TPixelFormat.pf24bit;
3587 TopBar.Canvas.Font.Assign(UniFont[ftNormal]);
3588 TopBar.Canvas.Brush.Style := TBrushStyle.bsClear;
3589 Buffer := TBitmap.Create;
3590 Buffer.PixelFormat := TPixelFormat.pf24bit;
3591 if 2 * lxmax > 3 * xSizeBig then Buffer.Width := 2 * lxmax
3592 else Buffer.Width := 3 * xSizeBig;
3593 if lymax > 3 * ySizeBig then Buffer.Height := lymax
3594 else Buffer.Height := 3 * ySizeBig;
3595 Buffer.Canvas.Font.Assign(UniFont[ftSmall]);
3596 for I := 0 to nPl - 1 do
3597 AILogo[I] := nil;
3598 Canvas.Font.Assign(UniFont[ftSmall]);
3599 InitButtons;
3600 EOT.Template := Templates.Data;
3601end;
3602
3603procedure TMainScreen.FormDestroy(Sender: TObject);
3604var
3605 I: Integer;
3606begin
3607 if Assigned(FWondersDlg) then FreeAndNil(FWondersDlg);
3608 if Assigned(FTechTreeDlg) then FreeAndNil(FTechTreeDlg);
3609 if Assigned(FEnhanceDlg) then FreeAndNil(FEnhanceDlg);
3610 if Assigned(FNegoDlg) then FreeAndNil(FNegoDlg);
3611 if Assigned(FCityTypeDlg) then FreeAndNil(FCityTypeDlg);
3612 if Assigned(FDiaDlg) then FreeAndNil(FDiaDlg);
3613 if Assigned(FCityDlg) then FreeAndNil(FCityDlg);
3614 if Assigned(FRatesDlg) then FreeAndNil(FRatesDlg);
3615 if Assigned(FBattleDlg) then FreeAndNil(FBattleDlg);
3616 if Assigned(FNatStatDlg) then FreeAndNil(FNatStatDlg);
3617 if Assigned(FUnitStatDlg) then FreeAndNil(FUnitStatDlg);
3618 if Assigned(FDraftDlg) then FreeAndNil(FDraftDlg);
3619 if Assigned(FModalSelectDlg) then FreeAndNil(FModalSelectDlg);
3620 if Assigned(FListDlg) then FreeAndNil(FListDlg);
3621 if Assigned(FMessgExDlg) then FreeAndNil(FMessgExDlg);
3622 if Assigned(FHelpDlg) then FreeAndNil(FHelpDlg);
3623
3624 MainFormKeyDown := nil;
3625 FreeAndNil(VerticalScrollBar);
3626 FreeAndNil(TopBar);
3627 FreeAndNil(MiniMap);
3628 FreeAndNil(Buffer);
3629 FreeAndNil(Panel);
3630 for I := 0 to nPl - 1 do
3631 if AILogo[I] <> nil then
3632 FreeAndNil(AILogo[I]);
3633 FreeAndNil(Offscreen);
3634 FreeAndNil(MainMap);
3635 FreeAndNil(NoMap);
3636 FreeAndNil(NoMapPanel);
3637end;
3638
3639procedure TMainScreen.FormMouseWheel(Sender: TObject; Shift: TShiftState;
3640 WheelDelta: Integer; MousePos: TPoint; var Handled: Boolean);
3641var
3642 MouseLoc: Integer;
3643begin
3644 if (MousePos.Y > ClientHeight - MidPanelHeight) and
3645 (MousePos.Y < ClientHeight) then begin
3646 if VerticalScrollBar.ProcessMouseWheel(WheelDelta) then begin
3647 PanelPaint;
3648 Update;
3649 end;
3650 end else begin
3651 if (WheelDelta > 0) and (MainMap.TileSize < High(TTileSize)) then begin
3652 MouseLoc := LocationOfScreenPixel(MousePos.X, MousePos.Y);
3653 SetTileSize(Succ(MainMap.TileSize), MouseLoc, Point(MousePos.X, MousePos.Y));
3654 end
3655 else if (WheelDelta < 0) and (MainMap.TileSize > Low(TTileSize)) then begin
3656 MouseLoc := LocationOfScreenPixel(MousePos.X, MousePos.Y);
3657 SetTileSize(Pred(MainMap.TileSize), MouseLoc, Point(MousePos.X, MousePos.Y));
3658 end;
3659 end;
3660end;
3661
3662procedure TMainScreen.mAfforestClick(Sender: TObject);
3663begin
3664 if UnFocus >= 0 then
3665 with TUn(MyUn[UnFocus]) do
3666 DoJob(jAfforest);
3667end;
3668
3669procedure TMainScreen.mAirBaseClick(Sender: TObject);
3670begin
3671 if UnFocus >= 0 then
3672 with TUn(MyUn[UnFocus]) do
3673 DoJob(jBase);
3674end;
3675
3676procedure TMainScreen.mCanalClick(Sender: TObject);
3677begin
3678 if UnFocus >= 0 then
3679 with TUn(MyUn[UnFocus]) do
3680 DoJob(jCanal);
3681end;
3682
3683procedure TMainScreen.mCancelClick(Sender: TObject);
3684begin
3685 if UnFocus >= 0 then
3686 with MyUn[UnFocus] do begin
3687 DestinationMarkON := False;
3688 PaintDestination;
3689 Status := Status and ($FFFF - usRecover - usGoto - usEnhance);
3690 if Job > jNone then
3691 Server(sStartJob + jNone shl 4, Me, UnFocus, nil^);
3692 end;
3693end;
3694
3695procedure TMainScreen.mCentreClick(Sender: TObject);
3696begin
3697 if UnFocus >= 0 then
3698 with TUn(MyUn[UnFocus]) do begin
3699 Centre(Loc);
3700 PaintAllMaps;
3701 end;
3702end;
3703
3704procedure TMainScreen.mcityClick(Sender: TObject);
3705var
3706 Loc0: Integer;
3707 cix: Integer;
3708 ServerResult: Integer;
3709begin
3710 if UnFocus >= 0 then
3711 with TUn(MyUn[UnFocus]) do begin
3712 Loc0 := Loc;
3713 if MyMap[Loc] and fCity = 0 then
3714 begin // build city
3715 if DoJob(jCity) = eCity then
3716 begin
3717 MapValid := False;
3718 PaintAll;
3719 ZoomToCity(Loc0, True, chFounded);
3720 end;
3721 end else begin
3722 CityOptimizer_BeforeRemoveUnit(UnFocus);
3723 ServerResult := Server(sAddToCity, Me, UnFocus, nil^);
3724 if ServerResult >= rExecuted then
3725 begin
3726 cix := MyRO.nCity - 1;
3727 while (cix >= 0) and (MyCity[cix].Loc <> Loc0) do
3728 Dec(cix);
3729 Assert(cix >= 0);
3730 CityOptimizer_CityChange(cix);
3731 CityOptimizer_AfterRemoveUnit; // does nothing here
3732 SetTroopLoc(Loc0);
3733 UpdateViews(True);
3734 DestinationMarkON := False;
3735 PaintDestination;
3736 UnFocus := -1;
3737 PaintLoc(Loc0);
3738 NextUnit(UnStartLoc, True);
3739 end
3740 else if ServerResult = eMaxSize then
3741 SimpleMessage(Phrases.Lookup('ADDTOMAXSIZE'));
3742 end;
3743 end;
3744end;
3745
3746procedure TMainScreen.mCityStatClick(Sender: TObject);
3747begin
3748 ListDlg.ShowNewContent(wmPersistent, kCities);
3749end;
3750
3751procedure TMainScreen.mCityTypesClick(Sender: TObject);
3752begin
3753 CityTypeDlg.ShowNewContent(wmModal);
3754 // must be modal because types are not saved before closing
3755end;
3756
3757procedure TMainScreen.mClearClick(Sender: TObject);
3758begin
3759 if UnFocus >= 0 then
3760 with TUn(MyUn[UnFocus]) do
3761 DoJob(jClear);
3762end;
3763
3764procedure TMainScreen.mDiagramClick(Sender: TObject);
3765begin
3766 DiaDlg.ShowNewContent_Charts(wmPersistent);
3767end;
3768
3769procedure TMainScreen.mEmpireClick(Sender: TObject);
3770begin
3771 RatesDlg.ShowNewContent(wmPersistent);
3772end;
3773
3774procedure TMainScreen.mFillMapClick(Sender: TObject);
3775var
3776 FillMapData: TFillMapData;
3777begin
3778 if not Edited or (SimpleQuery(mkYesNo, Phrases.Lookup('MAP_FILL'), '')
3779 = mrOK) then begin
3780 FillMapData.Tile := BrushType;
3781 Server(sFillMap, Me, 0, FillMapData);
3782 Edited := True;
3783 MapValid := False;
3784 PaintAllMaps;
3785 end;
3786end;
3787
3788procedure TMainScreen.mEnhanceClick(Sender: TObject);
3789begin
3790 if UnFocus >= 0 then
3791 with TUn(MyUn[UnFocus]) do
3792 DoJob(-1);
3793end;
3794
3795procedure TMainScreen.mEnhanceDefClick(Sender: TObject);
3796begin
3797 if UnFocus >= 0 then
3798 EnhanceDlg.ShowNewContent(wmPersistent,
3799 MyMap[MyUn[UnFocus].Loc] and fTerrain)
3800 else
3801 EnhanceDlg.ShowNewContent(wmPersistent);
3802end;
3803
3804procedure TMainScreen.mEUnitStatClick(Sender: TObject);
3805begin
3806 if MyRO.nEnemyModel > 0 then
3807 ListDlg.ShowNewContent(wmPersistent, kAllEModels);
3808end;
3809
3810procedure TMainScreen.mFarmClick(Sender: TObject);
3811begin
3812 if UnFocus >= 0 then
3813 with TUn(MyUn[UnFocus]) do
3814 DoJob(jFarm);
3815end;
3816
3817procedure TMainScreen.mfortClick(Sender: TObject);
3818begin
3819 if UnFocus >= 0 then
3820 with TUn(MyUn[UnFocus]) do
3821 DoJob(jFort);
3822end;
3823
3824procedure TMainScreen.mGoOnClick(Sender: TObject);
3825var
3826 Destination: Integer;
3827begin
3828 if UnFocus >= 0 then
3829 with TUn(MyUn[UnFocus]) do begin
3830 if Status shr 16 = $7FFF then
3831 Destination := maNextCity
3832 else
3833 Destination := Status shr 16;
3834 Status := Status and not (usStay or usRecover) or usWaiting;
3835 MoveToLoc(Destination, True);
3836 end;
3837end;
3838
3839procedure TMainScreen.mHelpClick(Sender: TObject);
3840begin
3841 if ClientMode = cEditMap then
3842 HelpDlg.ShowNewContent(wmPersistent, hkText, HelpDlg.TextIndex('MAPEDIT'))
3843 else
3844 HelpDlg.ShowNewContent(wmPersistent, hkMisc, Integer(miscMain));
3845end;
3846
3847procedure TMainScreen.mhomeClick(Sender: TObject);
3848var
3849 cixOldHome: Integer;
3850begin
3851 if UnFocus >= 0 then
3852 with TUn(MyUn[UnFocus]) do begin
3853 if MyMap[Loc] and fCity <> 0 then
3854 begin
3855 cixOldHome := Home;
3856 if Server(sSetUnitHome, Me, UnFocus, nil^) >= rExecuted then
3857 begin
3858 CityOptimizer_CityChange(cixOldHome);
3859 CityOptimizer_CityChange(Home);
3860 UpdateViews(True);
3861 end
3862 else
3863 Play('INVALID');
3864 end
3865 else
3866 begin
3867 Status := Status and not (usStay or usRecover or usEnhance);
3868 MoveToLoc(maNextCity, True);
3869 end;
3870 end;
3871end;
3872
3873procedure TMainScreen.mirrigationClick(Sender: TObject);
3874begin
3875 if UnFocus >= 0 then
3876 with TUn(MyUn[UnFocus]) do
3877 DoJob(jIrr);
3878end;
3879
3880procedure TMainScreen.mirrigationDrawItem(Sender: TObject; ACanvas: TCanvas;
3881 ARect: TRect; AState: TOwnerDrawState);
3882begin
3883
3884end;
3885
3886procedure TMainScreen.mJumpClick(Sender: TObject);
3887begin
3888 if Supervising then
3889 Jump[0] := 20
3890 else
3891 Jump[Me] := 20;
3892 EndTurn(True);
3893end;
3894
3895procedure TMainScreen.mLoadClick(Sender: TObject);
3896var
3897 I: Integer;
3898begin
3899 if UnFocus >= 0 then
3900 with MyUn[UnFocus] do begin
3901 I := Server(sLoadUnit, Me, UnFocus, nil^);
3902 if I >= rExecuted then
3903 begin
3904 if MyModel[mix].Domain = dAir then
3905 Play('MOVE_PLANELANDING')
3906 else
3907 Play('MOVE_LOAD');
3908 DestinationMarkON := False;
3909 PaintDestination;
3910 Status := Status and ($FFFF - usWaiting - usStay - usRecover - usGoto - usEnhance);
3911 NextUnit(UnStartLoc, True);
3912 end
3913 else if I = eNoTime_Load then
3914 if MyModel[mix].Domain = dAir then
3915 SoundMessage(Phrases.Lookup('NOTIMELOADAIR'), 'NOMOVE_TIME')
3916 else
3917 SoundMessage(Format(Phrases.Lookup('NOTIMELOADGROUND'),
3918 [MovementToString(MyModel[mix].speed)]), 'NOMOVE_TIME');
3919 end;
3920end;
3921
3922procedure TMainScreen.mmineClick(Sender: TObject);
3923begin
3924 if UnFocus >= 0 then
3925 with TUn(MyUn[UnFocus]) do
3926 DoJob(jMine);
3927end;
3928
3929procedure TMainScreen.mNationsClick(Sender: TObject);
3930begin
3931 NatStatDlg.ShowNewContent(wmPersistent);
3932end;
3933
3934procedure TMainScreen.mNextUnitClick(Sender: TObject);
3935begin
3936 if UnFocus >= 0 then
3937 with MyUn[UnFocus] do begin
3938 Status := Status and not usWaiting;
3939 FocusNextUnit(1);
3940 end;
3941end;
3942
3943procedure TMainScreen.mnoordersClick(Sender: TObject);
3944begin
3945 if UnFocus >= 0 then
3946 with MyUn[UnFocus] do begin
3947 Status := Status and not usWaiting;
3948 NextUnit(UnStartLoc, True);
3949 end;
3950end;
3951
3952procedure TMainScreen.mPillageClick(Sender: TObject);
3953begin
3954 DoJob(jPillage);
3955end;
3956
3957procedure TMainScreen.mpollutionClick(Sender: TObject);
3958begin
3959 if UnFocus >= 0 then
3960 with TUn(MyUn[UnFocus]) do
3961 DoJob(jPoll);
3962end;
3963
3964procedure TMainScreen.mPrevUnitClick(Sender: TObject);
3965begin
3966 if UnFocus >= 0 then
3967 with MyUn[UnFocus] do begin
3968 Status := Status and not usWaiting;
3969 FocusNextUnit(-1);
3970 end;
3971end;
3972
3973procedure TMainScreen.mRandomMapClick(Sender: TObject);
3974begin
3975 if not Edited or (SimpleQuery(mkYesNo, Phrases.Lookup('MAP_RANDOM'), '')
3976 = mrOK) then begin
3977 Server(sRandomMap, Me, 0, nil^);
3978 Edited := True;
3979 MapValid := False;
3980 PaintAllMaps;
3981 end;
3982end;
3983
3984procedure TMainScreen.mRecoverClick(Sender: TObject);
3985begin
3986 if UnFocus >= 0 then
3987 with MyUn[UnFocus] do begin
3988 DestinationMarkON := False;
3989 PaintDestination;
3990 Status := Status and ($FFFF - usStay - usGoto - usEnhance) or usRecover;
3991 if Job > jNone then
3992 Server(sStartJob + jNone shl 4, Me, UnFocus, nil^);
3993 NextUnit(UnStartLoc, True);
3994 end;
3995end;
3996
3997procedure TMainScreen.mResignClick(Sender: TObject);
3998var
3999 QueryText: string;
4000begin
4001 if ClientMode = cEditMap then begin
4002 if Edited then begin
4003 QueryText := Phrases.Lookup('MAP_CLOSE');
4004 case SimpleQuery(mkYesNoCancel, QueryText, '') of
4005 mrIgnore: Server(sAbandonMap, Me, 0, nil^);
4006 mrOK: Server(sSaveMap, Me, 0, nil^);
4007 end;
4008 end else
4009 Server(sAbandonMap, Me, 0, nil^);
4010 end else begin
4011 if Server(sGetGameChanged, 0, 0, nil^) = eOK then begin
4012 QueryText := Phrases.Lookup('RESIGN');
4013 case SimpleQuery(mkYesNoCancel, QueryText, '') of
4014 mrIgnore: Server(sResign, 0, 0, nil^);
4015 mrOK: Server(sBreak, 0, 0, nil^);
4016 end;
4017 end else
4018 Server(sResign, 0, 0, nil^);
4019 end;
4020end;
4021
4022procedure TMainScreen.mRevolutionClick(Sender: TObject);
4023var
4024 AltGovs: Boolean;
4025 RevolutionChanged: Boolean;
4026 I: Integer;
4027begin
4028 AltGovs := False;
4029 for I := 2 to nGov - 1 do
4030 if (GovPreq[I] <> preNA) and
4031 ((GovPreq[I] = preNone) or (MyRO.Tech[GovPreq[I]] >= tsApplicable)) then begin
4032 AltGovs := True;
4033 Break;
4034 end;
4035
4036 if not AltGovs then
4037 SoundMessage(Phrases.Lookup('NOALTGOVS'), 'MSG_DEFAULT')
4038 else
4039 begin
4040 RevolutionChanged := False;
4041 if MyRO.Happened and phChangeGov <> 0 then
4042 begin
4043 ModalSelectDlg.ShowNewContent(wmModal, kGov);
4044 if ModalSelectDlg.Result >= 0 then
4045 begin
4046 Play('NEWGOV');
4047 Server(sSetGovernment, Me, ModalSelectDlg.Result, nil^);
4048 CityOptimizer_BeginOfTurn;
4049 RevolutionChanged := True;
4050 end;
4051 end
4052 else
4053 with MessgExDlg do
4054 begin // revolution!
4055 MessgExDlg.MessgText := Tribe[Me].TPhrase('REVOLUTION');
4056 MessgExDlg.Kind := mkYesNo;
4057 MessgExDlg.IconKind := mikPureIcon;
4058 MessgExDlg.IconIndex := 72; // anarchy palace
4059 MessgExDlg.ShowModal;
4060 if ModalResult = mrOK then
4061 begin
4062 Play('REVOLUTION');
4063 Server(sRevolution, Me, 0, nil^);
4064 RevolutionChanged := True;
4065 if NatStatDlg.Visible then
4066 NatStatDlg.Close;
4067 if CityDlg.Visible then
4068 CityDlg.Close;
4069 end
4070 end;
4071 if RevolutionChanged then
4072 UpdateViews(True);
4073 end;
4074end;
4075
4076procedure TMainScreen.mroadClick(Sender: TObject);
4077begin
4078 if UnFocus >= 0 then
4079 with TUn(MyUn[UnFocus]) do
4080 DoJob(jRoad);
4081end;
4082
4083procedure TMainScreen.mRailRoadClick(Sender: TObject);
4084begin
4085 if UnFocus >= 0 then
4086 with TUn(MyUn[UnFocus]) do
4087 DoJob(jRR);
4088end;
4089
4090procedure TMainScreen.mRunClick(Sender: TObject);
4091begin
4092 if Supervising then
4093 Jump[0] := 999999
4094 else
4095 Jump[Me] := 999999;
4096 EndTurn(True);
4097end;
4098
4099procedure TMainScreen.mScienceStatClick(Sender: TObject);
4100begin
4101 ListDlg.ShowNewContent(wmPersistent, kScience);
4102end;
4103
4104procedure TMainScreen.mSelectTransportClick(Sender: TObject);
4105begin
4106 if UnFocus >= 0 then
4107 with TUn(MyUn[UnFocus]) do
4108 Server(sSelectTransport, Me, UnFocus, nil^);
4109end;
4110
4111procedure TMainScreen.mShipsClick(Sender: TObject);
4112begin
4113 DiaDlg.ShowNewContent_Ship(wmPersistent);
4114end;
4115
4116procedure TMainScreen.mstayClick(Sender: TObject);
4117begin
4118 if UnFocus >= 0 then
4119 with TUn(MyUn[UnFocus]) do begin
4120 DestinationMarkON := False;
4121 PaintDestination;
4122 Status := Status and ($FFFF - usRecover - usGoto - usEnhance) or usStay;
4123 if Job > jNone then
4124 Server(sStartJob + jNone shl 4, Me, UnFocus, nil^);
4125 NextUnit(UnStartLoc, True);
4126 end;
4127end;
4128
4129procedure TMainScreen.mTechTreeClick(Sender: TObject);
4130begin
4131 TechTreeDlg.ShowModal;
4132end;
4133
4134procedure TMainScreen.mtransClick(Sender: TObject);
4135begin
4136 if UnFocus >= 0 then
4137 with TUn(MyUn[UnFocus]) do
4138 DoJob(jTrans);
4139end;
4140
4141procedure TMainScreen.mUnitStatClick(Sender: TObject);
4142var
4143 I: Integer;
4144begin
4145 if G.Difficulty[Me] > 0 then
4146 ListDlg.ShowNewContent_MilReport(wmPersistent, Me)
4147 else
4148 begin
4149 I := 1;
4150 while (I < nPl) and (1 shl I and MyRO.Alive = 0) do
4151 Inc(I);
4152 if I < nPl then
4153 ListDlg.ShowNewContent_MilReport(wmPersistent, I);
4154 end;
4155end;
4156
4157procedure TMainScreen.mUnloadClick(Sender: TObject);
4158var
4159 I: Integer;
4160 OldMaster: Integer;
4161 NewFocus: Integer;
4162 uix: Integer;
4163begin
4164 if UnFocus >= 0 then
4165 with MyUn[UnFocus] do begin
4166 if Master >= 0 then begin
4167 OldMaster := Master;
4168 I := Server(sUnloadUnit, Me, UnFocus, nil^);
4169 if I >= rExecuted then
4170 begin
4171 if MyModel[mix].Domain = dAir then
4172 Play('MOVE_PLANESTART')
4173 else if (MyModel[MyUn[OldMaster].mix].Domain = dAir) and
4174 (MyMap[Loc] and fCity = 0) and (MyMap[Loc] and fTerImp <> tiBase)
4175 then
4176 Play('MOVE_PARACHUTE')
4177 else
4178 Play('MOVE_UNLOAD');
4179 Status := Status and not usWaiting;
4180 if MyModel[mix].Domain <> dAir then
4181 NextUnit(Loc, True)
4182 else
4183 PanelPaint;
4184 end
4185 else if I = eNoTime_Load then
4186 if MyModel[mix].Domain = dAir then
4187 SoundMessage(Phrases.Lookup('NOTIMELOADAIR'), 'NOMOVE_TIME')
4188 else
4189 SoundMessage(Format(Phrases.Lookup('NOTIMELOADGROUND'),
4190 [MovementToString(MyModel[mix].speed)]), 'NOMOVE_TIME');
4191 end else begin
4192 NewFocus := -1;
4193 uix := UnFocus;
4194 for I := 1 to MyRO.nUn - 1 do
4195 begin
4196 uix := (uix + MyRO.nUn - 1) mod MyRO.nUn;
4197 if (MyUn[uix].Master = UnFocus) and
4198 (MyUn[uix].Movement = Integer(MyModel[MyUn[uix].mix].speed)) then
4199 begin
4200 MyUn[uix].Status := MyUn[uix].Status or usWaiting;
4201 NewFocus := uix;
4202 end;
4203 end;
4204 if NewFocus >= 0 then
4205 begin
4206 SetUnFocus(NewFocus);
4207 SetTroopLoc(Loc);
4208 PanelPaint;
4209 end;
4210 end;
4211 end;
4212end;
4213
4214procedure TMainScreen.mwaitClick(Sender: TObject);
4215begin
4216 if UnFocus >= 0 then
4217 with MyUn[UnFocus] do begin
4218 DestinationMarkON := False;
4219 PaintDestination;
4220 Status := Status and ($FFFF - usStay - usRecover - usGoto - usEnhance) or usWaiting;
4221 end;
4222 NextUnit(-1, False);
4223end;
4224
4225procedure TMainScreen.mWebsiteClick(Sender: TObject);
4226begin
4227 OpenURL(CevoHomepage);
4228end;
4229
4230procedure TMainScreen.mWondersClick(Sender: TObject);
4231begin
4232 WondersDlg.ShowNewContent(wmPersistent);
4233end;
4234
4235procedure TMainScreen.FormResize(Sender: TObject);
4236begin
4237 if (LastResizeWidth = Width) and (LastResizeHeight = Height) and
4238 (LastWindowState = WindowState) then Exit
4239 else begin
4240 LastResizeWidth := Width;
4241 LastResizeHeight := Height;
4242 LastWindowState := WindowState;
4243 end;
4244
4245 ResizeControls;
4246end;
4247
4248procedure TMainScreen.FormCloseQuery(Sender: TObject; var CanClose: Boolean);
4249begin
4250 CanClose := Closable;
4251 if not Closable and Idle and (Me = 0) and (ClientMode < scContact) then
4252 mResign.Click;
4253end;
4254
4255procedure TMainScreen.OnScroll(var Msg: TMessage);
4256begin
4257 if VerticalScrollBar.Process(Msg) then begin
4258 PanelPaint;
4259 Update;
4260 end;
4261end;
4262
4263procedure TMainScreen.OnEOT(var Msg: TMessage);
4264begin
4265 EndTurn;
4266end;
4267
4268procedure TMainScreen.EOTClick(Sender: TObject);
4269begin
4270 if GameMode = cMovie then
4271 begin
4272 MessgExDlg.CancelMovie;
4273 Server(sBreak, Me, 0, nil^);
4274 end
4275 else if ClientMode < 0 then
4276 Skipped := True
4277 else if ClientMode >= scContact then
4278 NegoDlg.ShowNewContent(wmPersistent)
4279 else if Jump[pTurn] > 0 then
4280 begin
4281 Jump[pTurn] := 0;
4282 StartRunning := False;
4283 end
4284 else
4285 EndTurn;
4286end;
4287
4288// set xTerrain, xTroop, and TrRow
4289procedure TMainScreen.ArrangeMidPanel;
4290begin
4291 if ClientMode = cEditMap then
4292 xTroop := xMidPanel + 15
4293 else
4294 with MainMap do begin
4295 if Supervising then
4296 xTerrain := xMidPanel + 2 * xxt + 14
4297 else if ClientWidth < 1280 then
4298 xTerrain := ClientWidth div 2 + (1280 - ClientWidth) div 3
4299 else
4300 xTerrain := ClientWidth div 2;
4301 xTroop := xTerrain + 2 * xxt + 12;
4302 if SmallScreen and not Supervising then
4303 xTroop := xRightPanel + 10 - 3 * 66 -
4304 GetSystemMetrics(SM_CXVSCROLL) - 19 - 4;
4305 // not perfect but we assume almost no one is still playing on a 800x600 screen
4306 end;
4307 TrRow := (xRightPanel + 10 - xTroop - GetSystemMetrics(SM_CXVSCROLL) - 19)
4308 div TrPitch;
4309end;
4310
4311function TMainScreen.EndTurn(WasSkipped: Boolean): Boolean;
4312
4313 function IsResourceUnused(cix, NeedFood, NeedProd: Integer): Boolean;
4314 var
4315 dx, dy, fix: Integer;
4316 CityAreaInfo: TCityAreaInfo;
4317 TileInfo: TTileInfo;
4318 begin
4319 Server(sGetCityAreaInfo, Me, cix, CityAreaInfo);
4320 for dy := -3 to 3 do
4321 for dx := -3 to 3 do
4322 if ((dx + dy) and 1 = 0) and (dx * dx * dy * dy < 81) then
4323 begin
4324 fix := (dy + 3) shl 2 + (dx + 3) shr 1;
4325 if (MyCity[cix].Tiles and (1 shl fix) = 0) // not used yet
4326 and (CityAreaInfo.Available[fix] = faAvailable) then // usable
4327 begin
4328 TileInfo.ExplCity := cix;
4329 Server(sGetHypoCityTileInfo, Me, dLoc(MyCity[cix].Loc, dx, dy),
4330 TileInfo);
4331 if (TileInfo.Food >= NeedFood) and (TileInfo.Prod >= NeedProd) then
4332 begin
4333 Result := True;
4334 Exit;
4335 end;
4336 end
4337 end;
4338 Result := False;
4339 end;
4340
4341var
4342 p1, uix, cix, CenterLoc: Integer;
4343 MsgItem: string;
4344 CityReport: TCityReport;
4345 PlaneReturnData: TPlaneReturnData;
4346 Zoom: Boolean;
4347begin
4348 Result := False;
4349 if ClientMode >= scDipOffer then
4350 Exit;
4351
4352 if Supervising and (Me <> 0) then begin
4353 ApplyToVisibleForms(faClose);
4354 ItsMeAgain(0);
4355 end;
4356
4357 CityOptimizer_EndOfTurn;
4358
4359 if not WasSkipped then // check warnings
4360 begin
4361 // need to move planes home?
4362 for uix := 0 to MyRO.nUn - 1 do
4363 with MyUn[uix] do
4364 if (Loc >= 0) and (MyModel[mix].Domain = dAir) and
4365 (Status and usToldNoReturn = 0) and (Master < 0) and
4366 (MyMap[Loc] and fCity = 0) and (MyMap[Loc] and fTerImp <> tiBase) then
4367 begin
4368 PlaneReturnData.Fuel := Fuel;
4369 PlaneReturnData.Loc := Loc;
4370 PlaneReturnData.Movement := 0; // end turn without further movement?
4371 if Server(sGetPlaneReturn, Me, uix, PlaneReturnData) = eNoWay then
4372 begin
4373 CenterLoc := Loc + G.lx * 6;
4374 // centering the unit itself would make it covered by the query dialog
4375 while CenterLoc >= G.lx * G.ly do
4376 Dec(CenterLoc, G.lx * 2);
4377 Centre(CenterLoc);
4378 SetTroopLoc(-1);
4379 PaintAll;
4380
4381 if MyModel[mix].Kind = mkSpecial_Glider then
4382 MsgItem := 'LOWFUEL_GLIDER'
4383 else
4384 MsgItem := 'LOWFUEL';
4385 if SimpleQuery(mkYesNo, Phrases.Lookup(MsgItem),
4386 'WARNING_LOWSUPPORT') <> mrOK then
4387 begin
4388 SetUnFocus(uix);
4389 SetTroopLoc(Loc);
4390 PanelPaint;
4391 Exit;
4392 end;
4393 MyUn[uix].Status := MyUn[uix].Status or usToldNoReturn;
4394 end;
4395 end;
4396
4397 if not Supervising and (MyRO.TestFlags and tfImmImprove = 0) and
4398 (MyRO.Government <> gAnarchy) and (MyRO.Money + TaxSum < 0) and
4399 (MyRO.TaxRate < 100) then // low funds!
4400 with MessgExDlg do
4401 begin
4402 OpenSound := 'WARNING_LOWFUNDS';
4403 MessgText := Phrases.Lookup('LOWFUNDS');
4404 Kind := mkYesNo;
4405 IconKind := mikImp;
4406 IconIndex := imTrGoods;
4407 ShowModal;
4408 if ModalResult <> mrOK then
4409 Exit;
4410 end;
4411
4412 if MyRO.Government <> gAnarchy then
4413 for cix := 0 to MyRO.nCity - 1 do
4414 with MyCity[cix] do
4415 if (Loc >= 0) and (Flags and chCaptured = 0) then
4416 begin
4417 Zoom := False;
4418 CityReport.HypoTiles := -1;
4419 CityReport.HypoTax := -1;
4420 CityReport.HypoLux := -1;
4421 Server(sGetCityReport, Me, cix, CityReport);
4422
4423 if (CityReport.Working - CityReport.Happy > Size shr 1) and
4424 (Flags and chCaptured <= $10000) then
4425 with MessgExDlg do
4426 begin
4427 OpenSound := 'WARNING_DISORDER';
4428 if Status and csResourceWeightsMask = 0 then
4429 MsgItem := 'DISORDER'
4430 else
4431 MsgItem := 'DISORDER_UNREST';
4432 MessgText := Format(Phrases.Lookup(MsgItem), [CityName(ID)]);
4433 Kind := mkYesNo;
4434 // BigIcon := 29;
4435 ShowModal;
4436 Zoom := ModalResult <> mrOK;
4437 end;
4438 if not Zoom and (Food + CityReport.FoodRep - CityReport.Eaten < 0)
4439 then
4440 with MessgExDlg do
4441 begin
4442 OpenSound := 'WARNING_FAMINE';
4443 if Status and csResourceWeightsMask = 0 then
4444 MsgItem := 'FAMINE'
4445 else if (CityReport.Deployed <> 0) and
4446 IsResourceUnused(cix, 1, 0) then
4447 MsgItem := 'FAMINE_UNREST'
4448 else
4449 MsgItem := 'FAMINE_TILES';
4450 MessgText := Format(Phrases.Lookup(MsgItem), [CityName(ID)]);
4451 Kind := mkYesNo;
4452 IconKind := mikImp;
4453 IconIndex := 22;
4454 ShowModal;
4455 Zoom := ModalResult <> mrOK;
4456 end;
4457 if not Zoom and (CityReport.ProdRep < CityReport.Support) then
4458 with MessgExDlg do
4459 begin
4460 OpenSound := 'WARNING_LOWSUPPORT';
4461 if Status and csResourceWeightsMask = 0 then
4462 MsgItem := 'LOWSUPPORT'
4463 else if (CityReport.Deployed <> 0) and
4464 IsResourceUnused(cix, 0, 1) then
4465 MsgItem := 'LOWSUPPORT_UNREST'
4466 else
4467 MsgItem := 'LOWSUPPORT_TILES';
4468 MessgText := Format(Phrases.Lookup(MsgItem), [CityName(ID)]);
4469 Kind := mkYesNo;
4470 IconKind := mikImp;
4471 IconIndex := 29;
4472 ShowModal;
4473 Zoom := ModalResult <> mrOK;
4474 end;
4475 if Zoom then
4476 begin // zoom to city
4477 ZoomToCity(Loc);
4478 Exit;
4479 end;
4480 end;
4481
4482 if (MyRO.Happened and phTech <> 0) and (MyRO.ResearchTech < 0) and
4483 (MyData.FarTech <> adNexus) then
4484 if not ChooseResearch then
4485 Exit;
4486 end;
4487
4488 RememberPeaceViolation;
4489
4490 SetUnFocus(-1);
4491 for uix := 0 to MyRO.nUn - 1 do
4492 MyUn[uix].Status := MyUn[uix].Status and usPersistent;
4493
4494 CityDlg.CloseAction := None;
4495 if IsMultiPlayerGame then begin
4496 // Close windows for next player
4497 ApplyToVisibleForms(faClose);
4498 end else begin
4499 if CityDlg.Visible then
4500 CityDlg.Close;
4501 if UnitStatDlg.Visible then
4502 UnitStatDlg.Close;
4503 end;
4504 ApplyToVisibleForms(faDisable);
4505
4506 if Server(sTurn, pTurn, 0, nil^) >= rExecuted then
4507 begin
4508 if Jump[pTurn] > 0 then
4509 EOT.Hint := Phrases.Lookup('BTN_STOP')
4510 else
4511 EOT.Hint := Phrases.Lookup('BTN_SKIP');
4512 Result := True;
4513 SetTroopLoc(-1);
4514 pTurn := -1;
4515 pLogo := -1;
4516 UnitInfoBtn.Visible := False;
4517 UnitBtn.Visible := False;
4518 TerrainBtn.Visible := False;
4519 EOT.ButtonIndex := eotCancel;
4520 EOT.Visible := True;
4521 MapValid := False;
4522 PanelPaint;
4523 Update;
4524 ClientMode := -1;
4525 Idle := False;
4526 Skipped := WasSkipped;
4527 for p1 := 1 to nPl - 1 do
4528 if G.RO[p1] <> nil then begin
4529 Skipped := True; // don't show enemy moves in hotseat mode
4530 Break;
4531 end;
4532 end
4533 else
4534 PanelPaint;
4535end;
4536
4537procedure TMainScreen.EndNego;
4538begin
4539 if NegoDlg.Visible then
4540 NegoDlg.Close;
4541 HaveStrategyAdvice := False;
4542 EOT.ButtonIndex := eotCancel;
4543 EOT.Visible := True;
4544 PanelPaint;
4545 Update;
4546 ClientMode := -1;
4547 Idle := False;
4548end;
4549
4550procedure TMainScreen.ProcessRect(x0, y0, nx, ny, Options: Integer);
4551var
4552 xs, ys, xl, yl: Integer;
4553begin
4554 with MainMap do begin
4555 xl := nx * xxt + xxt;
4556 yl := ny * yyt + yyt * 2;
4557 xs := (x0 - xw) * (xxt * 2) + y0 and 1 * xxt - G.lx * (xxt * 2);
4558 // |xs+xl/2-MapWidth/2| -> min
4559 while Abs(2 * (xs + G.lx * (xxt * 2)) + xl - MapWidth) <
4560 Abs(2 * xs + xl - MapWidth) do
4561 Inc(xs, G.lx * (xxt * 2));
4562 ys := (y0 - yw) * yyt - yyt;
4563 if xs + xl > MapWidth then
4564 xl := MapWidth - xs;
4565 if ys + yl > MapHeight then
4566 yl := MapHeight - ys;
4567 if (xl <= 0) or (yl <= 0) then
4568 Exit;
4569 if Options and prPaint <> 0 then begin
4570 if Options and prAutoBounds <> 0 then
4571 MainMap.SetPaintBounds(xs, ys, xs + xl, ys + yl);
4572 MainMap.Paint(xs, ys, x0 + G.lx * y0, nx, ny, -1, -1);
4573 end;
4574 if Options and prInvalidate <> 0 then
4575 RectInvalidate(MapOffset + xs, TopBarHeight + ys, MapOffset + xs + xl,
4576 TopBarHeight + ys + yl)
4577 end;
4578end;
4579
4580procedure TMainScreen.PaintLoc(Loc: Integer; Radius: Integer = 0);
4581var
4582 yLoc, x0: Integer;
4583begin
4584 if MapValid then begin
4585 yLoc := (Loc + G.lx * 1024) div G.lx - 1024;
4586 x0 := (Loc + (yLoc and 1 - 2 * Radius + G.lx * 1024) div 2) mod G.lx;
4587 Offscreen.Canvas.Font.Assign(UniFont[ftSmall]);
4588 ProcessRect(x0, yLoc - 2 * Radius, 4 * Radius + 1, 4 * Radius + 1,
4589 prPaint or prAutoBounds or prInvalidate);
4590 Update;
4591 end;
4592end;
4593
4594procedure TMainScreen.PaintLocTemp(Loc: Integer; Style: TPaintLocTempStyle);
4595var
4596 y0, x0, xMap, yMap: Integer;
4597begin
4598 with NoMap do begin
4599 if not MapValid then
4600 Exit;
4601 Buffer.Canvas.Font.Assign(UniFont[ftSmall]);
4602 y0 := Loc div G.lx;
4603 x0 := Loc mod G.lx;
4604 xMap := (x0 - xw) * (xxt * 2) + y0 and 1 * xxt - G.lx * (xxt * 2);
4605 // |xMap+xxt-MapWidth/2| -> min
4606 while Abs(2 * (xMap + G.lx * (xxt * 2)) + 2 * xxt - MapWidth) <
4607 Abs(2 * xMap + 2 * xxt - MapWidth) do
4608 Inc(xMap, G.lx * (xxt * 2));
4609 yMap := (y0 - yw) * yyt - yyt;
4610 NoMap.SetOutput(Buffer);
4611 NoMap.SetPaintBounds(0, 0, 2 * xxt, 3 * yyt);
4612 NoMap.Paint(0, 0, Loc, 1, 1, -1, -1, Style = pltsBlink);
4613 PaintBufferToScreen(xMap, yMap, 2 * xxt, 3 * yyt);
4614 end;
4615end;
4616
4617// paint content of buffer directly to screen instead of offscreen
4618// panel protusions are added
4619// NoMap must be set to buffer and bounds before
4620procedure TMainScreen.PaintBufferToScreen(xMap, yMap, Width, Height: Integer);
4621begin
4622 if xMap + Width > MapWidth then
4623 Width := MapWidth - xMap;
4624 if yMap + Height > MapHeight then
4625 Height := MapHeight - yMap;
4626 if (Width <= 0) or (Height <= 0) or (Width + xMap <= 0) or (Height + yMap <= 0)
4627 then
4628 Exit;
4629
4630 NoMap.BitBltBitmapOutput(Panel, -xMap - MapOffset, -yMap + MapHeight - Overlap, xMidPanel,
4631 Overlap, 0, 0);
4632 NoMap.BitBltBitmapOutput(Panel, -xMap - MapOffset + xRightPanel,
4633 -yMap + MapHeight - Overlap, Panel.Width - xRightPanel, Overlap,
4634 xRightPanel, 0);
4635 if yMap < 0 then
4636 begin
4637 if xMap < 0 then
4638 BitBltCanvas(Canvas, MapOffset, TopBarHeight, Width + xMap,
4639 Height + yMap, Buffer.Canvas, -xMap, -yMap)
4640 else
4641 BitBltCanvas(Canvas, xMap + MapOffset, TopBarHeight, Width,
4642 Height + yMap, Buffer.Canvas, 0, -yMap);
4643 end
4644 else
4645 begin
4646 if xMap < 0 then
4647 BitBltCanvas(Canvas, MapOffset, TopBarHeight + yMap, Width + xMap,
4648 Height, Buffer.Canvas, -xMap, 0)
4649 else
4650 BitBltCanvas(Canvas, xMap + MapOffset, TopBarHeight + yMap, Width,
4651 Height, Buffer.Canvas, 0, 0);
4652 end;
4653end;
4654
4655procedure TMainScreen.PaintLoc_BeforeMove(FromLoc: Integer);
4656var
4657 yLoc, x0: Integer;
4658begin
4659 if MapValid then
4660 begin
4661 yLoc := (FromLoc + G.lx * 1024) div G.lx - 1024;
4662 x0 := (FromLoc + (yLoc and 1 + G.lx * 1024) div 2) mod G.lx;
4663 Offscreen.Canvas.Font.Assign(UniFont[ftSmall]);
4664 ProcessRect(x0, yLoc, 1, 1, prPaint or prAutoBounds);
4665 end
4666end;
4667
4668procedure TMainScreen.PaintDestination;
4669var
4670 Destination: Integer;
4671begin
4672 if (UnFocus >= 0) and (MyUn[UnFocus].Status and usGoto <> 0) then
4673 begin
4674 Destination := MyUn[UnFocus].Status shr 16;
4675 if (Destination <> $7FFF) and (Destination <> MyUn[UnFocus].Loc) then
4676 PaintLocTemp(Destination, pltsBlink);
4677 end;
4678end;
4679
4680{$IFDEF UNIX}
4681// Can't do scrolling of DC under Linux, then fallback into BitBlt.
4682function ScrollDC(Canvas: TCanvas; dx: longint; dy: longint; const lprcScroll:TRect; const lprcClip:TRect; hrgnUpdate:HRGN; lprcUpdate: PRect):Boolean;
4683begin
4684 Result := BitBltCanvas(Canvas, lprcScroll.Left + dx, lprcScroll.Top + dy, lprcScroll.Right - lprcScroll.Left, lprcScroll.Bottom - lprcScroll.Top,
4685 Canvas, lprcScroll.Left, lprcScroll.Top);
4686end;
4687{$ENDIF}
4688
4689procedure TMainScreen.MainOffscreenPaint;
4690var
4691 ProcessOptions: Integer;
4692 Rec: TRect;
4693 DoInvalidate: Boolean;
4694begin
4695 if Me < 0 then
4696 with Offscreen.Canvas do
4697 begin
4698 Brush.Color := $000000;
4699 FillRect(Rect(0, 0, MapWidth, MapHeight));
4700 Brush.Style := TBrushStyle.bsClear;
4701 OffscreenUser := Self;
4702 Exit;
4703 end;
4704
4705 MainMap.SetPaintBounds(0, 0, MapWidth, MapHeight);
4706 if OffscreenUser <> Self then
4707 begin
4708 if OffscreenUser <> nil then
4709 OffscreenUser.Update;
4710 // complete working with old owner to prevent rebound
4711 if MapValid and (xwd = xw) and (ywd = yw) then
4712 MainMap.SetPaintBounds(0, 0, UsedOffscreenWidth, UsedOffscreenHeight);
4713 MapValid := False;
4714 OffscreenUser := Self;
4715 end;
4716
4717 with MainMap do begin
4718 if xw - xwd > G.lx div 2 then
4719 xwd := xwd + G.lx
4720 else if xwd - xw > G.lx div 2 then
4721 xwd := xwd - G.lx;
4722 if not MapValid or (xw - xwd > MapWidth div (xxt * 2)) or
4723 (xwd - xw > MapWidth div (xxt * 2)) or (yw - ywd > MapHeight div yyt) or
4724 (ywd - yw > MapHeight div yyt) then
4725 begin
4726 Offscreen.Canvas.Font.Assign(UniFont[ftSmall]);
4727 ProcessRect(xw, yw, MapWidth div xxt, MapHeight div yyt,
4728 prPaint or prInvalidate);
4729 end else begin
4730 if (xwd = xw) and (ywd = yw) then
4731 Exit; { map window not moved }
4732 Offscreen.Canvas.Font.Assign(UniFont[ftSmall]);
4733 Rec := Rect(0, 0, MapWidth, MapHeight);
4734{$IFDEF WINDOWS}
4735 ScrollDC(Offscreen.Canvas.Handle, (xwd - xw) * (xxt * 2), (ywd - yw) * yyt,
4736 rec, rec, 0, nil);
4737{$ENDIF}
4738{$IFDEF UNIX}
4739 ScrollDC(Offscreen.Canvas, (xwd - xw) * (xxt * 2), (ywd - yw) * yyt,
4740 Rec, Rec, 0, nil);
4741{$ENDIF}
4742 for DoInvalidate := False to FastScrolling do begin
4743 if DoInvalidate then begin
4744 Rec.Bottom := MapHeight - Overlap;
4745{$IFDEF WINDOWS}
4746 ScrollDC(Canvas.Handle, (xwd - xw) * (xxt * 2), (ywd - yw) * yyt, rec,
4747 rec, 0, nil);
4748{$ENDIF}
4749{$IFDEF UNIX}
4750 ScrollDC(Canvas, (xwd - xw) * (xxt * 2), (ywd - yw) * yyt,
4751 Rec, Rec, 0, nil);
4752{$ENDIF}
4753 ProcessOptions := prInvalidate;
4754 end
4755 else ProcessOptions := prPaint or prAutoBounds;
4756 if yw < ywd then begin
4757 ProcessRect(xw, yw, MapWidth div xxt, ywd - yw - 1, ProcessOptions);
4758 if xw < xwd then
4759 ProcessRect(xw, ywd, (xwd - xw) * 2 - 1, MapHeight div yyt - ywd + yw,
4760 ProcessOptions)
4761 else if xw > xwd then
4762 ProcessRect((xwd + MapWidth div (xxt * 2)) mod G.lx, ywd,
4763 (xw - xwd) * 2 + 1, MapHeight div yyt - ywd + yw, ProcessOptions);
4764 end
4765 else if yw > ywd then begin
4766 if DoInvalidate then
4767 RectInvalidate(MapOffset, TopBarHeight + MapHeight - Overlap -
4768 (yw - ywd) * yyt, MapOffset + MapWidth, TopBarHeight + MapHeight
4769 - Overlap)
4770 else
4771 ProcessRect(xw, (ywd + MapHeight div (yyt * 2) * 2), MapWidth div xxt,
4772 yw - ywd + 1, ProcessOptions);
4773 if xw < xwd then
4774 ProcessRect(xw, yw, (xwd - xw) * 2 - 1, MapHeight div yyt - yw + ywd -
4775 2, ProcessOptions)
4776 else if xw > xwd then
4777 ProcessRect((xwd + MapWidth div (xxt * 2)) mod G.lx, yw,
4778 (xw - xwd) * 2 + 1, MapHeight div yyt - yw + ywd - 2,
4779 ProcessOptions);
4780 end
4781 else if xw < xwd then
4782 ProcessRect(xw, yw, (xwd - xw) * 2 - 1, MapHeight div yyt,
4783 ProcessOptions)
4784 else if xw > xwd then
4785 ProcessRect((xwd + MapWidth div (xxt * 2)) mod G.lx, yw,
4786 (xw - xwd) * 2 + 1, MapHeight div yyt, ProcessOptions);
4787 end;
4788 if not FastScrolling then
4789 RectInvalidate(MapOffset, TopBarHeight, MapOffset + MapWidth,
4790 TopBarHeight + MapHeight - Overlap);
4791 RectInvalidate(xMidPanel, TopBarHeight + MapHeight - Overlap, xRightPanel,
4792 TopBarHeight + MapHeight);
4793 end;
4794 end;
4795 // if (xwd<>xw) or (ywd<>yw) then
4796 // Server(sChangeSuperView,me,yw*G.lx+xw,nil^); // for synchronizing client side viewer, not used currently
4797 xwd := xw;
4798 ywd := yw;
4799 MapValid := True;
4800end;
4801
4802procedure TMainScreen.MiniMapPaint;
4803begin
4804 with MainMap do
4805 MiniMap.Paint(MyMap, MapWidth, ClientMode, xxt, xwMini);
4806end;
4807
4808procedure TMainScreen.PaintAll;
4809begin
4810 MainOffscreenPaint;
4811 xwMini := xw;
4812 ywMini := yw;
4813 MiniMapPaint;
4814 PanelPaint;
4815end;
4816
4817procedure TMainScreen.RepaintAll;
4818begin
4819 RectInvalidate(0, TopBarHeight, Width, TopBarHeight + MapHeight);
4820 MapValid := False;
4821 PaintAll;
4822end;
4823
4824procedure TMainScreen.PaintAllMaps;
4825begin
4826 MainOffscreenPaint;
4827 xwMini := xw;
4828 ywMini := yw;
4829 MiniMapPaint;
4830 CopyMiniToPanel;
4831 RectInvalidate(xMini + 2, TopBarHeight + MapHeight - Overlap + yMini + 2,
4832 xMini + 2 + G.lx * 2, TopBarHeight + MapHeight - Overlap + yMini +
4833 2 + G.ly);
4834end;
4835
4836procedure TMainScreen.CopyMiniToPanel;
4837begin
4838 with MainMap do begin
4839 BitBltBitmap(Panel, xMini + 2, yMini + 2, G.lx * 2, G.ly,
4840 MiniMap.Bitmap, 0, 0);
4841 if MarkCityLoc >= 0 then
4842 Sprite(Panel, HGrSystem, xMini - 2 + (4 * G.lx + 2 * (MarkCityLoc mod G.lx)
4843 + (G.lx - MapWidth div (xxt * 2)) - 2 * xwd) mod (2 * G.lx) +
4844 MarkCityLoc div G.lx and 1, yMini - 3 + MarkCityLoc div G.lx, CityMark2.Width,
4845 CityMark2.Height, CityMark2.Left, CityMark2.Top)
4846 else if ywmax <= 0 then
4847 Frame(Panel.Canvas,
4848 xMini + 2 + G.lx - MapWidth div (xxt * 2), yMini + 2,
4849 xMini + 1 + G.lx + MapWidth div (xxt * 2), yMini + 2 + G.ly - 1,
4850 MainTexture.ColorMark, MainTexture.ColorMark)
4851 else
4852 Frame(Panel.Canvas,
4853 xMini + 2 + G.lx - MapWidth div (xxt * 2), yMini + 2 + yw,
4854 xMini + 1 + G.lx + MapWidth div (xxt * 2), yMini + yw + MapHeight div yyt,
4855 MainTexture.ColorMark, MainTexture.ColorMark);
4856 end;
4857end;
4858
4859procedure TMainScreen.PanelPaint;
4860
4861 function MovementToString(var Un: TUn): string;
4862 begin
4863 Result := ScreenTools.MovementToString(Un.Movement);
4864 if Un.Master >= 0 then
4865 Result := '(' + Result + ')'
4866 else if (MyModel[Un.mix].Domain = dAir) and
4867 (MyModel[Un.mix].Kind <> mkSpecial_Glider) then
4868 Result := Format('%s(%d)', [Result, Un.Fuel]);
4869 end;
4870
4871var
4872 I, uix, uixDefender, X, CostFactor, Count,
4873 mixShow, xTreasurySection, xResearchSection, JobFocus, TrueMoney,
4874 TrueResearch: Integer;
4875 Src, SrcBase: TPoint;
4876 Tile: Cardinal;
4877 S: string;
4878 unx: TUn;
4879 UnitInfo: TUnitInfo;
4880 JobProgressData: TJobProgressData;
4881 Prio: Boolean;
4882begin
4883 if not Assigned(MyRO) then Exit;
4884 with Panel.Canvas do
4885 begin
4886 Fill(Panel.Canvas, 0, 3, xMidPanel + 7 - 10, PanelHeight - 3,
4887 MainTexture.Width - (xMidPanel + 7 - 10), MainTexture.Height - PanelHeight);
4888 Fill(Panel.Canvas, xRightPanel + 10 - 7, 3, Panel.Width - xRightPanel - 10 +
4889 7, PanelHeight - 3, -(xRightPanel + 10 - 7), MainTexture.Height - PanelHeight);
4890 FillLarge(Panel.Canvas, xMidPanel - 2, PanelHeight - MidPanelHeight,
4891 xRightPanel + 2, PanelHeight, ClientWidth div 2);
4892
4893 Brush.Style := TBrushStyle.bsClear;
4894 Pen.Color := $000000;
4895 MoveTo(0, 0);
4896 LineTo(xMidPanel + 7 - 8, 0);
4897 LineTo(xMidPanel + 7 - 8, PanelHeight - MidPanelHeight);
4898 LineTo(xRightPanel, PanelHeight - MidPanelHeight);
4899 LineTo(xRightPanel, 0);
4900 LineTo(ClientWidth, 0);
4901 Pen.Color := MainTexture.ColorBevelLight;
4902 MoveTo(xMidPanel + 7 - 9, PanelHeight - MidPanelHeight + 2);
4903 LineTo(xRightPanel + 10 - 8, PanelHeight - MidPanelHeight + 2);
4904 Pen.Color := MainTexture.ColorBevelLight;
4905 MoveTo(0, 1);
4906 LineTo(xMidPanel + 7 - 9, 1);
4907 Pen.Color := MainTexture.ColorBevelShade;
4908 LineTo(xMidPanel + 7 - 9, PanelHeight - MidPanelHeight + 1);
4909 Pen.Color := MainTexture.ColorBevelLight;
4910 LineTo(xRightPanel + 10 - 9, PanelHeight - MidPanelHeight + 1);
4911 Pen.Color := MainTexture.ColorBevelLight;
4912 LineTo(xRightPanel + 10 - 9, 1);
4913 LineTo(ClientWidth, 1);
4914 MoveTo(ClientWidth, 2);
4915 LineTo(xRightPanel + 10 - 8, 2);
4916 LineTo(xRightPanel + 10 - 8, PanelHeight);
4917 MoveTo(0, 2);
4918 LineTo(xMidPanel + 7 - 10, 2);
4919 Pen.Color := MainTexture.ColorBevelShade;
4920 LineTo(xMidPanel + 7 - 10, PanelHeight);
4921 Corner(Panel.Canvas, xMidPanel + 7 - 16, 1, 1, MainTexture);
4922 Corner(Panel.Canvas, xRightPanel + 10 - 9, 1, 0, MainTexture);
4923 if ClientMode <> cEditMap then
4924 begin
4925 if Supervising then
4926 begin
4927 ScreenTools.Frame(Panel.Canvas, ClientWidth - xPalace - 1, yPalace - 1,
4928 ClientWidth - xPalace + xSizeBig, yPalace + ySizeBig,
4929 $B0B0B0, $FFFFFF);
4930 RFrame(Panel.Canvas, ClientWidth - xPalace - 2, yPalace - 2,
4931 ClientWidth - xPalace + xSizeBig + 1, yPalace + ySizeBig + 1,
4932 $FFFFFF, $B0B0B0);
4933 BitBltBitmap(Panel, ClientWidth - xPalace, yPalace, xSizeBig,
4934 ySizeBig, HGrSystem2.Data, 70, 123);
4935 end
4936 else if MyRO.NatBuilt[imPalace] > 0 then
4937 ImpImage(Panel.Canvas, ClientWidth - xPalace, yPalace, imPalace, -1,
4938 GameMode <> cMovie
4939 { (GameMode <> cMovie) and (MyRO.Government <> gAnarchy) } )
4940 else
4941 ImpImage(Panel.Canvas, ClientWidth - xPalace, yPalace, 21, -1,
4942 GameMode <> cMovie
4943 { (GameMode <> cMovie) and (MyRO.Government <> gAnarchy) } );
4944 end;
4945
4946 if GameMode = cMovie then
4947 ScreenTools.Frame(Panel.Canvas, xMini + 1, yMini + 1,
4948 xMini + 2 + G.lx * 2, yMini + 2 + G.ly, $000000, $000000)
4949 else
4950 begin
4951 ScreenTools.Frame(Panel.Canvas, xMini + 1, yMini + 1,
4952 xMini + 2 + G.lx * 2, yMini + 2 + G.ly, $B0B0B0, $FFFFFF);
4953 RFrame(Panel.Canvas, xMini, yMini, xMini + 3 + G.lx * 2, yMini + 3 + G.ly,
4954 $FFFFFF, $B0B0B0);
4955 end;
4956 CopyMiniToPanel;
4957 if ClientMode <> cEditMap then // MapBtn icons
4958 for I := 0 to 5 do
4959 if I <> 3 then
4960 Dump(Panel, HGrSystem, xMini + G.lx - 42 + 16 * I, PanelHeight - 26,
4961 8, 8, 121 + I * 9, 61);
4962
4963 if ClientMode = cEditMap then begin
4964 for I := 0 to TrRow - 1 do
4965 trix[I] := -1;
4966 Count := 0;
4967 for I := 0 to Length(BrushTypes) - 1 do
4968 begin // display terrain types
4969 if (Count >= TrRow * VerticalScrollBar.Position) and (Count < TrRow * (VerticalScrollBar.Position + 1))
4970 then begin
4971 trix[Count - TrRow * VerticalScrollBar.Position] := BrushTypes[I];
4972 X := (Count - TrRow * VerticalScrollBar.Position) * TrPitch;
4973 SrcBase.X := -1;
4974 case BrushTypes[I] of
4975 0..8: Src := Point(BrushTypes[I], 0);
4976 9..30: begin
4977 SrcBase := Point(2, 2);
4978 Src := Point(0, 2 * Integer(BrushTypes[I]) - 15);
4979 end;
4980 fRiver: Src := Point(7, 14);
4981 fRoad: Src := Point(0, 9);
4982 fRR: Src := Point(0, 10);
4983 fCanal: Src := Point(0, 11);
4984 fPoll: Src := Point(6, 12);
4985 fDeadLands, fDeadLands or fCobalt, fDeadLands or fUranium,
4986 fDeadLands or fMercury:
4987 begin
4988 SrcBase := Point(6, 2);
4989 Src := Point(8, 12 + BrushTypes[I] shr 25);
4990 end;
4991 tiIrrigation, tiFarm, tiMine, tiBase:
4992 Src := Point(BrushTypes[I] shr 12 - 1, 12);
4993 tiFort: begin
4994 Src := Point(3, 12);
4995 SrcBase := Point(7, 12);
4996 end;
4997 fPrefStartPos: Src := Point(0, 1);
4998 fStartPos: Src := Point(0, 2);
4999 end;
5000 with MainMap do begin
5001 if SrcBase.X >= 0 then
5002 Sprite(Panel, HGrTerrain, xTroop + 2 + X, yTroop + 9 - yyt, xxt * 2,
5003 yyt * 3, 1 + SrcBase.X * (xxt * 2 + 1),
5004 1 + SrcBase.Y * (yyt * 3 + 1));
5005 Sprite(Panel, HGrTerrain, xTroop + 2 + X, yTroop + 9 - yyt, xxt * 2,
5006 yyt * 3, 1 + Src.X * (xxt * 2 + 1), 1 + Src.Y * (yyt * 3 + 1));
5007 if BrushTypes[I] = BrushType then begin
5008 ScreenTools.Frame(Panel.Canvas, xTroop + 2 + X,
5009 yTroop + 7 - yyt div 2, xTroop + 2 * xxt + X,
5010 yTroop + 2 * yyt + 11, $000000, $000000);
5011 ScreenTools.Frame(Panel.Canvas, xTroop + 1 + X,
5012 yTroop + 6 - yyt div 2, xTroop + 2 * xxt - 1 + X,
5013 yTroop + 2 * yyt + 10, MainTexture.ColorMark, MainTexture.ColorMark);
5014 end;
5015 end;
5016 end;
5017 Inc(Count);
5018 end;
5019 case BrushType of
5020 fDesert, fPrairie, fTundra, fArctic, fSwamp, fHills, fMountains:
5021 S := Phrases.Lookup('TERRAIN', BrushType);
5022 fShore:
5023 S := Format(Phrases.Lookup('TWOTERRAINS'),
5024 [Phrases.Lookup('TERRAIN', fOcean), Phrases.Lookup('TERRAIN',
5025 fShore)]);
5026 fGrass:
5027 S := Format(Phrases.Lookup('TWOTERRAINS'),
5028 [Phrases.Lookup('TERRAIN', fGrass), Phrases.Lookup('TERRAIN',
5029 fGrass + 12)]);
5030 fForest:
5031 S := Format(Phrases.Lookup('TWOTERRAINS'),
5032 [Phrases.Lookup('TERRAIN', fForest), Phrases.Lookup('TERRAIN',
5033 fJungle)]);
5034 fRiver:
5035 S := Phrases.Lookup('RIVER');
5036 fDeadLands, fDeadLands or fCobalt, fDeadLands or fUranium,
5037 fDeadLands or fMercury:
5038 S := Phrases.Lookup('TERRAIN', 3 * 12 + BrushType shr 25);
5039 fPrefStartPos:
5040 S := Phrases.Lookup('MAP_PREFSTART');
5041 fStartPos:
5042 S := Phrases.Lookup('MAP_START');
5043 fPoll:
5044 S := Phrases.Lookup('POLL');
5045 else // terrain improvements
5046 begin
5047 case BrushType of
5048 fRoad: I := 1;
5049 fRR: I := 2;
5050 tiIrrigation: I := 4;
5051 tiFarm: I := 5;
5052 tiMine: I := 7;
5053 fCanal: I := 8;
5054 tiFort: I := 10;
5055 tiBase: I := 12;
5056 end;
5057 S := Phrases.Lookup('JOBRESULT', I);
5058 end;
5059 end;
5060 LoweredTextOut(Panel.Canvas, -1, MainTexture, xTroop + 1,
5061 PanelHeight - 19, S);
5062 end
5063 else if TroopLoc >= 0 then
5064 begin
5065 Brush.Style := TBrushStyle.bsClear;
5066 if UnFocus >= 0 then
5067 with MyUn^[UnFocus] do
5068 with MyModel^[mix] do
5069 if (Me >= 0) and (Me < nPl) then
5070 begin { display info about selected unit }
5071 if Job = jCity then
5072 mixShow := -1 // building site
5073 else
5074 mixShow := mix;
5075 with Tribe[Me].ModelPicture[mixShow] do
5076 begin
5077 Sprite(Panel, HGr, xMidPanel + 7 + 12, yTroop + 1, 64, 48,
5078 pix mod 10 * 65 + 1, pix div 10 * 49 + 1);
5079 if MyUn[UnFocus].Flags and unFortified <> 0 then
5080 Sprite(Panel, HGrStdUnits, xMidPanel + 7 + 12, yTroop + 1,
5081 xxu * 2, yyu * 2, 1 + 6 * (xxu * 2 + 1), 1);
5082 end;
5083
5084 MakeBlue(Panel, xMidPanel + 7 + 12 + 10, yTroop - 13, 44, 12);
5085 S := MovementToString(MyUn[UnFocus]);
5086 RisedTextOut(Panel.Canvas, xMidPanel + 7 + 12 + 32 -
5087 BiColorTextWidth(Panel.Canvas, S) div 2, yTroop - 16, S);
5088
5089 S := IntToStr(Health) + '%';
5090 LightGradient(Panel.Canvas, xMidPanel + 7 + 12 + 7, PanelHeight - 22,
5091 (Health + 1) div 2, (ColorOfHealth(Health) and $FEFEFE shr 2) * 3);
5092 if Health < 100 then
5093 LightGradient(Panel.Canvas, xMidPanel + 7 + 12 + 7 + (Health + 1)
5094 div 2, PanelHeight - 22, 50 - (Health + 1) div 2, $000000);
5095 RisedTextOut(Panel.Canvas, xMidPanel + 7 + 12 + 32 -
5096 BiColorTextWidth(Panel.Canvas, S) div 2, PanelHeight - 23, S);
5097
5098 FrameImage(Panel.Canvas, HGrSystem.Data,
5099 xMidPanel + 7 + xUnitText, yTroop + 15, 12, 14,
5100 121 + Exp div ExpCost * 13, 28);
5101 if Job = jCity then
5102 S := Tribe[Me].ModelName[-1]
5103 else
5104 S := Tribe[Me].ModelName[mix];
5105 if Home >= 0 then
5106 begin
5107 LoweredTextOut(Panel.Canvas, -1, MainTexture,
5108 xMidPanel + 7 + xUnitText + 18, yTroop + 5, S);
5109 LoweredTextOut(Panel.Canvas, -1, MainTexture,
5110 xMidPanel + 7 + xUnitText + 18, yTroop + 21,
5111 '(' + CityName(MyCity[Home].ID) + ')');
5112 end
5113 else
5114 LoweredTextOut(Panel.Canvas, -1, MainTexture,
5115 xMidPanel + 7 + xUnitText + 18, yTroop + 13, S);
5116 end;
5117
5118 if (UnFocus >= 0) and (MyUn[UnFocus].Loc <> TroopLoc) then
5119 begin // divide panel
5120 if SmallScreen and not Supervising then
5121 X := xTroop - 8
5122 else
5123 X := xTroop - 152;
5124 Pen.Color := MainTexture.ColorBevelShade;
5125 MoveTo(X - 1, PanelHeight - MidPanelHeight + 2);
5126 LineTo(X - 1, PanelHeight);
5127 Pen.Color := MainTexture.ColorBevelLight;
5128 MoveTo(X, PanelHeight - MidPanelHeight + 2);
5129 LineTo(X, PanelHeight);
5130 end;
5131
5132 for I := 0 to 23 do
5133 trix[I] := -1;
5134 if MyMap[TroopLoc] and fUnit <> 0 then
5135 begin
5136 if MyMap[TroopLoc] and fOwned <> 0 then
5137 begin
5138 if (TrCnt > 1) or (UnFocus < 0) or (MyUn[UnFocus].Loc <> TroopLoc)
5139 then
5140 begin
5141 LoweredTextOut(Panel.Canvas, -1, MainTexture, xTroop + 10,
5142 PanelHeight - 24, Phrases.Lookup('PRESENT'));
5143 Server(sGetDefender, Me, TroopLoc, uixDefender);
5144 Count := 0;
5145 for Prio := True downto False do
5146 for uix := 0 to MyRO.nUn - 1 do
5147 if (uix = uixDefender) = Prio then
5148 begin // display own units
5149 unx := MyUn[uix];
5150 if unx.Loc = TroopLoc then
5151 begin
5152 if (Count >= TrRow * VerticalScrollBar.Position) and
5153 (Count < TrRow * (VerticalScrollBar.Position + 1)) and
5154 (Me >= 0) and (Me < nPl) then
5155 begin
5156 trix[Count - TrRow * VerticalScrollBar.Position] := uix;
5157 MakeUnitInfo(Me, unx, UnitInfo);
5158 X := (Count - TrRow * VerticalScrollBar.Position) * TrPitch;
5159 if uix = UnFocus then
5160 begin
5161 ScreenTools.Frame(Panel.Canvas, xTroop + 4 + X,
5162 yTroop + 3, xTroop + 64 + X, yTroop + 47,
5163 $000000, $000000);
5164 ScreenTools.Frame(Panel.Canvas, xTroop + 3 + X,
5165 yTroop + 2, xTroop + 63 + X, yTroop + 46,
5166 MainTexture.ColorMark, MainTexture.ColorMark);
5167 end
5168 else if (unx.Master >= 0) and (unx.Master = UnFocus) then
5169 begin
5170 CFrame(Panel.Canvas, xTroop + 4 + X, yTroop + 3,
5171 xTroop + 64 + X, yTroop + 47, 8, $000000);
5172 CFrame(Panel.Canvas, xTroop + 3 + X, yTroop + 2,
5173 xTroop + 63 + X, yTroop + 46, 8, MainTexture.ColorMark);
5174 end;
5175 NoMapPanel.SetOutput(Panel);
5176 NoMapPanel.PaintUnit(xTroop + 2 + X, yTroop + 1, UnitInfo,
5177 unx.Status);
5178 if (ClientMode < scContact) and
5179 ((unx.Job > jNone) or
5180 (unx.Status and (usStay or usRecover or usGoto) <> 0))
5181 then
5182 Sprite(Panel, HGrSystem, xTroop + 2 + 60 - 20 + X,
5183 yTroop + 35, 20, 20, 81, 25);
5184
5185 if not Supervising then
5186 begin
5187 MakeBlue(Panel, xTroop + 2 + 10 + X,
5188 yTroop - 13, 44, 12);
5189 S := MovementToString(unx);
5190 RisedTextOut(Panel.Canvas,
5191 xTroop + X + 34 - BiColorTextWidth(Panel.Canvas, S)
5192 div 2, yTroop - 16, S);
5193 end;
5194 end;
5195 Inc(Count);
5196 end;
5197 end; // for uix:=0 to MyRO.nUn-1
5198 Assert(Count = TrCnt);
5199 end;
5200 end
5201 else
5202 begin
5203 LoweredTextOut(Panel.Canvas, -1, MainTexture, xTroop + 8,
5204 PanelHeight - 24, Phrases.Lookup('PRESENT'));
5205 Server(sGetUnits, Me, TroopLoc, Count);
5206 for I := 0 to Count - 1 do
5207 if (I >= TrRow * VerticalScrollBar.Position) and (I < TrRow * (VerticalScrollBar.Position + 1)) then
5208 begin // display enemy units
5209 trix[I - TrRow * VerticalScrollBar.Position] := I;
5210 X := (I - TrRow * VerticalScrollBar.Position) * TrPitch;
5211 NoMapPanel.SetOutput(Panel);
5212 NoMapPanel.PaintUnit(xTroop + 2 + X, yTroop + 1,
5213 MyRO.EnemyUn[MyRO.nEnemyUn + I], 0);
5214 end;
5215 end;
5216 end;
5217 if not SmallScreen or Supervising then
5218 begin // show terrain and improvements
5219 with NoMapPanel do
5220 PaintZoomedTile(Panel, xTerrain - xxt * 2, 110 - yyt * 3, TroopLoc);
5221 if (UnFocus >= 0) and (MyUn[UnFocus].Job <> jNone) then begin
5222 JobFocus := MyUn[UnFocus].Job;
5223 Server(sGetJobProgress, Me, MyUn[UnFocus].Loc, JobProgressData);
5224 MakeBlue(Panel, xTerrain - 72, 148 - 17, 144, 31);
5225 PaintRelativeProgressBar(Panel.Canvas, 3, xTerrain - 68, 148 + 3, 63,
5226 JobProgressData[JobFocus].Done,
5227 JobProgressData[JobFocus].NextTurnPlus,
5228 JobProgressData[JobFocus].Required, True, MainTexture);
5229 S := Format('%s/%s',
5230 [ScreenTools.MovementToString(JobProgressData[JobFocus].Done),
5231 ScreenTools.MovementToString(JobProgressData[JobFocus].Required)]);
5232 RisedTextOut(Panel.Canvas, xTerrain + 6, 148 - 3, S);
5233 Tile := MyMap[MyUn[UnFocus].Loc];
5234 if (JobFocus = jRoad) and (Tile and fRiver <> 0) then
5235 JobFocus := nJob + 0
5236 else if (JobFocus = jRR) and (Tile and fRiver <> 0) then
5237 JobFocus := nJob + 1
5238 else if JobFocus = jClear then
5239 begin
5240 if Tile and fTerrain = fForest then
5241 JobFocus := nJob + 2
5242 else if Tile and fTerrain = fDesert then
5243 JobFocus := nJob + 3
5244 else
5245 JobFocus := nJob + 4;
5246 end;
5247 S := Phrases.Lookup('JOBRESULT', JobFocus);
5248 RisedTextOut(Panel.Canvas, xTerrain - BiColorTextWidth(Panel.Canvas,
5249 S) div 2, 148 - 19, S);
5250 end;
5251 if MyMap[TroopLoc] and (fTerrain or fSpecial) = fGrass or fSpecial1 then
5252 S := Phrases.Lookup('TERRAIN', fGrass + 12)
5253 else if MyMap[TroopLoc] and fDeadLands <> 0 then
5254 S := Phrases.Lookup('TERRAIN', 3 * 12)
5255 else if (MyMap[TroopLoc] and fTerrain = fForest) and
5256 IsJungle(TroopLoc div G.lx) then
5257 S := Phrases.Lookup('TERRAIN', fJungle)
5258 else
5259 S := Phrases.Lookup('TERRAIN', MyMap[TroopLoc] and fTerrain);
5260 RisedTextOut(Panel.Canvas, xTerrain - BiColorTextWidth(Panel.Canvas, S)
5261 div 2, 99, S);
5262 end;
5263
5264 if TerrainBtn.Visible then
5265 with TerrainBtn do
5266 RFrame(Panel.Canvas, Left - 1, Top - Self.ClientHeight +
5267 (PanelHeight - 1), Left + Width, Top + Height - Self.ClientHeight +
5268 PanelHeight, MainTexture.ColorBevelShade, MainTexture.ColorBevelLight)
5269 end; { if TroopLoc >= 0 }
5270 end;
5271
5272 for I := 0 to ControlCount - 1 do
5273 if Controls[I] is TButtonB then
5274 with TButtonB(Controls[I]) do
5275 begin
5276 if Visible then
5277 begin
5278 Dump(Panel, HGrSystem, Left, Top - Self.ClientHeight + PanelHeight,
5279 25, 25, 169, 243);
5280 Sprite(Panel, HGrSystem, Left, Top - Self.ClientHeight + PanelHeight,
5281 25, 25, 1 + 26 * ButtonIndex, 337);
5282 RFrame(Panel.Canvas, Left - 1, Top - Self.ClientHeight +
5283 (PanelHeight - 1), Left + Width, Top + Height - Self.ClientHeight +
5284 PanelHeight, MainTexture.ColorBevelShade, MainTexture.ColorBevelLight);
5285 end;
5286 end;
5287
5288 if ClientMode <> cEditMap then
5289 begin
5290 for I := 0 to ControlCount - 1 do
5291 if Controls[I] is TButtonC then
5292 with TButtonC(Controls[I]) do
5293 begin
5294 Dump(Panel, HGrSystem, Left, Top - Self.ClientHeight + PanelHeight,
5295 12, 12, 169, 178 + 13 * ButtonIndex);
5296 RFrame(Panel.Canvas, Left - 1, Top - Self.ClientHeight +
5297 (PanelHeight - 1), Left + Width, Top + Height - Self.ClientHeight +
5298 PanelHeight, MainTexture.ColorBevelShade, MainTexture.ColorBevelLight);
5299 end;
5300 end;
5301 EOT.SetBack(Panel.Canvas, EOT.Left, EOT.Top - (ClientHeight - PanelHeight));
5302 SmartRectInvalidate(0, ClientHeight - PanelHeight, ClientWidth, ClientHeight);
5303
5304 // topbar
5305 xTreasurySection := ClientWidth div 2 - 172;
5306 xResearchSection := ClientWidth div 2;
5307 // ClientWidth div 2 + 68 = maximum to right
5308 FillLarge(TopBar.Canvas, 0, 0, ClientWidth, TopBarHeight - 3,
5309 ClientWidth div 2);
5310 with TopBar.Canvas do
5311 begin
5312 Pen.Color := $000000;
5313 MoveTo(0, TopBarHeight - 1);
5314 LineTo(ClientWidth, TopBarHeight - 1);
5315 Pen.Color := MainTexture.ColorBevelShade;
5316 MoveTo(0, TopBarHeight - 2);
5317 LineTo(ClientWidth, TopBarHeight - 2);
5318 MoveTo(0, TopBarHeight - 3);
5319 LineTo(ClientWidth, TopBarHeight - 3);
5320 Pen.Color := MainTexture.ColorBevelLight;
5321 ScreenTools.Frame(TopBar.Canvas, 40, -1, xTreasurySection - 1,
5322 TopBarHeight - 7, MainTexture.ColorBevelShade, MainTexture.ColorBevelLight);
5323 ScreenTools.Frame(TopBar.Canvas, xResearchSection + 332, -1, ClientWidth,
5324 TopBarHeight - 7, MainTexture.ColorBevelShade, MainTexture.ColorBevelLight);
5325 end;
5326 if GameMode <> cMovie then
5327 ImageOp_BCC(TopBar, Templates.Data, Point(2, 1), MenuLogo.BoundsRect, $BFBF20, $4040DF);
5328 if MyRO.nCity > 0 then
5329 begin
5330 TrueMoney := MyRO.Money;
5331 TrueResearch := MyRO.Research;
5332 if Supervising then
5333 begin // normalize values from after-turn state
5334 Dec(TrueMoney, TaxSum);
5335 if TrueMoney < 0 then
5336 TrueMoney := 0; // shouldn't happen
5337 Dec(TrueResearch, ScienceSum);
5338 if TrueResearch < 0 then
5339 TrueResearch := 0; // shouldn't happen
5340 end;
5341
5342 // treasury section
5343 ImageOp_BCC(TopBar, Templates.Data, Point(xTreasurySection + 8, 1), TreasuryIcon.BoundsRect,
5344 $40A040, $4030C0);
5345 S := IntToStr(TrueMoney);
5346 LoweredTextOut(TopBar.Canvas, -1, MainTexture, xTreasurySection + 48, 0,
5347 S + '%c');
5348 if MyRO.Government <> gAnarchy then
5349 begin
5350 ImageOp_BCC(TopBar, Templates.Data, Point(xTreasurySection + 48, 22), ChangeIcon.BoundsRect,
5351 $0000C0, $0080C0);
5352 if TaxSum >= 0 then
5353 S := Format(Phrases.Lookup('MONEYGAINPOS'), [TaxSum])
5354 else
5355 S := Format(Phrases.Lookup('MONEYGAINNEG'), [TaxSum]);
5356 LoweredTextOut(TopBar.Canvas, -1, MainTexture, xTreasurySection + 48 +
5357 15, 18, S);
5358 end;
5359
5360 // research section
5361 ImageOp_BCC(TopBar, Templates.Data, Point(xResearchSection + 8, 1), ResearchIcon.BoundsRect,
5362 $FF0000, $00FFE0);
5363 if MyData.FarTech <> adNexus then
5364 begin
5365 if MyRO.ResearchTech < 0 then
5366 CostFactor := 2
5367 else if (MyRO.ResearchTech = adMilitary) or
5368 (MyRO.Tech[MyRO.ResearchTech] = tsSeen) then
5369 CostFactor := 1
5370 else if MyRO.ResearchTech in FutureTech then
5371 if MyRO.Government = gFuture then
5372 CostFactor := 4
5373 else
5374 CostFactor := 8
5375 else
5376 CostFactor := 2;
5377 Server(sGetTechCost, Me, 0, I);
5378 CostFactor := CostFactor * 22; // length of progress bar
5379 PaintRelativeProgressBar(TopBar.Canvas, 2, xResearchSection + 48 + 1, 26,
5380 CostFactor, TrueResearch, ScienceSum, I, True, MainTexture);
5381
5382 if MyRO.ResearchTech < 0 then
5383 S := Phrases.Lookup('SCIENCE')
5384 else if MyRO.ResearchTech = adMilitary then
5385 S := Phrases.Lookup('INITUNIT')
5386 else
5387 begin
5388 S := Phrases.Lookup('ADVANCES', MyRO.ResearchTech);
5389 if MyRO.ResearchTech in FutureTech then
5390 if MyRO.Tech[MyRO.ResearchTech] >= 1 then
5391 S := S + ' ' + IntToStr(MyRO.Tech[MyRO.ResearchTech] + 1)
5392 else
5393 S := S + ' 1';
5394 end;
5395 if ScienceSum > 0 then
5396 begin
5397 { j := (i - MyRO.Research - 1) div ScienceSum + 1;
5398 if J < 1 then J := 1;
5399 if J > 1 then
5400 S := Format(Phrases.Lookup('TECHWAIT'), [S, J]); }
5401 LoweredTextOut(TopBar.Canvas, -1, MainTexture,
5402 xResearchSection + 48, 0, S);
5403 end
5404 else
5405 LoweredTextOut(TopBar.Canvas, -1, MainTexture,
5406 xResearchSection + 48, 0, S);
5407 end
5408 else
5409 CostFactor := 0;
5410 if (MyData.FarTech <> adNexus) and (ScienceSum > 0) then
5411 begin
5412 ImageOp_BCC(TopBar, Templates.Data, Point(xResearchSection + 48 + CostFactor + 11,
5413 22), ChangeIcon.BoundsRect, $0000C0, $0080C0);
5414 S := Format(Phrases.Lookup('TECHGAIN'), [ScienceSum]);
5415 LoweredTextOut(TopBar.Canvas, -1, MainTexture, xResearchSection + 48 +
5416 CostFactor + 26, 18, S);
5417 end;
5418 end;
5419 if ClientMode <> cEditMap then
5420 begin
5421 TopBar.Canvas.Font.Assign(UniFont[ftCaption]);
5422 S := TurnToString(MyRO.Turn);
5423 RisedTextOut(TopBar.Canvas,
5424 40 + (xTreasurySection - 40 - BiColorTextWidth(TopBar.Canvas, S))
5425 div 2, 6, S);
5426 TopBar.Canvas.Font.Assign(UniFont[ftNormal]);
5427 end;
5428 RectInvalidate(0, 0, ClientWidth, TopBarHeight);
5429end;
5430
5431procedure TMainScreen.FocusNextUnit(Dir: Integer);
5432var
5433 I, uix, NewFocus: Integer;
5434begin
5435 if ClientMode >= scContact then
5436 Exit;
5437 DestinationMarkON := False;
5438 PaintDestination;
5439 NewFocus := -1;
5440 for I := 1 to MyRO.nUn do begin
5441 uix := (UnFocus + I * Dir + MyRO.nUn) mod MyRO.nUn;
5442 if (MyUn[uix].Loc >= 0) and (MyUn[uix].Status and usStay = 0) then begin
5443 NewFocus := uix;
5444 Break;
5445 end;
5446 end;
5447 if NewFocus >= 0 then begin
5448 SetUnFocus(NewFocus);
5449 SetTroopLoc(MyUn[NewFocus].Loc);
5450 FocusOnLoc(TroopLoc, flRepaintPanel);
5451 end;
5452end;
5453
5454procedure TMainScreen.FocusOnLoc(Loc: Integer; Options: Integer = 0);
5455var
5456 dx: Integer;
5457 Outside, Changed: Boolean;
5458begin
5459 with MainMap do begin
5460 dx := G.lx + 1 - (xw - Loc + G.lx * 1024 + 1) mod G.lx;
5461 Outside := (dx >= (MapWidth + 1) div (xxt * 2) - 2) or (ywmax > 0) and
5462 ((yw > 0) and (Loc div G.lx <= yw + 1) or (yw < ywmax) and
5463 (Loc div G.lx >= yw + (MapHeight - 1) div yyt - 2));
5464 end;
5465 Changed := True;
5466 if Outside then begin
5467 Centre(Loc);
5468 PaintAllMaps;
5469 end
5470 else if not MapValid then
5471 PaintAllMaps
5472 else
5473 Changed := False;
5474 if Options and flRepaintPanel <> 0 then
5475 PanelPaint;
5476 if Changed and (Options and flImmUpdate <> 0) then
5477 Update;
5478end;
5479
5480procedure TMainScreen.NextUnit(NearLoc: Integer; AutoTurn: Boolean);
5481var
5482 Dist, TestDist: Single;
5483 I, uix, NewFocus: Integer;
5484 GotoOnly: Boolean;
5485begin
5486 Dist := 0;
5487 if ClientMode >= scContact then
5488 Exit;
5489 DestinationMarkON := False;
5490 PaintDestination;
5491 for GotoOnly := GoOnPhase downto False do begin
5492 NewFocus := -1;
5493 for I := 1 to MyRO.nUn do begin
5494 uix := (UnFocus + I) mod MyRO.nUn;
5495 if (MyUn[uix].Loc >= 0) and (MyUn[uix].Job = jNone) and
5496 (MyUn[uix].Status and (usStay or usRecover or usWaiting) = usWaiting)
5497 and (not GotoOnly or (MyUn[uix].Status and usGoto <> 0)) then
5498 if NearLoc < 0 then begin
5499 NewFocus := uix;
5500 Break;
5501 end else begin
5502 TestDist := Distance(NearLoc, MyUn[uix].Loc);
5503 if (NewFocus < 0) or (TestDist < Dist) then begin
5504 NewFocus := uix;
5505 Dist := TestDist;
5506 end;
5507 end;
5508 end;
5509 if GotoOnly then
5510 if NewFocus < 0 then GoOnPhase := False
5511 else Break;
5512 end;
5513 if NewFocus >= 0 then begin
5514 SetUnFocus(NewFocus);
5515 SetTroopLoc(MyUn[NewFocus].Loc);
5516 FocusOnLoc(TroopLoc, flRepaintPanel);
5517 end else
5518 if AutoTurn and not mWaitTurn.Checked then begin
5519 TurnComplete := True;
5520 SetUnFocus(-1);
5521 SetTroopLoc(-1);
5522 PostMessage(Handle, WM_EOT, 0, 0);
5523 end else begin
5524 if { (UnFocus>=0) and } not TurnComplete and EOT.Visible then
5525 Play('TURNEND');
5526 TurnComplete := True;
5527 SetUnFocus(-1);
5528 SetTroopLoc(-1);
5529 PanelPaint;
5530 end;
5531end;
5532
5533procedure TMainScreen.Scroll(dx, dy: Integer);
5534begin
5535 xw := (xw + G.lx + dx) mod G.lx;
5536 if ywmax > 0 then
5537 begin
5538 yw := yw + 2 * dy;
5539 if yw < 0 then
5540 yw := 0
5541 else if yw > ywmax then
5542 yw := ywmax;
5543 end;
5544 MainOffscreenPaint;
5545 xwMini := xw;
5546 ywMini := yw;
5547 MiniMapPaint;
5548 CopyMiniToPanel;
5549 RectInvalidate(xMini + 2, TopBarHeight + MapHeight - Overlap + yMini + 2,
5550 xMini + 2 + G.lx * 2, TopBarHeight + MapHeight - Overlap + yMini +
5551 2 + G.ly);
5552 Update;
5553end;
5554
5555procedure TMainScreen.Timer1Timer(Sender: TObject);
5556var
5557 dx, dy, ScrollSpeed: Integer;
5558 MousePos: TPoint;
5559const
5560 ScrollWidth = 5;
5561begin
5562 if Idle and (Me >= 0) and (GameMode <> cMovie) then
5563 if Assigned(Screen.ActiveForm) and
5564 (TFormStateType.fsModal in Screen.ActiveForm.FormState) or
5565 (Screen.ActiveForm is TBufferedDrawDlg) and
5566 (TBufferedDrawDlg(Screen.ActiveForm).WindowMode <> wmPersistent) then
5567 begin
5568 BlinkTime := BlinkOnTime + BlinkOffTime - 1;
5569 if not BlinkON then
5570 begin
5571 BlinkON := True;
5572 if UnFocus >= 0 then
5573 PaintLocTemp(MyUn[UnFocus].Loc)
5574 else if TurnComplete and not Supervising then
5575 EOT.SetButtonIndexFast(eotBlinkOn);
5576 end;
5577 end else begin
5578 if Application.Active and not mScrollOff.Checked then
5579 begin
5580 if mScrollFast.Checked then ScrollSpeed := 2
5581 else ScrollSpeed := 1;
5582 dx := 0;
5583 dy := 0;
5584 MousePos := ScreenToClient(Mouse.CursorPos);
5585 if MousePos.Y < Height - PanelHeight then
5586 if (MousePos.X >= 0) and (MousePos.X <= ScrollWidth) then
5587 dx := -ScrollSpeed // scroll left
5588 else if (MousePos.X <= Width) and (MousePos.X >= Width - ScrollWidth - 1) then
5589 dx := ScrollSpeed; // scroll right
5590 if (MousePos.Y >= 0) and (MousePos.Y <= ScrollWidth) then
5591 dy := -ScrollSpeed // scroll up
5592 else if (MousePos.Y <= Height) and (MousePos.Y >= Height - ScrollWidth - 1) and
5593 (MousePos.X >= TerrainBtn.Left + TerrainBtn.Width) and
5594 (MousePos.X < xRightPanel + 10 - 8) then
5595 dy := ScrollSpeed; // scroll down
5596 if (dx <> 0) or (dy <> 0) then
5597 begin
5598 if Assigned(Screen.ActiveForm) and
5599 (Screen.ActiveForm <> MainScreen) and
5600 (@Screen.ActiveForm.OnDeactivate <> nil) then
5601 Screen.ActiveForm.OnDeactivate(nil);
5602 Scroll(dx, dy);
5603 end;
5604 end;
5605
5606 BlinkTime := (BlinkTime + 1) mod (BlinkOnTime + BlinkOffTime);
5607 BlinkON := BlinkTime >= BlinkOffTime;
5608 DestinationMarkON := True;
5609 if UnFocus >= 0 then
5610 begin
5611 if (BlinkTime = 0) or (BlinkTime = BlinkOffTime) then
5612 begin
5613 PaintLocTemp(MyUn[UnFocus].Loc, pltsBlink);
5614 PaintDestination;
5615 // if MoveHintToLoc >= 0 then
5616 // ShowMoveHint(MoveHintToLoc, True);
5617 end;
5618 end
5619 else if TurnComplete and not Supervising then
5620 begin
5621 if BlinkTime = 0 then
5622 EOT.SetButtonIndexFast(eotBlinkOff)
5623 else if BlinkTime = BlinkOffTime then
5624 EOT.SetButtonIndexFast(eotBlinkOn);
5625 end;
5626 end;
5627end;
5628
5629procedure TMainScreen.SetMapPos(Loc: Integer; MapPos: TPoint);
5630begin
5631 with MainMap do begin
5632 if FastScrolling and MapValid then
5633 Update;
5634 // necessary because ScrollDC for form canvas is called after
5635 xw := (Loc mod G.lx) - (MapPos.X - ((xxt * 2) * ((Loc div G.lx) and 1)) div 2)
5636 div (xxt * 2);
5637 xw := (xw + G.lx) mod G.lx;
5638 if ywmax <= 0 then yw := ywcenter
5639 else begin
5640 yw := (Loc div G.lx - (MapPos.Y * 2) div (yyt * 2) + 1) and not 1;
5641 if yw < 0 then yw := 0
5642 else if yw > ywmax then yw := ywmax;
5643 end;
5644 end;
5645end;
5646
5647procedure TMainScreen.Centre(Loc: Integer);
5648begin
5649 SetMapPos(Loc, Point(MapWidth div 2, MapHeight div 2));
5650end;
5651
5652function TMainScreen.ZoomToCity(Loc: Integer; NextUnitOnClose: Boolean = False;
5653 ShowEvent: Integer = 0): Boolean;
5654begin
5655 Result := MyMap[Loc] and (fOwned or fSpiedOut) <> 0;
5656 if Result then
5657 with CityDlg do
5658 begin
5659 if ClientMode >= scContact then
5660 begin
5661 CloseAction := None;
5662 RestoreUnFocus := -1;
5663 end
5664 else if NextUnitOnClose then
5665 begin
5666 CloseAction := StepFocus;
5667 RestoreUnFocus := -1;
5668 end
5669 else if not Visible then
5670 begin
5671 CloseAction := RestoreFocus;
5672 RestoreUnFocus := UnFocus;
5673 end;
5674 SetUnFocus(-1);
5675 SetTroopLoc(Loc);
5676 MarkCityLoc := Loc;
5677 PanelPaint;
5678 ShowNewContent(wmPersistent, Loc, ShowEvent);
5679 end;
5680end;
5681
5682function TMainScreen.LocationOfScreenPixel(X, Y: Integer): Integer;
5683var
5684 qx, qy: Integer;
5685begin
5686 with MainMap do begin
5687 qx := (X * (yyt * 2) + Y * (xxt * 2) + xxt * yyt * 2) div (xxt * yyt * 4) - 1;
5688 qy := (Y * (xxt * 2) - X * (yyt * 2) - xxt * yyt * 2 + 4000 * xxt * yyt)
5689 div (xxt * yyt * 4) - 999;
5690 Result := (xw + (qx - qy + 2048) div 2 - 1024 + G.lx) mod G.lx + G.lx *
5691 (yw + qx + qy);
5692 end;
5693end;
5694
5695function TMainScreen.GetCenterLoc: Integer;
5696begin
5697 Result := (xw + MapWidth div (MainMap.xxt * 4)) mod G.lx +
5698 (yw + MapHeight div (MainMap.yyt * 2)) * G.lx;
5699end;
5700
5701procedure TMainScreen.MapBoxMouseDown(Sender: TObject; Button: TMouseButton;
5702 Shift: TShiftState; X, Y: Integer);
5703var
5704 I, uix, emix, p1, dx, dy, MouseLoc: Integer;
5705 EditTileData: TEditTileData;
5706 M, m2: TMenuItem;
5707 MoveAdviceData: TMoveAdviceData;
5708 DoCenter: Boolean;
5709begin
5710 if GameMode = cMovie then
5711 Exit;
5712
5713 if Assigned(FCityDlg) and CityDlg.Visible then
5714 CityDlg.Close;
5715 if Assigned(FUnitStatDlg) and UnitStatDlg.Visible then
5716 UnitStatDlg.Close;
5717 MouseLoc := LocationOfScreenPixel(X, Y);
5718 if (MouseLoc < 0) or (MouseLoc >= G.lx * G.ly) then
5719 Exit;
5720 if (Button = TMouseButton.mbLeft) and not (ssShift in Shift) then
5721 begin
5722 DoCenter := True;
5723 if ClientMode = cEditMap then
5724 begin
5725 DoCenter := False;
5726 EditTileData.Loc := MouseLoc;
5727 if ssCtrl in Shift then // toggle special resource
5728 case MyMap[MouseLoc] and fTerrain of
5729 fOcean:
5730 EditTileData.NewTile := MyMap[MouseLoc];
5731 fGrass, fArctic:
5732 EditTileData.NewTile := MyMap[MouseLoc] and not fSpecial or
5733 ((MyMap[MouseLoc] shr 5 and 3 + 1) mod 2 shl 5);
5734 else
5735 EditTileData.NewTile := MyMap[MouseLoc] and not fSpecial or
5736 ((MyMap[MouseLoc] shr 5 and 3 + 1) mod 3 shl 5)
5737 end
5738 else if BrushType <= fTerrain then
5739 EditTileData.NewTile := MyMap[MouseLoc] and not fTerrain or fSpecial or
5740 BrushType
5741 else if BrushType and fDeadLands <> 0 then
5742 if MyMap[MouseLoc] and (fDeadLands or fModern) = BrushType and
5743 (fDeadLands or fModern) then
5744 EditTileData.NewTile := MyMap[MouseLoc] and not (fDeadLands or fModern)
5745 else
5746 EditTileData.NewTile := MyMap[MouseLoc] and not (fDeadLands or fModern)
5747 or BrushType
5748 else if BrushType and fTerImp <> 0 then
5749 if MyMap[MouseLoc] and fTerImp = BrushType then
5750 EditTileData.NewTile := MyMap[MouseLoc] and not fTerImp
5751 else
5752 EditTileData.NewTile := MyMap[MouseLoc] and not fTerImp or BrushType
5753 else if BrushType and (fPrefStartPos or fStartPos) <> 0 then
5754 if MyMap[MouseLoc] and (fPrefStartPos or fStartPos) = BrushType and
5755 (fPrefStartPos or fStartPos) then
5756 EditTileData.NewTile := MyMap[MouseLoc] and
5757 not (fPrefStartPos or fStartPos)
5758 else
5759 EditTileData.NewTile := MyMap[MouseLoc] and
5760 not (fPrefStartPos or fStartPos) or BrushType
5761 else
5762 EditTileData.NewTile := MyMap[MouseLoc] xor BrushType;
5763 Server(sEditTile, Me, 0, EditTileData);
5764 Edited := True;
5765 BrushLoc := MouseLoc;
5766 PaintLoc(MouseLoc, 2);
5767 MiniMapPaint;
5768 BitBltBitmap(Panel, xMini + 2, yMini + 2, G.lx * 2, G.ly,
5769 MiniMap.Bitmap, 0, 0);
5770 with MainMap do begin
5771 if ywmax <= 0 then
5772 Frame(Panel.Canvas, xMini + 2 + G.lx - MapWidth div (2 * xxt),
5773 yMini + 2, xMini + 1 + G.lx + MapWidth div (2 * xxt),
5774 yMini + 2 + G.ly - 1, MainTexture.ColorMark, MainTexture.ColorMark)
5775 else
5776 Frame(Panel.Canvas, xMini + 2 + G.lx - MapWidth div (2 * xxt),
5777 yMini + 2 + yw, xMini + 2 + G.lx + MapWidth div (2 * xxt) - 1,
5778 yMini + 2 + yw + MapHeight div yyt - 2, MainTexture.ColorMark,
5779 MainTexture.ColorMark);
5780 end;
5781 RectInvalidate(xMini + 2, TopBarHeight + MapHeight - Overlap + yMini + 2,
5782 xMini + 2 + G.lx * 2, TopBarHeight + MapHeight - Overlap + yMini
5783 + 2 + G.ly);
5784 end
5785 else if MyMap[MouseLoc] and fCity <> 0 then { city clicked }
5786 begin
5787 if MyMap[MouseLoc] and (fOwned or fSpiedOut) <> 0 then
5788 begin
5789 ZoomToCity(MouseLoc);
5790 DoCenter := False;
5791 end
5792 else
5793 begin
5794 UnitStatDlg.ShowNewContent_EnemyCity(wmPersistent, MouseLoc);
5795 DoCenter := False;
5796 end;
5797 end
5798 else if MyMap[MouseLoc] and fUnit <> 0 then { unit clicked }
5799 if MyMap[MouseLoc] and fOwned <> 0 then
5800 begin
5801 DoCenter := False;
5802 if not Supervising and (ClientMode < scContact) then
5803 begin // not in negotiation mode
5804 if (UnFocus >= 0) and (MyUn[UnFocus].Loc = MouseLoc) then
5805 begin // rotate
5806 uix := (UnFocus + 1) mod MyRO.nUn;
5807 I := MyRO.nUn - 1;
5808 while I > 0 do
5809 begin
5810 if (MyUn[uix].Loc = MouseLoc) and (MyUn[uix].Job = jNone) and
5811 (MyUn[uix].Status and (usStay or usRecover or usEnhance or
5812 usWaiting) = usWaiting) then
5813 Break;
5814 Dec(I);
5815 uix := (uix + 1) mod MyRO.nUn;
5816 end;
5817 if I = 0 then
5818 uix := UnFocus;
5819 end
5820 else
5821 Server(sGetDefender, Me, MouseLoc, uix);
5822 if uix <> UnFocus then
5823 SetUnFocus(uix);
5824 TurnComplete := False;
5825 EOT.ButtonIndex := eotGray;
5826 end;
5827 SetTroopLoc(MouseLoc);
5828 PanelPaint;
5829 end // own unit
5830 else if (MyMap[MouseLoc] and fSpiedOut <> 0) and not (ssCtrl in Shift) then
5831 begin
5832 DoCenter := False;
5833 SetTroopLoc(MouseLoc);
5834 PanelPaint;
5835 end
5836 else
5837 begin
5838 DoCenter := False;
5839 UnitStatDlg.ShowNewContent_EnemyLoc(wmPersistent, MouseLoc);
5840 end;
5841 if DoCenter then
5842 begin
5843 Centre(MouseLoc);
5844 PaintAllMaps;
5845 end;
5846 end
5847 else if (ClientMode <> cEditMap) and (Button = TMouseButton.mbRight) and
5848 not (ssShift in Shift) then
5849 begin
5850 if Supervising then
5851 begin
5852 EditLoc := MouseLoc;
5853 Server(sGetModels, Me, 0, nil^);
5854 EmptyMenu(mCreateUnit);
5855 for p1 := 0 to nPl - 1 do
5856 if 1 shl p1 and MyRO.Alive <> 0 then
5857 begin
5858 M := TMenuItem.Create(mCreateUnit);
5859 M.Caption := Tribe[p1].TPhrase('SHORTNAME');
5860 for emix := MyRO.nEnemyModel - 1 downto 0 do
5861 if (MyRO.EnemyModel[emix].Owner = p1) and
5862 (Server(sCreateUnit - sExecute + p1 shl 4, Me,
5863 MyRO.EnemyModel[emix].mix, MouseLoc) >= rExecuted) then
5864 begin
5865 if not Assigned(Tribe[p1].ModelPicture[MyRO.EnemyModel[emix].mix].HGr) then
5866 InitEnemyModel(emix);
5867 m2 := TMenuItem.Create(M);
5868 m2.Caption := Tribe[p1].ModelName[MyRO.EnemyModel[emix].mix];
5869 m2.Tag := p1 shl 16 + MyRO.EnemyModel[emix].mix;
5870 m2.OnClick := CreateUnitClick;
5871 M.Add(m2);
5872 end;
5873 M.Visible := M.Count > 0;
5874 mCreateUnit.Add(M);
5875 end;
5876 if FullScreen then
5877 EditPopup.Popup(Left + X, Top + Y)
5878 else
5879 EditPopup.Popup(Left + X + 4,
5880 Top + Y + GetSystemMetrics(SM_CYCAPTION) + 4);
5881 end
5882 else if (UnFocus >= 0) and (MyUn[UnFocus].Loc <> MouseLoc) then
5883 with MyUn[UnFocus] do
5884 begin
5885 dx := ((MouseLoc mod G.lx * 2 + MouseLoc div G.lx and 1) -
5886 (Loc mod G.lx * 2 + Loc div G.lx and 1) + 3 * G.lx)
5887 mod (2 * G.lx) - G.lx;
5888 dy := MouseLoc div G.lx - Loc div G.lx;
5889 if Abs(dx) + Abs(dy) < 3 then
5890 begin
5891 DestinationMarkON := False;
5892 PaintDestination;
5893 Status := Status and ($FFFF - usStay - usRecover - usGoto - usEnhance)
5894 or usWaiting;
5895 MoveUnit(dx, dy, muAutoNext); { simple move }
5896 end
5897 else if GetMoveAdvice(UnFocus, MouseLoc, MoveAdviceData) >= rExecuted
5898 then
5899 begin
5900 if MyMap[MouseLoc] and (fUnit or fOwned) = fUnit then
5901 begin // check for suicide mission before movement
5902 with MyUn[UnFocus], BattleDlg.Forecast do
5903 begin
5904 pAtt := Me;
5905 mixAtt := mix;
5906 HealthAtt := Health;
5907 ExpAtt := Exp;
5908 FlagsAtt := Flags;
5909 end;
5910 BattleDlg.Forecast.Movement := MyUn[UnFocus].Movement;
5911 if (Server(sGetBattleForecastEx, Me, MouseLoc, BattleDlg.Forecast)
5912 >= rExecuted) and (BattleDlg.Forecast.EndHealthAtt <= 0) then
5913 begin
5914 BattleDlg.uix := UnFocus;
5915 BattleDlg.ToLoc := MouseLoc;
5916 BattleDlg.IsSuicideQuery := True;
5917 BattleDlg.ShowModal;
5918 if BattleDlg.ModalResult <> mrOK then
5919 Exit;
5920 end;
5921 end;
5922 DestinationMarkON := False;
5923 PaintDestination;
5924 Status := Status and not (usStay or usRecover or usEnhance) or
5925 usWaiting;
5926 MoveToLoc(MouseLoc, False); { goto }
5927 end;
5928 end;
5929 end
5930 else if (Button = TMouseButton.mbMiddle) and (UnFocus >= 0) and
5931 (MyModel[MyUn[UnFocus].mix].Kind in [mkSettler, mkSlaves]) then
5932 begin
5933 DestinationMarkON := False;
5934 PaintDestination;
5935 MyUn[UnFocus].Status := MyUn[UnFocus].Status and
5936 ($FFFF - usStay - usRecover - usGoto) or usEnhance;
5937 uix := UnFocus;
5938 if MouseLoc <> MyUn[uix].Loc then
5939 MoveToLoc(MouseLoc, True); { goto }
5940 if (UnFocus = uix) and (MyUn[uix].Loc = MouseLoc) then
5941 mEnhance.Click;
5942 end
5943 else if (Button = TMouseButton.mbLeft) and (ssShift in Shift) and
5944 (MyMap[MouseLoc] and fTerrain <> fUNKNOWN) then
5945 HelpOnTerrain(MouseLoc, wmPersistent)
5946 else if (ClientMode <= cContinue) and (Button = TMouseButton.mbRight) and
5947 (ssShift in Shift) and (UnFocus >= 0) and
5948 (MyMap[MouseLoc] and (fUnit or fOwned) = fUnit) then
5949 begin // battle forecast
5950 with MyUn[UnFocus], BattleDlg.Forecast do
5951 begin
5952 pAtt := Me;
5953 mixAtt := mix;
5954 HealthAtt := Health;
5955 ExpAtt := Exp;
5956 FlagsAtt := Flags;
5957 end;
5958 BattleDlg.Forecast.Movement := MyUn[UnFocus].Movement;
5959 if Server(sGetBattleForecastEx, Me, MouseLoc, BattleDlg.Forecast) >= rExecuted
5960 then
5961 begin
5962 BattleDlg.uix := UnFocus;
5963 BattleDlg.ToLoc := MouseLoc;
5964 BattleDlg.Left := X - BattleDlg.Width div 2;
5965 if BattleDlg.Left < 0 then
5966 BattleDlg.Left := 0
5967 else if BattleDlg.Left + BattleDlg.Width > Screen.DesktopWidth then
5968 BattleDlg.Left := Screen.DesktopWidth - BattleDlg.Width;
5969 BattleDlg.Top := Y - BattleDlg.Height div 2;
5970 if BattleDlg.Top < 0 then
5971 BattleDlg.Top := 0
5972 else if BattleDlg.Top + BattleDlg.Height > Screen.DesktopHeight then
5973 BattleDlg.Top := Screen.DesktopHeight - BattleDlg.Height;
5974 BattleDlg.IsSuicideQuery := False;
5975 BattleDlg.Show;
5976 end;
5977 end;
5978end;
5979
5980function TMainScreen.MoveUnit(dx, dy: Integer; Options: Integer): Integer;
5981// move focused unit to adjacent tile
5982var
5983 I, cix, uix, euix, FromLoc, ToLoc, DirCode, UnFocus0, Defender, Mission, p1,
5984 NewTiles, cixChanged: Integer;
5985 OldToTile: Cardinal;
5986 CityCaptured, IsAttack, OldUnrest, NewUnrest, NeedEcoUpdate, NeedRepaintPanel,
5987 ToTransport, ToShip: Boolean;
5988 PlaneReturnData: TPlaneReturnData;
5989 QueryItem: string;
5990begin
5991 Result := eInvalid;
5992 UnFocus0 := UnFocus;
5993 FromLoc := MyUn[UnFocus].Loc;
5994 ToLoc := dLoc(FromLoc, dx, dy);
5995 if (ToLoc < 0) or (ToLoc >= G.lx * G.ly) then
5996 begin
5997 Result := eInvalid;
5998 Exit;
5999 end;
6000 if MyMap[ToLoc] and fStealthUnit <> 0 then
6001 begin
6002 SoundMessage(Phrases.Lookup('ATTACKSTEALTH'), '');
6003 Exit;
6004 end;
6005 if MyMap[ToLoc] and fHiddenUnit <> 0 then
6006 begin
6007 SoundMessage(Phrases.Lookup('ATTACKSUB'), '');
6008 Exit;
6009 end;
6010
6011 if MyMap[ToLoc] and (fUnit or fOwned) = fUnit then
6012 begin // attack -- search enemy unit
6013 if (MyModel[MyUn[UnFocus].mix].Attack = 0) and
6014 not ((MyModel[MyUn[UnFocus].mix].Cap[mcBombs] > 0) and
6015 (MyUn[UnFocus].Flags and unBombsLoaded <> 0)) then
6016 begin
6017 SoundMessage(Phrases.Lookup('NOATTACKER'), '');
6018 Exit;
6019 end;
6020 euix := MyRO.nEnemyUn - 1;
6021 while (euix >= 0) and (MyRO.EnemyUn[euix].Loc <> ToLoc) do
6022 Dec(euix);
6023 end;
6024
6025 DirCode := dx and 7 shl 4 + dy and 7 shl 7;
6026 Result := Server(sMoveUnit - sExecute + DirCode, Me, UnFocus, nil^);
6027 if (Result < rExecuted) and (MyUn[UnFocus].Job > jNone) then
6028 Server(sStartJob + jNone shl 4, Me, UnFocus, nil^);
6029 if (Result < rExecuted) and (Result <> eNoTime_Move) then
6030 begin
6031 case Result of
6032 eNoTime_Load:
6033 if MyModel[MyUn[UnFocus].mix].Domain = dAir then
6034 SoundMessage(Phrases.Lookup('NOTIMELOADAIR'), 'NOMOVE_TIME')
6035 else
6036 SoundMessage(Format(Phrases.Lookup('NOTIMELOADGROUND'),
6037 [MovementToString(MyModel[MyUn[UnFocus].mix].speed)]),
6038 'NOMOVE_TIME');
6039 eNoTime_Bombard:
6040 SoundMessage(Phrases.Lookup('NOTIMEBOMBARD'), 'NOMOVE_TIME');
6041 eNoTime_Expel:
6042 SoundMessage(Phrases.Lookup('NOTIMEEXPEL'), 'NOMOVE_TIME');
6043 eNoRoad:
6044 SoundMessage(Phrases.Lookup('NOROAD'), 'NOMOVE_DEFAULT');
6045 eNoNav:
6046 SoundMessage(Phrases.Lookup('NONAV'), 'NOMOVE_DEFAULT');
6047 eNoCapturer:
6048 SoundMessage(Phrases.Lookup('NOCAPTURER'), 'NOMOVE_DEFAULT');
6049 eNoBombarder:
6050 SoundMessage(Phrases.Lookup('NOBOMBARDER'), 'NOMOVE_DEFAULT');
6051 eZOC:
6052 ContextMessage(Phrases.Lookup('ZOC'), 'NOMOVE_ZOC', hkText,
6053 HelpDlg.TextIndex('MOVEMENT'));
6054 eTreaty:
6055 if MyMap[ToLoc] and (fUnit or fOwned) <> fUnit
6056 then { no enemy unit -- move }
6057 SoundMessage(Tribe[MyRO.Territory[ToLoc]].TPhrase('PEACE_NOMOVE'),
6058 'NOMOVE_TREATY')
6059 else
6060 SoundMessage(Tribe[MyRO.EnemyUn[euix].Owner]
6061 .TPhrase('PEACE_NOATTACK'), 'NOMOVE_TREATY');
6062 eDomainMismatch:
6063 begin
6064 if (MyModel[MyUn[UnFocus].mix].Domain < dSea) and
6065 (MyMap[ToLoc] and (fUnit or fOwned) = fUnit or fOwned) then
6066 begin // false load attempt
6067 ToShip := False;
6068 ToTransport := False;
6069 for uix := 0 to MyRO.nUn - 1 do
6070 if (MyUn[uix].Loc = ToLoc) and
6071 (MyModel[MyUn[uix].mix].Domain = dSea) then
6072 begin
6073 ToShip := True;
6074 if MyModel[MyUn[uix].mix].Cap[mcSeaTrans] > 0 then
6075 ToTransport := True;
6076 end;
6077 if ToTransport then
6078 SoundMessage(Phrases.Lookup('FULLTRANSPORT'), 'NOMOVE_DEFAULT')
6079 else if ToShip then
6080 SoundMessage(Phrases.Lookup('NOTRANSPORT'), 'NOMOVE_DEFAULT')
6081 else
6082 Play('NOMOVE_DOMAIN');
6083 end
6084 else
6085 Play('NOMOVE_DOMAIN');
6086 end
6087 else
6088 Play('NOMOVE_DEFAULT');
6089 end;
6090 Exit;
6091 end;
6092
6093 if ((Result = eWon) or (Result = eLost) or (Result = eBloody)) and
6094 (MyUn[UnFocus].Movement < 100) and
6095 (MyModel[MyUn[UnFocus].mix].Cap[mcWill] = 0) then
6096 begin
6097 if SimpleQuery(mkYesNo, Format(Phrases.Lookup('FASTATTACK'),
6098 [MyUn[UnFocus].Movement]), 'NOMOVE_TIME') <> mrOK then
6099 begin
6100 Result := eInvalid;
6101 Exit;
6102 end;
6103 Update; // remove message box from screen
6104 end;
6105
6106 OldUnrest := False;
6107 NewUnrest := False;
6108 if (Result >= rExecuted) and (Result and rUnitRemoved = 0) and
6109 (MyMap[ToLoc] and (fUnit or fOwned) <> fUnit) then
6110 begin
6111 OldUnrest := UnrestAtLoc(UnFocus, FromLoc);
6112 NewUnrest := UnrestAtLoc(UnFocus, ToLoc);
6113 if NewUnrest > OldUnrest then
6114 begin
6115 if MyRO.Government = gDemocracy then
6116 begin
6117 QueryItem := 'UNREST_NOTOWN';
6118 p1 := Me;
6119 end
6120 else
6121 begin
6122 QueryItem := 'UNREST_FOREIGN';
6123 p1 := MyRO.Territory[ToLoc];
6124 end;
6125 with MessgExDlg do
6126 begin
6127 MessgText := Format(Tribe[p1].TPhrase(QueryItem),
6128 [Phrases.Lookup('GOVERNMENT', MyRO.Government)]);
6129 Kind := mkYesNo;
6130 IconKind := mikImp;
6131 IconIndex := imPalace;
6132 ShowModal;
6133 if ModalResult <> mrOK then
6134 begin
6135 Result := eInvalid;
6136 Exit;
6137 end;
6138 end;
6139 Update; // remove message box from screen
6140 end;
6141 end;
6142
6143 if (Result >= rExecuted) and (MyModel[MyUn[UnFocus].mix].Domain = dAir) and
6144 (MyUn[UnFocus].Status and usToldNoReturn = 0) then
6145 begin // can plane return?
6146 PlaneReturnData.Fuel := MyUn[UnFocus].Fuel;
6147 if (MyMap[ToLoc] and (fUnit or fOwned) = fUnit) or
6148 (MyMap[ToLoc] and (fCity or fOwned) = fCity) then
6149 begin // attack/expel/bombard -> 100MP
6150 PlaneReturnData.Loc := FromLoc;
6151 PlaneReturnData.Movement := MyUn[UnFocus].Movement - 100;
6152 if PlaneReturnData.Movement < 0 then
6153 PlaneReturnData.Movement := 0;
6154 end
6155 else // move
6156 begin
6157 PlaneReturnData.Loc := ToLoc;
6158 if dx and 1 <> 0 then
6159 PlaneReturnData.Movement := MyUn[UnFocus].Movement - 100
6160 else
6161 PlaneReturnData.Movement := MyUn[UnFocus].Movement - 150;
6162 end;
6163 if Server(sGetPlaneReturn, Me, UnFocus, PlaneReturnData) = eNoWay then
6164 begin
6165 if MyModel[MyUn[UnFocus].mix].Kind = mkSpecial_Glider then
6166 QueryItem := 'LOWFUEL_GLIDER'
6167 else
6168 QueryItem := 'LOWFUEL';
6169 if SimpleQuery(mkYesNo, Phrases.Lookup(QueryItem), 'WARNING_LOWSUPPORT')
6170 <> mrOK then
6171 begin
6172 Result := eInvalid;
6173 Exit;
6174 end;
6175 Update; // remove message box from screen
6176 MyUn[UnFocus].Status := MyUn[UnFocus].Status or usToldNoReturn;
6177 end;
6178 end;
6179
6180 if Result = eMissionDone then
6181 begin
6182 ModalSelectDlg.ShowNewContent(wmModal, kMission);
6183 Update; // dialog still on screen
6184 Mission := ModalSelectDlg.Result;
6185 if Mission < 0 then
6186 Exit;
6187 Server(sSetSpyMission + Mission shl 4, Me, 0, nil^);
6188 end;
6189
6190 CityCaptured := False;
6191 if Result = eNoTime_Move then
6192 Play('NOMOVE_TIME')
6193 else
6194 begin
6195 NeedEcoUpdate := False;
6196 DestinationMarkON := False;
6197 PaintDestination;
6198 if Result and rUnitRemoved <> 0 then
6199 CityOptimizer_BeforeRemoveUnit(UnFocus);
6200 IsAttack := (Result = eBombarded) or (Result <> eMissionDone) and
6201 (MyMap[ToLoc] and (fUnit or fOwned) = fUnit);
6202 if not IsAttack then
6203 begin // move
6204 cix := MyRO.nCity - 1; { look for own city at dest location }
6205 while (cix >= 0) and (MyCity[cix].Loc <> ToLoc) do
6206 Dec(cix);
6207 if (Result <> eMissionDone) and (MyMap[ToLoc] and fCity <> 0) and (cix < 0)
6208 then
6209 CityCaptured := True;
6210 Result := Server(sMoveUnit + DirCode, Me, UnFocus, nil^);
6211 case Result of
6212 eHiddenUnit:
6213 begin
6214 Play('NOMOVE_SUBMARINE');
6215 PaintLoc(ToLoc);
6216 end;
6217 eStealthUnit:
6218 begin
6219 Play('NOMOVE_STEALTH');
6220 PaintLoc(ToLoc);
6221 end;
6222 eZOC_EnemySpotted:
6223 begin
6224 Play('NOMOVE_ZOC');
6225 PaintLoc(ToLoc, 1);
6226 end;
6227 rExecuted..MaxInt:
6228 begin
6229 if Result and rUnitRemoved <> 0 then
6230 UnFocus := -1 // unit died
6231 else
6232 begin
6233 Assert(UnFocus >= 0);
6234 MyUn[UnFocus].Status := MyUn[UnFocus].Status and
6235 not (usStay or usRecover);
6236 for uix := 0 to MyRO.nUn - 1 do
6237 if MyUn[uix].Master = UnFocus then
6238 MyUn[uix].Status := MyUn[uix].Status and not usWaiting;
6239 if CityCaptured and
6240 (MyRO.Government in [gRepublic, gDemocracy, gFuture]) then
6241 begin // borders have moved, unrest might have changed in any city
6242 CityOptimizer_BeginOfTurn;
6243 NeedEcoUpdate := True;
6244 end
6245 else
6246 begin
6247 if OldUnrest <> NewUnrest then
6248 begin
6249 CityOptimizer_CityChange(MyUn[UnFocus].Home);
6250 for uix := 0 to MyRO.nUn - 1 do
6251 if MyUn[uix].Master = UnFocus then
6252 CityOptimizer_CityChange(MyUn[uix].Home);
6253 NeedEcoUpdate := True;
6254 end;
6255 if (MyRO.Government = gDespotism) and
6256 (MyModel[MyUn[UnFocus].mix].Kind = mkSpecial_TownGuard) then
6257 begin
6258 if MyMap[FromLoc] and fCity <> 0 then
6259 begin // town guard moved out of city in despotism -- reoptimize!
6260 cixChanged := MyRO.nCity - 1;
6261 while (cixChanged >= 0) and
6262 (MyCity[cixChanged].Loc <> FromLoc) do
6263 Dec(cixChanged);
6264 Assert(cixChanged >= 0);
6265 if cixChanged >= 0 then
6266 begin
6267 CityOptimizer_CityChange(cixChanged);
6268 NeedEcoUpdate := True;
6269 end;
6270 end;
6271 if (MyMap[ToLoc] and fCity <> 0) and not CityCaptured then
6272 begin // town guard moved into city in despotism -- reoptimize!
6273 cixChanged := MyRO.nCity - 1;
6274 while (cixChanged >= 0) and
6275 (MyCity[cixChanged].Loc <> ToLoc) do
6276 Dec(cixChanged);
6277 Assert(cixChanged >= 0);
6278 if cixChanged >= 0 then
6279 begin
6280 CityOptimizer_CityChange(cixChanged);
6281 NeedEcoUpdate := True;
6282 end;
6283 end;
6284 end;
6285 end;
6286 end;
6287 end;
6288 else
6289 Assert(False);
6290 end;
6291 SetTroopLoc(ToLoc);
6292 end
6293 else
6294 begin { enemy unit -- attack }
6295 if Result = eBombarded then
6296 Defender := MyRO.Territory[ToLoc]
6297 else
6298 Defender := MyRO.EnemyUn[euix].Owner;
6299 { if MyRO.Treaty[Defender]=trCeaseFire then
6300 if SimpleQuery(mkYesNo,Phrases.Lookup('FRCANCELQUERY_CEASEFIRE'),
6301 'MSG_DEFAULT')<>mrOK then
6302 Exit; }
6303 if (Options and muNoSuicideCheck = 0) and (Result and rUnitRemoved <> 0)
6304 and (Result <> eMissionDone) then
6305 begin // suicide query
6306 with MyUn[UnFocus], BattleDlg.Forecast do
6307 begin
6308 pAtt := Me;
6309 mixAtt := mix;
6310 HealthAtt := Health;
6311 ExpAtt := Exp;
6312 FlagsAtt := Flags;
6313 end;
6314 BattleDlg.Forecast.Movement := MyUn[UnFocus].Movement;
6315 Server(sGetBattleForecastEx, Me, ToLoc, BattleDlg.Forecast);
6316 BattleDlg.uix := UnFocus;
6317 BattleDlg.ToLoc := ToLoc;
6318 BattleDlg.IsSuicideQuery := True;
6319 BattleDlg.ShowModal;
6320 if BattleDlg.ModalResult <> mrOK then
6321 Exit;
6322 end;
6323
6324 cixChanged := -1;
6325 if (Result and rUnitRemoved <> 0) and (MyRO.Government = gDespotism) and
6326 (MyModel[MyUn[UnFocus].mix].Kind = mkSpecial_TownGuard) and
6327 (MyMap[FromLoc] and fCity <> 0) then
6328 begin // town guard died in city in despotism -- reoptimize!
6329 cixChanged := MyRO.nCity - 1;
6330 while (cixChanged >= 0) and (MyCity[cixChanged].Loc <> FromLoc) do
6331 Dec(cixChanged);
6332 Assert(cixChanged >= 0);
6333 end;
6334
6335 for I := 0 to MyRO.nEnemyModel - 1 do
6336 LostArmy[I] := MyRO.EnemyModel[I].Lost;
6337 OldToTile := MyMap[ToLoc];
6338 Result := Server(sMoveUnit + DirCode, Me, UnFocus, nil^);
6339 nLostArmy := 0;
6340 for I := 0 to MyRO.nEnemyModel - 1 do
6341 begin
6342 LostArmy[I] := MyRO.EnemyModel[I].Lost - LostArmy[I];
6343 Inc(nLostArmy, LostArmy[I]);
6344 end;
6345 if Result and rUnitRemoved <> 0 then
6346 begin
6347 UnFocus := -1;
6348 SetTroopLoc(FromLoc);
6349 end;
6350 if (OldToTile and not MyMap[ToLoc] and fCity <> 0) and
6351 (MyRO.Government in [gRepublic, gDemocracy, gFuture]) then
6352 begin // city was destroyed, borders have moved, unrest might have changed in any city
6353 CityOptimizer_BeginOfTurn;
6354 NeedEcoUpdate := True;
6355 end else begin
6356 if cixChanged >= 0 then
6357 begin
6358 CityOptimizer_CityChange(cixChanged);
6359 NeedEcoUpdate := True;
6360 end;
6361 if (Result = eWon) or (Result = eBloody) or (Result = eExpelled) then
6362 begin
6363 CityOptimizer_TileBecomesAvailable(ToLoc);
6364 NeedEcoUpdate := True;
6365 end;
6366 end;
6367 if nLostArmy > 1 then
6368 begin
6369 with MessgExDlg do
6370 begin
6371 Kind := mkOk;
6372 IconKind := mikEnemyArmy;
6373 MessgText := Tribe[Defender].TString(Phrases.Lookup('ARMYLOST',
6374 MyRO.EnemyModel[MyRO.EnemyUn[euix].emix].Domain));
6375 ShowModal;
6376 end;
6377 end;
6378 end;
6379 if Result and rUnitRemoved <> 0 then
6380 begin
6381 CityOptimizer_AfterRemoveUnit;
6382 ListDlg.RemoveUnit;
6383 NeedEcoUpdate := True;
6384 end;
6385 if NeedEcoUpdate then
6386 begin
6387 UpdateViews(True);
6388 Update;
6389 end;
6390 end;
6391
6392 if Result = eMissionDone then
6393 begin
6394 p1 := MyRO.Territory[ToLoc];
6395 case Mission of
6396 smStealMap:
6397 begin
6398 MapValid := False;
6399 PaintAllMaps
6400 end;
6401 smStealCivilReport:
6402 TribeMessage(p1, Tribe[p1].TPhrase('DOSSIER_PREPARED'), '');
6403 smStealMilReport:
6404 ListDlg.ShowNewContent_MilReport(wmPersistent, p1);
6405 end;
6406 end;
6407
6408 if UnFocus >= 0 then
6409 CheckToldNoReturn(UnFocus);
6410
6411 NeedRepaintPanel := False;
6412 if Result >= rExecuted then
6413 begin
6414 if CityCaptured and (MyMap[ToLoc] and fCity = 0) then
6415 begin // city destroyed
6416 for I := 0 to nWonder - 1 do { tell about destroyed wonders }
6417 if (MyRO.Wonder[I].CityID = WonderDestroyed) and (MyData.ToldWonders[I].CityID <> WonderDestroyed)
6418 then
6419 with MessgExDlg do
6420 begin
6421 if WondersDlg.Visible then
6422 WondersDlg.SmartUpdateContent(False);
6423 OpenSound := 'WONDER_DESTROYED';
6424 MessgText := Format(Phrases.Lookup('WONDERDEST'),
6425 [Phrases.Lookup('IMPROVEMENTS', I)]);
6426 Kind := mkOkHelp;
6427 HelpKind := hkImp;
6428 HelpNo := I;
6429 IconKind := mikImp;
6430 IconIndex := I;
6431 ShowModal;
6432 MyData.ToldWonders[I] := MyRO.Wonder[I];
6433 end;
6434 end;
6435 if CityCaptured and (MyMap[ToLoc] and fCity <> 0) then
6436 begin // city captured
6437 ListDlg.AddCity;
6438 for I := 0 to nWonder - 1 do { tell about capture of wonders }
6439 if MyRO.City[MyRO.nCity - 1].Built[I] > 0 then
6440 with MessgExDlg do
6441 begin
6442 if WondersDlg.Visible then
6443 WondersDlg.SmartUpdateContent(False);
6444 OpenSound := 'WONDER_CAPTURED';
6445 MessgText := Format(Tribe[Me].TPhrase('WONDERCAPTOWN'),
6446 [Phrases.Lookup('IMPROVEMENTS', I)]);
6447 Kind := mkOkHelp;
6448 HelpKind := hkImp;
6449 HelpNo := I;
6450 IconKind := mikImp;
6451 IconIndex := I;
6452 ShowModal;
6453 MyData.ToldWonders[I] := MyRO.Wonder[I];
6454 end;
6455
6456 if MyRO.Happened and phStealTech <> 0 then
6457 begin { Temple of Zeus -- choose advance to steal }
6458 ModalSelectDlg.ShowNewContent(wmModal, kStealTech);
6459 Server(sStealTech, Me, ModalSelectDlg.Result, nil^);
6460 end;
6461 TellNewModels;
6462
6463 cix := MyRO.nCity - 1;
6464 while (cix >= 0) and (MyCity[cix].Loc <> ToLoc) do
6465 Dec(cix);
6466 Assert(cix >= 0);
6467 MyCity[cix].Status := MyCity[cix].Status and not csResourceWeightsMask or
6468 (3 shl 4);
6469 // captured city, set to maximum growth
6470 NewTiles := 1 shl 13; { exploit central tile only }
6471 Server(sSetCityTiles, Me, cix, NewTiles);
6472 end
6473 else
6474 NeedRepaintPanel := True;
6475 end;
6476 TellNewContacts;
6477
6478 if (UnFocus >= 0) and (MyUn[UnFocus].Master >= 0) then
6479 with MyUn[MyUn[UnFocus].Master] do
6480 if Status and usStay <> 0 then
6481 begin
6482 Status := Status and not usStay;
6483 if (Movement >= 100) and (Status and (usRecover or usGoto) = 0) then
6484 Status := Status or usWaiting;
6485 end;
6486 if Options and (muAutoNoWait or muAutoNext) <> 0 then
6487 begin
6488 if (UnFocus >= 0) and ((Result = eNoTime_Move) or UnitExhausted(UnFocus) or
6489 (MyUn[UnFocus].Master >= 0) or (MyModel[MyUn[UnFocus].mix].Domain = dAir)
6490 and ((MyMap[MyUn[UnFocus].Loc] and fCity <> 0)
6491 { aircrafts stop in cities }
6492 or (MyMap[MyUn[UnFocus].Loc] and fTerImp = tiBase))) then
6493 begin
6494 MyUn[UnFocus].Status := MyUn[UnFocus].Status and not usWaiting;
6495 if Options and muAutoNext <> 0 then
6496 if CityCaptured and (MyMap[ToLoc] and fCity <> 0) then
6497 begin
6498 UnFocus := -1;
6499 PaintLoc(ToLoc); // don't show unit in city if not selected
6500 end
6501 else
6502 NextUnit(UnStartLoc, True);
6503 end
6504 else if (UnFocus < 0) and (Options and muAutoNext <> 0) then
6505 NextUnit(UnStartLoc, Result <> eMissionDone);
6506 end;
6507
6508 if NeedRepaintPanel and (UnFocus = UnFocus0) then
6509 if IsAttack then
6510 PanelPaint
6511 else
6512 begin
6513 Assert(Result <> eMissionDone);
6514 CheckTerrainBtnVisible;
6515 FocusOnLoc(ToLoc, flRepaintPanel or flImmUpdate);
6516 end;
6517
6518 if (Result >= rExecuted) and CityCaptured and (MyMap[ToLoc] and fCity <> 0)
6519 then
6520 ZoomToCity(ToLoc, UnFocus < 0, chCaptured); // show captured city
6521end;
6522
6523procedure TMainScreen.MoveOnScreen(ShowMove: TShowMove;
6524 Step0, Step1, nStep: Integer; Restore: Boolean = True);
6525var
6526 ToLoc, xFromLoc, yFromLoc, xToLoc, yToLoc, xFrom, yFrom, xTo, yTo, xMin, yMin,
6527 xRange, yRange, xw1, Step, xMoving, yMoving, SliceCount: Integer;
6528 UnitInfo: TUnitInfo;
6529 Ticks0, Ticks: TDateTime;
6530begin
6531 Timer1.Enabled := False;
6532 Ticks0 := NowPrecise;
6533 with ShowMove do
6534 begin
6535 UnitInfo.Owner := Owner;
6536 UnitInfo.mix := mix;
6537 UnitInfo.Health := Health;
6538 UnitInfo.Job := jNone;
6539 UnitInfo.Flags := Flags;
6540 if Owner <> Me then
6541 UnitInfo.emix := emix;
6542
6543 ToLoc := dLoc(FromLoc, dx, dy);
6544 xToLoc := ToLoc mod G.lx;
6545 yToLoc := ToLoc div G.lx;
6546 xFromLoc := FromLoc mod G.lx;
6547 yFromLoc := FromLoc div G.lx;
6548 if xToLoc > xFromLoc + 2 then
6549 xToLoc := xToLoc - G.lx
6550 else if xToLoc < xFromLoc - 2 then
6551 xToLoc := xToLoc + G.lx;
6552
6553 xw1 := xw + G.lx;
6554 // ((xFromLoc-xw1)*2+yFromLoc and 1+1)*xxt+dx*xxt/2-MapWidth/2 -> min
6555 with MainMap do begin
6556 while Abs(((xFromLoc - xw1 + G.lx) * 2 + yFromLoc and 1 + 1) * xxt * 2 + dx
6557 * xxt - MapWidth) < Abs(((xFromLoc - xw1) * 2 + yFromLoc and 1 + 1) * xxt
6558 * 2 + dx * xxt - MapWidth) do
6559 Dec(xw1, G.lx);
6560
6561 xTo := (xToLoc - xw1) * (xxt * 2) + yToLoc and 1 * xxt + (xxt - xxu);
6562 yTo := (yToLoc - yw) * yyt + (yyt - yyu_anchor);
6563 xFrom := (xFromLoc - xw1) * (xxt * 2) + yFromLoc and 1 * xxt + (xxt - xxu);
6564 yFrom := (yFromLoc - yw) * yyt + (yyt - yyu_anchor);
6565 if xFrom < xTo then begin
6566 xMin := xFrom;
6567 xRange := xTo - xFrom;
6568 end else begin
6569 xMin := xTo;
6570 xRange := xFrom - xTo;
6571 end;
6572 if yFrom < yTo then begin
6573 yMin := yFrom;
6574 yRange := yTo - yFrom;
6575 end else begin
6576 yMin := yTo;
6577 yRange := yFrom - yTo;
6578 end;
6579 Inc(xRange, xxt * 2);
6580 Inc(yRange, yyt * 3);
6581 end;
6582
6583 MainOffscreenPaint;
6584 NoMap.SetOutput(Buffer);
6585 NoMap.SetPaintBounds(0, 0, xRange, yRange);
6586 for Step := 0 to Abs(Step1 - Step0) do
6587 begin
6588 BitBltBitmap(Buffer, 0, 0, xRange, yRange, Offscreen, xMin, yMin);
6589 if Step1 <> Step0 then
6590 begin
6591 xMoving := xFrom +
6592 Round((Step0 + Step * (Step1 - Step0) div Abs(Step1 - Step0)) *
6593 (xTo - xFrom) / nStep);
6594 yMoving := yFrom +
6595 Round((Step0 + Step * (Step1 - Step0) div Abs(Step1 - Step0)) *
6596 (yTo - yFrom) / nStep);
6597 end
6598 else
6599 begin
6600 xMoving := xFrom;
6601 yMoving := yFrom;
6602 end;
6603 NoMap.PaintUnit(xMoving - xMin, yMoving - yMin, UnitInfo, 0);
6604 PaintBufferToScreen(xMin, yMin, xRange, yRange);
6605 {$IFDEF UNIX}
6606 // TODO: Force animation under UNIX. Causes also to process keyboard events
6607 // which can lead to unexpected errors.
6608 KeyboardDisabled := True;
6609 try
6610 Application.ProcessMessages;
6611 finally
6612 KeyboardDisabled := False;
6613 end;
6614 {$ENDIF}
6615
6616 SliceCount := 0;
6617 Ticks := Ticks0;
6618 repeat
6619 if (SliceCount = 0) or
6620 (Round(((Ticks - Ticks0) * 12) / OneMillisecond) * (SliceCount + 1) div SliceCount
6621 < MoveTime) then
6622 begin
6623 if not Idle or (GameMode = cMovie) then
6624 Application.ProcessMessages;
6625 Sleep(1);
6626 Inc(SliceCount);
6627 end;
6628 Ticks := NowPrecise;
6629 until (((Ticks - Ticks0) * 12) / OneMillisecond) >= MoveTime;
6630 Ticks0 := Ticks;
6631 end;
6632 end;
6633 if Restore then
6634 begin
6635 BitBltBitmap(Buffer, 0, 0, xRange, yRange, Offscreen, xMin, yMin);
6636 PaintBufferToScreen(xMin, yMin, xRange, yRange);
6637 end;
6638 BlinkTime := -1;
6639 Timer1.Enabled := True;
6640end;
6641
6642procedure TMainScreen.MoveToLoc(Loc: Integer; CheckSuicide: Boolean);
6643// path finder: move focused unit to loc, start multi-turn goto if too far
6644var
6645 uix, I, MoveOptions, NextLoc, MoveResult: Integer;
6646 MoveAdviceData: TMoveAdviceData;
6647 StopReason: (None, Arrived, Dead, NoTime, EnemySpotted, MoveError);
6648begin
6649 if MyUn[UnFocus].Job > jNone then
6650 Server(sStartJob + jNone shl 4, Me, UnFocus, nil^);
6651 if GetMoveAdvice(UnFocus, Loc, MoveAdviceData) >= rExecuted then
6652 begin
6653 uix := UnFocus;
6654 StopReason := None;
6655 repeat
6656 for I := 0 to MoveAdviceData.nStep - 1 do
6657 begin
6658 if I = MoveAdviceData.nStep - 1 then
6659 MoveOptions := muAutoNext
6660 else
6661 MoveOptions := 0;
6662 NextLoc := dLoc(MyUn[uix].Loc, MoveAdviceData.dx[I],
6663 MoveAdviceData.dy[I]);
6664 if (NextLoc = Loc) or (Loc = maNextCity) and
6665 (MyMap[NextLoc] and fCity <> 0) then
6666 StopReason := Arrived;
6667 if not CheckSuicide and (NextLoc = Loc) then
6668 MoveOptions := MoveOptions or muNoSuicideCheck;
6669 MoveResult := MoveUnit(MoveAdviceData.dx[I], MoveAdviceData.dy[I],
6670 MoveOptions);
6671 if MoveResult < rExecuted then
6672 StopReason := MoveError
6673 else if MoveResult and rUnitRemoved <> 0 then
6674 StopReason := Dead
6675 else if (StopReason = None) and (MoveResult and rEnemySpotted <> 0) then
6676 StopReason := EnemySpotted;
6677 if StopReason <> None then
6678 Break;
6679 end;
6680 if (StopReason = None) and ((MoveAdviceData.nStep < 25) or
6681 (MyRO.Wonder[woShinkansen].EffectiveOwner <> Me)) then
6682 StopReason := NoTime;
6683 if StopReason <> None then
6684 Break;
6685 if GetMoveAdvice(UnFocus, Loc, MoveAdviceData) < rExecuted then
6686 begin
6687 Assert(False);
6688 Break;
6689 end;
6690 until False;
6691
6692 case StopReason of
6693 None:
6694 Assert(False);
6695 Arrived:
6696 MyUn[uix].Status := MyUn[uix].Status and ($FFFF - usGoto);
6697 Dead:
6698 if UnFocus < 0 then
6699 NextUnit(UnStartLoc, False);
6700 else
6701 begin // multi-turn goto
6702 if Loc = maNextCity then
6703 MyUn[uix].Status := MyUn[uix].Status and ($FFFF - usStay - usRecover)
6704 or usGoto + $7FFF shl 16
6705 else
6706 MyUn[uix].Status := MyUn[uix].Status and ($FFFF - usStay - usRecover)
6707 or usGoto + Loc shl 16;
6708 PaintLoc(MyUn[uix].Loc);
6709 if (StopReason = NoTime) and (UnFocus = uix) then
6710 begin
6711 MyUn[uix].Status := MyUn[uix].Status and not usWaiting;
6712 NextUnit(UnStartLoc, True);
6713 end;
6714 end;
6715 end;
6716 end;
6717end;
6718
6719procedure TMainScreen.PanelBoxMouseDown(Sender: TObject; Button: TMouseButton;
6720 Shift: TShiftState; X, Y: Integer);
6721var
6722 I, xMouse, MouseLoc, p1: Integer;
6723begin
6724 if GameMode = cMovie then
6725 Exit;
6726
6727 if Button = TMouseButton.mbLeft then
6728 begin
6729 if (X >= xMini + 2) and (Y >= yMini + 2) and (X < xMini + 2 + 2 * G.lx) and
6730 (Y < yMini + 2 + G.ly) then
6731 if ssShift in Shift then
6732 begin
6733 with MainMap do
6734 xMouse := (xwMini + (X - (xMini + 2) + MapWidth div (xxt * 2) + G.lx)
6735 div 2) mod G.lx;
6736 MouseLoc := xMouse + G.lx * (Y - (yMini + 2));
6737 if MyMap[MouseLoc] and fTerrain <> fUNKNOWN then
6738 begin
6739 p1 := MyRO.Territory[MouseLoc];
6740 if (p1 = Me) or (p1 >= 0) and (MyRO.Treaty[p1] >= trNone) then
6741 NatStatDlg.ShowNewContent(wmPersistent, p1);
6742 end;
6743 end
6744 else
6745 begin
6746 if CityDlg.Visible then
6747 CityDlg.Close;
6748 if UnitStatDlg.Visible then
6749 UnitStatDlg.Close;
6750 Tracking := True;
6751 PanelBoxMouseMove(Sender, Shift + [ssLeft], X, Y);
6752 end
6753 else if (ClientMode <> cEditMap) and (X >= ClientWidth - xPalace) and
6754 (Y >= yPalace) and (X < ClientWidth - xPalace + xSizeBig) and
6755 (Y < yPalace + ySizeBig) then
6756 begin
6757 InitPopup(StatPopup);
6758 if FullScreen then
6759 StatPopup.Popup(Left + ClientWidth - xPalace + xSizeBig + 2,
6760 Top + ClientHeight - PanelHeight + yPalace - 1)
6761 else
6762 StatPopup.Popup(Left + ClientWidth - xPalace + 6,
6763 Top + ClientHeight - PanelHeight + yPalace + ySizeBig +
6764 GetSystemMetrics(SM_CYCAPTION) + 3);
6765 end
6766 (* else if (X >= xAdvisor - 3) and (Y >= yAdvisor - 3)
6767 and (X < xAdvisor + 16 + 3) and (Y < yAdvisor + 16 + 3) and HaveStrategyAdvice then
6768 AdviceBtnClick *)
6769 else if (X >= xTroop + 1) and (Y >= yTroop + 1) and
6770 (X < xTroop + TrRow * TrPitch) and (Y <= yTroop + 55) then
6771 begin
6772 I := (X - xTroop - 1) div TrPitch;
6773 if trix[I] >= 0 then
6774 if ClientMode = cEditMap then
6775 begin
6776 BrushType := trix[I];
6777 UpdateInterface;
6778 PanelPaint;
6779 end
6780 else if (TroopLoc >= 0) then
6781 if MyMap[TroopLoc] and fOwned <> 0 then
6782 begin
6783 if ssShift in Shift then
6784 UnitStatDlg.ShowNewContent_OwnModel(wmPersistent,
6785 MyUn[trix[I]].mix)
6786 else if not Supervising and (ClientMode < scContact) and
6787 (X - xTroop - 1 - I * TrPitch >= 60 - 20) and (Y >= yTroop + 35)
6788 and ((MyUn[trix[I]].Job > jNone) or (MyUn[trix[I]].Status and
6789 (usStay or usRecover or usGoto) <> 0)) then
6790 begin // wake up
6791 MyUn[trix[I]].Status := MyUn[trix[I]].Status and
6792 ($FFFF - usStay - usRecover - usGoto - usEnhance) or usWaiting;
6793 if MyUn[trix[I]].Job > jNone then
6794 Server(sStartJob + jNone shl 4, Me, trix[I], nil^);
6795 if (UnFocus < 0) and not CityDlg.Visible then
6796 begin
6797 SetUnFocus(trix[I]);
6798 SetTroopLoc(MyUn[trix[I]].Loc);
6799 FocusOnLoc(TroopLoc, flRepaintPanel);
6800 end
6801 else
6802 begin
6803 if CityDlg.Visible and (CityDlg.RestoreUnFocus < 0) then
6804 CityDlg.RestoreUnFocus := trix[I];
6805 PanelPaint;
6806 end;
6807 end
6808 else if (ClientMode < scContact) then
6809 begin
6810 if Supervising then
6811 UnitStatDlg.ShowNewContent_OwnUnit(wmPersistent, trix[I])
6812 else if CityDlg.Visible then
6813 begin
6814 CityDlg.CloseAction := None;
6815 CityDlg.Close;
6816 SumCities(TaxSum, ScienceSum);
6817 SetUnFocus(trix[I]);
6818 end
6819 else
6820 begin
6821 DestinationMarkON := False;
6822 PaintDestination;
6823 UnFocus := trix[I];
6824 UnStartLoc := TroopLoc;
6825 BlinkTime := 0;
6826 BlinkON := False;
6827 PaintLoc(TroopLoc);
6828 end;
6829 if UnFocus >= 0 then
6830 begin
6831 UnitInfoBtn.Visible := True;
6832 UnitBtn.Visible := True;
6833 TurnComplete := False;
6834 EOT.ButtonIndex := eotGray;
6835 end;
6836 CheckTerrainBtnVisible;
6837 PanelPaint;
6838 end;
6839 end
6840 else if Server(sGetUnits, Me, TroopLoc, TrCnt) >= rExecuted then
6841 if ssShift in Shift then
6842 UnitStatDlg.ShowNewContent_EnemyModel(wmPersistent,
6843 MyRO.EnemyUn[MyRO.nEnemyUn + trix[I]].emix) // model info
6844 else
6845 UnitStatDlg.ShowNewContent_EnemyUnit(wmPersistent,
6846 MyRO.nEnemyUn + trix[I]); // unit info
6847 end;
6848 end;
6849end;
6850
6851procedure TMainScreen.SetTroopLoc(Loc: Integer);
6852var
6853 trixFocus, uix, uixDefender: Integer;
6854 Prio: Boolean;
6855 RowsCount: Integer;
6856begin
6857 TroopLoc := Loc;
6858 TrRow := (xRightPanel + 10 - xTroop - GetSystemMetrics(SM_CXVSCROLL) - 19)
6859 div TrPitch;
6860 TrCnt := 0;
6861 trixFocus := -1;
6862 if ClientMode = cEditMap then
6863 TrCnt := Length(BrushTypes)
6864 else if (Loc >= 0) and (MyMap[Loc] and fUnit <> 0) then
6865 if MyMap[Loc] and fOwned <> 0 then
6866 begin // count own units here
6867 Server(sGetDefender, Me, TroopLoc, uixDefender);
6868 for Prio := True downto False do
6869 for uix := 0 to MyRO.nUn - 1 do
6870 if ((uix = uixDefender) = Prio) and (MyUn[uix].Loc = Loc) then
6871 begin
6872 if uix = UnFocus then
6873 trixFocus := TrCnt;
6874 Inc(TrCnt);
6875 end;
6876 end
6877 else // count enemy units here
6878 Server(sGetUnits, Me, Loc, TrCnt);
6879 if TrCnt = 0 then
6880 VerticalScrollBar.Init(0, 1)
6881 else
6882 begin
6883 {$IFDEF UNIX}
6884 RowsCount := Ceil(TrCnt / TrRow);
6885 {$ELSE}
6886 RowsCount := (TrCnt + TrRow - 1) div TrRow - 1;
6887 {$ENDIF}
6888 VerticalScrollBar.Init(RowsCount, 1);
6889 if (VerticalScrollBar.Max >= VerticalScrollBar.PageSize) and (trixFocus >= 0) then
6890 VerticalScrollBar.Position := trixFocus div TrRow;
6891 end;
6892end;
6893
6894(* procedure TMainScreen.ShowMoveHint(ToLoc: Integer; Force: Boolean = False);
6895 var
6896 Step,Loc,x0,y0,xs,ys: Integer;
6897 Info: string;
6898 InfoSize: TSize;
6899 MoveAdvice: TMoveAdviceData;
6900 begin
6901 if (ToLoc<0) or (ToLoc>=G.lx*G.ly)
6902 or (UnFocus<0) or (MyUn[UnFocus].Loc=ToLoc) then
6903 ToLoc:=-1
6904 else
6905 begin
6906 MoveAdvice.ToLoc:=ToLoc;
6907 MoveAdvice.MoreTurns:=0;
6908 MoveAdvice.MaxHostile_MovementLeft:=MyUn[UnFocus].Health-50;
6909 if Server(sGetMoveAdvice,Me,UnFocus,MoveAdvice)<rExecuted then
6910 ToLoc:=-1
6911 end;
6912 if (ToLoc=MoveHintToLoc) and not Force then Exit;
6913 if (ToLoc<>MoveHintToLoc) and (MoveHintToLoc>=0) then
6914 begin invalidate; update end; // clear old hint from screen
6915 MoveHintToLoc:=ToLoc;
6916 if ToLoc<0 then Exit;
6917
6918 with Canvas do
6919 begin
6920 Pen.Color:=$80C0FF;
6921 Pen.Width:=3;
6922 Loc:=MyUn[UnFocus].Loc;
6923 for Step:=0 to MoveAdvice.nStep do
6924 begin
6925 y0:=(Loc+G.lx*1024) div G.lx -1024;
6926 x0:=(Loc+(y0 and 1+G.lx*1024) div 2) mod G.lx;
6927 xs:=(x0-xw)*66+y0 and 1*33-G.lx*66;
6928 while Abs(2*(xs+G.lx*66)-MapWidth)<Abs(2*xs-MapWidth) do
6929 Inc(xs,G.lx*66);
6930 ys:=(y0-yw)*16;
6931 if Step=0 then moveto(xs+33,ys+16)
6932 else lineto(xs+33,ys+16);
6933 if Step<MoveAdvice.nStep then
6934 Loc:=dLoc(Loc,MoveAdvice.dx[Step],MoveAdvice.dy[Step]);
6935 end;
6936 Brush.Color:=$80C0FF;
6937 Info:=' '+IntToStr(88)+' ';
6938 InfoSize:=TextExtent(Info);
6939 TextOut(xs+33-InfoSize.cx div 2, ys+16-InfoSize.cy div 2, Info);
6940 Brush.Style:=bsClear;
6941 end
6942 end; *)
6943
6944procedure TMainScreen.SetDebugMap(P: Integer);
6945begin
6946 MainMap.pDebugMap := P;
6947 MapOptions := MapOptions - [moLocCodes];
6948 mLocCodes.Checked := False;
6949 MapValid := False;
6950 MainOffscreenPaint;
6951end;
6952
6953procedure TMainScreen.SetViewpoint(P: Integer);
6954var
6955 I: Integer;
6956begin
6957 if Supervising and (G.RO[0].Turn > 0) and
6958 ((P = 0) or (1 shl P and G.RO[0].Alive <> 0)) then
6959 begin
6960 ApplyToVisibleForms(faClose);
6961 ItsMeAgain(P);
6962 SumCities(TaxSum, ScienceSum);
6963 for I := 0 to MyRO.nModel - 1 do
6964 if not Assigned(Tribe[Me].ModelPicture[I].HGr) then
6965 InitMyModel(I, True);
6966
6967 SetTroopLoc(-1);
6968 PanelPaint;
6969 MapValid := False;
6970 PaintAllMaps;
6971 end;
6972end;
6973
6974procedure TMainScreen.UpdateKeyShortcuts;
6975begin
6976 mHelp.ShortCut := BHelp.ShortCut;
6977 mUnitStat.ShortCut := BUnitStat.ShortCut;
6978 mCityStat.ShortCut := BCityStat.ShortCut;
6979 mScienceStat.ShortCut := BScienceStat.ShortCut;
6980 mEUnitStat.ShortCut := BEUnitStat.ShortCut;;
6981 mDiagram.ShortCut := BDiagram.ShortCut;
6982 mWonders.ShortCut := BWonders.ShortCut;
6983 mShips.ShortCut := BShips.ShortCut;
6984 mNations.ShortCut := BNations.ShortCut;
6985 mEmpire.ShortCut := BEmpire.ShortCut;
6986 mResign.ShortCut := BResign.ShortCut;
6987 mRandomMap.ShortCut := BRandomMap.ShortCut;
6988 mFillMap.ShortCut := BFillMap.ShortCut;
6989 mDisband.ShortCut := BDisbandUnit.ShortCut;
6990 mFort.ShortCut := BFortify.ShortCut;
6991 mCentre.ShortCut := BCenterUnit.ShortCut;
6992 mStay.ShortCut := BStay.ShortCut;
6993 mNoOrders.ShortCut := BNoOrders.ShortCut;
6994 mPrevUnit.ShortCut := BPrevUnit.ShortCut;
6995 mNextUnit.ShortCut := BNextUnit.ShortCut;
6996 mCancel.ShortCut := BCancel.ShortCut;
6997 mPillage.ShortCut := BPillage.ShortCut;
6998 mTechTree.ShortCut := BTechTree.ShortCut;
6999 mWait.ShortCut := BWait.ShortCut;
7000 mJump.ShortCut := BJump.ShortCut;;
7001 mDebugMap.ShortCut := BDebugMap.ShortCut;
7002 mLocCodes.ShortCut := BLocCodes.ShortCut;
7003 mNames.ShortCut := BNames.ShortCut;
7004 mRun.ShortCut := BRun.ShortCut;
7005 mAirBase.ShortCut := BAirBase.ShortCut;
7006 mCity.ShortCut := BBuildCity.ShortCut;
7007 mEnhance.ShortCut := BEnhance.ShortCut;
7008 mGoOn.ShortCut := BGoOn.ShortCut;
7009 mHome.ShortCut := BHome.ShortCut;
7010 mFarm.ShortCut := BFarmClearIrrigation.ShortCut;
7011 mClear.ShortCut := BFarmClearIrrigation.ShortCut;
7012 mIrrigation.ShortCut := BFarmClearIrrigation.ShortCut;
7013 mLoad.ShortCut := BLoad.ShortCut;
7014 mAfforest.ShortCut := BAfforestMine.ShortCut;
7015 mMine.ShortCut := BAfforestMine.ShortCut;
7016 mCanal.ShortCut := BCanal.ShortCut;
7017 MTrans.ShortCut := BTrans.ShortCut;
7018 mPollution.ShortCut := BPollution.ShortCut;
7019 mRailRoad.ShortCut := BRailRoad.ShortCut;
7020 mRoad.ShortCut := BRailRoad.ShortCut;
7021 mUnload.ShortCut := BUnload.ShortCut;
7022 mRecover.ShortCut := BRecover.ShortCut;
7023 mUtilize.ShortCut := BUtilize.ShortCut;
7024end;
7025
7026procedure TMainScreen.SetFullScreen(Active: Boolean);
7027var
7028 Form: TForm;
7029begin
7030 Form := Self;
7031 if Active then begin
7032 FormFullScreen := True;
7033 if Form.WindowState = TWindowState.wsMaximized then begin
7034 FormRestoredSize := Bounds(Form.RestoredLeft, Form.RestoredTop, Form.RestoredWidth,
7035 Form.RestoredHeight);
7036 end else
7037 if Form.WindowState = TWindowState.wsNormal then begin
7038 FormRestoredSize := Bounds(Form.Left, Form.Top, Form.Width, Form.Height);
7039 end;
7040 FormWindowState := Form.WindowState;
7041 Form.WindowState := TWindowState.wsMaximized;
7042 Form.WindowState := TWindowState.wsNormal;
7043 ShowWindow(Form.Handle, SW_SHOWFULLSCREEN);
7044 {$IFDEF WINDOWS}
7045 Form.BorderStyle := TBorderStyle.bsNone;
7046 {$ENDIF}
7047 end else begin
7048 FormFullScreen := False;
7049 {$IFDEF WINDOWS}
7050 Form.BorderStyle := TBorderStyle.bsSizeable;
7051 {$ENDIF}
7052 ShowWindow(Form.Handle, SW_SHOWNORMAL);
7053 if FormWindowState = TWindowState.wsNormal then begin
7054 Form.WindowState := TWindowState.wsNormal;
7055 Form.BoundsRect := FormRestoredSize;
7056 end else
7057 if FormWindowState = TWindowState.wsMaximized then begin
7058 Form.BoundsRect := FormRestoredSize;
7059 Form.WindowState := TWindowState.wsMaximized;
7060 end;
7061 end;
7062end;
7063
7064procedure TMainScreen.FormKeyDown(Sender: TObject; var Key: Word;
7065 Shift: TShiftState);
7066
7067 procedure MenuClick_Check(Popup: TPopupMenu; Item: TMenuItem);
7068 begin
7069 InitPopup(Popup);
7070 if Item.Visible and Item.Enabled then
7071 Item.Click;
7072 end;
7073
7074 procedure SetViewpointMe(P: Integer);
7075 begin
7076 if P = Me then SetViewpoint(P)
7077 else SetViewpoint(P);
7078 end;
7079
7080 procedure DoMoveUnit(X, Y: Integer);
7081 begin
7082 DestinationMarkON := False;
7083 PaintDestination;
7084 MyUn[UnFocus].Status := MyUn[UnFocus].Status and
7085 ($FFFF - usStay - usRecover - usGoto - usEnhance) or usWaiting;
7086 MoveUnit(X, Y, muAutoNext);
7087 end;
7088
7089 procedure DoMoveMap(X, Y: Integer);
7090 var
7091 ScrollSpeed: Integer;
7092 begin
7093 if mScrollFast.Checked then ScrollSpeed := 2
7094 else ScrollSpeed := 1;
7095 if (X <> 0) or (Y <> 0) then begin
7096 Scroll(X * ScrollSpeed, Y * ScrollSpeed);
7097 end;
7098 end;
7099
7100var
7101 Time0, Time1: TDateTime;
7102 ShortCut: TShortCut;
7103begin
7104 if KeyboardDisabled then Exit;
7105
7106 ShortCut := KeyToShortCut(Key, Shift);
7107
7108 if BFullScreen.Test(ShortCut) then begin
7109 FullScreen := not FullScreen;
7110 SetFullScreen(FullScreen);
7111 end;
7112
7113 if GameMode = cMovie then begin
7114 if BScienceStat.Test(ShortCut) then MenuClick_Check(StatPopup, mScienceStat)
7115 else if BDiagram.Test(ShortCut) then MenuClick_Check(StatPopup, mDiagram)
7116 else if BWonders.Test(ShortCut) then MenuClick_Check(StatPopup, mWonders)
7117 else if BShips.Test(ShortCut) then MenuClick_Check(StatPopup, mShips);
7118 Exit;
7119 end;
7120
7121 if not Idle then Exit;
7122
7123 if ClientMode = cEditMap then begin
7124 if BResign.Test(ShortCut) then mResign.Click
7125 else if BRandomMap.Test(ShortCut) then mRandomMap.Click
7126 else if BFillMap.Test(ShortCut) then mFillMap.Click
7127 else if BHelp.Test(ShortCut) then mHelp.Click;
7128 Exit;
7129 end;
7130
7131 if BEndTurn.Test(ShortCut) then EndTurn
7132 else if BHelp.Test(ShortCut) then mHelp.Click
7133 else if BUnitStat.Test(ShortCut) then MenuClick_Check(StatPopup, mUnitStat)
7134 else if BCityStat.Test(ShortCut) then MenuClick_Check(StatPopup, mCityStat)
7135 else if BScienceStat.Test(ShortCut) then MenuClick_Check(StatPopup, mScienceStat)
7136 else if BEUnitStat.Test(ShortCut) then MenuClick_Check(StatPopup, mEUnitStat)
7137 else if BDiagram.Test(ShortCut) then MenuClick_Check(StatPopup, mDiagram)
7138 else if BWonders.Test(ShortCut) then MenuClick_Check(StatPopup, mWonders)
7139 else if BShips.Test(ShortCut) then MenuClick_Check(StatPopup, mShips)
7140 else if BNations.Test(ShortCut) then MenuClick_Check(StatPopup, mNations)
7141 else if BEmpire.Test(ShortCut) then MenuClick_Check(StatPopup, mEmpire)
7142
7143 else if BSetDebugMap0.Test(ShortCut) then SetDebugMap(-1)
7144 else if BSetDebugMap1.Test(ShortCut) then SetDebugMap(1)
7145 else if BSetDebugMap2.Test(ShortCut) then SetDebugMap(2)
7146 else if BSetDebugMap3.Test(ShortCut) then SetDebugMap(3)
7147 else if BSetDebugMap4.Test(ShortCut) then SetDebugMap(4)
7148 else if BSetDebugMap5.Test(ShortCut) then SetDebugMap(5)
7149 else if BSetDebugMap6.Test(ShortCut) then SetDebugMap(6)
7150 else if BSetDebugMap7.Test(ShortCut) then SetDebugMap(7)
7151 else if BSetDebugMap8.Test(ShortCut) then SetDebugMap(8)
7152 else if BSetDebugMap9.Test(ShortCut) then SetDebugMap(9)
7153
7154 else if BJump.Test(ShortCut) then mJump.Click
7155 else if BDebugMap.Test(ShortCut) then mShowClick(mDebugMap)
7156 else if BLocCodes.Test(ShortCut) then mShowClick(mLocCodes)
7157 else if BLogDlg.Test(ShortCut) then begin
7158 if LogDlg.Visible then LogDlg.Close
7159 else LogDlg.Show;
7160 end
7161 else if BNames.Test(ShortCut) then mNamesClick(mNames)
7162 else if BResign.Test(ShortCut) then MenuClick_Check(GamePopup, mResign)
7163 else if BRun.Test(ShortCut) then mRun.Click
7164 else if BTestMapRepaint.Test(ShortCut) then begin // test map repaint time
7165 Time0 := NowPrecise;
7166 MapValid := False;
7167 MainOffscreenPaint;
7168 Time1 := NowPrecise;
7169 SimpleMessage(Format('Map repaint time: %.3f ms',
7170 [(Time1 - Time0) / OneMillisecond]));
7171 end
7172 else if BSetViewpoint0.Test(ShortCut) then SetViewpointMe(0)
7173 else if BSetViewpoint1.Test(ShortCut) then SetViewpointMe(1)
7174 else if BSetViewpoint2.Test(ShortCut) then SetViewpointMe(2)
7175 else if BSetViewpoint3.Test(ShortCut) then SetViewpointMe(3)
7176 else if BSetViewpoint4.Test(ShortCut) then SetViewpointMe(4)
7177 else if BSetViewpoint5.Test(ShortCut) then SetViewpointMe(5)
7178 else if BSetViewpoint6.Test(ShortCut) then SetViewpointMe(6)
7179 else if BSetViewpoint7.Test(ShortCut) then SetViewpointMe(7)
7180 else if BSetViewpoint8.Test(ShortCut) then SetViewpointMe(8)
7181 else if BSetViewpoint9.Test(ShortCut) then SetViewpointMe(9)
7182
7183 else if BMapBtn0.Test(ShortCut) then MapBtnClick(MapBtn0)
7184 else if BMapBtn1.Test(ShortCut) then MapBtnClick(MapBtn1)
7185 else if BMapBtn4.Test(ShortCut) then MapBtnClick(MapBtn4)
7186 else if BMapBtn5.Test(ShortCut) then MapBtnClick(MapBtn5)
7187 else if BMapBtn6.Test(ShortCut) then MapBtnClick(MapBtn6)
7188 else if BTechTree.Test(ShortCut) then mTechTree.Click
7189 else if BWait.Test(ShortCut) then mWait.Click;
7190
7191 if UnFocus >= 0 then begin
7192 if BDisbandUnit.Test(ShortCut) then mDisband.Click
7193 else if BFortify.Test(ShortCut) then MenuClick_Check(TerrainPopup, mFort)
7194 else if BCenterUnit.Test(ShortCut) then mCentre.Click
7195 else if BStay.Test(ShortCut) then mStay.Click
7196 else if BNoOrders.Test(ShortCut) then mNoOrders.Click
7197 else if BPrevUnit.Test(ShortCut) then mPrevUnit.Click
7198 else if BNextUnit.Test(ShortCut) then mNextUnit.Click
7199 else if BCancel.Test(ShortCut) then MenuClick_Check(UnitPopup, mCancel)
7200 else if BPillage.Test(ShortCut) then MenuClick_Check(UnitPopup, mPillage)
7201 else if BSelectTransport.Test(ShortCut) then MenuClick_Check(UnitPopup, mSelectTransport)
7202 else if BAirBase.Test(ShortCut) then MenuClick_Check(TerrainPopup, mAirBase)
7203 else if BBuildCity.Test(ShortCut) then MenuClick_Check(UnitPopup, mCity)
7204 else if BEnhance.Test(ShortCut) then begin
7205 InitPopup(TerrainPopup);
7206 if mEnhance.Visible and mEnhance.Enabled then mEnhance.Click
7207 else mEnhanceDef.Click;
7208 end
7209 else if BGoOn.Test(ShortCut) then MenuClick_Check(UnitPopup, mGoOn)
7210 else if BHome.Test(ShortCut) then MenuClick_Check(UnitPopup, mHome)
7211 else if BFarmClearIrrigation.Test(ShortCut) then begin
7212 if JobTest(UnFocus, jFarm, [eTreaty]) then
7213 mFarm.Click
7214 else if JobTest(UnFocus, jClear, [eTreaty]) then
7215 mClear.Click
7216 else MenuClick_Check(TerrainPopup, mIrrigation);
7217 end
7218 else if BLoad.Test(ShortCut) then MenuClick_Check(UnitPopup, mLoad)
7219 else if BAfforestMine.Test(ShortCut) then begin
7220 if JobTest(UnFocus, jAfforest, [eTreaty]) then mAfforest.Click
7221 else MenuClick_Check(TerrainPopup, mMine);
7222 end
7223 else if BCanal.Test(ShortCut) then MenuClick_Check(TerrainPopup, mCanal)
7224 else if BTrans.Test(ShortCut) then MenuClick_Check(TerrainPopup, MTrans)
7225 else if BPollution.Test(ShortCut) then MenuClick_Check(TerrainPopup, mPollution)
7226 else if BRailRoad.Test(ShortCut) then begin
7227 if JobTest(UnFocus, jRR, [eTreaty]) then mRailRoad.Click
7228 else MenuClick_Check(TerrainPopup, mRoad);
7229 end
7230 else if BUnload.Test(ShortCut) then MenuClick_Check(UnitPopup, mUnload)
7231 else if BRecover.Test(ShortCut) then MenuClick_Check(UnitPopup, mRecover)
7232 else if BUtilize.Test(ShortCut) then MenuClick_Check(UnitPopup, mUtilize)
7233 // Unit move
7234 else if BUnitMoveLeftDown.Test(ShortCut) then DoMoveUnit(-1, 1)
7235 else if BUnitMoveDown.Test(ShortCut) then DoMoveUnit(0, 2)
7236 else if BUnitMoveRightDown.Test(ShortCut) then DoMoveUnit(1, 1)
7237 else if BUnitMoveLeft.Test(ShortCut) then DoMoveUnit(-2, 0)
7238 else if BUnitMoveRight.Test(ShortCut) then DoMoveUnit(2, 0)
7239 else if BUnitMoveLeftUp.Test(ShortCut) then DoMoveUnit(-1, -1)
7240 else if BUnitMoveUp.Test(ShortCut) then DoMoveUnit(0, -2)
7241 else if BUnitMoveRightUp.Test(ShortCut) then DoMoveUnit(1, -1)
7242 // Map move
7243 else if BMapMoveLeftDown.Test(ShortCut) then DoMoveMap(-1, 1)
7244 else if BMapMoveDown.Test(ShortCut) then DoMoveMap(0, 2)
7245 else if BMapMoveRightDown.Test(ShortCut) then DoMoveMap(1, 1)
7246 else if BMapMoveLeft.Test(ShortCut) then DoMoveMap(-2, 0)
7247 else if BMapMoveRight.Test(ShortCut) then DoMoveMap(2, 0)
7248 else if BMapMoveLeftUp.Test(ShortCut) then DoMoveMap(-1, -1)
7249 else if BMapMoveUp.Test(ShortCut) then DoMoveMap(0, -2)
7250 else if BMapMoveRightUp.Test(ShortCut) then DoMoveMap(1, -1);
7251 end;
7252end;
7253
7254function TMainScreen.DoJob(j0: Integer): Integer;
7255var
7256 Loc0, Movement0: Integer;
7257begin
7258 with MyUn[UnFocus] do
7259 begin
7260 DestinationMarkON := False;
7261 PaintDestination;
7262 Loc0 := Loc;
7263 Movement0 := Movement;
7264 if j0 < 0 then
7265 Result := ProcessEnhancement(UnFocus, MyData.EnhancementJobs)
7266 // terrain enhancement
7267 else
7268 Result := Server(sStartJob + j0 shl 4, Me, UnFocus, nil^);
7269 if Result >= rExecuted then
7270 begin
7271 if Result = eDied then
7272 UnFocus := -1;
7273 PaintLoc(Loc0);
7274 if UnFocus >= 0 then
7275 begin
7276 if (j0 < 0) and (Result <> eJobDone) then
7277 // multi-turn terrain enhancement
7278 Status := Status and ($FFFF - usStay - usRecover - usGoto) or
7279 usEnhance
7280 else
7281 Status := Status and
7282 ($FFFF - usStay - usRecover - usGoto - usEnhance);
7283 if (Job <> jNone) or (Movement0 < 100) then
7284 begin
7285 Status := Status and not usWaiting;
7286 NextUnit(UnStartLoc, True);
7287 end
7288 else
7289 PanelPaint;
7290 end
7291 else
7292 NextUnit(UnStartLoc, True);
7293 end;
7294 end;
7295 case Result of
7296 eNoBridgeBuilding:
7297 SoundMessage(Phrases.Lookup('NOBB'), 'INVALID');
7298 eNoCityTerrain:
7299 SoundMessage(Phrases.Lookup('NOCITY'), 'INVALID');
7300 eTreaty:
7301 SoundMessage(Tribe[MyRO.Territory[Loc0]].TPhrase('PEACE_NOWORK'),
7302 'NOMOVE_TREATY');
7303 else
7304 if Result < rExecuted then
7305 Play('INVALID');
7306 end;
7307end;
7308
7309function TMainScreen.GetBattleDlg: TBattleDlg;
7310begin
7311 if not Assigned(FBattleDlg) then FBattleDlg := TBattleDlg.Create(nil);
7312 Result := FBattleDlg;
7313end;
7314
7315function TMainScreen.GetCityDlg: TCityDlg;
7316begin
7317 if not Assigned(FCityDlg) then begin
7318 FCityDlg := TCityDlg.Create(nil);
7319 FCityDlg.CheckAge;
7320 end;
7321 Result := FCityDlg;
7322end;
7323
7324function TMainScreen.GetCityTypeDlg: TCityTypeDlg;
7325begin
7326 if not Assigned(FCityTypeDlg) then FCityTypeDlg := TCityTypeDlg.Create(nil);
7327 Result := FCityTypeDlg;
7328end;
7329
7330function TMainScreen.GetDiaDlg: TDiaDlg;
7331begin
7332 if not Assigned(FDiaDlg) then begin
7333 FDiaDlg := TDiaDlg.Create(nil);
7334 ArrangeDialog(FDiaDlg);
7335 end;
7336 Result := FDiaDlg;
7337end;
7338
7339function TMainScreen.GetDraftDlg: TDraftDlg;
7340begin
7341 if not Assigned(FDraftDlg) then FDraftDlg := TDraftDlg.Create(nil);
7342 Result := FDraftDlg;
7343end;
7344
7345function TMainScreen.GetEnhanceDlg: TEnhanceDlg;
7346begin
7347 if not Assigned(FEnhanceDlg) then FEnhanceDlg := TEnhanceDlg.Create(nil);
7348 Result := FEnhanceDlg;
7349end;
7350
7351function TMainScreen.GetHelpDlg: THelpDlg;
7352begin
7353 if not Assigned(FHelpDlg) then begin
7354 FHelpDlg := THelpDlg.Create(nil);
7355 ArrangeDialog(FHelpDlg);
7356 if Me <> -1 then
7357 FHelpDlg.Difficulty := G.Difficulty[Me]
7358 else FHelpDlg.Difficulty := 0;
7359 end;
7360 Result := FHelpDlg;
7361end;
7362
7363function TMainScreen.GetListDlg: TListDlg;
7364begin
7365 if not Assigned(FListDlg) then begin
7366 FListDlg := TListDlg.Create(nil);
7367 ArrangeDialog(FListDlg);
7368 end;
7369 Result := FListDlg;
7370end;
7371
7372function TMainScreen.GetMessgExDlg: TMessgExDlg;
7373begin
7374 if not Assigned(FMessgExDlg) then FMessgExDlg := TMessgExDlg.Create(nil);
7375 Result := FMessgExDlg;
7376end;
7377
7378function TMainScreen.GetModalSelectDlg: TModalSelectDlg;
7379begin
7380 if not Assigned(FModalSelectDlg) then FModalSelectDlg := TModalSelectDlg.Create(nil);
7381 Result := FModalSelectDlg;
7382end;
7383
7384function TMainScreen.GetNatStatDlg: TNatStatDlg;
7385begin
7386 if not Assigned(FNatStatDlg) then begin
7387 FNatStatDlg := TNatStatDlg.Create(nil);
7388 ArrangeDialog(FNatStatDlg);
7389 FNatStatDlg.CheckAge;
7390 end;
7391 Result := FNatStatDlg;
7392end;
7393
7394function TMainScreen.GetNegoDlg: TNegoDlg;
7395begin
7396 if not Assigned(FNegoDlg) then FNegoDlg := TNegoDlg.Create(nil);
7397 Result := FNegoDlg;
7398end;
7399
7400function TMainScreen.GetRatesDlg: TRatesDlg;
7401begin
7402 if not Assigned(FRatesDlg) then FRatesDlg := TRatesDlg.Create(nil);
7403 Result := FRatesDlg;
7404end;
7405
7406function TMainScreen.GetTechTreeDlg: TTechTreeDlg;
7407begin
7408 if not Assigned(FTechTreeDlg) then FTechTreeDlg := TTechTreeDlg.Create(nil);
7409 Result := FTechTreeDlg;
7410end;
7411
7412procedure TMainScreen.mDisbandOrUtilizeClick(Sender: TObject);
7413var
7414 Loc0: Integer;
7415begin
7416 if UnFocus >= 0 then
7417 with TUn(MyUn[UnFocus]) do begin
7418 if (Sender = mUtilize) and
7419 not (Server(sRemoveUnit - sExecute, Me, UnFocus, nil^) = eUtilized) then
7420 begin
7421 SimpleMessage(Phrases2.Lookup('SHIP_UTILIZE'));
7422 // freight for colony ship is the only case in which the command is
7423 // available to player though not valid
7424 Exit;
7425 end;
7426 if (Sender = mUtilize) and (Health < 100) then
7427 if SimpleQuery(mkYesNo, Phrases.Lookup('DAMAGED_UTILIZE'), '') <> mrOK
7428 then
7429 Exit;
7430 Loc0 := Loc;
7431 CityOptimizer_BeforeRemoveUnit(UnFocus);
7432 if Server(sRemoveUnit, Me, UnFocus, nil^) = eUtilized then
7433 Play('CITY_UTILIZE')
7434 else
7435 Play('DISBAND');
7436 CityOptimizer_AfterRemoveUnit;
7437 SetTroopLoc(Loc0);
7438 UpdateViews(True);
7439 DestinationMarkON := False;
7440 PaintDestination;
7441 UnFocus := -1;
7442 PaintLoc(Loc0);
7443 NextUnit(UnStartLoc, True);
7444 end;
7445end;
7446
7447procedure TMainScreen.InitPopup(Popup: TPopupMenu);
7448var
7449 I, p1, Tile, Test: Integer;
7450 NoSuper, Extended, Multi, NeedSep, HaveCities: Boolean;
7451 LastSep, M: TMenuItem;
7452 mox: ^TModel;
7453begin
7454 NoSuper := not Supervising and (1 shl Me and MyRO.Alive <> 0);
7455 HaveCities := False;
7456 for I := 0 to MyRO.nCity - 1 do
7457 if MyCity[I].Loc >= 0 then
7458 begin
7459 HaveCities := True;
7460 Break;
7461 end;
7462 if Popup = GamePopup then
7463 begin
7464 mTechTree.Visible := ClientMode <> cEditMap;
7465 mResign.Enabled := Supervising or (Me = 0) and (ClientMode < scContact);
7466 mRandomMap.Visible := (ClientMode = cEditMap) and
7467 (Server(sMapGeneratorRequest, Me, 0, nil^) = eOK);
7468 mFillMap.Visible := mRandomMap.Visible;
7469 mOptions.Visible := ClientMode <> cEditMap;
7470 mManip.Visible := ClientMode <> cEditMap;
7471 if ClientMode <> cEditMap then
7472 begin
7473 mWaitTurn.Visible := NoSuper;
7474 mRep.Visible := NoSuper;
7475 mRepList.Visible := NoSuper;
7476 mRepScreens.Visible := NoSuper;
7477 N10.Visible := NoSuper;
7478 mOwnMovement.Visible := NoSuper;
7479 mAllyMovement.Visible := NoSuper;
7480 case SoundMode of
7481 smOff:
7482 mSoundOff.Checked := True;
7483 smOn:
7484 mSoundOn.Checked := True;
7485 smOnAlt:
7486 mSoundOnAlt.Checked := True;
7487 end;
7488
7489 for I := 0 to nTestFlags - 1 do
7490 mManip[I].Checked := MyRO.TestFlags and (1 shl I) <> 0;
7491 mManip.Enabled := Supervising or (Me = 0);
7492
7493 Multi := False;
7494 for p1 := 1 to nPl - 1 do
7495 if G.RO[p1] <> nil then begin
7496 Multi := True;
7497 Break;
7498 end;
7499 mEnemyMovement.Visible := not Multi;
7500 end;
7501 mMacro.Visible := NoSuper and (ClientMode < scContact);
7502 if NoSuper and (ClientMode < scContact) then
7503 begin
7504 mCityTypes.Enabled := False;
7505 // check if city types already usefull:
7506 if MyRO.nCity > 0 then
7507 for I := nWonder to nImp - 1 do
7508 if (I <> imTrGoods) and (Imp[I].Kind = ikCommon) and
7509 (Imp[I].Preq <> preNA) and
7510 ((Imp[I].Preq = preNone) or (MyRO.Tech[Imp[I].Preq] >= tsApplicable))
7511 then
7512 begin
7513 mCityTypes.Enabled := True;
7514 Break
7515 end;
7516 end;
7517 mViewpoint.Visible := (ClientMode <> cEditMap) and Supervising;
7518 mViewpoint.Enabled := G.RO[0].Turn > 0;
7519 if Supervising then
7520 begin
7521 EmptyMenu(mViewpoint);
7522 for p1 := 0 to nPl - 1 do
7523 if (p1 = 0) or (1 shl p1 and G.RO[0].Alive <> 0) then
7524 begin
7525 M := TMenuItem.Create(mViewpoint);
7526 if p1 = 0 then
7527 M.Caption := Phrases.Lookup('SUPER')
7528 else
7529 M.Caption := Tribe[p1].TString(Phrases2.Lookup('BELONG'));
7530 M.Tag := p1;
7531 M.OnClick := ViewpointClick;
7532 if p1 < 10 then
7533 M.ShortCut := ShortCut(48 + p1, [ssCtrl]);
7534 M.RadioItem := True;
7535 if p1 = Me then
7536 M.Checked := True;
7537 mViewpoint.Add(M);
7538 end
7539 end;
7540 mDebugMap.Visible := (ClientMode <> cEditMap) and Supervising;
7541 if Supervising then
7542 begin
7543 EmptyMenu(mDebugMap);
7544 for p1 := 0 to nPl - 1 do
7545 if (p1 = 0) or (1 shl p1 and G.RO[0].Alive <> 0) then
7546 begin
7547 M := TMenuItem.Create(mDebugMap);
7548 if p1 = 0 then
7549 M.Caption := Phrases2.Lookup('MENU_DEBUGMAPOFF')
7550 else
7551 M.Caption := Tribe[p1].TString(Phrases2.Lookup('BELONG'));
7552 if p1 = 0 then
7553 M.Tag := -1
7554 else
7555 M.Tag := p1;
7556 M.OnClick := DebugMapClick;
7557 if p1 < 10 then
7558 M.ShortCut := ShortCut(48 + p1, [ssAlt]);
7559 M.RadioItem := True;
7560 if M.Tag = MainMap.pDebugMap then
7561 M.Checked := True;
7562 mDebugMap.Add(M);
7563 end;
7564 end;
7565 mSmallTiles.Checked := MainMap.TileSize = tsSmall;
7566 mNormalTiles.Checked := MainMap.TileSize = tsMedium;
7567 mBigTiles.Checked := MainMap.TileSize = tsBig;
7568 end
7569 else if Popup = StatPopup then
7570 begin
7571 mEmpire.Visible := NoSuper;
7572 mEmpire.Enabled := MyRO.Government <> gAnarchy;
7573 mRevolution.Visible := NoSuper;
7574 mRevolution.Enabled := (MyRO.Government <> gAnarchy) and
7575 (ClientMode < scContact);
7576 mUnitStat.Enabled := NoSuper or (MyRO.Turn > 0);
7577 mCityStat.Visible := 1 shl Me and MyRO.Alive <> 0;
7578 mCityStat.Enabled := HaveCities;
7579 mScienceStat.Visible := True;
7580 mScienceStat.Enabled := not NoSuper or (MyRO.ResearchTech >= 0) or
7581 (MyRO.Happened and phTech <> 0) or (MyRO.Happened and phGameEnd <> 0)
7582 // no researchtech in case just completed
7583 or (MyRO.TestFlags and (tfAllTechs or tfUncover or tfAllContact) <> 0);
7584 mEUnitStat.Enabled := MyRO.nEnemyModel > 0;
7585 { mWonders.Enabled:= False;
7586 for I := 0 to nWonder - 1 do if MyRO.Wonder[I].CityID <> WonderNotBuiltYet then
7587 mWonders.Enabled := True; }
7588 mDiagram.Enabled := MyRO.Turn >= 2;
7589 mShips.Enabled := False;
7590 for p1 := 0 to nPl - 1 do
7591 if MyRO.Ship[p1].Parts[spComp] + MyRO.Ship[p1].Parts[spPow] +
7592 MyRO.Ship[p1].Parts[spHab] > 0 then begin
7593 mShips.Enabled := True;
7594 Break;
7595 end;
7596 end
7597 else if Popup = UnitPopup then
7598 begin
7599 mox := @MyModel[MyUn[UnFocus].mix];
7600 Tile := MyMap[MyUn[UnFocus].Loc];
7601 Extended := Tile and fCity = 0;
7602 if Extended then
7603 begin
7604 mCity.Caption := Phrases.Lookup('BTN_FOUND');
7605 mHome.Caption := Phrases.Lookup('BTN_MOVEHOME')
7606 end
7607 else
7608 begin
7609 mCity.Caption := Phrases.Lookup('BTN_ADD');
7610 mHome.Caption := Phrases.Lookup('BTN_SETHOME')
7611 end;
7612
7613 Extended := Extended and ((mox.Kind = mkSettler) or (mox.Kind = mkSlaves)
7614 and (MyRO.Wonder[woPyramids].EffectiveOwner >= 0)) and
7615 (MyUn[UnFocus].Master < 0) and (Tile and fDeadLands = 0);
7616 if (mox.Kind = mkFreight) and (Tile and fCity <> 0) and
7617 not Phrases2FallenBackToEnglish or
7618 (Server(sRemoveUnit - sExecute, Me, UnFocus, nil^) = eUtilized) then
7619 begin
7620 mDisband.Visible := False;
7621 mUtilize.Visible := True;
7622 if mox.Kind = mkFreight then
7623 mUtilize.Caption := Phrases.Lookup('UTILIZE')
7624 else
7625 mUtilize.Caption := Phrases.Lookup('INTEGRATE')
7626 end
7627 else
7628 begin
7629 mDisband.Visible := True;
7630 mUtilize.Visible := False
7631 end;
7632 mGoOn.Visible := MyUn[UnFocus].Status and (usGoto or usWaiting) = usGoto or
7633 usWaiting;
7634 mHome.Visible := HaveCities;
7635 mRecover.Visible := (MyUn[UnFocus].Health < 100) and
7636 (Tile and fTerrain >= fGrass) and
7637 ((MyRO.Wonder[woGardens].EffectiveOwner = Me) or
7638 (Tile and fTerrain <> fArctic) and (Tile and fTerrain <> fDesert)) and
7639 not ((mox.Domain = dAir) and (Tile and fCity = 0) and
7640 (Tile and fTerImp <> tiBase));
7641 mStay.Visible := not ((mox.Domain = dAir) and (Tile and fCity = 0) and
7642 (Tile and fTerImp <> tiBase));
7643 mCity.Visible := Extended and (mox.Kind = mkSettler) or
7644 (Tile and fCity <> 0) and ((mox.Kind in [mkSettler, mkSlaves]) or
7645 (MyUn[UnFocus].Flags and unConscripts <> 0));
7646 mPillage.Visible := (Tile and (fRoad or fRR or fCanal or fTerImp) <> 0) and
7647 (MyUn[UnFocus].Master < 0) and (mox.Domain = dGround);
7648 mCancel.Visible := (MyUn[UnFocus].Job > jNone) or
7649 (MyUn[UnFocus].Status and (usRecover or usGoto) <> 0);
7650
7651 Test := Server(sLoadUnit - sExecute, Me, UnFocus, nil^);
7652 mLoad.Visible := (Test >= rExecuted) or (Test = eNoTime_Load);
7653 mUnload.Visible := (MyUn[UnFocus].Master >= 0) or
7654 (MyUn[UnFocus].TroopLoad + MyUn[UnFocus].AirLoad > 0);
7655 mSelectTransport.Visible := Server(sSelectTransport - sExecute, Me, UnFocus,
7656 nil^) >= rExecuted;
7657 end
7658 else { if Popup=TerrainPopup then }
7659 begin
7660 mox := @MyModel[MyUn[UnFocus].mix];
7661 Tile := MyMap[MyUn[UnFocus].Loc];
7662 Extended := Tile and fCity = 0;
7663
7664 if (Tile and fRiver <> 0) and (MyRO.Tech[adBridgeBuilding] >= tsApplicable)
7665 then
7666 begin
7667 mRoad.Caption := Phrases.Lookup('BTN_BUILDBRIDGE');
7668 mRailRoad.Caption := Phrases.Lookup('BTN_BUILDRRBRIDGE');
7669 end
7670 else
7671 begin
7672 mRoad.Caption := Phrases.Lookup('BTN_BUILDROAD');
7673 mRailRoad.Caption := Phrases.Lookup('BTN_BUILDRR');
7674 end;
7675 if Tile and fTerrain = fForest then
7676 mClear.Caption := Phrases.Lookup('BTN_CLEAR')
7677 else if Tile and fTerrain = fDesert then
7678 mClear.Caption := Phrases.Lookup('BTN_UNDESERT')
7679 else
7680 mClear.Caption := Phrases.Lookup('BTN_DRAIN');
7681
7682 Extended := Extended and ((mox.Kind = mkSettler) or (mox.Kind = mkSlaves)
7683 and (MyRO.Wonder[woPyramids].EffectiveOwner >= 0)) and
7684 (MyUn[UnFocus].Master < 0);
7685 if Extended then
7686 begin
7687 mRoad.Visible := JobTest(UnFocus, jRoad, [eNoBridgeBuilding, eTreaty]);
7688 mRailRoad.Visible := JobTest(UnFocus, jRR, [eNoBridgeBuilding, eTreaty]);
7689 mClear.Visible := JobTest(UnFocus, jClear, [eTreaty]);
7690 mIrrigation.Visible := JobTest(UnFocus, jIrr, [eTreaty]);
7691 mFarm.Visible := JobTest(UnFocus, jFarm, [eTreaty]);
7692 mAfforest.Visible := JobTest(UnFocus, jAfforest, [eTreaty]);
7693 mMine.Visible := JobTest(UnFocus, jMine, [eTreaty]);
7694 MTrans.Visible := JobTest(UnFocus, jTrans, [eTreaty]);
7695 mCanal.Visible := JobTest(UnFocus, jCanal, [eTreaty]);
7696 mFort.Visible := JobTest(UnFocus, jFort, [eTreaty]);
7697 mAirBase.Visible := JobTest(UnFocus, jBase, [eTreaty]);
7698 mPollution.Visible := JobTest(UnFocus, jPoll, [eTreaty]);
7699 mEnhance.Visible := (Tile and fDeadLands = 0) and
7700 (MyData.EnhancementJobs[MyMap[MyUn[UnFocus].Loc] and fTerrain, 0]
7701 <> jNone);
7702 end
7703 else
7704 begin
7705 for I := 0 to Popup.Items.Count - 1 do
7706 Popup.Items[I].Visible := False;
7707 end;
7708 end;
7709
7710 // set menu seperators
7711 LastSep := nil;
7712 NeedSep := False;
7713 for I := 0 to Popup.Items.Count - 1 do
7714 if Popup.Items[I].Caption = '-' then
7715 begin
7716 Popup.Items[I].Visible := NeedSep;
7717 if NeedSep then
7718 LastSep := Popup.Items[I];
7719 NeedSep := False;
7720 end
7721 else if Popup.Items[I].Visible then
7722 NeedSep := True;
7723 if (LastSep <> nil) and not NeedSep then
7724 LastSep.Visible := False;
7725end;
7726
7727procedure TMainScreen.PanelBtnClick(Sender: TObject);
7728var
7729 Popup: TPopupMenu;
7730begin
7731 if Sender = UnitBtn then
7732 Popup := UnitPopup
7733 else { if Sender=TerrainBtn then }
7734 Popup := TerrainPopup;
7735 InitPopup(Popup);
7736 if FullScreen then
7737 Popup.Popup(Left + TControl(Sender).Left, Top + TControl(Sender).Top)
7738 else
7739 Popup.Popup(Left + TControl(Sender).Left + 4, Top + TControl(Sender).Top +
7740 GetSystemMetrics(SM_CYCAPTION) + 4);
7741end;
7742
7743procedure TMainScreen.CityClosed(Activateuix: Integer; StepFocus: Boolean;
7744 SelectFocus: Boolean);
7745begin
7746 if Supervising then
7747 begin
7748 SetTroopLoc(-1);
7749 PanelPaint;
7750 end
7751 else
7752 begin
7753 if Activateuix >= 0 then
7754 begin
7755 SetUnFocus(Activateuix);
7756 SetTroopLoc(MyUn[Activateuix].Loc);
7757 if SelectFocus then
7758 FocusOnLoc(TroopLoc, flRepaintPanel)
7759 else
7760 PanelPaint;
7761 end
7762 else if StepFocus then
7763 NextUnit(TroopLoc, True)
7764 else
7765 begin
7766 SetTroopLoc(-1);
7767 PanelPaint;
7768 end;
7769 end;
7770end;
7771
7772procedure TMainScreen.Toggle(Sender: TObject);
7773begin
7774 TMenuItem(Sender).Checked := not TMenuItem(Sender).Checked;
7775end;
7776
7777procedure TMainScreen.PanelBoxMouseMove(Sender: TObject; Shift: TShiftState;
7778 X, Y: Integer);
7779var
7780 xCentre, yCentre: Integer;
7781begin
7782 if Tracking and (ssLeft in Shift) then
7783 with MainMap do begin
7784 if (X >= xMini + 2) and (Y >= yMini + 2) and (X < xMini + 2 + 2 * G.lx) and
7785 (Y < yMini + 2 + G.ly) then
7786 begin
7787 xCentre := (xwMini + (X - xMini - 2) div 2 + G.lx div 2 +
7788 MapWidth div (xxt * 4)) mod G.lx;
7789 yCentre := (Y - yMini - 2);
7790 xw := (xCentre - MapWidth div (xxt * 4) + G.lx) mod G.lx;
7791 if ywmax <= 0 then
7792 yw := ywcenter
7793 else
7794 begin
7795 yw := (yCentre - MapHeight div (yyt * 2) + 1) and not 1;
7796 if yw < 0 then
7797 yw := 0
7798 else if yw > ywmax then
7799 yw := ywmax;
7800 end;
7801 BitBltBitmap(Buffer, 0, 0, G.lx * 2, G.ly, MiniMap.Bitmap, 0, 0);
7802 if ywmax <= 0 then
7803 Frame(Buffer.Canvas, X - xMini - 2 - MapWidth div (xxt * 2), 0,
7804 X - xMini - 2 + MapWidth div (xxt * 2) - 1, G.ly - 1,
7805 MainTexture.ColorMark, MainTexture.ColorMark)
7806 else
7807 Frame(Buffer.Canvas, X - xMini - 2 - MapWidth div (xxt * 2), yw,
7808 X - xMini - 2 + MapWidth div (xxt * 2) - 1, yw + MapHeight div yyt -
7809 2, MainTexture.ColorMark, MainTexture.ColorMark);
7810 BitBltBitmap(Panel, xMini + 2, yMini + 2, G.lx * 2, G.ly,
7811 Buffer, 0, 0);
7812 MainOffscreenPaint;
7813 RectInvalidate(xMini + 2, TopBarHeight + MapHeight - Overlap + yMini + 2,
7814 xMini + 2 + G.lx * 2, TopBarHeight + MapHeight - Overlap + yMini +
7815 2 + G.ly);
7816 Update;
7817 end;
7818 end
7819 else
7820 Tracking := False;
7821end;
7822
7823procedure TMainScreen.PanelBoxMouseUp(Sender: TObject; Button: TMouseButton;
7824 Shift: TShiftState; X, Y: Integer);
7825begin
7826 if Tracking then
7827 begin
7828 Tracking := False;
7829 xwMini := xw;
7830 ywMini := yw;
7831 MiniMapPaint;
7832 PanelPaint;
7833 end;
7834end;
7835
7836procedure TMainScreen.MapBoxMouseMove(Sender: TObject; Shift: TShiftState;
7837 X, Y: Integer);
7838var
7839 MouseLoc: Integer;
7840begin
7841 xMouse := X;
7842 yMouse := Y;
7843 if (ClientMode = cEditMap) and (ssLeft in Shift) and not Tracking then
7844 begin
7845 MouseLoc := LocationOfScreenPixel(X, Y);
7846 if MouseLoc <> BrushLoc then
7847 MapBoxMouseDown(nil, TMouseButton.mbLeft, Shift, X, Y);
7848 end
7849 (* else if Idle and (UnFocus >= 0) then
7850 begin
7851 qx := (xMouse * 32 + yMouse * 66 + 16 * 66) div(32 * 66) - 1;
7852 qy := (yMouse * 66 - xMouse * 32 - 16 * 66 + 2000 * 33 * 32) div(32 * 66) - 999;
7853 MouseLoc := (xw + (qx - qy + 2048) div 2 - 1024 + G.lx) mod G.lx + G.lx * (yw + qx + qy);
7854 ShowMoveHint(MouseLoc);
7855 end *)
7856end;
7857
7858procedure TMainScreen.mShowClick(Sender: TObject);
7859begin
7860 TMenuItem(Sender).Checked := not TMenuItem(Sender).Checked;
7861 SetMapOptions;
7862 MapValid := False;
7863 PaintAllMaps;
7864end;
7865
7866procedure TMainScreen.mNamesClick(Sender: TObject);
7867var
7868 p1: Integer;
7869begin
7870 mNames.Checked := not mNames.Checked;
7871 GenerateNames := mNames.Checked;
7872 for p1 := 0 to nPl - 1 do
7873 if Tribe[p1] <> nil then
7874 if GenerateNames then
7875 Tribe[p1].NumberName := -1
7876 else
7877 Tribe[p1].NumberName := p1;
7878 MapValid := False;
7879 PaintAll;
7880end;
7881
7882function TMainScreen.IsPanelPixel(X, Y: Integer): Boolean;
7883begin
7884 Result := (Y >= TopBarHeight + MapHeight) or (Y >= ClientHeight - PanelHeight)
7885 and ((X < xMidPanel) or (X >= xRightPanel));
7886end;
7887
7888procedure TMainScreen.FormMouseDown(Sender: TObject; Button: TMouseButton;
7889 Shift: TShiftState; X, Y: Integer);
7890begin
7891 if Idle then
7892 if (X < 40) and (Y < 40) then
7893 begin
7894 if GameMode <> cMovie then
7895 begin
7896 InitPopup(GamePopup);
7897 if FullScreen then
7898 GamePopup.Popup(Left, Top + TopBarHeight - 1)
7899 else
7900 GamePopup.Popup(Left + 4, Top + GetSystemMetrics(SM_CYCAPTION) + 4 +
7901 TopBarHeight - 1);
7902 end;
7903 end
7904 else if IsPanelPixel(X, Y) then
7905 PanelBoxMouseDown(Sender, Button, Shift, X,
7906 Y - (ClientHeight - PanelHeight))
7907 else if (Y >= TopBarHeight) and (X >= MapOffset) and
7908 (X < MapOffset + MapWidth) then
7909 MapBoxMouseDown(Sender, Button, Shift, X - MapOffset, Y - TopBarHeight);
7910end;
7911
7912procedure TMainScreen.FormMouseMove(Sender: TObject; Shift: TShiftState;
7913 X, Y: Integer);
7914begin
7915 if Idle then
7916 if IsPanelPixel(X, Y) then
7917 PanelBoxMouseMove(Sender, Shift, X, Y - (ClientHeight - PanelHeight))
7918 else if (Y >= TopBarHeight) and (X >= MapOffset) and
7919 (X < MapOffset + MapWidth) then
7920 MapBoxMouseMove(Sender, Shift, X - MapOffset, Y - TopBarHeight);
7921end;
7922
7923procedure TMainScreen.FormMouseUp(Sender: TObject; Button: TMouseButton;
7924 Shift: TShiftState; X, Y: Integer);
7925begin
7926 if Idle then
7927 PanelBoxMouseUp(Sender, Button, Shift, X, Y - (ClientHeight - PanelHeight));
7928end;
7929
7930procedure TMainScreen.FormPaint(Sender: TObject);
7931begin
7932 MainOffscreenPaint;
7933 if (MapOffset > 0) or (MapOffset + MapWidth < ClientWidth) then
7934 with Canvas do
7935 begin // pillarbox, make left and right border black
7936 if Me < 0 then
7937 Brush.Color := $000000
7938 else
7939 Brush.Color := EmptySpaceColor;
7940 if xMidPanel > MapOffset then
7941 FillRect(Rect(0, TopBarHeight, MapOffset, TopBarHeight + MapHeight
7942 - Overlap))
7943 else
7944 begin
7945 FillRect(Rect(0, TopBarHeight, xMidPanel, TopBarHeight + MapHeight -
7946 Overlap));
7947 FillRect(Rect(xMidPanel, TopBarHeight, MapOffset,
7948 TopBarHeight + MapHeight));
7949 end;
7950 if xRightPanel < MapOffset + MapWidth then
7951 FillRect(Rect(MapOffset + MapWidth, TopBarHeight, ClientWidth,
7952 TopBarHeight + MapHeight - Overlap))
7953 else
7954 begin
7955 FillRect(Rect(MapOffset + MapWidth, TopBarHeight, xRightPanel,
7956 TopBarHeight + MapHeight));
7957 FillRect(Rect(xRightPanel, TopBarHeight, ClientWidth,
7958 TopBarHeight + MapHeight - Overlap));
7959 end;
7960 Brush.Style := TBrushStyle.bsClear;
7961 end;
7962 BitBltCanvas(Canvas, MapOffset, TopBarHeight, MapWidth, MapHeight - Overlap,
7963 Offscreen.Canvas, 0, 0);
7964 BitBltCanvas(Canvas, 0, 0, ClientWidth, TopBarHeight, TopBar.Canvas,
7965 0, 0);
7966 if xMidPanel > MapOffset then
7967 BitBltCanvas(Canvas, xMidPanel, TopBarHeight + MapHeight - Overlap,
7968 ClientWidth div 2 - xMidPanel, Overlap, Offscreen.Canvas,
7969 xMidPanel - MapOffset, MapHeight - Overlap)
7970 else
7971 BitBltCanvas(Canvas, MapOffset, TopBarHeight + MapHeight - Overlap,
7972 ClientWidth div 2 - MapOffset, Overlap, Offscreen.Canvas, 0,
7973 MapHeight - Overlap);
7974 if xRightPanel < MapOffset + MapWidth then
7975 BitBltCanvas(Canvas, ClientWidth div 2, TopBarHeight + MapHeight - Overlap,
7976 xRightPanel - ClientWidth div 2, Overlap, Offscreen.Canvas,
7977 ClientWidth div 2 - MapOffset, MapHeight - Overlap)
7978 else
7979 BitBltCanvas(Canvas, ClientWidth div 2, TopBarHeight + MapHeight - Overlap,
7980 MapOffset + MapWidth - ClientWidth div 2, Overlap,
7981 Offscreen.Canvas, ClientWidth div 2 - MapOffset,
7982 MapHeight - Overlap);
7983 BitBltCanvas(Canvas, 0, TopBarHeight + MapHeight - Overlap, xMidPanel,
7984 Overlap, Panel.Canvas, 0, 0);
7985 BitBltCanvas(Canvas, xRightPanel, TopBarHeight + MapHeight - Overlap,
7986 Panel.Width - xRightPanel, Overlap, Panel.Canvas, xRightPanel, 0);
7987 BitBltCanvas(Canvas, 0, TopBarHeight + MapHeight, Panel.Width,
7988 PanelHeight - Overlap, Panel.Canvas, 0, Overlap);
7989 if (pLogo >= 0) and (G.RO[pLogo] = nil) and (AILogo[pLogo] <> nil) then
7990 BitBltCanvas(Canvas, xRightPanel + 10 - (16 + 64),
7991 ClientHeight - PanelHeight, 64, 64, AILogo[pLogo].Canvas, 0, 0);
7992end;
7993
7994procedure TMainScreen.RectInvalidate(Left, Top, Rigth, Bottom: Integer);
7995var
7996 r0: HRgn;
7997begin
7998 r0 := CreateRectRgn(Left, Top, Rigth, Bottom);
7999 InvalidateRgn(Handle, r0, False);
8000 DeleteObject(r0);
8001end;
8002
8003procedure TMainScreen.SmartRectInvalidate(Left, Top, Rigth, Bottom: Integer);
8004var
8005 I: Integer;
8006 r0, r1: HRgn;
8007begin
8008 r0 := CreateRectRgn(Left, Top, Rigth, Bottom);
8009 for I := 0 to ControlCount - 1 do
8010 if not (Controls[I] is TArea) and Controls[I].Visible then
8011 begin
8012 with Controls[I].BoundsRect do
8013 r1 := CreateRectRgn(Left, Top, Right, Bottom);
8014 CombineRgn(r0, r0, r1, RGN_DIFF);
8015 DeleteObject(r1);
8016 end;
8017 InvalidateRgn(Handle, r0, False);
8018 DeleteObject(r0);
8019end;
8020
8021procedure TMainScreen.LoadSettings;
8022var
8023 Reg: TRegistry;
8024 DefaultOptionChecked: TSaveOptions;
8025begin
8026 DefaultOptionChecked := [soEnMoves, soSlowMoves, soNames, soRepScreens,
8027 soSoundOn, soScrollSlow, soAlSlowMoves];
8028 Reg := TRegistry.Create;
8029 with Reg do try
8030 OpenKey(AppRegistryKey, False);
8031 if ValueExists('TileSize') then MainMap.TileSize := TTileSize(ReadInteger('TileSize'))
8032 else MainMap.TileSize := tsMedium;
8033 NoMap.TileSize := MainMap.TileSize;
8034 if ValueExists('OptionChecked') then OptionChecked := TSaveOptions(ReadInteger('OptionChecked'))
8035 else OptionChecked := DefaultOptionChecked;
8036 if ValueExists('MapOptionChecked') then MapOptionChecked := TMapOptions(ReadInteger('MapOptionChecked'))
8037 else MapOptionChecked := [moCityNames];
8038 if ValueExists('CityReport') then CityRepMask := Cardinal(ReadInteger('CityReport'))
8039 else CityRepMask := Cardinal(not chPopIncrease and not chNoGrowthWarning and
8040 not chCaptured);
8041 if (not (soScrollFast in OptionChecked)) and (not (soScrollSlow in OptionChecked)) and
8042 (not (soScrollOff in OptionChecked)) then
8043 OptionChecked := OptionChecked + [soScrollSlow];
8044 // old regver with no scrolling
8045 finally
8046 Free;
8047 end;
8048
8049 if soSoundOff in OptionChecked then
8050 SoundMode := smOff
8051 else if soSoundOnAlt in OptionChecked then
8052 SoundMode := smOnAlt
8053 else
8054 SoundMode := smOn;
8055end;
8056
8057procedure TMainScreen.mRepClicked(Sender: TObject);
8058begin
8059 with TMenuItem(Sender) do
8060 begin
8061 Checked := not Checked;
8062 if Checked then
8063 CityRepMask := CityRepMask or (1 shl (Tag shr 8))
8064 else
8065 CityRepMask := CityRepMask and not (1 shl (Tag shr 8));
8066 end;
8067end;
8068
8069procedure TMainScreen.mLogClick(Sender: TObject);
8070begin
8071 LogDlg.Show;
8072end;
8073
8074procedure TMainScreen.FormShow(Sender: TObject);
8075begin
8076 if FullScreen then BoundsRect := Screen.PrimaryMonitor.BoundsRect
8077 else BoundsRect := TermBounds;
8078 FormRestoredSize := TermBounds;
8079 SetFullScreen(FullScreen);
8080 FormResize(nil); // place mini map correctly according to its size
8081 Timer1.Enabled := True;
8082end;
8083
8084procedure TMainScreen.FormClose(Sender: TObject; var Action: TCloseAction);
8085begin
8086 if FullScreen then TermBounds := FormRestoredSize
8087 else TermBounds := BoundsRect;
8088 Timer1.Enabled := False;
8089end;
8090
8091procedure TMainScreen.Radio(Sender: TObject);
8092begin
8093 TMenuItem(Sender).Checked := True;
8094end;
8095
8096procedure TMainScreen.mManipClick(Sender: TObject);
8097var
8098 Flag: Integer;
8099begin
8100 with TMenuItem(Sender) do
8101 begin
8102 Flag := 1 shl (Tag shr 8);
8103 if Checked then
8104 Server(sClearTestFlag, 0, Flag, nil^)
8105 else
8106 begin
8107 Server(sSetTestFlag, 0, Flag, nil^);
8108 Play('CHEAT');
8109 end;
8110 if not Supervising then
8111 begin
8112 if Flag = tfUncover then
8113 begin
8114 MapValid := False;
8115 PaintAllMaps;
8116 end
8117 else if Flag = tfAllTechs then
8118 TellNewModels;
8119 end;
8120 end;
8121end;
8122
8123procedure TMainScreen.MapBtnClick(Sender: TObject);
8124begin
8125 with TButtonC(Sender) do
8126 begin
8127 MapOptionChecked := TMapOptions(Integer(MapOptionChecked) xor (1 shl (Tag shr 8)));
8128 SetMapOptions;
8129 ButtonIndex := Integer(MapOptionChecked) shr (Tag shr 8) and 1 + 2;
8130 end;
8131 if Sender = MapBtn0 then
8132 begin
8133 MiniMapPaint;
8134 PanelPaint;
8135 end // update mini map only
8136 else
8137 begin
8138 MapValid := False;
8139 PaintAllMaps;
8140 end; // update main map
8141end;
8142
8143procedure TMainScreen.GrWallBtnDownChanged(Sender: TObject);
8144begin
8145 if TButtonBase(Sender).Down then
8146 begin
8147 MapOptionChecked := MapOptionChecked + [moGreatWall];
8148 TButtonBase(Sender).Hint := '';
8149 end
8150 else
8151 begin
8152 MapOptionChecked := MapOptionChecked - [moGreatWall];
8153 TButtonBase(Sender).Hint := Phrases.Lookup('CONTROLS',
8154 -1 + TButtonBase(Sender).Tag and $FF);
8155 end;
8156 SetMapOptions;
8157 MapValid := False;
8158 PaintAllMaps;
8159end;
8160
8161procedure TMainScreen.BareBtnDownChanged(Sender: TObject);
8162begin
8163 if TButtonBase(Sender).Down then
8164 begin
8165 MapOptionChecked := MapOptionChecked + [moBareTerrain];
8166 TButtonBase(Sender).Hint := '';
8167 end
8168 else
8169 begin
8170 MapOptionChecked := MapOptionChecked - [moBareTerrain];
8171 TButtonBase(Sender).Hint := Phrases.Lookup('CONTROLS',
8172 -1 + TButtonBase(Sender).Tag and $FF);
8173 end;
8174 SetMapOptions;
8175 MapValid := False;
8176 PaintAllMaps;
8177end;
8178
8179procedure TMainScreen.FormKeyUp(Sender: TObject; var Key: Word;
8180 Shift: TShiftState);
8181begin
8182 if KeyboardDisabled then Exit;
8183
8184 if Idle and (Key = VK_APPS) then
8185 begin
8186 InitPopup(GamePopup);
8187 if FullScreen then
8188 GamePopup.Popup(Left, Top + TopBarHeight - 1)
8189 else
8190 GamePopup.Popup(Left + 4, Top + GetSystemMetrics(SM_CYCAPTION) + 4 +
8191 TopBarHeight - 1);
8192 Exit;
8193 end; // windows menu button calls game menu
8194end;
8195
8196procedure TMainScreen.CreateUnitClick(Sender: TObject);
8197var
8198 p1, mix: Integer;
8199begin
8200 p1 := TComponent(Sender).Tag shr 16;
8201 mix := TComponent(Sender).Tag and $FFFF;
8202 if Server(sCreateUnit + p1 shl 4, Me, mix, EditLoc) >= rExecuted then
8203 PaintLoc(EditLoc);
8204end;
8205
8206procedure TMainScreen.mSoundOffClick(Sender: TObject);
8207begin
8208 SoundMode := smOff;
8209end;
8210
8211procedure TMainScreen.mSoundOnClick(Sender: TObject);
8212begin
8213 SoundMode := smOn;
8214end;
8215
8216procedure TMainScreen.mSoundOnAltClick(Sender: TObject);
8217begin
8218 SoundMode := smOnAlt;
8219end;
8220
8221procedure TMainScreen.UnitInfoBtnClick(Sender: TObject);
8222begin
8223 if UnFocus >= 0 then
8224 UnitStatDlg.ShowNewContent_OwnModel(wmPersistent, MyUn[UnFocus].mix)
8225end;
8226
8227procedure TMainScreen.ViewpointClick(Sender: TObject);
8228begin
8229 SetViewpoint(TMenuItem(Sender).Tag);
8230end;
8231
8232procedure TMainScreen.DebugMapClick(Sender: TObject);
8233begin
8234 SetDebugMap(TMenuItem(Sender).Tag);
8235end;
8236
8237procedure TMainScreen.mSmallTilesClick(Sender: TObject);
8238begin
8239 SetTileSizeCenter(tsSmall);
8240end;
8241
8242procedure TMainScreen.mNormalTilesClick(Sender: TObject);
8243begin
8244 SetTileSizeCenter(tsMedium);
8245end;
8246
8247procedure TMainScreen.mBigTilesClick(Sender: TObject);
8248begin
8249 SetTileSizeCenter(tsBig);
8250end;
8251
8252procedure TMainScreen.SetTileSizeCenter(TileSize: TTileSize);
8253begin
8254 SetTileSize(TileSize, GetCenterLoc, Point(MapWidth div 2, MapHeight div 2));
8255end;
8256
8257procedure TMainScreen.SetTileSize(TileSize: TTileSize; Loc: Integer; MapPos: TPoint);
8258begin
8259 MainMap.TileSize := TileSize;
8260 NoMap.TileSize := TileSize;
8261 LastResizeWidth := -1; // Force repaint on
8262 ResizeControls;
8263 SetMapPos(Loc, MapPos);
8264 PaintAllMaps;
8265 ApplyToVisibleForms(faSmartUpdateContent);
8266end;
8267
8268procedure TMainScreen.SaveMenuItemsState;
8269var
8270 I, J: Integer;
8271begin
8272 if soTellAI in OptionChecked then OptionChecked := [soTellAI]
8273 else OptionChecked := [];
8274 for I := 0 to ComponentCount - 1 do
8275 if Components[I] is TMenuItem then
8276 for J := 0 to Length(SaveOption) - 1 do
8277 if TMenuItem(Components[I]).Checked and
8278 (TMenuItem(Components[I]).Tag = SaveOption[J]) then
8279 OptionChecked := OptionChecked + [TSaveOption(J)];
8280end;
8281
8282procedure TMainScreen.SaveSettings;
8283var
8284 Reg: TRegistry;
8285begin
8286 SaveMenuItemsState;
8287
8288 Reg := TRegistry.Create;
8289 with Reg do
8290 try
8291 OpenKey(AppRegistryKey, True);
8292 WriteInteger('TileSize', Integer(MainMap.TileSize));
8293 WriteInteger('OptionChecked', Integer(OptionChecked));
8294 WriteInteger('MapOptionChecked', Integer(MapOptionChecked));
8295 WriteInteger('CityReport', Integer(CityRepMask));
8296 finally
8297 Free;
8298 end;
8299end;
8300
8301procedure TMainScreen.MovieSpeedBtnClick(Sender: TObject);
8302begin
8303 MovieSpeed := TButtonB(Sender).Tag shr 8;
8304 CheckMovieSpeedBtnState;
8305end;
8306
8307procedure TMainScreen.ResizeControls;
8308var
8309 MiniFrame, MaxMapWidth: Integer;
8310begin
8311 BaseWin.CreateOffscreen(Offscreen, Width, Height);
8312 SmallScreen := Width < 1024;
8313 with MainMap do begin
8314 MaxMapWidth := (G.lx * 2 - 3) * xxt;
8315 // Avoid the same tile being visible left and right
8316 if Width <= MaxMapWidth then begin
8317 MapWidth := Width;
8318 MapOffset := 0;
8319 end else begin
8320 MapWidth := MaxMapWidth;
8321 MapOffset := (Width - MapWidth) div 2;
8322 end;
8323 MapHeight := Height - TopBarHeight - PanelHeight + Overlap;
8324 Panel.SetSize(Width, PanelHeight);
8325 TopBar.SetSize(Width, TopBarHeight);
8326 MiniFrame := (lxmax_xxx - G.ly) div 2;
8327 xMidPanel := (G.lx + MiniFrame) * 2 + 1;
8328 xRightPanel := Width - LeftPanelWidth - 10;
8329 if ClientMode = cEditMap then
8330 TrPitch := 2 * xxt
8331 else
8332 TrPitch := 66;
8333 xMini := MiniFrame - 5;
8334 yMini := (PanelHeight - 26 - lxmax_xxx) div 2 + MiniFrame;
8335 ywmax := (G.ly - MapHeight div yyt + 1) and not 1;
8336 ywcenter := -((MapHeight - yyt * (G.ly - 1)) div (4 * yyt)) * 2;
8337 end;
8338 // only for ywmax <= 0
8339 if ywmax <= 0 then
8340 yw := ywcenter
8341 else if yw < 0 then
8342 yw := 0
8343 else if yw > ywmax then
8344 yw := ywmax;
8345 UnitInfoBtn.Top := Height - 29;
8346 UnitInfoBtn.Left := xMidPanel + 7 + 99;
8347 UnitBtn.Top := Height - 29;
8348 UnitBtn.Left := xMidPanel + 7 + 99 + 31;
8349 TerrainBtn.Top := Height - 29;
8350 TerrainBtn.Left := xMidPanel + 7 + 99 + 62;
8351 MovieSpeed1Btn.Top := Height - 91;
8352 MovieSpeed1Btn.Left := Width div 2 - 62;
8353 MovieSpeed2Btn.Top := Height - 91;
8354 MovieSpeed2Btn.Left := Width div 2 - 62 + 29;
8355 MovieSpeed3Btn.Top := Height - 91;
8356 MovieSpeed3Btn.Left := Width div 2 - 62 + 2 * 29;
8357 MovieSpeed4Btn.Top := Height - 91;
8358 MovieSpeed4Btn.Left := Width div 2 - 62 + 3 * 29 + 12;
8359 EOT.Top := Height - 64;
8360 EOT.Left := Width - 62;
8361 VerticalScrollBar.SetBorderSpacing(Height - yTroop - 24, Width - xRightPanel + 8, 8);
8362 SetWindowPos(VerticalScrollBar.ScrollBar.Handle, 0, xRightPanel + 10 - 14 - GetSystemMetrics(SM_CXVSCROLL),
8363 Height - MidPanelHeight + 8, 0, 0, SWP_NOSIZE or SWP_NOZORDER);
8364 MapBtn0.Left := xMini + G.lx - 44;
8365 MapBtn0.Top := Height - 15;
8366 MapBtn1.Left := xMini + G.lx - 28;
8367 MapBtn1.Top := Height - 15;
8368 { MapBtn2.Left := xMini + G.lx - 20;
8369 MapBtn2.Top := Height - 15;
8370 MapBtn3.Left := xMini + G.lx - 4;
8371 MapBtn3.Top := Height - 15; }
8372 MapBtn5.Left := xMini + G.lx - 12;
8373 MapBtn5.Top := Height - 15;
8374 MapBtn4.Left := xMini + G.lx + 20;
8375 MapBtn4.Top := Height - 15;
8376 MapBtn6.Left := xMini + G.lx + 36;
8377 MapBtn6.Top := Height - 15;
8378 TreasuryArea.Left := Width div 2 - 172;
8379 ResearchArea.Left := Width div 2;
8380 ManagementArea.Left := Width - xPalace;
8381 ManagementArea.Top := TopBarHeight + MapHeight - Overlap + yPalace;
8382 ArrangeMidPanel;
8383 if RepaintOnResize then RepaintAll;
8384 SetTroopLoc(TroopLoc);
8385end;
8386
8387procedure TMainScreen.ArrangeDialogs;
8388begin
8389 ArrangeDialog(FListDlg);
8390 ArrangeDialog(FHelpDlg);
8391 ArrangeDialog(FUnitStatDlg);
8392 ArrangeDialog(FDiaDlg);
8393 ArrangeDialog(FNatStatDlg);
8394end;
8395
8396procedure TMainScreen.ArrangeDialog(Form: TBufferedDrawDlg);
8397begin
8398 if not Assigned(Form) then Exit;
8399
8400 if Form is TListDlg then begin;
8401 ListDlg.UserLeft := Screen.PrimaryMonitor.Left + 8;
8402 ListDlg.UserTop := Screen.PrimaryMonitor.Top + TopBarHeight + 8;
8403 end;
8404 if Form is THelpDlg then begin
8405 HelpDlg.UserLeft := Screen.PrimaryMonitor.Left + Screen.PrimaryMonitor.Width - HelpDlg.Width - 8;
8406 HelpDlg.UserTop := Screen.PrimaryMonitor.Top + TopBarHeight + 8;
8407 end;
8408 if Form is TUnitStatDlg then begin
8409 UnitStatDlg.UserLeft := Screen.PrimaryMonitor.Left + 397;
8410 UnitStatDlg.UserTop := Screen.PrimaryMonitor.Top + TopBarHeight + 64;
8411 end;
8412 if Form is TDiaDlg then begin
8413 DiaDlg.UserLeft := Screen.PrimaryMonitor.Left + (Screen.PrimaryMonitor.Width - DiaDlg.Width) div 2;
8414 DiaDlg.UserTop := Screen.PrimaryMonitor.Top + (Screen.PrimaryMonitor.Height - DiaDlg.Height) div 2;
8415 end;
8416 if Form is TNatStatDlg then begin
8417 NatStatDlg.UserLeft := Screen.PrimaryMonitor.Left + Screen.PrimaryMonitor.Width - NatStatDlg.Width - 8;
8418 NatStatDlg.UserTop := Screen.PrimaryMonitor.Top + Screen.PrimaryMonitor.Height - PanelHeight - NatStatDlg.Height - 8;
8419 if NatStatDlg.UserTop < 8 then
8420 NatStatDlg.UserTop := 8;
8421 end;
8422end;
8423
8424procedure TMainScreen.ScrollBarUpdate(Sender: TObject);
8425begin
8426 {$IFDEF UNIX}
8427 // Do not scroll with mouse wheel to and above max value
8428 if VerticalScrollBar.Position > VerticalScrollBar.Max - 1 then
8429 VerticalScrollBar.Position := VerticalScrollBar.Max - 1;
8430 {$ENDIF}
8431 PanelPaint;
8432 Update;
8433end;
8434
8435end.
8436
Note: See TracBrowser for help on using the repository browser.