source: tags/1.3.7/GameServer.pas

Last change on this file was 649, checked in by chronos, 6 weeks ago
  • Modified: Improved brains libraries enumeration.
File size: 152.3 KB
Line 
1{$INCLUDE Switches.inc}
2// {$DEFINE TEXTLOG}
3// {$DEFINE LOADPERF}
4unit GameServer;
5
6interface
7
8uses
9 Protocol, Database, DynLibs, Platform, DateUtils, LazFileUtils, Brain, Global,
10 Map,
11 {$IFDEF DPI}Graphics{$ELSE}Graphics{$ENDIF};
12
13const
14 FirstAICompatibleVersion = $000D00;
15 FirstBookCompatibleVersion = $010103;
16
17 maxBrain = 255;
18
19type
20 // notifications
21 TNotify = (
22 ntCreateWorld,
23 ntInitModule,
24 ntInitLocalHuman,
25 ntDLLError,
26 ntAIError,
27 ntClientError,
28 ntInitPlayers,
29 ntDeactivationMissing,
30 ntSetAIName,
31 ntException,
32 ntLoadBegin,
33 ntLoadState,
34 ntEndInfo,
35 ntBackOn,
36 ntBackOff,
37 ntLoadError,
38 ntStartDone,
39 ntStartGo,
40 ntStartGoRefresh,
41 ntStartGoRefreshMaps,
42 ntChangeClient,
43 ntNextPlayer,
44 ntDeinitModule
45 );
46
47 TNotifyFunction = procedure(ID: TNotify; Index: Integer = 0);
48
49var
50 // PARAMETERS
51 PlayersBrain: TBrains; { brain of the players view }
52 Difficulty: array [0 .. nPl - 1] of Integer absolute Database.Difficulty;
53 { difficulty }
54
55 // READ ONLY
56 DotNetClient: TClientCall;
57 Brains: TBrains; // { available brains }
58 NotifyMessage: string;
59
60 BrainNoTerm: TBrain;
61 BrainSuperVirtual: TBrain;
62 BrainTerm: TBrain;
63 BrainRandom: TBrain;
64 BrainNetworkClient: TBrain;
65 BrainNetworkServer: TBrain;
66
67 NetworkEnabled: Boolean;
68
69procedure Init(NotifyFunction: TNotifyFunction);
70procedure Done;
71
72procedure StartNewGame(const FileName, AMapFileName: string;
73 Newlx, Newly, NewLandMass, NewMaxTurn: Integer);
74function LoadGame(const FileName: string; Turn: Integer;
75 MovieMode: Boolean): Boolean;
76procedure EditMap(const AMapFileName: string; Newlx, Newly, NewLandMass: Integer);
77procedure FillMap(TerrainType: Cardinal; SpecialResources: Boolean);
78procedure DirectHelp(Command: Integer);
79function ToAutoSaveFileName(FileName: string): string;
80function IsAutoSaveFileName(FileName: string): Boolean;
81
82procedure ChangeClient;
83procedure NextPlayer;
84function PreviewMap(ALandMass: Integer): Pointer;
85
86
87implementation
88
89uses
90 Directories, CityProcessing, UnitProcessing, CmdList, LCLIntf, LCLType,
91 LMessages, Classes, SysUtils;
92
93resourcestring
94 SNoAiFound = 'No AI libraries found in directory %s';
95
96var
97 MaxTurn: Integer;
98 LoadTurn: Integer; { turn where to stop loading }
99 nLogOpened: Integer; { nLog of opened book }
100{$IFOPT O-}nHandoverStack, {$ENDIF}
101 LastEndClientCommand: Integer;
102 pContacted: Integer; // player contacted for negotiation
103 pDipActive: Integer; // player who's to speak in a negotiation
104 pTurn: Integer; { player who's turn it is }
105 GWinner: Integer;
106 GColdWarStart: Integer;
107 GStealFrom: Integer;
108 SpyMission: Integer;
109 ZOCTile: Integer;
110 CCCommand: Integer;
111 CCPlayer: Integer;
112 DebugMap: array [0 .. nPl - 1] of Pointer;
113 ExeInfo: TSearchRec;
114 Stat: array [0 .. nStat - 1, 0 .. nPl - 1] of ^TChart;
115 AutoSaveState: TCmdListState;
116 MapField: ^Cardinal; // predefined map
117 LastOffer: TOffer;
118 CCData: array [0 .. 14] of Integer;
119 bix: TBrains; { brain of the players }
120 DevModelTurn: array [0 .. nPl - 1] of Integer; { turn of last call to sResetModel }
121 OriginalDataVersion: array [0 .. nPl - 1] of Integer;
122 SavedTiles { , SavedResourceWeights } : array [0 .. ncmax - 1] of Cardinal;
123 SavedData: array [0 .. nPl - 1] of Pointer;
124 LogFileName: string;
125 SavePath: string; { name of file for saving the current game }
126 MapFileName: string; // name of map to use, empty for random
127 AICredits: string;
128 AIInfo: array [0 .. nPl - 1] of string;
129 Notify: TNotifyFunction;
130 LastClientTime: TDateTime;
131{$IFOPT O-}HandoverStack: array [0 .. 31] of Cardinal; {$ENDIF}
132 AutoSaveExists: Boolean;
133 LoadOK: Boolean;
134 WinOnAlone: Boolean;
135 PreviewElevation: Boolean;
136 MovieStopped: Boolean;
137
138const
139 PreviewRND = 41601260; { randseed for preview map }
140
141function Server(Command, Player, Subject: Integer; var Data): Integer;
142 stdcall; forward;
143
144procedure CallPlayer(Command, P: Integer; var Data);
145begin
146 if ((Mode <> moMovie) or (P = 0)) then
147 begin
148{$IFOPT O-}
149 HandoverStack[nHandoverStack] := P;
150 HandoverStack[nHandoverStack + 1] := Command;
151 Inc(nHandoverStack, 2);
152 bix[P].Client(Command, P, Data);
153 Dec(nHandoverStack, 2);
154{$ELSE}
155 try
156 bix[P].Client(Command, P, Data);
157 except
158 Notify(ntException + bix[P]);
159 end;
160{$ENDIF}
161 end;
162end;
163
164procedure CallAllPlayers(Command: Integer; var Data);
165var
166 I: Integer;
167begin
168 for I := 0 to nPl - 1 do
169 if Assigned(bix[I]) then
170 CallPlayer(Command, I, Data);
171end;
172
173procedure CallClient(bix, Command: Integer; var Data);
174begin
175 if ((Mode <> moMovie) or (bix = Brains.IndexOf(GameServer.bix[0]))) then
176 begin
177{$IFOPT O-}
178 HandoverStack[nHandoverStack] := bix;
179 HandoverStack[nHandoverStack + 1] := Command;
180 Inc(nHandoverStack, 2);
181 Brains[bix].Client(Command, -1, Data);
182 Dec(nHandoverStack, 2);
183{$ELSE}
184 try
185 Brain[bix].Client(Command, -1, Data);
186 except
187 Notify(ntException + bix);
188 end;
189{$ENDIF}
190 end
191end;
192
193procedure Init(NotifyFunction: TNotifyFunction);
194var
195 F: TSearchRec;
196 BasePath: string;
197 NewBrain: TBrain;
198 I: Integer;
199begin
200 Notify := NotifyFunction;
201 PreviewElevation := False;
202 PlayersBrain := TBrains.Create(False);
203 PlayersBrain.Count := nPl;
204 for I := 0 to nPl - 1 do
205 PlayersBrain[I] := nil;
206
207 bix := TBrains.Create(False);
208 bix.Count := nPl;
209 for I := 0 to nPl - 1 do
210 bix[I] := nil;
211
212 { get available brains }
213 Brains := TBrains.Create;
214 BrainNoTerm := Brains.AddNew;
215 BrainNoTerm.FileName := ':AIT';
216 BrainNoTerm.Flags := 0;
217 BrainNoTerm.Initialized := False;
218 BrainNoTerm.Kind := btNoTerm;
219 BrainSuperVirtual := Brains.AddNew;
220 BrainSuperVirtual.FileName := ':Supervisor';
221 BrainSuperVirtual.Flags := 0;
222 BrainSuperVirtual.Initialized := False;
223 BrainSuperVirtual.Kind := btSuperVirtual;
224 if NetworkEnabled then begin
225 BrainNetworkClient := Brains.AddNew;
226 BrainNetworkClient.FileName := ':NetworkClient';
227 BrainNetworkClient.Flags := fMultiple;
228 BrainNetworkClient.Initialized := False;
229 BrainNetworkClient.ServerVersion := CevoVersion;
230 BrainNetworkClient.Kind := btNetworkClient;
231 end;
232 BrainTerm := Brains.AddNew;
233 BrainTerm.FileName := ':StdIntf';
234 BrainTerm.Flags := fMultiple;
235 BrainTerm.Initialized := False;
236 BrainTerm.ServerVersion := CevoVersion;
237 BrainTerm.Kind := btTerm;
238 BrainRandom := Brains.AddNew;
239 BrainRandom.FileName := ':Random';
240 BrainRandom.Flags := fMultiple;
241 BrainRandom.Initialized := False;
242 BrainRandom.Kind := btRandom;
243 if NetworkEnabled then begin
244 BrainNetworkServer := Brains.AddNew;
245 BrainNetworkServer.FileName := ':NetworkServer';
246 BrainNetworkServer.Flags := fMultiple;
247 BrainNetworkServer.Initialized := False;
248 BrainNetworkServer.ServerVersion := CevoVersion;
249 BrainNetworkServer.Kind := btNetworkServer;
250 end;
251
252 if FindFirst(GetAiDir + DirectorySeparator + '*', faDirectory or faArchive or faReadOnly, F) = 0 then
253 repeat
254 BasePath := GetAiDir + DirectorySeparator + F.Name;
255 if (F.Name <> '.') and (F.Name <> '..') and DirectoryExists(BasePath) then begin
256 NewBrain := TBrain.Create;
257 NewBrain.Kind := btAI;
258 NewBrain.LoadFromFile(BasePath + DirectorySeparator + F.Name + '.ai.txt');
259 if (NewBrain.ServerVersion >= FirstAICompatibleVersion) and
260 (NewBrain.ServerVersion <= CevoVersion) and
261 ((NewBrain.Flags and fDotNet = 0) or (@DotNetClient <> nil)) and
262 (NewBrain.DLLName <> '') then begin
263 Brains.Add(NewBrain);
264 end else NewBrain.Free;
265 end;
266 until FindNext(F) <> 0;
267 FindClose(F);
268
269 if Brains.GetKindCount(btAI) = 0 then
270 raise Exception.Create(Format(SNoAiFound, [GetAiDir]));
271end;
272
273procedure Done;
274var
275 I: Integer;
276begin
277 for I := 0 to Brains.Count - 1 do
278 with Brains[I] do
279 if Initialized then begin
280 CallClient(I, cReleaseModule, nil^);
281 if (Kind = btAI) and ((Flags and fDotNet) = 0) then
282 FreeLibrary(hm);
283 end;
284 FreeAndNil(PlayersBrain);
285 FreeAndNil(bix);
286 FreeAndNil(Brains);
287end;
288
289function PreviewMap(ALandMass: Integer): Pointer;
290begin
291 lx := lxmax;
292 ly := lymax;
293 MapSize := lx * ly;
294 LandMass := ALandMass;
295 DelphiRandSeed := PreviewRND;
296 if not PreviewElevation then begin
297 CreateElevation;
298 PreviewElevation := True;
299 end;
300 CreateMap(True);
301 Result := @RealMap;
302end;
303
304procedure ChangeClientWhenDone(Command, Player: Integer; var Data;
305 DataSize: Integer);
306begin
307 CCCommand := Command;
308 CCPlayer := Player;
309 if DataSize > 0 then
310 Move(Data, CCData, DataSize);
311 Notify(ntChangeClient);
312end;
313
314procedure PutMessage(Level: Integer; Text: string);
315begin
316 bix[0].Client(cDebugMessage, Level, PChar(Text)^);
317end;
318
319procedure ForceClientDeactivation;
320var
321 NullOffer: TOffer;
322begin
323 if pDipActive < 0 then
324 Server(sTurn, pTurn, 0, nil^) // no nego mode
325 else
326 case LastEndClientCommand of // nego mode
327 scContact:
328 Server(scReject, pDipActive, 0, nil^);
329 scDipCancelTreaty, scDipBreak:
330 Server(scDipNotice, pDipActive, 0, nil^);
331 else
332 begin // make null offer
333 NullOffer.nDeliver := 0;
334 NullOffer.nCost := 0;
335 Server(scDipOffer, pDipActive, 0, NullOffer);
336 end;
337 end;
338end;
339
340procedure ChangeClient;
341// hand over control to other client (as specified by CC...)
342var
343 P: Integer;
344 T: TDateTime;
345begin
346 T := NowPrecise;
347 PutMessage(1 shl 16 + 2, Format('CLIENT: took %.1f ms',
348 [(T - LastClientTime) / OneMillisecond]));
349 LastClientTime := T;
350 PutMessage(1 shl 16 + 2, Format('CLIENT: calling %d (%s)',
351 [CCPlayer, bix[CCPlayer].Name]));
352 if CCCommand = cTurn then
353 for P := 0 to nPl - 1 do
354 if (P <> CCPlayer) and (1 shl P and GWatching <> 0) then
355 CallPlayer(cShowTurnChange, P, CCPlayer);
356
357 P := CCPlayer;
358 CCPlayer := -1;
359 CallPlayer(CCCommand, P, CCData);
360 if (Mode = moPlaying) and (bix[P].Flags and aiThreaded = 0) and
361 (CCPlayer < 0) then
362 begin
363 Notify(ntDeactivationMissing, P);
364 ForceClientDeactivation;
365 end;
366end;
367
368procedure Inform(P: Integer);
369var
370 I, p1: Integer;
371begin
372 RW[P].Turn := GTurn;
373 if (GTurn = MaxTurn) and (P = pTurn) and (P = 0) then
374 RW[P].Happened := RW[P].Happened or phTimeUp;
375 if (GWinner > 0) and (P = pTurn) and (P = 0) then
376 RW[P].Happened := RW[P].Happened or phShipComplete;
377 RW[P].Alive := GAlive;
378 Move(GWonder, RW[P].Wonder, SizeOf(GWonder));
379 Move(GShip, RW[P].Ship, SizeOf(GShip));
380 for p1 := 0 to nPl - 1 do
381 if (p1 <> P) and Assigned(bix[p1]) and (Difficulty[p1] > 0) then
382 RW[P].EnemyReport[p1].Credibility := RW[p1].Credibility;
383 for p1 := 0 to nPl - 1 do
384 if (p1 <> P) and (1 shl p1 and GAlive <> 0) then
385 begin
386 if (GTestFlags and tfUncover <> 0) or (Difficulty[P] = 0) or
387 (RW[P].Treaty[p1] >= trFriendlyContact) then
388 GiveCivilReport(P, p1);
389 if (GTestFlags and tfUncover <> 0) or (Difficulty[P] = 0) or
390 (RW[P].Treaty[p1] = trAlliance) then
391 GiveMilReport(P, p1)
392 end;
393 for I := 0 to RW[P].nEnemyModel - 1 do
394 with RW[P].EnemyModel[I] do
395 Lost := Destroyed[P, Owner, mix];
396end;
397
398const
399 AutoSavePrefix = '~';
400
401function ToAutoSaveFileName(FileName: string): string;
402begin
403 if Copy(ExtractFileName(FileName), 1, 1) <> AutoSavePrefix then
404 Result := ExtractFileDir(FileName) + DirectorySeparator + AutoSavePrefix + ExtractFileName(FileName)
405 else Result := FileName;
406end;
407
408function FromAutoSaveFileName(FileName: string): string;
409begin
410 if Copy(ExtractFileName(FileName), 1, 1) = AutoSavePrefix then
411 Result := ExtractFileDir(FileName) + DirectorySeparator + Copy(ExtractFileName(FileName), 2, MaxInt)
412 else Result := FileName;
413end;
414
415function IsAutoSaveFileName(FileName: string): Boolean;
416begin
417 FileName := ExtractFileName(FileName);
418 Result := (Length(FileName) > 0) and (FileName[1] = AutoSavePrefix);
419end;
420
421procedure LogChanges;
422var
423 P, ix: Integer;
424begin
425 for P := 0 to nPl - 1 do
426 if (1 shl P and GWatching <> 0) and ProcessClientData[P] then
427 begin
428 // log unit status changes
429 for ix := 0 to RW[P].nUn - 1 do
430 with RW[P].Un[ix] do
431 if (Loc >= 0) and (SavedStatus <> Status) then
432 begin
433 CL.Put(sIntSetUnitStatus, P, ix, @Status);
434 SavedStatus := Status;
435 end;
436 // log city status changes
437 for ix := 0 to RW[P].nCity - 1 do
438 with RW[P].City[ix] do
439 if (Loc >= 0) and (SavedStatus <> Status) then
440 begin
441 CL.Put(sIntSetCityStatus, P, ix, @Status);
442 SavedStatus := Status;
443 end;
444 // log model status changes
445 for ix := 0 to RW[P].nModel - 1 do
446 with RW[P].Model[ix] do
447 if SavedStatus <> Status then
448 begin
449 CL.Put(sIntSetModelStatus, P, ix, @Status);
450 SavedStatus := Status;
451 end;
452 // log enemy city status changes
453 for ix := 0 to RW[P].nEnemyCity - 1 do
454 with RW[P].EnemyCity[ix] do
455 if (Loc >= 0) and (SavedStatus <> Status) then
456 begin
457 CL.Put(sIntSetECityStatus, P, ix, @Status);
458 SavedStatus := Status;
459 end;
460 // log data changes
461 if bix[P].DataSize > 0 then
462 begin
463 CL.PutDataChanges(sIntDataChange, P, SavedData[P], RW[P].Data,
464 bix[P].DataSize);
465 Move(RW[P].Data^, SavedData[P]^, bix[P].DataSize * 4);
466 end;
467 end;
468end;
469
470procedure NoLogChanges;
471var
472 P, ix: Integer;
473begin
474 for P := 0 to nPl - 1 do
475 if (1 shl P and GWatching <> 0) and ProcessClientData[P] then
476 begin
477 for ix := 0 to RW[P].nUn - 1 do
478 with RW[P].Un[ix] do
479 SavedStatus := Status;
480 for ix := 0 to RW[P].nCity - 1 do
481 with RW[P].City[ix] do
482 SavedStatus := Status;
483 for ix := 0 to RW[P].nModel - 1 do
484 with RW[P].Model[ix] do
485 SavedStatus := Status;
486 for ix := 0 to RW[P].nEnemyCity - 1 do
487 with RW[P].EnemyCity[ix] do
488 SavedStatus := Status;
489 if bix[P].DataSize > 0 then
490 Move(RW[P].Data^, SavedData[P]^, bix[P].DataSize * 4);
491 end;
492end;
493
494function HasChanges(P: Integer): Boolean;
495type
496 TDWordList = array [0 .. INFIN] of Cardinal;
497 PDWortList = ^TDWordList;
498var
499 ix: Integer;
500begin
501 Result := False;
502 for ix := 0 to RW[P].nUn - 1 do
503 with RW[P].Un[ix] do
504 if (Loc >= 0) and (SavedStatus <> Status) then begin
505 Result := True;
506 Exit;
507 end;
508 for ix := 0 to RW[P].nCity - 1 do
509 with RW[P].City[ix] do
510 if (Loc >= 0) and (SavedStatus <> Status) then begin
511 Result := True;
512 Exit;
513 end;
514 for ix := 0 to RW[P].nModel - 1 do
515 with RW[P].Model[ix] do
516 if SavedStatus <> Status then begin
517 Result := True;
518 Exit;
519 end;
520 for ix := 0 to RW[P].nEnemyCity - 1 do
521 with RW[P].EnemyCity[ix] do
522 if (Loc >= 0) and (SavedStatus <> Status) then begin
523 Result := True;
524 Exit;
525 end;
526 if RW[P].Data <> nil then
527 for ix := 0 to bix[P].DataSize - 1 do
528 if PDWortList(SavedData[P])[ix] <> PDWortList(RW[P].Data)[ix] then begin
529 Result := True;
530 Exit;
531 end;
532end;
533
534procedure InitBrain(bix: TBrain);
535var
536 InitModuleData: TInitModuleData;
537begin
538 Assert(bix.Kind <> btSuperVirtual);
539 with bix do begin
540 if Initialized then
541 Exit;
542 if Kind = btAI then
543 begin { get client function }
544 Notify(ntInitModule, Brains.IndexOf(bix));
545 if Flags and fDotNet > 0 then
546 Client := DotNetClient
547 else
548 begin
549 hm := LoadLibrary(PChar(DLLName));
550 if hm = 0 then
551 begin
552 Client := nil;
553 Notify(ntDLLError, Brains.IndexOf(bix));
554 end
555 else
556 begin
557 Client := GetProcAddress(hm, 'client');
558 if @Client = nil then
559 Notify(ntClientError, Brains.IndexOf(bix));
560 end;
561 end;
562 end;
563 if @Client <> nil then
564 begin
565 Initialized := True;
566 InitModuleData.Server := @Server;
567 InitModuleData.DataVersion := 0;
568 InitModuleData.DataSize := 0;
569 InitModuleData.Flags := 0;
570 CallClient(Brains.IndexOf(bix), cInitModule, InitModuleData);
571 DataVersion := InitModuleData.DataVersion;
572 DataSize := (InitModuleData.DataSize + 3) div 4;
573 if DataSize > MaxDataSize then
574 DataSize := 0;
575 Flags := Flags or InitModuleData.Flags;
576 end;
577 end;
578end;
579
580procedure SaveGame(FileName: string; Auto: Boolean);
581var
582 X, Y, I, Zero, Tile, nLocal: Integer;
583 LogFile: TFileStream;
584 S: string[255];
585 SaveMap: array [0 .. lxmax * lymax - 1] of Byte;
586begin
587 nLocal := 0;
588 for I := 0 to nPl - 1 do
589 if Assigned(bix[I]) and (bix[I].Kind = btTerm) then
590 Inc(nLocal);
591 if Difficulty[0] = 0 then
592 nLocal := 0;
593 if nLocal <= 1 then
594 for Y := 0 to ly - 1 do
595 for X := 0 to lx - 1 do
596 begin
597 Tile := RW[0].Map[(X + SaveMapCenterLoc + lx shr 1) mod lx + lx * Y];
598 SaveMap[X + lx * Y] := Tile and fTerrain + Tile and
599 (fCity or fUnit or fOwned) shr 16;
600 end;
601
602 if Auto and AutoSaveExists then // append to existing file
603 LogFile := TFileStream.Create(FileName, fmOpenReadWrite or fmShareExclusive)
604 else // create new file
605 LogFile := TFileStream.Create(FileName, fmCreate or fmShareExclusive);
606
607 Zero := 0;
608 LogFile.Position := 0;
609 S := 'cEvoBook';
610 LogFile.Write(S[1], 8); { file id }
611 I := CevoVersion;
612 LogFile.Write(I, 4); { c-evo version }
613 LogFile.Write(ExeInfo.Time, 4);
614 LogFile.Write(lx, 4);
615 LogFile.Write(ly, 4);
616 LogFile.Write(LandMass, 4);
617 if LandMass = 0 then
618 LogFile.Write(MapField^, MapSize * 4);
619
620 LogFile.Write(MaxTurn, 4);
621 LogFile.Write(RND, 4);
622 LogFile.Write(GTurn, 4);
623 if nLocal > 1 then // multiplayer game -- no quick view
624 begin
625 I := $80;
626 LogFile.Write(I, 4);
627 end
628 else
629 LogFile.Write(SaveMap, ((MapSize - 1) div 4 + 1) * 4);
630 for I := 0 to nPl - 1 do
631 if not Assigned(bix[I]) then
632 LogFile.Write(Zero, 4)
633 else
634 begin
635 if PlayersBrain[I].Kind in [btRandom, btAI] then
636 S := bix[I].FileName
637 else
638 S := PlayersBrain[I].FileName;
639 Move(Zero, S[Length(S) + 1], 4);
640 LogFile.Write(S, (Length(S) div 4 + 1) * 4);
641 LogFile.Write(OriginalDataVersion[I], 4);
642 S := ''; { behavior }
643 Move(Zero, S[Length(S) + 1], 4);
644 LogFile.Write(S, (Length(S) div 4 + 1) * 4);
645 LogFile.Write(Difficulty[I], 4);
646 end;
647
648 if Auto and AutoSaveExists then
649 CL.AppendToFile(LogFile, AutoSaveState)
650 else
651 CL.SaveToFile(LogFile);
652 FreeAndNil(LogFile);
653 if Auto then
654 begin
655 AutoSaveState := CL.State;
656 AutoSaveExists := True;
657 end;
658end;
659
660procedure StartGame;
661var
662 I, P, p1, Human, nAlive, bixUni: Integer;
663 Game: TNewGameData;
664 // GameEx: TNewGameExData;
665 Path: shortstring;
666 BrainUsed: Set of 0 .. 254; { used brains }
667 AIBrains: TBrains;
668 Map: TMap;
669begin
670 for p1 := 0 to nPl - 1 do begin
671 if Assigned(PlayersBrain[p1]) and (PlayersBrain[p1].Kind = btSuperVirtual) then
672 bix[p1] := BrainTerm // supervisor and local human use same module
673 else if Assigned(PlayersBrain[p1]) and (PlayersBrain[p1].Kind = btRandom) then
674 if Brains.GetKindCount(btAI) = 0 then
675 bix[p1] := nil
676 else begin
677 AIBrains := TBrains.Create(False);
678 Brains.GetByKind(btAI, AIBrains);
679 bix[p1] := AIBrains[DelphiRandom(AIBrains.Count)];
680 FreeAndNil(AIBrains);
681 end
682 else
683 bix[p1] := PlayersBrain[p1];
684 if not Assigned(PlayersBrain[p1]) then
685 Difficulty[p1] := -1;
686 end;
687
688 if bix[0].Kind <> btNoTerm then
689 Notify(ntInitLocalHuman);
690
691 BrainUsed := [];
692 for P := 0 to nPl - 1 do
693 if Assigned(bix[P]) and ((Mode <> moMovie) or (P = 0)) then
694 begin { initiate selected control module }
695 AIInfo[P] := bix[P].Name + #0;
696 InitBrain(bix[P]);
697 if Mode = moPlaying then
698 begin // new game, this data version is original
699 OriginalDataVersion[P] := bix[P].DataVersion;
700 ProcessClientData[P] := True;
701 end
702 else // loading game, compare with data version read from file
703 ProcessClientData[P] := ProcessClientData[P] and
704 (OriginalDataVersion[P] = bix[P].DataVersion);
705 if @bix[P].Client = nil then // client function not found
706 if bix[0].Kind = btNoTerm then
707 bix[P] := nil
708 else
709 begin
710 bix[P] := BrainTerm;
711 OriginalDataVersion[P] := -1;
712 ProcessClientData[P] := False;
713 end;
714 if Assigned(bix[P]) then
715 Include(BrainUsed, Brains.IndexOf(bix[P]));
716 end;
717
718 Notify(ntCreateWorld);
719 nAlive := 0;
720 GAlive := 0;
721 if Mode = moMovie then
722 GWatching := 1
723 else
724 GWatching := 0;
725 GAI := 0;
726 for p1 := 0 to nPl - 1 do
727 if Assigned(bix[p1]) then
728 begin
729 if Mode <> moMovie then
730 Inc(GWatching, 1 shl p1);
731 if bix[p1].Kind = btAI then
732 Inc(GAI, 1 shl p1);
733 if Difficulty[p1] > 0 then
734 begin
735 Inc(GAlive, 1 shl p1);
736 Inc(nAlive);
737 end;
738 ServerVersion[p1] := bix[p1].ServerVersion;
739 end;
740 WinOnAlone := (bix[0].Kind = btNoTerm) and (nAlive > 1);
741 GWinner := 0;
742 GColdWarStart := -ColdWarTurns - 1;
743 uixSelectedTransport := -1;
744 SpyMission := smSabotageProd;
745 for p1 := 0 to nPl - 1 do
746 DebugMap[p1] := nil;
747
748 GTurn := 0;
749 for I := 0 to nWonder - 1 do
750 with GWonder[I] do
751 begin
752 CityID := -1;
753 EffectiveOwner := -1
754 end;
755 FillChar(GShip, SizeOf(GShip), 0);
756
757 for P := 0 to nPl - 1 do
758 if 1 shl P and (GAlive or GWatching) <> 0 then
759 with RW[P] do
760 begin
761 Government := gDespotism;
762 Money := StartMoney;
763 TaxRate := 30;
764 LuxRate := 0;
765 Research := 0;
766 ResearchTech := -2;
767 AnarchyStart := -AnarchyTurns - 1;
768 Happened := 0;
769 LastValidStat[P] := -1;
770 Worked[P] := 0;
771 Founded[P] := 0;
772 DevModelTurn[P] := -1;
773 OracleIncome := 0;
774
775 if bix[P].DataSize > 0 then
776 begin
777 GetMem(SavedData[P], bix[P].DataSize * 4);
778 GetMem(Data, bix[P].DataSize * 4);
779 FillChar(SavedData[P]^, bix[P].DataSize * 4, 0);
780 FillChar(Data^, bix[P].DataSize * 4, 0);
781 end
782 else
783 begin
784 Data := nil;
785 SavedData[P] := nil;
786 end;
787 nBattleHistory := 0;
788 BattleHistory := nil;
789 { if bix[p]=bixTerm then
790 begin
791 GetMem(BorderHelper,MapSize);
792 FillChar(BorderHelper^,MapSize,0);
793 end
794 else } BorderHelper := nil;
795 for I := 0 to nStat - 1 do
796 GetMem(Stat[I, P], 4 * (MaxTurn + 1));
797 if bix[P].Flags and fDotNet <> 0 then
798 begin
799 GetMem(RW[P].DefaultDebugMap, MapSize * 4);
800 FillChar(RW[P].DefaultDebugMap^, MapSize * 4, 0);
801 DebugMap[P] := RW[P].DefaultDebugMap;
802 end
803 else
804 RW[P].DefaultDebugMap := nil;
805
806 { !!!for i:=0 to nShipPart-1 do GShip[p].Parts[i]:=Delphirandom((3-i)*2); }
807 end;
808
809 if LandMass > 0 then
810 begin // random map
811 InitRandomGame;
812 PreviewElevation := False;
813 MapField := nil;
814 end
815 else
816 begin // predefined map
817 if Mode = moPlaying then begin
818 Map := TMap.Create;
819 Map.LoadFromFile(MapFileName); // new game -- load map from file
820 MapSize := Map.MapSize;
821 lx := Map.Size.X;
822 ly := Map.Size.Y;
823 Move(Map.Tiles[0], RealMap, MapSize * 4);
824 FreeAndNil(Map);
825 end;
826 GetMem(MapField, MapSize * 4);
827 Move(RealMap, MapField^, MapSize * 4);
828 Human := 0;
829 for p1 := 0 to nPl - 1 do
830 if Assigned(bix[p1]) and (bix[p1].Kind = btTerm) then
831 Inc(Human, 1 shl p1);
832 InitMapGame(Human);
833 end;
834 CityProcessing.InitGame;
835 UnitProcessing.InitGame;
836 for P := 0 to nPl - 1 do
837 if 1 shl P and (GAlive or GWatching) <> 0 then
838 Inform(P);
839
840 pTurn := -1;
841 if bix[0].Kind <> btNoTerm then
842 Notify(ntInitLocalHuman);
843 Game.lx := lx;
844 Game.ly := ly;
845 Game.LandMass := LandMass;
846 Game.MaxTurn := MaxTurn;
847 Move(Difficulty, Game.Difficulty, SizeOf(Difficulty));
848 // GameEx.lx:=lx; GameEx.ly:=ly; GameEx.LandMass:=LandMass;
849 // GameEx.MaxTurn:=MaxTurn; GameEx.RND:=RND;
850 // move(Difficulty,GameEx.Difficulty,SizeOf(Difficulty));
851 AICredits := '';
852 for I := 0 to Brains.Count - 1 do
853 with Brains[I] do begin
854 if Initialized then
855 if I in BrainUsed then
856 begin
857 if Kind = btAI then
858 Notify(ntInitPlayers);
859 for P := 0 to nPl - 1 do
860 begin
861 if Brains.IndexOf(bix[P]) = I then
862 Game.RO[P] := @RW[P]
863 else
864 Game.RO[P] := nil;
865 if (Kind = btTerm) and (Difficulty[0] = 0) and Assigned(bix[P]) then
866 Game.SuperVisorRO[P] := @RW[P]
867 else
868 Game.SuperVisorRO[P] := nil;
869 end;
870 if Flags and fDotNet > 0 then
871 begin
872 Path := DLLName;
873 Move(Path[1], Game.AssemblyPath, Length(Path));
874 Game.AssemblyPath[Length(Path)] := #0;
875 end
876 else
877 Game.AssemblyPath[0] := #0;
878 case Mode of
879 moLoading, moLoading_Fast:
880 CallClient(I, cLoadGame, Game);
881 moMovie:
882 CallClient(I, cMovie, Game);
883 moPlaying:
884 CallClient(I, cNewGame, Game);
885 end;
886 if (Kind = btAI) and (Credits <> '') then
887 if AICredits = '' then
888 AICredits := Credits
889 else
890 AICredits := AICredits + '\' + Credits;
891 end
892 else
893 begin { module no longer used -- unload }
894 CallClient(I, cReleaseModule, nil^);
895 if Kind = btAI then
896 begin
897 if Flags and fDotNet = 0 then
898 FreeLibrary(hm);
899 Client := nil;
900 end;
901 Initialized := False;
902 end;
903 end;
904 AICredits := AICredits + #0;
905
906 if bix[0].Kind <> btNoTerm then
907 begin
908 // uni ai?
909 bixUni := -1;
910 for p1 := 0 to nPl - 1 do
911 if Assigned(bix[p1]) and (bix[p1].Kind = btAI) then
912 if bixUni = -1 then
913 bixUni := Brains.IndexOf(bix[p1])
914 else if bixUni <> Brains.IndexOf(bix[p1]) then
915 bixUni := -2;
916 for p1 := 0 to nPl - 1 do
917 if Assigned(bix[p1]) and (bix[p1].Kind = btAI) then
918 begin
919 if bixUni = -2 then
920 NotifyMessage := bix[p1].FileName
921 else
922 NotifyMessage := '';
923 Notify(ntSetAIName, p1);
924 end;
925 end;
926
927 CheckBorders(-1);
928{$IFOPT O-}InvalidTreatyMap := 0; {$ENDIF}
929 AutoSaveExists := False;
930 pDipActive := -1;
931 pTurn := 0;
932
933 if Mode >= moMovie then
934 Notify(ntEndInfo);
935end;
936
937procedure EndGame;
938var
939 I, p1: Integer;
940begin
941 if LandMass = 0 then
942 FreeMem(MapField);
943 for p1 := 0 to nPl - 1 do
944 if Assigned(bix[p1]) then
945 begin
946 for I := 0 to nStat - 1 do
947 FreeMem(Stat[I, p1]);
948 if RW[p1].BattleHistory <> nil then
949 FreeMem(RW[p1].BattleHistory);
950 { if RW[p1].BorderHelper<>nil then FreeMem(RW[p1].BorderHelper); }
951 FreeMem(RW[p1].Data);
952 FreeMem(SavedData[p1]);
953 if RW[p1].DefaultDebugMap <> nil then
954 FreeMem(RW[p1].DefaultDebugMap);
955 end;
956 UnitProcessing.ReleaseGame;
957 CityProcessing.ReleaseGame;
958 Database.ReleaseGame;
959 FreeAndNil(CL);
960end;
961
962procedure GenerateStat(P: Integer);
963var
964 cix, uix: Integer;
965begin
966 if Difficulty[P] > 0 then
967 with RW[P] do
968 begin
969 Stat[stPop, P, GTurn] := 0;
970 for cix := 0 to nCity - 1 do
971 if City[cix].Loc >= 0 then
972 Inc(Stat[stPop, P, GTurn], City[cix].Size);
973 Stat[stScience, P, GTurn] := Researched[P] * 50;
974 if (RW[P].ResearchTech >= 0) and (RW[P].ResearchTech <> adMilitary) then
975 Inc(Stat[stScience, P, GTurn], Research * 100 div TechBaseCost(nTech[P],
976 Difficulty[P]));
977 Stat[stMil, P, GTurn] := 0;
978 for uix := 0 to nUn - 1 do
979 if Un[uix].Loc >= 0 then
980 with Model[Un[uix].mix] do
981 begin
982 if (Kind <= mkEnemyDeveloped) and (Un[uix].mix <> 1) then
983 Inc(Stat[stMil, P, GTurn], Weight * MStrength *
984 Un[uix].Health div 100)
985 else if Domain = dGround then
986 Inc(Stat[stMil, P, GTurn], (Attack + 2 * Defense) *
987 Un[uix].Health div 100)
988 else
989 Inc(Stat[stMil, P, GTurn], (Attack + Defense) *
990 Un[uix].Health div 100);
991 case Kind of
992 mkSlaves:
993 Inc(Stat[stPop, P, GTurn]);
994 mkSettler:
995 Inc(Stat[stPop, P, GTurn], 2);
996 end;
997 end;
998 Stat[stMil, P, GTurn] := Stat[stMil, P, GTurn] div 16;
999 Stat[stExplore, P, GTurn] := Discovered[P];
1000 Stat[stTerritory, P, GTurn] := TerritoryCount[P];
1001 Stat[stWork, P, GTurn] := Worked[P];
1002 LastValidStat[P] := GTurn;
1003 end;
1004end;
1005
1006procedure LogCityTileChanges;
1007var
1008 cix: Integer;
1009begin
1010 for cix := 0 to RW[pTurn].nCity - 1 do
1011 with RW[pTurn].City[cix] do
1012 if Loc >= 0 then
1013 begin
1014 { if SavedResourceWeights[cix]<>ResourceWeights then
1015 begin // log city resource weight changes
1016 CL.Put(sSetCityResourceWeights, pTurn, cix, @ResourceWeights);
1017 SavedResourceWeights[cix]:=ResourceWeights;
1018 end; }
1019 if SavedTiles[cix] <> Tiles then
1020 begin // log city tile changes
1021 CL.Put(sSetCityTiles, pTurn, cix, @Tiles);
1022 SavedTiles[cix] := Tiles;
1023 end;
1024 end;
1025end;
1026
1027procedure NoLogCityTileChanges;
1028var
1029 cix: Integer;
1030begin
1031 for cix := 0 to RW[pTurn].nCity - 1 do
1032 with RW[pTurn].City[cix] do
1033 if Loc >= 0 then
1034 begin
1035 // SavedResourceWeights[cix]:=ResourceWeights;
1036 SavedTiles[cix] := Tiles;
1037 end;
1038end;
1039
1040function HasCityTileChanges: Boolean;
1041var
1042 cix: Integer;
1043begin
1044 Result := False;
1045 for cix := 0 to RW[pTurn].nCity - 1 do
1046 with RW[pTurn].City[cix] do
1047 if Loc >= 0 then begin
1048 // if SavedResourceWeights[cix] <> ResourceWeights then Result := True;
1049 if SavedTiles[cix] <> Tiles then begin
1050 Result := True;
1051 Break;
1052 end;
1053 end;
1054end;
1055
1056procedure BeforeTurn0;
1057var
1058 p1, uix: Integer;
1059begin
1060 for uix := 0 to RW[pTurn].nUn - 1 do { init movement points for first turn }
1061 with RW[pTurn].Un[uix] do
1062 Movement := RW[pTurn].Model[mix].Speed;
1063
1064 if Difficulty[pTurn] > 0 then
1065 DiscoverViewAreas(pTurn)
1066 else { supervisor }
1067 begin
1068 DiscoverAll(pTurn, lObserveSuper);
1069 for p1 := 1 to nPl - 1 do
1070 if 1 shl p1 and GAlive <> 0 then
1071 begin
1072 GiveCivilReport(pTurn, p1);
1073 GiveMilReport(pTurn, p1);
1074 end;
1075 end;
1076 // CheckContact;
1077end;
1078
1079function LoadGame(const FileName: string; Turn: Integer; MovieMode: Boolean): Boolean;
1080var
1081 J: TBrain;
1082 I, ix, D, p1, Command, Subject: Integer;
1083 K: Integer;
1084{$IFDEF TEXTLOG}LoadPos0: Integer; {$ENDIF}
1085 Data: Pointer;
1086 LogFile: TFileStream;
1087 FormerCLState: TCmdListState;
1088 S: string[255];
1089 SaveMap: array [0 .. lxmax * lymax - 1] of Byte;
1090 Started, StatRequest: Boolean;
1091begin
1092 SavePath := ExtractFileDir(FileName);
1093 LogFileName := FileName;
1094 LoadTurn := Turn;
1095 LogFile := TFileStream.Create(LogFileName, fmOpenRead or
1096 fmShareExclusive);
1097 LogFile.Position := 0;
1098 LogFile.Read(S[1], 8); { file id }
1099 LogFile.Read(I, 4); { c-evo version }
1100 LogFile.Read(J, 4); { exe time }
1101
1102 if (I >= FirstBookCompatibleVersion) and (I <= CevoVersion) then
1103 begin
1104 Result := True;
1105 LogFile.Read(lx, 4);
1106 LogFile.Read(ly, 4);
1107 MapSize := lx * ly;
1108 LogFile.Read(LandMass, 4);
1109 if LandMass = 0 then
1110 LogFile.Read(RealMap, MapSize * 4); // use predefined map
1111 LogFile.Read(MaxTurn, 4);
1112 LogFile.Read(RND, 4);
1113 LogFile.Read(GTurn, 4);
1114 LogFile.Read(SaveMap, 4);
1115 if SaveMap[0] <> $80 then
1116 LogFile.Read(SaveMap[4], ((MapSize - 1) div 4 + 1) * 4 - 4);
1117 for p1 := 0 to nPl - 1 do
1118 begin
1119 LogFile.Read(S[0], 4);
1120 if S[0] = #0 then
1121 PlayersBrain[p1] := nil
1122 else
1123 begin
1124 LogFile.Read(S[4], Byte(S[0]) div 4 * 4);
1125 LogFile.Read(OriginalDataVersion[p1], 4);
1126 LogFile.Read(D, 4); { behavior }
1127 LogFile.Read(Difficulty[p1], 4);
1128 J := Brains.Last;
1129 while Assigned(J) and (AnsiCompareFileName(J.FileName, S) <> 0) do begin
1130 K := Brains.IndexOf(J) - 1;
1131 if K >= 0 then J := Brains[K]
1132 else J := nil;
1133 end;
1134 if not Assigned(J) then
1135 begin // ai not found -- replace by local player
1136 ProcessClientData[p1] := False;
1137 NotifyMessage := S;
1138 Notify(ntAIError);
1139 J := BrainTerm;
1140 end
1141 else
1142 ProcessClientData[p1] := True;
1143 if J.Kind = btNoTerm then
1144 J := BrainSuperVirtual;
1145 // crashed tournament -- load as supervisor
1146 PlayersBrain[p1] := J;
1147 end;
1148 end;
1149 end
1150 else
1151 Result := False;
1152
1153 if Result then begin
1154 CL := TCmdList.Create;
1155 CL.LoadFromFile(LogFile);
1156 end;
1157 FreeAndNil(LogFile);
1158 if not Result then
1159 Exit;
1160
1161 Notify(ntStartDone);
1162 if LoadTurn < 0 then
1163 LoadTurn := GTurn;
1164 if MovieMode then
1165 Mode := moMovie
1166 else if LoadTurn = 0 then
1167 Mode := moLoading
1168 else
1169 Mode := moLoading_Fast;
1170{$IFDEF TEXTLOG}AssignFile(TextLog, LogFileName + '.txt');
1171 Rewrite(TextLog); {$ENDIF}
1172 LoadOK := True;
1173 StartGame;
1174 if MovieMode then
1175 begin
1176 bix[0].Client(cShowGame, 0, nil^);
1177 Notify(ntBackOff);
1178 end
1179 else
1180 Notify(ntLoadBegin);
1181
1182 Started := False;
1183 StatRequest := False;
1184 MovieStopped := False;
1185{$IFDEF LOADPERF}QueryPerformanceCounter(time_total0);
1186 time_a := 0;
1187 time_b := 0;
1188 time_c := 0; {$ENDIF}
1189 while not MovieStopped and (CL.Progress < 1000) do
1190 begin
1191 FormerCLState := CL.State;
1192 CL.Get(Command, p1, Subject, Data);
1193 if p1 < 0 then
1194 p1 := pTurn;
1195 if StatRequest and (Command and (sctMask or sExecute) <> sctInternal or
1196 sExecute) then
1197 begin
1198 GenerateStat(pTurn);
1199 StatRequest := False;
1200 end;
1201 // complete all internal commands following an sTurn before generating statistics
1202 if (Command = sTurn) and not Started then
1203 begin
1204{$IFDEF TEXTLOG}WriteLn(TextLog, '---Turn 0 P0---'); {$ENDIF}
1205 for p1 := 0 to nPl - 1 do
1206 if Assigned(bix[p1]) and ((Mode <> moMovie) or (p1 = 0)) then
1207 CallPlayer(cReplay, p1, nil^);
1208 BeforeTurn0;
1209 if MovieMode then
1210 begin
1211 Inform(pTurn);
1212 CallPlayer(cMovieTurn, 0, nil^);
1213 end;
1214 StatRequest := True;
1215 Started := True;
1216 end
1217 else if (Command = sTurn) and (pTurn = 0) and (GTurn = LoadTurn) then
1218 begin
1219 Assert(CL.State.LoadPos = FormerCLState.LoadPos + 4); // size of sTurn
1220 CL.State := FormerCLState;
1221 CL.Cut;
1222 Break;
1223 end
1224 else if Command = sIntDataChange then
1225 begin
1226{$IFDEF TEXTLOG}LoadPos0 := CL.State.LoadPos; {$ENDIF}
1227 if ProcessClientData[p1] then
1228 CL.GetDataChanges(RW[p1].Data, bix[p1].DataSize)
1229 else
1230 CL.GetDataChanges(nil, 0);
1231{$IFDEF TEXTLOG}WriteLn(TextLog, Format('Data Changes P%d (%d Bytes)', [p1, CL.State.LoadPos - LoadPos0])); {$ENDIF}
1232 end
1233 else
1234 begin
1235{$IFDEF TEXTLOG}CmdInfo := Format('Command %x', [Command]); {$ENDIF}
1236 if Command and (sctMask or sExecute) = sctInternal or sExecute then
1237 IntServer(Command, p1, Subject, Data^) // internal command
1238 else
1239 begin
1240 StatRequest := Command = sTurn;
1241 Server(Command, p1, Subject, Data^);
1242 end;
1243{$IFDEF TEXTLOG}WriteLn(TextLog, CmdInfo); {$ENDIF}
1244 end;
1245 if not MovieMode then
1246 Notify(ntLoadState, CL.Progress * 128 div 1000);
1247 end;
1248
1249 if MovieMode then
1250 begin
1251 Notify(ntBackOn);
1252 bix[0].Client(cBreakGame, -1, nil^);
1253 EndGame;
1254 Notify(ntStartGo);
1255 Result := False;
1256 Exit;
1257 end;
1258
1259 if StatRequest then
1260 GenerateStat(pTurn);
1261 Assert(Started);
1262{$IFDEF TEXTLOG}CloseFile(TextLog); {$ENDIF}
1263{$IFDEF LOADPERF}QueryPerformanceCounter(time_total); { time in s is: (time_total-time_total0)/PerfFreq }{$ENDIF}
1264 NoLogChanges;
1265 NoLogCityTileChanges;
1266 if IsAutoSaveFileName(LogFileName) then begin
1267 LogFileName := FromAutoSaveFileName(LogFileName);
1268 nLogOpened := -1;
1269 end
1270 else
1271 nLogOpened := CL.State.nLog;
1272
1273 Mode := moPlaying;
1274 LastEndClientCommand := -1;
1275 if (GTestFlags and tfUncover <> 0) or (Difficulty[pTurn] = 0) then
1276 DiscoverAll(pTurn, lObserveSuper) { supervisor - all tiles visible }
1277 else
1278 DiscoverViewAreas(pTurn);
1279
1280 for p1 := 0 to nPl - 1 do
1281 if 1 shl p1 and (GAlive or GWatching) <> 0 then
1282 begin
1283 RecalcPeaceMap(p1);
1284 for ix := 0 to RW[p1].nEnemyUn - 1 do
1285 with RW[p1].EnemyUn[ix] do
1286 emix := RWemix[p1, Owner, mix];
1287 Inform(p1);
1288 end;
1289{$IFOPT O-}CheckBorders(-2); {$ENDIF} // for testing only
1290 Notify(ntEndInfo);
1291 if not LoadOK then
1292 begin
1293 NotifyMessage := LogFileName;
1294 Notify(ntLoadError);
1295 end;
1296 bix[0].Client(cShowGame, 0, nil^);
1297 Notify(ntBackOff);
1298 Inform(pTurn);
1299 ChangeClientWhenDone(cResume, 0, nil^, 0);
1300end;
1301
1302procedure InsertTerritoryUpdateCommands;
1303var
1304 p1, Command, Subject: Integer;
1305 Data: Pointer;
1306 FormerCLState: TCmdListState;
1307begin
1308 while CL.Progress < 1000 do
1309 begin
1310 FormerCLState := CL.State;
1311 CL.Get(Command, p1, Subject, Data);
1312 if (Command = sIntExpandTerritory) and (p1 = pTurn) then
1313 begin
1314 IntServer(Command, p1, Subject, Data^);
1315{$IFDEF TEXTLOG}WriteLn(TextLog, 'AfterTurn - ExpandTerritory'); {$ENDIF}
1316 end
1317 else
1318 begin
1319 CL.State := FormerCLState;
1320 Break;
1321 end;
1322 end;
1323{$IFOPT O-}InvalidTreatyMap := 0; {$ENDIF}
1324end;
1325
1326procedure StartNewGame(const FileName, AMapFileName: string;
1327 Newlx, Newly, NewLandMass, NewMaxTurn: Integer);
1328var
1329 I: Integer;
1330begin
1331 Notify(ntStartDone);
1332 SavePath := ExtractFileDir(FileName);
1333 LogFileName := FileName;
1334 MapFileName := AMapFileName;
1335 {$IFDEF FastContact}
1336 lx := 24;
1337 ly := 42;
1338 {$ELSE}
1339 lx := Newlx;
1340 ly := Newly;
1341 {$ENDIF}
1342 MapSize := lx * ly;
1343 if MapFileName <> '' then
1344 LandMass := 0
1345 else
1346 LandMass := NewLandMass;
1347 MaxTurn := NewMaxTurn;
1348 DelphiRandomize;
1349 RND := DelphiRandSeed;
1350 Mode := moPlaying;
1351 CL := TCmdList.Create;
1352 StartGame;
1353 NoLogChanges;
1354 CallAllPlayers(cGetReady, nil^);
1355 LogChanges;
1356 CL.Put(sTurn, 0, 0, nil);
1357 BeforeTurn0;
1358 NoLogCityTileChanges;
1359 GenerateStat(pTurn);
1360 nLogOpened := -1;
1361 LastEndClientCommand := -1;
1362 CallPlayer(cShowGame, 0, nil^);
1363 for I := 0 to nPl - 1 do
1364 if Assigned(bix[I]) and (bix[I].Kind = btNetworkServer) then
1365 CallPlayer(cShowGame, I, nil^);
1366 Notify(ntBackOff);
1367 Inform(pTurn);
1368 ChangeClientWhenDone(cTurn, 0, nil^, 0);
1369end;
1370
1371procedure FillMap(TerrainType: Cardinal; SpecialResources: Boolean);
1372var
1373 Loc1: Integer;
1374begin
1375 if SpecialResources then begin
1376 for Loc1 := 0 to MapSize - 1 do
1377 EditTile(Loc1, TerrainType or ($F shl 27));
1378 end else begin
1379 for Loc1 := 0 to MapSize - 1 do
1380 RealMap[Loc1] := TerrainType or ($F shl 27);
1381 end;
1382end;
1383
1384procedure DirectHelp(Command: Integer);
1385begin
1386 InitBrain(BrainTerm);
1387 BrainTerm.Client(Command, -1, nil^);
1388 AICredits := #0;
1389end;
1390
1391procedure EditMap(const AMapFileName: string; Newlx, Newly, NewLandMass: Integer);
1392var
1393 p1: Integer;
1394 Game: TNewGameData;
1395 Map: TMap;
1396begin
1397 Notify(ntStartDone);
1398 Notify(ntInitLocalHuman);
1399 MapFileName := AMapFileName;
1400 lx := Newlx;
1401 ly := Newly;
1402 MapSize := lx * ly;
1403 LandMass := NewLandMass;
1404 bix[0] := BrainTerm;
1405 Difficulty[0] := 0;
1406 InitBrain(BrainTerm);
1407
1408 DelphiRandomize;
1409 GAlive := 0;
1410 GWatching := 1;
1411 if FileExists(MapFileName) then begin
1412 Map := TMap.Create;
1413 Map.LoadFromFile(MapFileName);
1414 MapSize := Map.MapSize;
1415 lx := Map.Size.X;
1416 ly := Map.Size.Y;
1417 Move(Map.Tiles[0], RealMap, MapSize * 4);
1418 FreeAndNil(Map);
1419 end else begin
1420 FillMap(fOcean, False);
1421 end;
1422 CL := nil;
1423 InitMapEditor;
1424 RW[0].Data := nil;
1425 RW[0].BorderHelper := nil;
1426 RW[0].Alive := 0;
1427 Game.lx := lx;
1428 Game.ly := ly;
1429 Game.RO[0] := @RW[0];
1430 Game.Difficulty[0] := 0;
1431 for p1 := 1 to nPl - 1 do begin
1432 Game.RO[p1] := nil;
1433 Game.Difficulty[p1] := -1;
1434 end;
1435 BrainTerm.Client(cNewMap, -1, Game);
1436
1437 DiscoverAll(0, lObserveSuper);
1438 Notify(ntEndInfo);
1439 bix[0].Client(cShowGame, 0, nil^);
1440 Notify(ntBackOff);
1441 ChangeClientWhenDone(cEditMap, 0, nil^, 0);
1442end;
1443
1444procedure DestroySpacePort_TellPlayers(P, pCapturer: Integer);
1445var
1446 cix, I, p1: Integer;
1447 ShowShipChange: TShowShipChange;
1448begin
1449 // stop ship part production
1450 for cix := 0 to RW[P].nCity - 1 do
1451 with RW[P].City[cix] do
1452 if (Loc >= 0) and (Project and cpImp <> 0) and
1453 ((Project and cpIndex = woMIR) or
1454 (Imp[Project and cpIndex].Kind = ikShipPart)) then
1455 begin
1456 Inc(RW[P].Money, Prod0);
1457 Prod := 0;
1458 Prod0 := 0;
1459 Project := cpImp + imTrGoods;
1460 Project0 := cpImp + imTrGoods;
1461 end;
1462
1463 // destroy ship
1464 with GShip[P] do
1465 if Parts[0] + Parts[1] + Parts[2] > 0 then
1466 begin
1467 for I := 0 to nShipPart - 1 do
1468 begin
1469 ShowShipChange.Ship1Change[I] := -Parts[I];
1470 if pCapturer >= 0 then
1471 begin
1472 ShowShipChange.Ship2Change[I] := Parts[I];
1473 Inc(GShip[pCapturer].Parts[I], Parts[I]);
1474 end;
1475 Parts[I] := 0;
1476 end;
1477 if Mode >= moMovie then
1478 begin
1479 if pCapturer >= 0 then
1480 ShowShipChange.Reason := scrCapture
1481 else
1482 ShowShipChange.Reason := scrDestruction;
1483 ShowShipChange.Ship1Owner := P;
1484 ShowShipChange.Ship2Owner := pCapturer;
1485 for p1 := 0 to nPl - 1 do
1486 if 1 shl p1 and (GAlive or GWatching) <> 0 then
1487 begin
1488 Move(GShip, RW[p1].Ship, SizeOf(GShip));
1489 if 1 shl p1 and GWatching <> 0 then
1490 CallPlayer(cShowShipChange, p1, ShowShipChange);
1491 end;
1492 end;
1493 end;
1494end;
1495
1496procedure DestroyCity_TellPlayers(P, cix: Integer; SaveUnits: Boolean);
1497begin
1498 if RW[P].City[cix].Built[imSpacePort] > 0 then
1499 DestroySpacePort_TellPlayers(P, -1);
1500 DestroyCity(P, cix, SaveUnits);
1501end;
1502
1503procedure ChangeCityOwner_TellPlayers(pOld, cixOld, pNew: Integer);
1504begin
1505 if RW[pOld].City[cixOld].Built[imSpacePort] > 0 then
1506 if RW[pNew].NatBuilt[imSpacePort] > 0 then
1507 DestroySpacePort_TellPlayers(pOld, pNew)
1508 else
1509 DestroySpacePort_TellPlayers(pOld, -1);
1510 ChangeCityOwner(pOld, cixOld, pNew);
1511end;
1512
1513procedure CheckWin(P: Integer);
1514var
1515 I: Integer;
1516 ShipComplete: Boolean;
1517begin
1518 ShipComplete := True;
1519 for I := 0 to nShipPart - 1 do
1520 if GShip[P].Parts[I] < ShipNeed[I] then begin
1521 ShipComplete := False;
1522 Break;
1523 end;
1524 if ShipComplete then
1525 GWinner := GWinner or 1 shl P; // game won!
1526end;
1527
1528procedure BeforeTurn;
1529var
1530 I, p1, uix, cix, V21, Loc1, Cost, Job0, nAlive, nAppliers, ad, OldLoc,
1531 SiegedTiles, nUpdateLoc: Integer;
1532 UpdateLoc: array [0 .. numax - 1] of Integer;
1533 Radius: TVicinity21Loc;
1534 ShowShipChange: TShowShipChange;
1535 TribeExtinct, JobDone, MirBuilt: Boolean;
1536begin
1537{$IFOPT O-}Assert(1 shl pTurn and InvalidTreatyMap = 0); {$ENDIF}
1538 Assert(1 shl pTurn and (GAlive or GWatching) <> 0);
1539 if (1 shl pTurn and GAlive = 0) and (Difficulty[pTurn] > 0) then
1540 Exit;
1541
1542 if (GWonder[woGrLibrary].EffectiveOwner = pTurn) and (GWinner = 0) then
1543 begin // check great library effect
1544 nAlive := 0;
1545 for p1 := 0 to nPl - 1 do
1546 if 1 shl p1 and GAlive <> 0 then
1547 Inc(nAlive);
1548 ad := 0;
1549 while ad <= (nAdv - 5) do begin
1550 if RW[pTurn].Tech[ad] < tsSeen then
1551 begin
1552 nAppliers := 0;
1553 for p1 := 0 to nPl - 1 do
1554 if (p1 <> pTurn) and (1 shl p1 and GAlive <> 0) and
1555 (RW[p1].Tech[ad] >= tsApplicable) then
1556 Inc(nAppliers);
1557 if nAppliers * 2 > nAlive then
1558 begin
1559 SeeTech(pTurn, ad);
1560 Inc(nTech[pTurn]);
1561 if Mode >= moMovie then
1562 CallPlayer(cShowGreatLibTech, pTurn, ad);
1563 // do not call CallPlayer(pTurn) while map is invalid
1564 end;
1565 end;
1566 Inc(ad);
1567 end;
1568 end;
1569
1570 MaskD(ObserveLevel, MapSize, not Cardinal(3 shl (2 * pTurn)));
1571 if Mode > moLoading_Fast then
1572 MaskD(RW[pTurn].Map^, MapSize, Cardinal(not Cardinal(fUnit or fHiddenUnit or
1573 fStealthUnit or fObserved or fSpiedOut or fOwned or fOwnZoCUnit or
1574 fInEnemyZoC)));
1575 RW[pTurn].nEnemyUn := 0;
1576
1577 MirBuilt := False;
1578 if (Difficulty[pTurn] > 0) and (GWinner = 0) then
1579 with RW[pTurn] do
1580 begin
1581 if nCity > 0 then
1582 for p1 := 0 to nPl - 1 do
1583 if GTurn = EvaStart[p1] + PeaceEvaTurns then
1584 begin // peace contract -- remove all units from p1's territory
1585 Loc1 := City[0].Loc; // search destination for homeless units
1586 for cix := 1 to nCity - 1 do
1587 if (City[cix].Loc >= 0) and
1588 ((Loc1 < 0) or (City[cix].Built[imPalace] > 0)) then
1589 Loc1 := City[cix].Loc;
1590 for uix := 0 to nUn - 1 do
1591 with Un[uix] do
1592 if (Loc >= 0) and (Model[mix].Kind <> mkDiplomat) and
1593 ((Home >= 0) or (Loc1 >= 0)) and
1594 (RealMap[Loc] shr 27 = Cardinal(p1)) then
1595 begin
1596 OldLoc := Loc;
1597 if Master >= 0 then
1598 begin // transport unload
1599 if Model[mix].Domain = dAir then
1600 Dec(Un[Master].AirLoad)
1601 else
1602 Dec(Un[Master].TroopLoad);
1603 Master := -1;
1604 end
1605 else
1606 FreeUnit(pTurn, uix);
1607
1608 if Home >= 0 then
1609 Loc := City[Home].Loc
1610 else
1611 Loc := Loc1;
1612 PlaceUnit(pTurn, uix);
1613 UpdateUnitMap(OldLoc);
1614 UpdateUnitMap(Loc);
1615 Flags := Flags or unWithdrawn;
1616 Happened := Happened or phPeaceEvacuation;
1617 end;
1618 end;
1619
1620 if Mode >= moMovie then
1621 FillChar(ShowShipChange, SizeOf(ShowShipChange), 0);
1622 TribeExtinct := True;
1623 nUpdateLoc := 0;
1624 for cix := 0 to nCity - 1 do
1625 with City[cix] do
1626 if Loc >= 0 then
1627 begin { next turn for all cities - city loop 1 }
1628 // if ServerVersion[pTurn]>=$000EF0 then
1629 // Flags:=Flags and (chFounded or chCaptured or chProductionSabotaged or chDisorder)
1630 // else Flags:=Flags and (chCaptured or chProductionSabotaged or chDisorder);
1631 // check for siege
1632 SiegedTiles := 0;
1633 V21_to_Loc(Loc, Radius);
1634 for V21 := 1 to 26 do
1635 if Tiles and (1 shl V21) and not (1 shl CityOwnTile) <> 0 then
1636 begin
1637 Loc1 := Radius[V21];
1638 Assert((Loc1 >= 0) and (Loc1 < MapSize) and
1639 (UsedByCity[Loc1] = Loc));
1640 p1 := RealMap[Loc1] shr 27;
1641 if (RealMap[Loc1] and fCity <> 0) or (p1 < nPl) and
1642 (p1 <> pTurn) and (RW[pTurn].Treaty[p1] >= trPeace) or
1643 (ZoCMap[Loc1] > 0) and (Occupant[Loc1] <> pTurn) and
1644 (Treaty[Occupant[Loc1]] < trPeace) then
1645 begin
1646 Tiles := Tiles and not (1 shl V21);
1647 UsedByCity[Loc1] := -1;
1648 Flags := Flags or chSiege;
1649 Inc(SiegedTiles);
1650 end;
1651 end;
1652 while SiegedTiles > 0 do // replace sieged tiles
1653 begin
1654 if not AddBestCityTile(pTurn, cix) then
1655 Break;
1656 Dec(SiegedTiles);
1657 end;
1658
1659 if Flags and chFounded = 0 then
1660 begin
1661 // CollectCityResources(pTurn,cix); // old style
1662
1663 if CityTurn(pTurn, cix) then
1664 TribeExtinct := False
1665 else
1666 begin // city is erased
1667 RemoveDomainUnits(dSea, pTurn, Loc);
1668 RemoveDomainUnits(dAir, pTurn, Loc);
1669 Map[Loc] := Map[Loc] and not fCity; // !!! do this in inner core
1670 UpdateLoc[nUpdateLoc] := Loc;
1671 Inc(nUpdateLoc);
1672 DestroyCity_TellPlayers(pTurn, cix, True);
1673 end;
1674
1675 if (Flags and chProduction <> 0) and (Project0 and cpImp <> 0)
1676 then
1677 begin
1678 if Project0 and cpIndex = woMIR then // MIR completed
1679 MirBuilt := True
1680 else if Project0 and cpIndex = woManhattan then
1681 GColdWarStart := GTurn
1682 else if Imp[Project0 and cpIndex].Kind = ikShipPart
1683 then { ship parts produced }
1684 Inc(ShowShipChange.Ship1Change[Project0 and cpIndex -
1685 imShipComp]);
1686 end;
1687 end;
1688 end; { city loop 1 }
1689 if nUpdateLoc > 0 then
1690 begin
1691 CheckBorders(-1, pTurn);
1692 for I := 0 to nUpdateLoc - 1 do
1693 UpdateUnitMap(UpdateLoc[I], True);
1694 if Mode >= moMovie then
1695 for p1 := 0 to nPl - 1 do
1696 if (1 shl p1 and GWatching <> 0) and (p1 <> pTurn) then
1697 for I := 0 to nUpdateLoc - 1 do
1698 if ObserveLevel[UpdateLoc[I]] shr (2 * p1) and 3 >= lObserveUnhidden
1699 then
1700 CallPlayer(cShowCityChanged, p1, UpdateLoc[I]);
1701 end;
1702
1703 for uix := 0 to nUn - 1 do
1704 with Un[uix] do
1705 if Loc >= 0 then
1706 begin // unit loop 2
1707 if Health < 100 then
1708 Recover(pTurn, uix);
1709
1710 if Flags and unMountainDelay <> 0 then
1711 begin
1712 Movement := 0;
1713 Flags := Flags and not unMountainDelay;
1714 end
1715 else
1716 Movement := UnitSpeed(pTurn, mix, Health); { refresh movement }
1717
1718 Assert(Loc >= 0);
1719 if Model[mix].Kind <> mkDiplomat then
1720 begin // check treaty violation
1721 p1 := RealMap[Loc] shr 27;
1722 if (p1 < nPl) and (p1 <> pTurn) and (Treaty[p1] >= trPeace) then
1723 begin
1724 if (Job in [jCity, jPillage, jClear, jAfforest, jTrans]) or
1725 (Job in [jIrr, jMine, jFort, jBase]) and
1726 (RealMap[Loc] and fTerImp <> 0) then
1727 Job := jNone;
1728 if (GTurn > EvaStart[p1] + PeaceEvaTurns) and
1729 (Treaty[p1] <> trAlliance) then
1730 begin
1731 EvaStart[p1] := GTurn;
1732 Happened := Happened or phPeaceViolation;
1733 if Mode >= moMovie then
1734 CallPlayer(cShowPeaceViolation, p1, pTurn);
1735 end;
1736 end;
1737 end;
1738
1739 if ServerVersion[pTurn] >= $000EF0 then
1740 begin
1741 if (Health <= 0) or TribeExtinct then
1742 RemoveUnit_UpdateMap(pTurn, uix);
1743 end;
1744 end;
1745
1746 if ServerVersion[pTurn] < $000EF0 then
1747 for uix := 0 to nUn - 1 do
1748 with Un[uix] do
1749 if Loc >= 0 then
1750 begin // unit loop 3
1751 Loc1 := Loc;
1752 Job0 := Job;
1753 if Job <> jNone then
1754 JobDone := Work(pTurn, uix);
1755 { settlers do terrain improvement jobs }
1756 if (Health <= 0) or TribeExtinct then
1757 RemoveUnit_UpdateMap(pTurn, uix);
1758
1759 if (Job0 = jCity) and JobDone then // new city
1760 begin
1761 AddBestCityTile(pTurn, RW[pTurn].nCity - 1);
1762 UpdateUnitMap(Loc1, True);
1763 if Mode >= moMovie then // tell enemies
1764 for p1 := 0 to nPl - 1 do
1765 if (1 shl p1 and GWatching <> 0) and (p1 <> pTurn) and
1766 (ObserveLevel[Loc1] and (3 shl (2 * p1)) > 0) then
1767 CallPlayer(cShowCityChanged, p1, Loc1);
1768 end;
1769 end;
1770
1771 { pollution - city loop 3 }
1772 for cix := 0 to nCity - 1 do
1773 with City[cix] do
1774 if (Loc >= 0) and (Pollution >= MaxPollution) then
1775 Pollute(pTurn, cix);
1776
1777 CompactLists(pTurn);
1778 if (nUn = 0) and (nCity = 0) then
1779 begin // nation made extinct
1780 Happened := Happened or phExtinct;
1781 GAlive := GAlive and not (1 shl pTurn);
1782 Stat[stPop, pTurn, GTurn] := 0;
1783 Stat[stMil, pTurn, GTurn] := 0;
1784 Stat[stScience, pTurn, GTurn] := 0;
1785 Stat[stExplore, pTurn, GTurn] := 0;
1786 Stat[stTerritory, pTurn, GTurn] := 0;
1787 Stat[stWork, pTurn, GTurn] := 0;
1788 for p1 := 0 to nPl - 1 do
1789 if 1 shl p1 and (GAlive or GWatching) <> 0 then
1790 begin
1791 if p1 <> pTurn then
1792 begin
1793 GiveCivilReport(p1, pTurn);
1794 if (GTestFlags and tfUncover <> 0) or (Difficulty[p1] = 0) or
1795 (RW[p1].Treaty[pTurn] = trAlliance) then
1796 GiveMilReport(p1, pTurn);
1797 end;
1798 with RW[p1] do
1799 begin
1800 Alive := GAlive;
1801 for Loc1 := 0 to MapSize - 1 do
1802 if Territory[Loc1] = pTurn then
1803 // remove territory of extinct nation from player maps
1804 begin
1805 Territory[Loc1] := -1;
1806 Map[Loc1] := Map[Loc1] and not fPeace;
1807 end;
1808 end;
1809 end;
1810 Exit;
1811 end;
1812
1813 // check research
1814 Cost := TechCost(pTurn);
1815 if GTestFlags and tfImmAdvance <> 0 then
1816 Research := Cost;
1817 if (Happened and phTech = 0) and (Research >= Cost) then
1818 begin
1819 if ResearchTech = adMilitary then
1820 EnableDevModel(pTurn) { new Unit class initiated }
1821 else if ResearchTech >= 0 then
1822 DiscoverTech(pTurn, ResearchTech);
1823
1824 Dec(Research, Cost);
1825 Happened := Happened or phTech;
1826 ResearchTech := -1;
1827 end
1828 else if (ResearchTech = -2) and (nCity > 0) then
1829 begin
1830 Happened := Happened or phTech;
1831 ResearchTech := -1;
1832 end;
1833
1834 if Credibility < MaxCredibility then
1835 for p1 := 0 to nPl - 1 do
1836 if (p1 <> pTurn) and (1 shl p1 and GAlive <> 0) and
1837 (Treaty[p1] >= trPeace) then
1838 begin
1839 Inc(Credibility);
1840 Break;
1841 end;
1842
1843 if GWinner = 0 then
1844 CheckWin(pTurn);
1845 if (Mode >= moMovie) and (GWinner = 0) and
1846 ((ShowShipChange.Ship1Change[0] > 0) or
1847 (ShowShipChange.Ship1Change[1] > 0) or
1848 (ShowShipChange.Ship1Change[2] > 0)) then
1849 begin
1850 ShowShipChange.Reason := scrProduction;
1851 ShowShipChange.Ship1Owner := pTurn;
1852 ShowShipChange.Ship2Owner := -1;
1853 for p1 := 0 to nPl - 1 do
1854 if (p1 <> pTurn) and (1 shl p1 and (GAlive or GWatching) <> 0) then
1855 begin
1856 Move(GShip, RW[p1].Ship, SizeOf(GShip));
1857 if 1 shl p1 and GWatching <> 0 then
1858 CallPlayer(cShowShipChange, p1, ShowShipChange);
1859 end;
1860 end;
1861 if WinOnAlone and (GAlive and not (1 shl pTurn or 1) = 0) then
1862 GWinner := 1 shl pTurn; // break if only one nation left
1863
1864 if GTurn = AnarchyStart + AnarchyTurns then
1865 begin
1866 AnarchyStart := -AnarchyTurns - 1;
1867 Government := gDespotism;
1868 for p1 := 0 to nPl - 1 do
1869 if (p1 <> pTurn) and ((GAlive or GWatching) and (1 shl p1) <> 0) then
1870 RW[p1].EnemyReport[pTurn].Government := gDespotism;
1871 Inc(Happened, phChangeGov);
1872 end;
1873 end; // if Difficulty[pTurn]>0
1874
1875 if (pTurn = 0) and (GWinner > 0) then
1876 begin // game over, give world map and all reports to player 0
1877 DiscoverAll(pTurn, lObserveSuper);
1878 for p1 := 1 to nPl - 1 do
1879 if 1 shl p1 and GAlive <> 0 then
1880 begin
1881 if RW[pTurn].Treaty[p1] < trNone then
1882 begin
1883 RW[pTurn].Treaty[p1] := trNone;
1884 RW[p1].Treaty[pTurn] := trNone;
1885 end;
1886 GiveCivilReport(pTurn, p1);
1887 GiveMilReport(pTurn, p1);
1888 end;
1889 end
1890 else
1891 begin
1892 // show observed areas
1893 if (GTestFlags and tfUncover <> 0) or (Difficulty[pTurn] = 0)
1894 then { supervisor - all tiles visible }
1895 begin
1896 if (bix[pTurn].Kind <> btNoTerm) and
1897 ((Difficulty[pTurn] > 0) or (Mode > moLoading_Fast)) then
1898 DiscoverAll(pTurn, lObserveSuper);
1899 end
1900 else
1901 begin
1902 DiscoverViewAreas(pTurn);
1903 if MirBuilt then
1904 DiscoverAll(pTurn, lObserveUnhidden);
1905 end;
1906 end;
1907 // CheckContact;
1908end;
1909
1910procedure AfterTurn;
1911var
1912 cix, uix, p1, Loc1, Job0: Integer;
1913 JobDone: Boolean;
1914begin
1915 with RW[pTurn] do
1916 begin
1917 for cix := 0 to nCity - 1 do
1918 if City[cix].Loc >= 0 then
1919 begin
1920 // City[cix].Flags:=City[cix].Flags and not chProductionSabotaged;
1921 City[cix].Flags := City[cix].Flags and (chCaptured or chDisorder);
1922 CollectCityResources(pTurn, cix); // new style
1923 end;
1924
1925 Inc(Money, OracleIncome);
1926 OracleIncome := 0;
1927 if GWonder[woOracle].EffectiveOwner = pTurn then
1928 begin
1929 for p1 := 0 to nPl - 1 do
1930 if (1 shl p1 and GAlive <> 0) and
1931 ((p1 = pTurn) or (RW[pTurn].Treaty[p1] > trNoContact)) then
1932 for cix := 0 to RW[p1].nCity - 1 do
1933 if (RW[p1].City[cix].Loc >= 0) and
1934 (RW[p1].City[cix].Built[imTemple] > 0) then
1935 Inc(OracleIncome);
1936 end;
1937
1938 if (GTestFlags and tfImmImprove = 0) and (Government <> gAnarchy) then
1939 for cix := 0 to nCity - 1 do
1940 if (City[cix].Loc >= 0) and (City[cix].Flags and chCaptured = 0) then
1941 PayCityMaintenance(pTurn, cix);
1942
1943 if ServerVersion[pTurn] >= $000EF0 then
1944 begin // let settlers work
1945 for cix := 0 to nCity - 1 do
1946 City[cix].Flags := City[cix].Flags and not chFounded;
1947 for uix := 0 to nUn - 1 do
1948 with Un[uix] do
1949 if Loc >= 0 then
1950 begin
1951 Loc1 := Loc;
1952 Job0 := Job;
1953 if Job <> jNone then
1954 JobDone := Work(pTurn, uix);
1955 { settlers do terrain improvement jobs }
1956 if Health <= 0 then
1957 RemoveUnit_UpdateMap(pTurn, uix);
1958
1959 if (Job0 = jCity) and JobDone then // new city
1960 begin
1961 AddBestCityTile(pTurn, RW[pTurn].nCity - 1);
1962 UpdateUnitMap(Loc1, True);
1963 if Mode >= moMovie then // tell enemies
1964 for p1 := 0 to nPl - 1 do
1965 if (1 shl p1 and GWatching <> 0) and (p1 <> pTurn) and
1966 (ObserveLevel[Loc1] and (3 shl (2 * p1)) > 0) then
1967 CallPlayer(cShowCityChanged, p1, Loc1);
1968 end;
1969 end;
1970 end;
1971
1972 for uix := 0 to nUn - 1 do
1973 with Un[uix] do
1974 if Loc >= 0 then
1975 begin { next turn for all units }
1976 if Model[mix].Domain = dAir then
1977 if (Master >= 0) or (RealMap[Loc] and fCity <> 0) or
1978 (RealMap[Loc] and fTerImp = tiBase) then
1979 begin
1980 Fuel := Model[mix].Cap[mcFuel];
1981 Flags := Flags or unBombsLoaded;
1982 end
1983 else if Model[mix].Kind = mkSpecial_Glider then { glider }
1984 begin
1985 if RealMap[Loc] and fTerrain < fGrass then
1986 begin
1987 RemoveUnit_UpdateMap(pTurn, uix); // unit lost
1988 Happened := Happened or phGliderLost;
1989 end;
1990 end
1991 else
1992 begin
1993 Dec(Fuel);
1994 if Fuel < 0 then
1995 begin
1996 RemoveUnit_UpdateMap(pTurn, uix); // unit lost
1997 Happened := Happened or phPlaneLost;
1998 end
1999 end
2000 else if (Master < 0) and (Movement > 0) then // check HostileDamage
2001 begin
2002 Health := Health - HostileDamage(pTurn, mix, Loc, Movement);
2003 if Health < 0 then
2004 RemoveUnit_UpdateMap(pTurn, uix);
2005 end;
2006 end; { unit loop 1 }
2007
2008 for uix := 0 to nUn - 1 do
2009 with Un[uix] do
2010 begin
2011 Flags := Flags and not unWithdrawn;
2012 if (Loc >= 0) and (Model[mix].Domain = dGround) and (Master < 0) and
2013 ((Integer(Movement) = Model[mix].Speed) or
2014 (Model[mix].Cap[mcAcademy] > 0) and (Movement * 2 >= Model[mix].Speed))
2015 then
2016 Flags := Flags or unFortified; // fortify unmoved units
2017 end;
2018
2019 if (GTestFlags and tfUncover = 0) and (Difficulty[pTurn] > 0) then
2020 begin // restrict view area to current positions
2021 MaskD(ObserveLevel, MapSize, not Cardinal(3 shl (2 * pTurn)));
2022 if Mode > moLoading_Fast then
2023 MaskD(RW[pTurn].Map^, MapSize, Cardinal(not Cardinal(fUnit or fHiddenUnit or
2024 fStealthUnit or fObserved or fSpiedOut or fOwned or fOwnZoCUnit or
2025 fInEnemyZoC)));
2026 RW[pTurn].nEnemyUn := 0;
2027 DiscoverViewAreas(pTurn);
2028 end;
2029
2030 if GWinner = 0 then
2031 for p1 := 0 to nPl - 1 do
2032 if 1 shl p1 and GAlive <> 0 then
2033 CheckWin(p1);
2034 end;
2035end;
2036
2037procedure NextPlayer;
2038begin
2039 if GTurn = 0 then
2040 BeforeTurn0
2041 else
2042 BeforeTurn;
2043 NoLogCityTileChanges;
2044 GenerateStat(pTurn);
2045 Inform(pTurn);
2046 ChangeClient;
2047end;
2048
2049function ExecuteMove(P, uix, ToLoc: Integer; var MoveInfo: TMoveInfo;
2050 ShowMove: TShowMove): Integer;
2051var
2052 I, p1, FromLoc, uix1, nUpdateLoc: Integer;
2053 MinLevel, MissionResult: Cardinal;
2054 PModel: ^TModel;
2055 UpdateLoc: array [0 .. numax - 1] of Integer;
2056 SeeFrom, SeeTo, ExtDiscover: Boolean;
2057begin
2058 Result := 0;
2059 with RW[P], Un[uix] do
2060 begin
2061 PModel := @Model[mix];
2062 FromLoc := Loc;
2063
2064 if Master < 0 then
2065 FreeUnit(P, uix);
2066 if (MoveInfo.MoveType in [mtMove, mtCapture]) and MoveInfo.MountainDelay
2067 then
2068 begin
2069 Flags := Flags or unMountainDelay;
2070 end;
2071 Loc := -2;
2072 if TroopLoad + AirLoad > 0 then
2073 for I := 0 to nUn - 1 do
2074 if (Un[I].Loc >= 0) and (Un[I].Master = uix) then
2075 Un[I].Loc := -2;
2076 UpdateUnitMap(FromLoc);
2077
2078 if Mode >= moMovie then { show move in interface modules }
2079 begin
2080 ShowMove.EndHealth := MoveInfo.EndHealth;
2081 ShowMove.EndHealthDef := -1;
2082 if Master >= 0 then
2083 if Model[Un[Master].mix].Domain = dAir then
2084 ShowMove.Flags := ShowMove.Flags or umPlaneUnloading
2085 else
2086 ShowMove.Flags := ShowMove.Flags or umShipUnloading;
2087 if MoveInfo.ToMaster >= 0 then
2088 if Model[Un[MoveInfo.ToMaster].mix].Domain = dAir then
2089 ShowMove.Flags := ShowMove.Flags or umPlaneLoading
2090 else
2091 ShowMove.Flags := ShowMove.Flags or umShipLoading;
2092 for p1 := 0 to nPl - 1 do
2093 if (1 shl p1 and GWatching <> 0) and ((p1 <> P) or (bix[p1].Kind = btTerm))
2094 then
2095 begin
2096 if PModel.Cap[mcStealth] > 0 then
2097 MinLevel := lObserveSuper
2098 else if PModel.Cap[mcSub] > 0 then
2099 MinLevel := lObserveAll
2100 else
2101 MinLevel := lObserveUnhidden;
2102 SeeFrom := (p1 = P) or (ObserveLevel[FromLoc] shr (2 * p1) and
2103 3 >= MinLevel);
2104 SeeTo := (p1 = P) or (ObserveLevel[ToLoc] shr (2 * p1) and
2105 3 >= MinLevel);
2106 if SeeFrom and SeeTo then
2107 begin
2108 TellAboutModel(p1, P, mix);
2109 if p1 = P then
2110 ShowMove.emix := -1
2111 else
2112 ShowMove.emix := emixSafe(p1, P, mix);
2113 if MoveInfo.MoveType = mtCapture then
2114 CallPlayer(cShowCapturing, p1, ShowMove)
2115 else
2116 CallPlayer(cShowMoving, p1, ShowMove);
2117 end
2118 else if SeeFrom then
2119 CallPlayer(cShowUnitChanged, p1, FromLoc);
2120 end;
2121 end;
2122
2123 if MoveInfo.MoveType <> mtSpyMission then
2124 Loc := ToLoc;
2125 if TroopLoad + AirLoad > 0 then
2126 for I := 0 to nUn - 1 do
2127 if Un[I].Loc = -2 then
2128 Un[I].Loc := ToLoc;
2129
2130 ExtDiscover := False;
2131 nUpdateLoc := 0;
2132 if MoveInfo.MoveType = mtCapture then
2133 begin
2134 Assert(Occupant[ToLoc] < 0);
2135 for uix1 := 0 to RW[MoveInfo.Defender].nUn - 1 do
2136 with RW[MoveInfo.Defender].Un[uix1] do
2137 if (Loc >= 0) and (Home = MoveInfo.Dcix) then
2138 begin
2139 UpdateLoc[nUpdateLoc] := Loc;
2140 Inc(nUpdateLoc);
2141 end;
2142 // unit will be removed -- remember position and update for all players
2143
2144 if (RW[MoveInfo.Defender].City[MoveInfo.Dcix].Size > 2) and (nCity < ncmax)
2145 then
2146 begin // city captured
2147 ChangeCityOwner_TellPlayers(MoveInfo.Defender, MoveInfo.Dcix, P);
2148 City[nCity - 1].Flags := CaptureTurns shl 16;
2149 CityShrink(P, nCity - 1);
2150 if Mode = moPlaying then
2151 with RW[P].City[nCity - 1] do
2152 begin
2153 // SavedResourceWeights[nCity-1]:=ResourceWeights;
2154 SavedTiles[nCity - 1] := Tiles;
2155 end;
2156 ExtDiscover := True;
2157
2158 // Temple of Zeus effect
2159 if GWonder[woZeus].EffectiveOwner = P then
2160 begin
2161 GiveCivilReport(P, MoveInfo.Defender);
2162 for I := 0 to nAdv - 1 do
2163 if not (I in FutureTech) and (RW[P].Tech[I] < tsSeen) and
2164 (RW[MoveInfo.Defender].Tech[I] >= tsApplicable) then
2165 begin
2166 Happened := Happened or phStealTech;
2167 GStealFrom := MoveInfo.Defender;
2168 Break;
2169 end;
2170 end;
2171 if Mode = moPlaying then
2172 LogCheckBorders(P, nCity - 1, MoveInfo.Defender);
2173{$IFOPT O-}
2174 if Mode < moPlaying then
2175 InvalidTreatyMap := not (1 shl P);
2176{$ENDIF}
2177 // territory should not be considered for the rest of the command
2178 // execution, because during loading a game it's incorrect before
2179 // subsequent sIntExpandTerritory is processed
2180 end
2181 else // city destroyed
2182 begin
2183 DestroyCity_TellPlayers(MoveInfo.Defender, MoveInfo.Dcix, False);
2184 CheckBorders(ToLoc, MoveInfo.Defender);
2185 end;
2186 RecalcPeaceMap(P);
2187 if Mode >= moMovie then
2188 Move(GWonder, Wonder, SizeOf(GWonder));
2189 end; { if MoveInfo.MoveType=mtCapture }
2190
2191 if MoveInfo.MoveType = mtSpyMission then
2192 begin
2193 MissionResult := DoSpyMission(P, MoveInfo.Defender, MoveInfo.Dcix,
2194 SpyMission);
2195 if (Mode = moPlaying) and (SpyMission = smStealForeignReports) then
2196 CallPlayer(cShowMissionResult, P, MissionResult);
2197 end;
2198
2199 Health := MoveInfo.EndHealth;
2200 Dec(Movement, MoveInfo.Cost);
2201 // transport unload
2202 if Master >= 0 then
2203 begin
2204 if PModel.Domain = dAir then
2205 Dec(Un[Master].AirLoad)
2206 else
2207 begin
2208 Dec(Un[Master].TroopLoad);
2209 Assert(Movement <= 0);
2210 end;
2211 Master := -1;
2212 end;
2213
2214 if (Health <= 0) or (MoveInfo.MoveType = mtSpyMission) then
2215 RemoveUnit(P, uix) // spy mission or victim of HostileDamage
2216 else
2217 begin // transport load
2218 Master := MoveInfo.ToMaster;
2219 if MoveInfo.ToMaster >= 0 then
2220 begin
2221 if PModel.Domain = dAir then
2222 Inc(Un[MoveInfo.ToMaster].AirLoad)
2223 else
2224 Inc(Un[MoveInfo.ToMaster].TroopLoad);
2225 end
2226 else
2227 PlaceUnit(P, uix);
2228 end;
2229
2230 if (MoveInfo.MoveType = mtCapture) and (nUpdateLoc > 0) then
2231 RecalcMapZoC(P);
2232 UpdateUnitMap(ToLoc, MoveInfo.MoveType = mtCapture);
2233 for I := 0 to nUpdateLoc - 1 do
2234 UpdateUnitMap(UpdateLoc[I]);
2235 // tell about lost units of defender
2236
2237 if (MoveInfo.MoveType <> mtSpyMission) and (Master < 0) then
2238 begin
2239 if (PModel.Kind = mkDiplomat) or (PModel.Domain = dAir) or
2240 (PModel.Cap[mcRadar] + PModel.Cap[mcCarrier] + PModel.Cap[mcAcademy] >
2241 0) or (RealMap[ToLoc] and fTerrain = fMountains) or
2242 (RealMap[ToLoc] and fTerImp = tiFort) or
2243 (RealMap[ToLoc] and fTerImp = tiBase) then
2244 ExtDiscover := True;
2245 if (PModel.Kind = mkDiplomat) or (PModel.Cap[mcSpy] > 0) then
2246 I := lObserveSuper
2247 else if (PModel.Domain = dAir) or
2248 (PModel.Cap[mcRadar] + PModel.Cap[mcCarrier] > 0) then
2249 I := lObserveAll
2250 else
2251 I := lObserveUnhidden;
2252 if ExtDiscover then
2253 begin
2254 if Discover21(ToLoc, P, I, True, PModel.Domain = dGround) then
2255 Result := Result or rEnemySpotted;
2256 end
2257 else
2258 begin
2259 if Discover9(ToLoc, P, I, True, PModel.Domain = dGround) then
2260 Result := Result or rEnemySpotted;
2261 end;
2262 end;
2263
2264 if Mode >= moMovie then { show after-move in interface modules }
2265 for p1 := 0 to nPl - 1 do
2266 if (1 shl p1 and GWatching <> 0) and ((p1 <> P) or (bix[p1].Kind = btTerm))
2267 then
2268 begin
2269 if PModel.Cap[mcStealth] > 0 then
2270 MinLevel := lObserveSuper
2271 else if PModel.Cap[mcSub] > 0 then
2272 MinLevel := lObserveAll
2273 else
2274 MinLevel := lObserveUnhidden;
2275 SeeFrom := (p1 = P) or (ObserveLevel[FromLoc] shr (2 * p1) and
2276 3 >= MinLevel);
2277 SeeTo := (p1 = P) or (ObserveLevel[ToLoc] shr (2 * p1) and
2278 3 >= MinLevel);
2279 if SeeTo and (MoveInfo.MoveType = mtCapture) then
2280 CallPlayer(cShowCityChanged, p1, ToLoc);
2281 if SeeFrom and SeeTo then
2282 CallPlayer(cShowAfterMove, p1, ToLoc)
2283 else if (MoveInfo.MoveType <> mtSpyMission) and SeeTo then
2284 CallPlayer(cShowUnitChanged, p1, ToLoc);
2285 for I := 0 to nUpdateLoc - 1 do
2286 if ObserveLevel[UpdateLoc[I]] shr (2 * p1) and 3 >= lObserveUnhidden
2287 then
2288 CallPlayer(cShowUnitChanged, p1, UpdateLoc[I]);
2289 end;
2290 end;
2291end;
2292
2293function ExecuteAttack(P, uix, ToLoc: Integer; var MoveInfo: TMoveInfo;
2294 ShowMove: TShowMove): Integer;
2295
2296 procedure WriteBattleHistory(ToLoc, FromLoc, Attacker, Defender, mixAttacker,
2297 mixDefender: Integer; AttackerLost, DefenderLost: Boolean);
2298 var
2299 AttackerBattle, DefenderBattle: ^TBattle;
2300 begin
2301 with RW[Attacker] do
2302 begin
2303 if nBattleHistory = 0 then
2304 ReallocMem(BattleHistory, 16 * SizeOf(TBattle))
2305 else if (nBattleHistory >= 16) and
2306 (nBattleHistory and (nBattleHistory - 1) = 0) then
2307 ReallocMem(BattleHistory, nBattleHistory * (2 * SizeOf(TBattle)));
2308 AttackerBattle := @BattleHistory[nBattleHistory];
2309 Inc(nBattleHistory);
2310 end;
2311 with RW[Defender] do
2312 begin
2313 if nBattleHistory = 0 then
2314 ReallocMem(BattleHistory, 16 * SizeOf(TBattle))
2315 else if (nBattleHistory >= 16) and
2316 (nBattleHistory and (nBattleHistory - 1) = 0) then
2317 ReallocMem(BattleHistory, nBattleHistory * (2 * SizeOf(TBattle)));
2318 DefenderBattle := @BattleHistory[nBattleHistory];
2319 Inc(nBattleHistory);
2320 end;
2321 AttackerBattle.Enemy := Defender;
2322 AttackerBattle.Flags := 0;
2323 AttackerBattle.Turn := GTurn;
2324 AttackerBattle.mix := mixAttacker;
2325 AttackerBattle.mixEnemy := mixDefender;
2326 AttackerBattle.ToLoc := ToLoc;
2327 AttackerBattle.FromLoc := FromLoc;
2328 DefenderBattle.Enemy := Attacker;
2329 DefenderBattle.Flags := bhEnemyAttack;
2330 DefenderBattle.Turn := GTurn;
2331 DefenderBattle.mix := mixDefender;
2332 DefenderBattle.mixEnemy := mixAttacker;
2333 DefenderBattle.ToLoc := ToLoc;
2334 DefenderBattle.FromLoc := FromLoc;
2335 if AttackerLost then
2336 begin
2337 AttackerBattle.Flags := AttackerBattle.Flags or bhMyUnitLost;
2338 DefenderBattle.Flags := DefenderBattle.Flags or bhEnemyUnitLost;
2339 end;
2340 if DefenderLost then
2341 begin
2342 AttackerBattle.Flags := AttackerBattle.Flags or bhEnemyUnitLost;
2343 DefenderBattle.Flags := DefenderBattle.Flags or bhMyUnitLost;
2344 end;
2345 end;
2346
2347var
2348 I, p1, FromLoc, uix1, nUpdateLoc, ExpGain, ExpelToLoc, cix1: Integer;
2349 PModel: ^TModel;
2350 UpdateLoc: array [0 .. numax - 1] of Integer;
2351 LoseCityPop, CityDestroyed, SeeFrom, SeeTo, ZoCDefenderDestroyed: Boolean;
2352begin
2353 Result := 0;
2354 with RW[P].Un[uix] do
2355 begin
2356 PModel := @RW[P].Model[mix];
2357 FromLoc := Loc;
2358
2359 ShowMove.EndHealth := MoveInfo.EndHealth;
2360 ShowMove.EndHealthDef := MoveInfo.EndHealthDef;
2361 if MoveInfo.MoveType = mtAttack then
2362 WriteBattleHistory(ToLoc, FromLoc, P, MoveInfo.Defender, mix,
2363 RW[MoveInfo.Defender].Un[MoveInfo.Duix].mix, MoveInfo.EndHealth <= 0,
2364 MoveInfo.EndHealthDef <= 0);
2365
2366 { if RW[p].Treaty[MoveInfo.Defender]=trCeaseFire then
2367 begin
2368 if Mode>=moMovie then
2369 CallPlayer(cShowCancelTreaty,MoveInfo.Defender,P);
2370 CancelTreaty(P,MoveInfo.Defender)
2371 end; }
2372 if Mode >= moMovie then { show attack in interface modules }
2373 for p1 := 0 to nPl - 1 do
2374 if (1 shl p1 and GWatching <> 0) and ((p1 <> P) or (bix[p1].Kind = btTerm))
2375 then
2376 begin
2377 SeeFrom := ObserveLevel[FromLoc] shr (2 * p1) and
2378 3 >= lObserveUnhidden;
2379 SeeTo := ObserveLevel[ToLoc] shr (2 * p1) and 3 >= lObserveUnhidden;
2380 if SeeFrom and SeeTo then
2381 begin
2382 TellAboutModel(p1, P, mix);
2383 if p1 = P then
2384 ShowMove.emix := -1
2385 else
2386 ShowMove.emix := emixSafe(p1, P, mix);
2387 CallPlayer(cShowAttacking, p1, ShowMove);
2388 end;
2389 end;
2390
2391 LoseCityPop := False;
2392 if (RealMap[ToLoc] and fCity <> 0) and
2393 ((MoveInfo.MoveType = mtAttack) and (MoveInfo.EndHealthDef <= 0) or
2394 (MoveInfo.MoveType = mtBombard) and (BombardmentDestroysCity or
2395 (RW[MoveInfo.Defender].City[MoveInfo.Dcix].Size > 2))) then
2396 case PModel.Domain of
2397 dGround:
2398 LoseCityPop := (PModel.Cap[mcArtillery] > 0) or
2399 (RW[MoveInfo.Defender].City[MoveInfo.Dcix].Built[imWalls] = 0) and
2400 (Continent[ToLoc] <> GrWallContinent[MoveInfo.Defender]);
2401 dSea:
2402 LoseCityPop := RW[MoveInfo.Defender].City[MoveInfo.Dcix].Built
2403 [imCoastalFort] = 0;
2404 dAir:
2405 LoseCityPop := RW[MoveInfo.Defender].City[MoveInfo.Dcix].Built
2406 [imMissileBat] = 0;
2407 end;
2408 CityDestroyed := LoseCityPop and
2409 (RW[MoveInfo.Defender].City[MoveInfo.Dcix].Size <= 2);
2410
2411 if MoveInfo.MoveType = mtBombard then
2412 begin
2413 Assert(Movement >= 100);
2414 if PModel.Attack = 0 then
2415 Flags := Flags and not unBombsLoaded;
2416 Dec(Movement, 100);
2417 end
2418 else if MoveInfo.MoveType = mtExpel then
2419 begin
2420 Assert(Movement >= 100);
2421 Job := jNone;
2422 Flags := Flags and not unFortified;
2423 Dec(Movement, 100);
2424 end
2425 else
2426 begin
2427 Assert(MoveInfo.MoveType = mtAttack);
2428 if MoveInfo.EndHealth = 0 then
2429 RemoveUnit(P, uix, MoveInfo.Defender) // destroy attacker
2430 else
2431 begin // update attacker
2432 ExpGain := (Health - MoveInfo.EndHealth + 1) shr 1;
2433 if Exp + ExpGain > (nExp - 1) * ExpCost then
2434 Exp := (nExp - 1) * ExpCost
2435 else
2436 Inc(Exp, ExpGain);
2437 Health := MoveInfo.EndHealth;
2438 Job := jNone;
2439 if RW[MoveInfo.Defender].Model[RW[MoveInfo.Defender].Un[MoveInfo.Duix]
2440 .mix].Domain < dAir then
2441 Flags := Flags and not unBombsLoaded;
2442 Flags := Flags and not unFortified;
2443 if Movement > 100 then
2444 Dec(Movement, 100)
2445 else
2446 Movement := 0;
2447 end;
2448 end;
2449
2450 ZoCDefenderDestroyed := False;
2451 nUpdateLoc := 0;
2452 if MoveInfo.MoveType = mtExpel then
2453 with RW[MoveInfo.Defender], Un[MoveInfo.Duix] do
2454 begin // expel friendly unit
2455 if Home >= 0 then
2456 ExpelToLoc := City[Home].Loc
2457 else
2458 begin
2459 ExpelToLoc := City[0].Loc; // search destination for homeless units
2460 for cix1 := 1 to nCity - 1 do
2461 if (City[cix1].Loc >= 0) and
2462 ((ExpelToLoc < 0) or (City[cix1].Built[imPalace] > 0)) then
2463 ExpelToLoc := City[cix1].Loc;
2464 end;
2465 if ExpelToLoc >= 0 then
2466 begin
2467 FreeUnit(MoveInfo.Defender, MoveInfo.Duix);
2468 Loc := ExpelToLoc;
2469 PlaceUnit(MoveInfo.Defender, MoveInfo.Duix);
2470 UpdateLoc[nUpdateLoc] := Loc;
2471 Inc(nUpdateLoc);
2472 Flags := Flags or unWithdrawn;
2473 end;
2474 end
2475 else if (MoveInfo.MoveType = mtAttack) and (MoveInfo.EndHealthDef > 0) then
2476 with RW[MoveInfo.Defender].Un[MoveInfo.Duix] do
2477 begin // update defender
2478 ExpGain := (Health - MoveInfo.EndHealthDef + 1) shr 1;
2479 if Exp + ExpGain > (nExp - 1) * ExpCost then
2480 Exp := (nExp - 1) * ExpCost
2481 else
2482 Inc(Exp, ExpGain);
2483 Health := MoveInfo.EndHealthDef;
2484 end
2485 else
2486 begin // destroy defenders
2487 if MoveInfo.MoveType <> mtBombard then
2488 begin
2489 ZoCDefenderDestroyed := RW[MoveInfo.Defender].Model
2490 [RW[MoveInfo.Defender].Un[MoveInfo.Duix].mix].Flags and mdZOC <> 0;
2491 if ((RealMap[ToLoc] and fCity = 0) and
2492 (RealMap[ToLoc] and fTerImp <> tiBase) and
2493 (RealMap[ToLoc] and fTerImp <> tiFort)) or LoseCityPop and
2494 (RW[MoveInfo.Defender].City[MoveInfo.Dcix].Size = 2) then
2495 RemoveAllUnits(MoveInfo.Defender, ToLoc, P)
2496 { no city, base or fortress }
2497 else
2498 RemoveUnit(MoveInfo.Defender, MoveInfo.Duix, P);
2499 end;
2500
2501 if LoseCityPop then // city defender defeated -- shrink city
2502 if not CityDestroyed then
2503 CityShrink(MoveInfo.Defender, MoveInfo.Dcix)
2504 else
2505 begin
2506 for uix1 := 0 to RW[MoveInfo.Defender].nUn - 1 do
2507 with RW[MoveInfo.Defender].Un[uix1] do
2508 if (Loc >= 0) and (Home = MoveInfo.Dcix) then
2509 begin
2510 UpdateLoc[nUpdateLoc] := Loc;
2511 Inc(nUpdateLoc);
2512 end;
2513 // unit will be removed -- remember position and update for all players
2514 DestroyCity_TellPlayers(MoveInfo.Defender, MoveInfo.Dcix, False);
2515 CheckBorders(ToLoc, MoveInfo.Defender);
2516 RecalcPeaceMap(P);
2517 end;
2518 end;
2519
2520 if CityDestroyed and (nUpdateLoc > 0) then
2521 RecalcMapZoC(P)
2522 else if ZoCDefenderDestroyed then
2523 RecalcV8ZoC(P, ToLoc);
2524 UpdateUnitMap(FromLoc);
2525 UpdateUnitMap(ToLoc, LoseCityPop);
2526 for I := 0 to nUpdateLoc - 1 do
2527 UpdateUnitMap(UpdateLoc[I]);
2528 // tell about lost units of defender
2529
2530 if Mode >= moMovie then
2531 begin
2532 for I := 0 to RW[P].nEnemyModel - 1 do
2533 with RW[P].EnemyModel[I] do
2534 Lost := Destroyed[P, Owner, mix];
2535 for p1 := 0 to nPl - 1 do { show after-attack in interface modules }
2536 if (1 shl p1 and GWatching <> 0) and ((p1 <> P) or (bix[p1].Kind = btTerm))
2537 then
2538 begin
2539 SeeFrom := ObserveLevel[FromLoc] shr (2 * p1) and
2540 3 >= lObserveUnhidden;
2541 SeeTo := ObserveLevel[ToLoc] shr (2 * p1) and 3 >= lObserveUnhidden;
2542 if SeeTo and CityDestroyed then
2543 CallPlayer(cShowCityChanged, p1, ToLoc); // city was destroyed
2544 if SeeFrom and SeeTo then
2545 begin
2546 CallPlayer(cShowAfterAttack, p1, ToLoc);
2547 CallPlayer(cShowAfterAttack, p1, FromLoc);
2548 end
2549 else
2550 begin
2551 if SeeTo then
2552 CallPlayer(cShowUnitChanged, p1, ToLoc);
2553 if SeeFrom then
2554 CallPlayer(cShowUnitChanged, p1, FromLoc);
2555 end;
2556 if SeeTo and (MoveInfo.MoveType = mtExpel) and (ExpelToLoc >= 0) then
2557 CallPlayer(cShowUnitChanged, p1, ExpelToLoc);
2558 end;
2559 end;
2560 end;
2561end;
2562
2563function MoveUnit(P, uix, dx, dy: Integer; TestOnly: Boolean): Integer;
2564var
2565 ToLoc: Integer;
2566 MoveInfo: TMoveInfo;
2567 ShowMove: TShowMove;
2568begin
2569{$IFOPT O-}Assert(1 shl P and InvalidTreatyMap = 0); {$ENDIF}
2570 with RW[P].Un[uix] do
2571 begin
2572 ToLoc := dLoc(Loc, dx, dy);
2573 if (ToLoc < 0) or (ToLoc >= MapSize) then
2574 begin
2575 Result := eInvalid;
2576 Exit;
2577 end;
2578 Result := CalculateMove(P, uix, ToLoc, 3 - dy and 1, TestOnly, MoveInfo);
2579 if Result = eZOC_EnemySpotted then
2580 ZOCTile := ToLoc;
2581 if (Result >= rExecuted) and not TestOnly then
2582 begin
2583 ShowMove.dx := dx;
2584 ShowMove.dy := dy;
2585 ShowMove.FromLoc := Loc;
2586 ShowMove.mix := mix;
2587 ShowMove.Health := Health;
2588 ShowMove.Fuel := Fuel;
2589 ShowMove.Exp := Exp;
2590 ShowMove.Load := TroopLoad + AirLoad;
2591 ShowMove.Owner := P;
2592 if (TroopLoad > 0) or (AirLoad > 0) then
2593 ShowMove.Flags := unMulti
2594 else
2595 ShowMove.Flags := 0;
2596 case MoveInfo.MoveType of
2597 mtCapture:
2598 ShowMove.Flags := ShowMove.Flags or umCapturing;
2599 mtSpyMission:
2600 ShowMove.Flags := ShowMove.Flags or umSpyMission;
2601 mtBombard:
2602 ShowMove.Flags := ShowMove.Flags or umBombarding;
2603 mtExpel:
2604 ShowMove.Flags := ShowMove.Flags or umExpelling;
2605 end;
2606 case MoveInfo.MoveType of
2607 mtMove, mtCapture, mtSpyMission:
2608 Result := ExecuteMove(P, uix, ToLoc, MoveInfo, ShowMove) or Result;
2609 mtAttack, mtBombard, mtExpel:
2610 Result := ExecuteAttack(P, uix, ToLoc, MoveInfo, ShowMove) or Result;
2611 end;
2612 end;
2613 end;
2614end;
2615
2616function Server(Command, Player, Subject: Integer; var Data): Integer; stdcall;
2617
2618 function CountPrice(const Offer: TOffer; PriceType: Integer): Integer;
2619 var
2620 I: Integer;
2621 begin
2622 Result := 0;
2623 for I := 0 to Offer.nDeliver + Offer.nCost - 1 do
2624 if Offer.Price[I] and $FFFF0000 = Cardinal(PriceType) then
2625 Inc(Result);
2626 end;
2627
2628{ procedure UpdateBorderHelper;
2629 var
2630 X, Y, Loc, Loc1, dx, dy, ObserveMask: Integer;
2631 begin
2632 ObserveMask:=3 shl (2*pTurn);
2633 for X:=0 to lx-1 do for Y:=0 to ly shr 1-1 do
2634 begin
2635 Loc:=lx*(Y*2)+X;
2636 if ObserveLevel[Loc] and ObserveMask<>0 then
2637 begin
2638 for dy:=0 to 1 do for dx:=0 to 1 do
2639 begin
2640 Loc1:=(Loc+dx-1+lx) mod lx +lx*((Y+dy)*2-1);
2641 if (Loc1>=0) and (Loc1<MapSize)
2642 and (ObserveLevel[Loc1] and ObserveMask<>0) then
2643 if RealMap[Loc1] and $78000000=RealMap[Loc] and $78000000 then
2644 begin
2645 RW[pTurn].BorderHelper[Loc]:=RW[pTurn].BorderHelper[Loc] and not (1 shl (dy*2+dx));
2646 RW[pTurn].BorderHelper[Loc1]:=RW[pTurn].BorderHelper[Loc1] and not (8 shr (dy*2+dx))
2647 end
2648 else
2649 begin
2650 RW[pTurn].BorderHelper[Loc]:=RW[pTurn].BorderHelper[Loc] or (1 shl (dy*2+dx));
2651 RW[pTurn].BorderHelper[Loc1]:=RW[pTurn].BorderHelper[Loc1] or (8 shr (dy*2+dx));
2652 end
2653 end
2654 end
2655 end
2656 end; }
2657
2658const
2659 ptSelect = 0;
2660 ptTrGoods = 1;
2661 ptUn = 2;
2662 ptCaravan = 3;
2663 ptImp = 4;
2664 ptWonder = 6;
2665 ptShip = 7;
2666 ptInvalid = 8;
2667
2668 function ProjectType(Project: Integer): Integer;
2669 begin
2670 if Project and cpCompleted <> 0 then
2671 Result := ptSelect
2672 else if Project and (cpImp + cpIndex) = cpImp + imTrGoods then
2673 Result := ptTrGoods
2674 else if Project and cpImp = 0 then
2675 if RW[Player].Model[Project and cpIndex].Kind = mkCaravan then
2676 Result := ptCaravan
2677 else
2678 Result := ptUn
2679 else if Project and cpIndex >= nImp then
2680 Result := ptInvalid
2681 else if Imp[Project and cpIndex].Kind = ikWonder then
2682 Result := ptWonder
2683 else if Imp[Project and cpIndex].Kind = ikShipPart then
2684 Result := ptShip
2685 else
2686 Result := ptImp;
2687 end;
2688
2689var
2690 D, I, J, p1, p2, pt0, pt1, uix1, cix1, Loc0, Loc1, dx, dy, NewCap, MinCap,
2691 MaxCap, CapWeight, Cost, NextProd, Preq, TotalFood, TotalProd, CheckSum,
2692 StopTurn, FutureMCost, NewProject, OldImp, mix, V8, V21, AStr, DStr,
2693 ABaseDamage, DBaseDamage: Integer;
2694 CityReport: TCityReport;
2695 FormerCLState: TCmdListState;
2696 Adjacent: TVicinity8Loc;
2697 Radius: TVicinity21Loc;
2698 ShowShipChange: TShowShipChange;
2699 ShowNegoData: TShowNegoData;
2700 Logged, Ok, HasShipChanged, AllHumansDead, OfferFullySupported: Boolean;
2701 Map: TMap;
2702begin
2703 if Command = sTurn then
2704 begin
2705 p2 := -1;
2706 for p1 := 0 to nPl - 1 do
2707 if (p1 <> Player) and (1 shl p1 and GWatching <> 0) then
2708 CallPlayer(cShowTurnChange, p1, p2);
2709 end;
2710
2711 Assert(MapSize = lx * ly);
2712 Assert(Command and (sctMask or sExecute) <> sctInternal or sExecute);
2713 // not for internal commands
2714 if (Command < 0) or (Command >= $10000) then
2715 begin
2716 Result := eUnknown;
2717 Exit;
2718 end;
2719
2720 if (Player < 0) or (Player >= nPl) or
2721 ((Command and (sctMask or sExecute) <> sctInfo) and
2722 ((Subject < 0) or (Subject >= $1000))) then
2723 begin
2724 Result := eInvalid;
2725 Exit;
2726 end;
2727
2728 if (1 shl Player and (GAlive or GWatching) = 0) and
2729 not ((Command = sTurn) or (Command = sBreak) or (Command = sResign) or
2730 (Command = sGetAIInfo) or (Command = sGetAICredits) or
2731 (Command = sGetVersion) or (Command and $FF0F = sGetChart)) then
2732 begin
2733 PutMessage(1 shl 16 + 1, Format('NOT Alive: %d', [Player]));
2734 Result := eNoTurn;
2735 Exit;
2736 end;
2737
2738 Result := eOK;
2739
2740 // check if command allowed now
2741 if (Mode = moPlaying) and not ((Command >= cClientEx) or (Command = sMessage)
2742 or (Command = sSetDebugMap) or (Command = sGetDebugMap) or
2743 (Command = sGetAIInfo) or (Command = sGetAICredits) or
2744 (Command = sGetVersion) or (Command = sGetTechCost) or
2745 (Command = sGetDefender) or (Command = sGetUnitReport) or
2746 (Command = sGetCityReport) or (Command = sGetCityTileInfo) or
2747 (Command = sGetCity) or (Command = sGetEnemyCityReport) or
2748 (Command = sGetEnemyCityAreaInfo) or (Command = sGetCityReportNew) or
2749 (Command and $FF0F = sGetChart) or (Command and $FF0F = sSetAttitude))
2750 // commands always allowed
2751 and not ((Player = pTurn) and (Command < $1000))
2752 // info request always allowed for pTurn
2753 and ((pDipActive < 0) and (Player <> pTurn) // not his turn
2754 or (pDipActive >= 0) and (Player <> pDipActive)
2755 // not active in negotiation mode
2756 or (pDipActive >= 0) and (Command and sctMask <> sctEndClient)) then
2757 // no nego command
2758 begin
2759 PutMessage(1 shl 16 + 1, Format('No Turn: %d calls %x',
2760 [Player, Command shr 4]));
2761 Result := eNoTurn;
2762 Exit;
2763 end;
2764
2765 // do not use EXIT hereafter!
2766
2767{$IFOPT O-}
2768 HandoverStack[nHandoverStack] := Player + $1000;
2769 HandoverStack[nHandoverStack + 1] := Command;
2770 Inc(nHandoverStack, 2);
2771
2772 InvalidTreatyMap := 0;
2773 // new command, sIntExpandTerritory of previous command was processed
2774{$ENDIF}
2775 if (Mode = moPlaying) and (Command >= sExecute) and
2776 ((Command and sctMask <> sctEndClient) or (Command = sTurn)) and
2777 (Command and sctMask <> sctModel) and (Command <> sCancelTreaty) and
2778 (Command <> sSetCityTiles) and (Command <> sBuyCityProject) and
2779 ((Command < cClientEx) or ProcessClientData[Player]) then
2780 begin { log command }
2781 FormerCLState := CL.State;
2782 CL.Put(Command, Player, Subject, @Data);
2783 Logged := True;
2784 end
2785 else
2786 Logged := False;
2787
2788 case Command of
2789
2790 {
2791 Info Request Commands
2792 ____________________________________________________________________
2793 }
2794 sMessage:
2795 bix[0].Client(cDebugMessage, Subject, Data);
2796
2797 sSetDebugMap:
2798 DebugMap[Player] := @Data;
2799
2800 sGetDebugMap:
2801 Pointer(Data) := DebugMap[Subject];
2802
2803 { sChangeSuperView:
2804 if Difficulty[Player]=0 then
2805 begin
2806 for I:=0 to nBrain-1 do if Brain[I].Initialized then
2807 CallClient(I, cShowSuperView, Subject)
2808 end
2809 else Result:=eInvalid; }
2810
2811 sRefreshDebugMap:
2812 bix[0].Client(cRefreshDebugMap, -1, Player);
2813
2814 sGetChart .. sGetChart + (nStat - 1) shl 4:
2815 if (Subject >= 0) and (Subject < nPl) and Assigned(bix[Subject]) then
2816 begin
2817 StopTurn := 0;
2818 if (Difficulty[Player] = 0) or (GTestFlags and tfUncover <> 0)
2819 // supervisor
2820 or (Subject = Player) // own chart
2821 or (GWinner > 0) // game end chart
2822 or (1 shl Subject and GAlive = 0) then // chart of extinct nation
2823 if Subject > Player then
2824 StopTurn := GTurn
2825 else
2826 StopTurn := GTurn + 1
2827 else if RW[Player].Treaty[Subject] > trNoContact then
2828 if Command shr 4 and $F = stMil then
2829 StopTurn := RW[Player].EnemyReport[Subject].TurnOfMilReport + 1
2830 else
2831 StopTurn := RW[Player].EnemyReport[Subject].TurnOfCivilReport + 1;
2832 Move(Stat[Command shr 4 and $F, Subject]^, Data,
2833 StopTurn * SizeOf(Integer));
2834 FillChar(TChart(Data)[StopTurn], (GTurn - StopTurn) *
2835 SizeOf(Integer), 0);
2836 end
2837 else
2838 Result := eInvalid;
2839
2840 sGetTechCost:
2841 Integer(Data) := TechCost(Player);
2842
2843 sGetAIInfo:
2844 if AIInfo[Subject] = '' then
2845 PChar(Data) := nil
2846 else
2847 PChar(Data) := @AIInfo[Subject][1];
2848
2849 sGetAICredits:
2850 if AICredits = '' then
2851 PChar(Data) := nil
2852 else
2853 PChar(Data) := @AICredits[1];
2854
2855 sGetVersion:
2856 Integer(Data) := CevoVersion;
2857
2858 sGetGameChanged:
2859 if Player <> 0 then
2860 Result := eInvalid
2861 else if (CL <> nil) and (CL.State.nLog = nLogOpened) and
2862 (CL.State.MoveCode = 0) and not HasCityTileChanges and
2863 not HasChanges(Player) then
2864 Result := eNotChanged;
2865
2866 sGetTileInfo:
2867 if (Subject >= 0) and (Subject < MapSize) then
2868 Result := GetTileInfo(Player, -2, Subject, TTileInfo(Data))
2869 else
2870 Result := eInvalid;
2871
2872 sGetCityTileInfo:
2873 if (Subject >= 0) and (Subject < MapSize) then
2874 Result := GetTileInfo(Player, -1, Subject, TTileInfo(Data))
2875 else
2876 Result := eInvalid;
2877
2878 sGetHypoCityTileInfo:
2879 if (Subject >= 0) and (Subject < MapSize) then
2880 begin
2881 if (TTileInfo(Data).ExplCity < 0) or
2882 (TTileInfo(Data).ExplCity >= RW[Player].nCity) then
2883 Result := eInvalid
2884 else if ObserveLevel[Subject] shr (Player * 2) and 3 = 0 then
2885 Result := eNoPreq
2886 else
2887 Result := GetTileInfo(Player, TTileInfo(Data).ExplCity, Subject,
2888 TTileInfo(Data));
2889 end
2890 else
2891 Result := eInvalid;
2892
2893 sGetJobProgress:
2894 if (Subject >= 0) and (Subject < MapSize) then
2895 begin
2896 if ObserveLevel[Subject] shr (Player * 2) and 3 = 0 then
2897 Result := eNoPreq
2898 else
2899 Result := GetJobProgress(Player, Subject, TJobProgressData(Data));
2900 end
2901 else
2902 Result := eInvalid;
2903
2904 sGetModels:
2905 if (GTestFlags and tfUncover <> 0) or (Difficulty[Player] = 0)
2906 then { supervisor only command }
2907 begin
2908 for p1 := 0 to nPl - 1 do
2909 if (p1 <> Player) and (1 shl p1 and GAlive <> 0) then
2910 for mix := 0 to RW[p1].nModel - 1 do
2911 TellAboutModel(Player, p1, mix);
2912 end
2913 else
2914 Result := eInvalid;
2915
2916 sGetUnits:
2917 if (Subject >= 0) and (Subject < MapSize) and
2918 (ObserveLevel[Subject] shr (Player * 2) and 3 = lObserveSuper) then
2919 Integer(Data) := GetUnitStack(Player, Subject)
2920 else
2921 Result := eNoPreq;
2922
2923 sGetDefender:
2924 if (Subject >= 0) and (Subject < MapSize) and (Occupant[Subject] = Player)
2925 then
2926 Strongest(Subject, Integer(Data), D, I, J)
2927 else
2928 Result := eInvalid;
2929
2930 sGetBattleForecast, sGetBattleForecastEx:
2931 if (Subject >= 0) and (Subject < MapSize) and
2932 (ObserveLevel[Subject] and (3 shl (Player * 2)) > 0) then
2933 with TBattleForecast(Data) do
2934 if (1 shl pAtt and GAlive <> 0) and (mixAtt >= 0) and
2935 (mixAtt < RW[pAtt].nModel) and
2936 ((pAtt = Player) or (RWemix[Player, pAtt, mixAtt] >= 0)) then
2937 begin
2938 Result := GetBattleForecast(Subject, TBattleForecast(Data), uix1,
2939 cix1, AStr, DStr, ABaseDamage, DBaseDamage);
2940 if Command = sGetBattleForecastEx then
2941 begin
2942 TBattleForecastEx(Data).AStr := (AStr + 200) div 400;
2943 TBattleForecastEx(Data).DStr := (DStr + 200) div 400;
2944 TBattleForecastEx(Data).ABaseDamage := ABaseDamage;
2945 TBattleForecastEx(Data).DBaseDamage := DBaseDamage;
2946 end;
2947 if Result = eOK then
2948 Result := eInvalid; // no enemy unit there!
2949 end
2950 else
2951 Result := eInvalid
2952 else
2953 Result := eInvalid;
2954
2955 sGetUnitReport:
2956 if (Subject < 0) or (Subject >= RW[Player].nUn) or
2957 (RW[Player].Un[Subject].Loc < 0) then
2958 Result := eInvalid
2959 else
2960 GetUnitReport(Player, Subject, TUnitReport(Data));
2961
2962 sGetMoveAdvice:
2963 if (Subject < 0) or (Subject >= RW[Player].nUn) or
2964 (RW[Player].Un[Subject].Loc < 0) then
2965 Result := eInvalid
2966 else
2967 Result := GetMoveAdvice(Player, Subject, TMoveAdviceData(Data));
2968
2969 sGetPlaneReturn:
2970 if (Subject < 0) or (Subject >= RW[Player].nUn) or
2971 (RW[Player].Un[Subject].Loc < 0) or
2972 (RW[Player].Model[RW[Player].Un[Subject].mix].Domain <> dAir) then
2973 Result := eInvalid
2974 else
2975 begin
2976 if CanPlaneReturn(Player, Subject, TPlaneReturnData(Data)) then
2977 Result := eOK
2978 else
2979 Result := eNoWay;
2980 end;
2981
2982 sGetCity:
2983 if (Subject >= 0) and (Subject < MapSize) and
2984 (ObserveLevel[Subject] shr (Player * 2) and 3 = lObserveSuper) and
2985 (RealMap[Subject] and fCity <> 0) then
2986 with TGetCityData(Data) do
2987 begin
2988 Owner := Player;
2989 SearchCity(Subject, Owner, cix1);
2990 C := RW[Owner].City[cix1];
2991 if (Owner <> Player) and (C.Project and cpImp = 0) then
2992 TellAboutModel(Player, Owner, C.Project and cpIndex);
2993 end
2994 else
2995 Result := eInvalid;
2996
2997 sGetCityReport:
2998 if (Subject < 0) or (Subject >= RW[Player].nCity) or
2999 (RW[Player].City[Subject].Loc < 0) then
3000 Result := eInvalid
3001 else
3002 Result := GetCityReport(Player, Subject, TCityReport(Data));
3003
3004 sGetCityReportNew:
3005 if (Subject < 0) or (Subject >= RW[Player].nCity) or
3006 (RW[Player].City[Subject].Loc < 0) then
3007 Result := eInvalid
3008 else
3009 GetCityReportNew(Player, Subject, TCityReportNew(Data));
3010
3011 sGetCityAreaInfo:
3012 if (Subject < 0) or (Subject >= RW[Player].nCity) or
3013 (RW[Player].City[Subject].Loc < 0) then
3014 Result := eInvalid
3015 else
3016 GetCityAreaInfo(Player, RW[Player].City[Subject].Loc,
3017 TCityAreaInfo(Data));
3018
3019 sGetEnemyCityReport:
3020 if (Subject >= 0) and (Subject < MapSize) and
3021 (ObserveLevel[Subject] shr (Player * 2) and 3 = lObserveSuper) and
3022 (RealMap[Subject] and fCity <> 0) then
3023 begin
3024 p1 := Occupant[Subject];
3025 if p1 < 0 then
3026 p1 := 1;
3027 SearchCity(Subject, p1, cix1);
3028 TCityReport(Data).HypoTiles := -1;
3029 TCityReport(Data).HypoTax := -1;
3030 TCityReport(Data).HypoLux := -1;
3031 GetCityReport(p1, cix1, TCityReport(Data));
3032 end
3033 else
3034 Result := eInvalid;
3035
3036 sGetEnemyCityReportNew:
3037 if (Subject >= 0) and (Subject < MapSize) and
3038 (ObserveLevel[Subject] shr (Player * 2) and 3 = lObserveSuper) and
3039 (RealMap[Subject] and fCity <> 0) then
3040 begin
3041 p1 := Occupant[Subject];
3042 if p1 < 0 then
3043 p1 := 1;
3044 SearchCity(Subject, p1, cix1);
3045 TCityReport(Data).HypoTiles := -1;
3046 TCityReport(Data).HypoTax := -1;
3047 TCityReport(Data).HypoLux := -1;
3048 GetCityReportNew(p1, cix1, TCityReportNew(Data));
3049 end
3050 else
3051 Result := eInvalid;
3052
3053 sGetEnemyCityAreaInfo:
3054 if (Subject >= 0) and (Subject < MapSize) and
3055 (ObserveLevel[Subject] shr (Player * 2) and 3 = lObserveSuper) and
3056 (RealMap[Subject] and fCity <> 0) then
3057 begin
3058 p1 := Occupant[Subject];
3059 if p1 < 0 then
3060 p1 := 1;
3061 SearchCity(Subject, p1, cix1);
3062 GetCityAreaInfo(p1, Subject, TCityAreaInfo(Data));
3063 end
3064 else
3065 Result := eInvalid;
3066
3067 sGetCityTileAdvice:
3068 if (Subject < 0) or (Subject >= RW[Player].nCity) or
3069 (RW[Player].City[Subject].Loc < 0) then
3070 Result := eInvalid
3071 else
3072 GetCityTileAdvice(Player, Subject, TCityTileAdviceData(Data));
3073
3074 {
3075 Map Editor Commands
3076 ____________________________________________________________________
3077 }
3078 sEditTile:
3079 if Player = 0 then
3080 with TEditTileData(Data) do
3081 EditTile(Loc, NewTile)
3082 else
3083 Result := eInvalid;
3084
3085 sRandomMap:
3086 if (Player = 0) and MapGeneratorAvailable then begin
3087 CreateElevation;
3088 PreviewElevation := False;
3089 CreateMap(False);
3090 FillChar(ObserveLevel, MapSize * 4, 0);
3091 DiscoverAll(Player, lObserveSuper);
3092 end else Result := eInvalid;
3093
3094 sFillMap:
3095 if Player = 0 then begin
3096 FillMap(TFillMapData(Data).Tile, True);
3097 FillChar(ObserveLevel, MapSize * 4, 0);
3098 DiscoverAll(Player, lObserveSuper);
3099 end else Result := eInvalid;
3100
3101 sMapGeneratorRequest:
3102 if not MapGeneratorAvailable then
3103 Result := eInvalid;
3104
3105 {
3106 Client Deactivation Commands
3107 ____________________________________________________________________
3108 }
3109 sTurn, sTurn - sExecute:
3110 begin
3111 AllHumansDead := True;
3112 for p1 := 0 to nPl - 1 do
3113 if (1 shl p1 and GAlive <> 0) and (bix[p1].Kind = btTerm) then begin
3114 AllHumansDead := False;
3115 Break;
3116 end;
3117 if (pDipActive >= 0) // still in negotiation mode
3118 or (pTurn = 0) and ((GWinner > 0) or (GTurn = MaxTurn) or
3119 (Difficulty[0] > 0) and AllHumansDead) then // game end reached
3120 Result := eViolation
3121 else if Command >= sExecute then
3122 begin
3123 if Mode = moPlaying then
3124 begin
3125 CL.State := FormerCLState;
3126 LogCityTileChanges;
3127{$IFNDEF SCR}
3128 if pTurn = 0 then
3129 begin
3130 LogChanges;
3131 SaveGame(ToAutoSaveFileName(LogFileName), True);
3132 end;
3133{$ENDIF}
3134 end
3135 else if (Mode = moMovie) and (pTurn = 0) then
3136 CallPlayer(cMovieEndTurn, 0, nil^);
3137 GWatching := GWatching and GAlive or 1;
3138 RW[pTurn].Happened := 0;
3139 uixSelectedTransport := -1;
3140 SpyMission := smSabotageProd;
3141 if 1 shl pTurn and GAlive <> 0 then
3142 begin
3143 // calculate checksum
3144 TotalFood := 0;
3145 TotalProd := 0;
3146 for I := 0 to RW[pTurn].nCity - 1 do
3147 if RW[pTurn].City[I].Loc >= 0 then
3148 begin
3149 Inc(TotalFood, RW[pTurn].City[I].Food);
3150 Inc(TotalProd, RW[pTurn].City[I].Prod);
3151 end;
3152 CheckSum := TotalFood and 7 + TotalProd and 7 shl 3 +
3153 RW[pTurn].Money and 7 shl 6 + Worked[pTurn] div 100 and 7 shl 9;
3154 end
3155 else
3156 CheckSum := 0;
3157
3158 if Mode < moPlaying then // check checksum
3159 begin
3160 if CheckSum <> Subject then
3161 LoadOK := False;
3162 end
3163 else // save checksum
3164 CL.Put(Command, Player, CheckSum, @Data);
3165{$IFDEF TEXTLOG}
3166 CmdInfo := '';
3167 if CheckSum and 7 <> Subject and 7 then
3168 CmdInfo := Format('***ERROR (Food %d) ',
3169 [(CheckSum and 7 - Subject and 7 + 12) mod 8 - 4]) + CmdInfo;
3170 if CheckSum shr 3 and 7 <> Subject shr 3 and 7 then
3171 CmdInfo := '***ERROR (Prod) ' + CmdInfo;
3172 if CheckSum shr 6 and 7 <> Subject shr 6 and 7 then
3173 CmdInfo := '***ERROR (Research) ' + CmdInfo;
3174 if CheckSum shr 9 and 7 <> Subject shr 9 and 7 then
3175 CmdInfo := '***ERROR (Work) ' + CmdInfo;
3176{$ENDIF}
3177 if 1 shl pTurn and GAlive <> 0 then
3178 begin
3179 AfterTurn;
3180 if Mode < moPlaying then
3181 InsertTerritoryUpdateCommands;
3182 // if bix[pTurn]=bixTerm then UpdateBorderHelper;
3183 end;
3184
3185 repeat
3186 pTurn := (pTurn + 1) mod nPl;
3187 if pTurn = 0 then
3188 Inc(GTurn);
3189 if Assigned(bix[pTurn]) and ((1 shl pTurn) and GAlive = 0) then
3190 begin // already made extinct -- continue statistics
3191 Stat[stExplore, pTurn, GTurn] := 0;
3192 Stat[stPop, pTurn, GTurn] := 0;
3193 Stat[stTerritory, pTurn, GTurn] := 0;
3194 Stat[stScience, pTurn, GTurn] := 0;
3195 Stat[stWork, pTurn, GTurn] := 0;
3196 Stat[stMil, pTurn, GTurn] := 0;
3197 end;
3198 until (pTurn = 0) or ((1 shl pTurn and (GAlive or GWatching) <> 0) and
3199 (GWinner = 0));
3200 if (Mode = moLoading_Fast) and
3201 ((GTurn = LoadTurn) or (GTurn = LoadTurn - 1) and (pTurn > 0)) then
3202 Mode := moLoading;
3203 if Mode = moPlaying then
3204 begin
3205 CCCommand := cTurn;
3206 CCPlayer := pTurn;
3207 Notify(ntNextPlayer);
3208 end
3209 else
3210 begin
3211 if GTurn = 0 then
3212 BeforeTurn0
3213 else
3214 BeforeTurn;
3215 if (Mode = moMovie) and (pTurn = 0) then
3216 begin
3217 Inform(pTurn);
3218 CallPlayer(cMovieTurn, 0, nil^);
3219 end;
3220 end;
3221{$IFDEF TEXTLOG}CmdInfo := CmdInfo + Format('---Turn %d P%d---', [GTurn, pTurn]); {$ENDIF}
3222 end;
3223 end; // sTurn
3224
3225 sBreak, sResign, sNextRound, sReload:
3226 if Mode = moMovie then
3227 MovieStopped := True
3228 else
3229 begin
3230 if Command = sReload then
3231 begin
3232 Ok := (Difficulty[0] = 0) and (bix[0].Kind <> btNoTerm) and
3233 (Integer(Data) >= 0) and (Integer(Data) < GTurn);
3234 for p1 := 1 to nPl - 1 do
3235 if bix[p1].Kind = btTerm then begin
3236 Ok := False;
3237 Break;
3238 end;
3239 // allow reload in AI-only games only
3240 end
3241 else
3242 Ok := Player = 0;
3243 if Ok then
3244 begin
3245 if (Command = sBreak) or (Command = sResign) then
3246 Notify(ntBackOn);
3247 for I := 0 to Brains.Count - 1 do
3248 if Brains[I].Initialized then
3249 begin
3250 if Brains[I].Kind = btAI then
3251 Notify(ntDeinitModule, I);
3252 CallClient(I, cBreakGame, nil^);
3253 end;
3254 Notify(ntEndInfo);
3255 if (Command = sBreak) or (Command = sReload) then
3256 begin
3257 LogCityTileChanges;
3258 LogChanges;
3259 SaveGame(LogFileName, False);
3260 end;
3261 DeleteFile(ToAutoSaveFileName(LogFileName));
3262 EndGame;
3263 case Command of
3264 sBreak:
3265 Notify(ntStartGoRefresh);
3266 sResign:
3267 Notify(ntStartGo);
3268 sNextRound:
3269 StartNewGame(LogFileName, MapFileName, lx, ly, LandMass, MaxTurn);
3270 sReload:
3271 LoadGame(LogFileName, Integer(Data), False);
3272 end;
3273 end
3274 else
3275 Result := eInvalid;
3276 end;
3277
3278 sAbandonMap, sSaveMap:
3279 if Player = 0 then
3280 begin
3281 if Command = sSaveMap then begin
3282 Map := TMap.Create;
3283 Map.Size := Point(lx, ly);
3284 Map.MaxTurn := MaxTurn;
3285 Move(RealMap, Map.Tiles[0], MapSize * 4);
3286 Map.SaveToFile(MapFileName);
3287 FreeAndNil(Map);
3288 end;
3289 Notify(ntBackOn);
3290 BrainTerm.Client(cBreakGame, -1, nil^);
3291 ReleaseMapEditor;
3292 if Command = sSaveMap then
3293 Notify(ntStartGoRefreshMaps)
3294 else
3295 Notify(ntStartGo);
3296 end
3297 else
3298 Result := eInvalid;
3299
3300 scContact .. scContact + (nPl - 1) shl 4, scContact - sExecute .. scContact
3301 - sExecute + (nPl - 1) shl 4:
3302 if (pDipActive >= 0) or (1 shl (Command shr 4 and $F) and GAlive = 0) then
3303 Result := eInvalid
3304 else if GWinner > 0 then
3305 Result := eViolation // game end reached
3306 else if RW[Player].Treaty[Command shr 4 and $F] = trNoContact then
3307 Result := eNoPreq
3308 else if GTurn < GColdWarStart + ColdWarTurns then
3309 Result := eColdWar
3310 else if RW[Player].Government = gAnarchy then
3311 Result := eAnarchy
3312 else if RW[Command shr 4 and $F].Government = gAnarchy then
3313 begin
3314 Result := eAnarchy;
3315 LastEndClientCommand := scReject; // enable cancel treaty
3316 pContacted := Command shr 4 and $F;
3317 end
3318 else if Command >= sExecute then
3319 begin // contact request
3320 pContacted := Command shr 4 and $F;
3321 pDipActive := pContacted;
3322 Assert(Mode = moPlaying);
3323 Inform(pDipActive);
3324 ChangeClientWhenDone(scContact, pDipActive, pTurn, 4);
3325 end;
3326
3327 scReject, scReject - sExecute:
3328 if LastEndClientCommand and $FF0F = scContact then
3329 begin
3330 if Command >= sExecute then
3331 begin // contact requested and not accepted yet
3332 pDipActive := -1;
3333 Assert(Mode = moPlaying);
3334 ChangeClientWhenDone(cContinue, pTurn, nil^, 0);
3335 end;
3336 end
3337 else
3338 Result := eInvalid;
3339
3340 scDipStart, scDipStart - sExecute:
3341 if LastEndClientCommand and $FF0F = scContact then
3342 begin
3343 if Command >= sExecute then
3344 begin // accept contact
3345 pContacted := pDipActive;
3346 RW[pContacted].EnemyReport[pTurn].Credibility :=
3347 RW[pTurn].Credibility;
3348 pDipActive := pTurn;
3349 Assert(Mode = moPlaying);
3350 IntServer(sIntHaveContact, pTurn, pContacted, nil^);
3351 ChangeClientWhenDone(scDipStart, pDipActive, nil^, 0);
3352 end;
3353 end
3354 else
3355 Result := eInvalid;
3356
3357 scDipNotice, scDipAccept, scDipCancelTreaty, scDipBreak,
3358 scDipNotice - sExecute, scDipAccept - sExecute,
3359 scDipCancelTreaty - sExecute, scDipBreak - sExecute:
3360 if pDipActive >= 0 then
3361 begin
3362 Assert(Mode = moPlaying);
3363 if pDipActive = pTurn then
3364 p1 := pContacted
3365 else
3366 p1 := pTurn;
3367 if (Command and not sExecute = scDipBreak and not sExecute) and
3368 (LastEndClientCommand <> scDipBreak) then // Ok
3369 else if (Command and not sExecute = scDipNotice and not sExecute) and
3370 ((LastEndClientCommand = scDipCancelTreaty) or
3371 (LastEndClientCommand = scDipBreak)) then // Ok
3372 else if (Command and not sExecute = scDipAccept and not sExecute) and
3373 (LastEndClientCommand = scDipOffer) then
3374 with LastOffer do
3375 begin
3376 // check if offer can be accepted
3377 if nDeliver + nCost = 0 then
3378 Result := eOfferNotAcceptable;
3379 for I := 0 to nDeliver + nCost - 1 do
3380 if Price[I] = opChoose then begin
3381 Result := eOfferNotAcceptable;
3382 Break;
3383 end;
3384 for I := 0 to nCost - 1 do
3385 if not PayPrice(pDipActive, p1, Price[nDeliver + I], False) then
3386 Result := eOfferNotAcceptable;
3387 if (Command >= sExecute) and (Result >= rExecuted) then
3388 begin
3389 IntServer(sIntPayPrices + nDeliver + nCost, p1, pDipActive,
3390 LastOffer);
3391 // CheckContact;
3392
3393 // tell other players about ship part trades
3394 HasShipChanged := False;
3395 FillChar(ShowShipChange, SizeOf(ShowShipChange), 0);
3396 for I := 0 to nDeliver + nCost - 1 do
3397 if Price[I] and opMask = opShipParts then
3398 begin
3399 HasShipChanged := True;
3400 if I >= nDeliver then
3401 begin // p1 has demanded from pDipActive
3402 ShowShipChange.Ship1Change[Price[I] shr 16 and 3] :=
3403 +Integer(Price[I] and $FFFF);
3404 ShowShipChange.Ship2Change[Price[I] shr 16 and 3] :=
3405 -Integer(Price[I] and $FFFF);
3406 end
3407 else
3408 begin // p1 has delivered to pDipActive
3409 ShowShipChange.Ship1Change[Price[I] shr 16 and 3] :=
3410 -Integer(Price[I] and $FFFF);
3411 ShowShipChange.Ship2Change[Price[I] shr 16 and 3] :=
3412 +Integer(Price[I] and $FFFF);
3413 end;
3414 end;
3415 if HasShipChanged then
3416 begin
3417 ShowShipChange.Reason := scrTrade;
3418 ShowShipChange.Ship1Owner := p1;
3419 ShowShipChange.Ship2Owner := pDipActive;
3420 for p2 := 0 to nPl - 1 do
3421 if (p2 <> p1) and (p2 <> pDipActive) and
3422 (1 shl p2 and (GAlive or GWatching) <> 0) then
3423 begin
3424 Move(GShip, RW[p2].Ship, SizeOf(GShip));
3425 if 1 shl p2 and GWatching <> 0 then
3426 CallPlayer(cShowShipChange, p2, ShowShipChange);
3427 end;
3428 end;
3429 end;
3430 end
3431 else if (Command and not sExecute = scDipCancelTreaty and not sExecute)
3432 and (RW[pDipActive].Treaty[p1] >= trPeace) then
3433 begin
3434 if (ServerVersion[pDipActive] >= $010100) and
3435 (GTurn < RW[pDipActive].LastCancelTreaty[p1] + CancelTreatyTurns)
3436 then
3437 Result := eCancelTreatyRush
3438 else if Command >= sExecute then
3439 begin
3440 IntServer(sIntCancelTreaty, pDipActive, p1, nil^);
3441 for p2 := 0 to nPl - 1 do
3442 if (p2 <> p1) and (1 shl p2 and PeaceEnded <> 0) then
3443 begin
3444 I := p1 shl 4 + pDipActive;
3445 CallPlayer(cShowSupportAllianceAgainst, p2, I);
3446 end;
3447 for p2 := 0 to nPl - 1 do
3448 if (p2 <> p1) and (1 shl p2 and PeaceEnded <> 0) then
3449 begin
3450 I := p2;
3451 CallPlayer(cShowCancelTreatyByAlliance, pDipActive, I);
3452 end;
3453 end;
3454 end
3455 else
3456 Result := eInvalid;
3457 if (Command >= sExecute) and (Result >= rExecuted) then
3458 if LastEndClientCommand = scDipBreak then
3459 begin // break negotiation
3460 pDipActive := -1;
3461 CallPlayer(cShowEndContact, pContacted, nil^);
3462 ChangeClientWhenDone(cContinue, pTurn, nil^, 0);
3463 end
3464 else
3465 begin
3466 if (GTestFlags and tfUncover <> 0) or (Difficulty[0] = 0) then
3467 with ShowNegoData do
3468 begin // display negotiation in log window
3469 pSender := pDipActive;
3470 pTarget := p1;
3471 Action := Command;
3472 bix[0].Client(cShowNego, 1 shl 16 + 3, ShowNegoData);
3473 end;
3474 pDipActive := p1;
3475 ChangeClientWhenDone(Command, pDipActive, nil^, 0);
3476 end;
3477 end
3478 else
3479 Result := eInvalid;
3480
3481 scDipOffer, scDipOffer - sExecute:
3482 if (pDipActive >= 0) and (LastEndClientCommand <> scDipCancelTreaty) and
3483 (LastEndClientCommand <> scDipBreak) then
3484 if (LastEndClientCommand = scDipOffer) and
3485 (LastOffer.nDeliver + LastOffer.nCost + TOffer(Data).nDeliver +
3486 TOffer(Data).nCost = 0) then
3487 begin
3488 if Command >= sExecute then
3489 begin // agreed discussion end
3490 pDipActive := -1;
3491 CallPlayer(cShowEndContact, pContacted, nil^);
3492 Assert(Mode = moPlaying);
3493 ChangeClientWhenDone(cContinue, pTurn, nil^, 0);
3494 end;
3495 end
3496 else
3497 begin
3498 // check if offer can be made
3499 if pDipActive = pTurn then
3500 p1 := pContacted
3501 else
3502 p1 := pTurn;
3503 if RW[pDipActive].Treaty[p1] < trPeace then
3504 begin // no tribute allowed!
3505 for I := 0 to TOffer(Data).nDeliver + TOffer(Data).nCost - 1 do
3506 if (TOffer(Data).Price[I] and opMask = opTribute) then begin
3507 Result := eInvalidOffer;
3508 Break;
3509 end;
3510 for I := 0 to TOffer(Data).nDeliver + TOffer(Data).nCost - 1 do
3511 if (TOffer(Data).Price[I] = opTreaty + trPeace) then begin
3512 Result := eOK;
3513 Break;
3514 end;
3515 end;
3516 for I := 0 to TOffer(Data).nDeliver - 1 do
3517 if (TOffer(Data).Price[I] <> opChoose) and
3518 not PayPrice(pDipActive, p1, TOffer(Data).Price[I], False) then
3519 Result := eInvalidOffer;
3520 if CountPrice(TOffer(Data), opTreaty) > 1 then
3521 Result := eInvalidOffer;
3522 for I := 0 to nShipPart - 1 do
3523 if CountPrice(TOffer(Data), opShipParts + I shl 16) > 1 then begin
3524 Result := eInvalidOffer;
3525 Break;
3526 end;
3527 if CountPrice(TOffer(Data), opMoney) > 1 then
3528 Result := eInvalidOffer;
3529 if CountPrice(TOffer(Data), opTribute) > 1 then
3530 Result := eInvalidOffer;
3531 case CountPrice(TOffer(Data), opChoose) of
3532 0:
3533 ;
3534 1:
3535 if (TOffer(Data).nCost = 0) or (TOffer(Data).nDeliver = 0) then
3536 Result := eInvalidOffer;
3537 else
3538 Result := eInvalidOffer;
3539 end;
3540
3541 // !!! check here if cost can be demanded
3542
3543 if (Command >= sExecute) and (Result >= rExecuted) then
3544 begin
3545 OfferFullySupported := (TOffer(Data).nDeliver <= 2) and
3546 (TOffer(Data).nCost <= 2); // >2 no more allowed
3547 for I := 0 to TOffer(Data).nDeliver + TOffer(Data).nCost - 1 do
3548 begin
3549 if TOffer(Data).Price[I] and opMask = opTribute then
3550 OfferFullySupported := False;
3551 // tribute no more part of the game
3552 if (TOffer(Data).Price[I] and opMask = opTreaty) and
3553 (TOffer(Data).Price[I] - opTreaty <= RW[pDipActive].Treaty[p1])
3554 then
3555 OfferFullySupported := False;
3556 // agreed treaty end no more part of the game
3557 if TOffer(Data).Price[I] = opTreaty + trCeaseFire then
3558 OfferFullySupported := False;
3559 // ceasefire no more part of the game
3560 end;
3561 if not OfferFullySupported then
3562 begin
3563 // some elements have been removed from the game -
3564 // automatically respond will null-offer
3565 LastOffer.nDeliver := 0;
3566 LastOffer.nCost := 0;
3567 ChangeClientWhenDone(scDipOffer, pDipActive, LastOffer,
3568 SizeOf(LastOffer));
3569 end
3570 else
3571 begin
3572 if (GTestFlags and tfUncover <> 0) or (Difficulty[0] = 0) then
3573 with ShowNegoData do
3574 begin // display negotiation in log window
3575 pSender := pDipActive;
3576 pTarget := p1;
3577 Action := Command;
3578 Offer := TOffer(Data);
3579 bix[0].Client(cShowNego, 1 shl 16 + 3, ShowNegoData);
3580 end;
3581 LastOffer := TOffer(Data);
3582 // show offered things to receiver
3583 for I := 0 to LastOffer.nDeliver - 1 do
3584 ShowPrice(pDipActive, p1, LastOffer.Price[I]);
3585 pDipActive := p1;
3586 Assert(Mode = moPlaying);
3587 ChangeClientWhenDone(scDipOffer, pDipActive, LastOffer,
3588 SizeOf(LastOffer));
3589 end;
3590 end;
3591 end
3592 else
3593 Result := eInvalid;
3594
3595 {
3596 General Commands
3597 ____________________________________________________________________
3598 }
3599 sClearTestFlag:
3600 if Player = 0 then
3601 begin
3602{$IFDEF TEXTLOG}CmdInfo := Format('ClearTestFlag %x', [Subject]); {$ENDIF}
3603 ClearTestFlags(Subject);
3604 end
3605 else
3606 Result := eInvalid;
3607
3608 sSetTestFlag:
3609 if Player = 0 then
3610 begin
3611{$IFDEF TEXTLOG}CmdInfo := Format('SetTestFlag %x', [Subject]); {$ENDIF}
3612 SetTestFlags(Player, Subject);
3613 // CheckContact;
3614 end
3615 else
3616 Result := eInvalid;
3617
3618 sSetGovernment, sSetGovernment - sExecute:
3619 begin
3620{$IFDEF TEXTLOG}CmdInfo := Format('SetGovernment P%d: %d', [Player, Subject]); {$ENDIF}
3621 if RW[Player].Happened and phChangeGov = 0 then
3622 Result := eViolation
3623 else if RW[Player].Government = Subject then
3624 Result := eNotChanged
3625 else if (Subject >= nGov) then
3626 Result := eInvalid
3627 else if (Subject >= gMonarchy) and
3628 (RW[Player].Tech[GovPreq[Subject]] < tsApplicable) then
3629 Result := eNoPreq
3630 else if Command >= sExecute then
3631 begin
3632 RW[Player].Government := Subject;
3633 for p1 := 0 to nPl - 1 do
3634 if (p1 <> Player) and ((GAlive or GWatching) and (1 shl p1) <> 0)
3635 then
3636 RW[p1].EnemyReport[Player].Government := Subject;
3637 end;
3638 end;
3639
3640 sSetRates, sSetRates - sExecute:
3641 begin
3642{$IFDEF TEXTLOG}CmdInfo := Format('SetRates P%d: %d/%d', [Player, Subject and $F * 10, Subject shr 4 * 10]); {$ENDIF}
3643 if Subject and $F + Subject shr 4 > 10 then
3644 Result := eInvalid
3645 else if (RW[Player].TaxRate = Subject and $F * 10) and
3646 (RW[Player].LuxRate = Subject shr 4 * 10) then
3647 Result := eNotChanged
3648 else if Command >= sExecute then
3649 begin
3650 RW[Player].TaxRate := Subject and $F * 10;
3651 RW[Player].LuxRate := Subject shr 4 * 10;
3652 end;
3653 end;
3654
3655 sRevolution:
3656 begin
3657{$IFDEF TEXTLOG}CmdInfo := Format('Revolution P%d', [Player]); {$ENDIF}
3658 if RW[Player].Government = gAnarchy then
3659 Result := eInvalid
3660 else
3661 begin
3662 RW[Player].Government := gAnarchy;
3663 for p1 := 0 to nPl - 1 do
3664 if (p1 <> Player) and ((GAlive or GWatching) and (1 shl p1) <> 0)
3665 then
3666 RW[p1].EnemyReport[Player].Government := gAnarchy;
3667 RW[Player].AnarchyStart := GTurn;
3668 end;
3669 end;
3670
3671 sSetResearch, sSetResearch - sExecute:
3672 with RW[Player] do
3673 begin
3674{$IFDEF TEXTLOG}CmdInfo := Format('SetResearch P%d: %d', [Player, Subject]);
3675 {$ENDIF}
3676 if (Happened and phTech <> 0) and
3677 ((Subject < nAdv) or (Subject = adMilitary)) then
3678 begin
3679 if (Mode = moPlaying) and (Subject = adMilitary) and
3680 (DevModelTurn[Player] <> GTurn) then
3681 Result := eNoModel
3682 else if Subject <> adMilitary then
3683 begin
3684 if Subject = futComputingTechnology then
3685 begin
3686 if Tech[Subject] >= MaxFutureTech_Computing then
3687 Result := eInvalid;
3688 end
3689 else if Subject in FutureTech then
3690 begin
3691 if Tech[Subject] >= MaxFutureTech then
3692 Result := eInvalid;
3693 end
3694 else if Tech[Subject] >= tsApplicable then
3695 Result := eInvalid; // already discovered
3696 if Tech[Subject] <> tsSeen then // look if preqs met
3697 if AdvPreq[Subject, 2] <> preNone then
3698 begin // 2 of 3 required
3699 I := 0;
3700 for J := 0 to 2 do
3701 if Tech[AdvPreq[Subject, J]] >= tsApplicable then
3702 Inc(I);
3703 if I < 2 then
3704 Result := eNoPreq;
3705 end
3706 else if (AdvPreq[Subject, 0] <> preNone) and
3707 (Tech[AdvPreq[Subject, 0]] < tsApplicable) or
3708 (AdvPreq[Subject, 1] <> preNone) and
3709 (Tech[AdvPreq[Subject, 1]] < tsApplicable) then
3710 Result := eNoPreq;
3711 end;
3712 if (Result = eOK) and (Command >= sExecute) then
3713 begin
3714 if (Mode = moPlaying) and (Subject = adMilitary) then
3715 IntServer(sIntSetDevModel, Player, 0, DevModel.Kind);
3716 // save DevModel, because sctModel commands are not Logged
3717 ResearchTech := Subject;
3718 end;
3719 end
3720 else
3721 Result := eViolation;
3722 end;
3723
3724 sStealTech, sStealTech - sExecute:
3725 begin
3726{$IFDEF TEXTLOG}CmdInfo := Format('StealTech P%d: %d', [Player, Subject]);
3727 {$ENDIF}
3728 if RW[Player].Happened and phStealTech = 0 then
3729 Result := eInvalid
3730 else if (Subject >= nAdv) or (Subject in FutureTech) or
3731 (RW[Player].Tech[Subject] >= tsSeen) or
3732 (RW[GStealFrom].Tech[Subject] < tsApplicable) then
3733 Result := eInvalid
3734 else if Command >= sExecute then
3735 begin
3736 SeeTech(Player, Subject);
3737 Dec(RW[Player].Happened, phStealTech);
3738 end;
3739 end;
3740
3741 sSetAttitude .. sSetAttitude + (nPl - 1) shl 4,
3742 sSetAttitude - sExecute .. sSetAttitude - sExecute + (nPl - 1) shl 4:
3743 begin
3744 p1 := Command shr 4 and $F;
3745{$IFDEF TEXTLOG}CmdInfo := Format('SetAttitude P%d to P%d: %d', [Player, p1, Subject]); {$ENDIF}
3746 if (Subject >= nAttitude) or (p1 >= nPl) or
3747 (RW[Player].EnemyReport[p1] = nil) then
3748 Result := eInvalid
3749 else if RW[Player].Treaty[p1] = trNoContact then
3750 Result := eNoPreq
3751 else if RW[Player].Attitude[p1] = Subject then
3752 Result := eNotChanged
3753 else if Command >= sExecute then
3754 begin
3755 RW[Player].Attitude[p1] := Subject;
3756 RW[p1].EnemyReport[Player].Attitude := Subject;
3757 end;
3758 end;
3759
3760 sCancelTreaty, sCancelTreaty - sExecute:
3761 if (LastEndClientCommand <> scReject) or
3762 (RW[Player].Treaty[pContacted] < trPeace) then
3763 Result := eInvalid
3764 else if (ServerVersion[Player] >= $010100) and
3765 (GTurn < RW[Player].LastCancelTreaty[pContacted] + CancelTreatyTurns)
3766 then
3767 Result := eCancelTreatyRush
3768 else if Command >= sExecute then
3769 begin
3770 CallPlayer(cShowCancelTreaty, pContacted, Player);
3771 IntServer(sIntCancelTreaty, Player, pContacted, nil^);
3772 for p2 := 0 to nPl - 1 do
3773 if (p2 <> pContacted) and (1 shl p2 and PeaceEnded <> 0) then
3774 begin
3775 I := pContacted shl 4 + Player;
3776 CallPlayer(cShowSupportAllianceAgainst, p2, I);
3777 end;
3778 for p2 := 0 to nPl - 1 do
3779 if (p2 <> pContacted) and (1 shl p2 and PeaceEnded <> 0) then
3780 begin
3781 I := p2;
3782 CallPlayer(cShowCancelTreatyByAlliance, Player, I);
3783 end;
3784 LastEndClientCommand := sTurn;
3785 end;
3786
3787 {
3788 Model Related Commands
3789 ____________________________________________________________________
3790 }
3791 sCreateDevModel, sCreateDevModel - sExecute:
3792 begin
3793{$IFDEF TEXTLOG}CmdInfo := Format('CreateDevModel P%d', [Player]); {$ENDIF}
3794 if Subject >= 4 then
3795 Result := eInvalid
3796 else if (upgrade[Subject, 0].Preq <> preNone) and
3797 (RW[Player].Tech[upgrade[Subject, 0].Preq] < tsApplicable) then
3798 Result := eNoPreq
3799 else if Command >= sExecute then
3800 begin
3801 with RW[Player].DevModel do
3802 begin
3803 Domain := Subject;
3804 MStrength := 0;
3805 MTrans := 0;
3806 MCost := 0;
3807 Upgrades := 0;
3808 FutureMCost := 0;
3809 for I := 0 to nUpgrade - 1 do
3810 with upgrade[Domain, I] do
3811 if (Preq = preNone) or (Preq >= 0) and
3812 ((RW[Player].Tech[Preq] >= tsApplicable) or
3813 (Preq in FutureTech) and (RW[Player].Tech[Preq] >= 0)) then
3814 begin
3815 if Preq in FutureTech then
3816 begin
3817 J := RW[Player].Tech[Preq];
3818 Inc(FutureMCost, J * Cost);
3819 end
3820 else
3821 begin
3822 J := 1;
3823 if Cost > MCost then
3824 MCost := Cost;
3825 end;
3826 Inc(Upgrades, 1 shl I);
3827 Inc(MStrength, J * Strength);
3828 Inc(MTrans, J * Trans);
3829 end;
3830 Inc(MCost, FutureMCost);
3831 FillChar(Cap, SizeOf(Cap), 0);
3832 Cap[mcOffense] := 2;
3833 Cap[mcDefense] := 1;
3834 for I := 0 to nFeature - 1 do
3835 with Feature[I] do
3836 if (1 shl Domain and Domains <> 0) and
3837 ((Preq = preNone) or (Preq = preSun) and
3838 (GWonder[woSun].EffectiveOwner = Player) or (Preq >= 0) and
3839 (RW[Player].Tech[Preq] >= tsApplicable)) and (I in AutoFeature)
3840 then
3841 Cap[I] := 1;
3842 MaxWeight := 5;
3843 if (WeightPreq7[Domain] <> preNA) and
3844 (RW[Player].Tech[WeightPreq7[Domain]] >= tsApplicable) then
3845 MaxWeight := 7;
3846 if (WeightPreq10[Domain] <> preNA) and
3847 (RW[Player].Tech[WeightPreq10[Domain]] >= tsApplicable) then
3848 if Domain = dSea then
3849 MaxWeight := 9
3850 else
3851 MaxWeight := 10;
3852 end;
3853 CalculateModel(RW[Player].DevModel);
3854 DevModelTurn[Player] := GTurn;
3855 end
3856 end;
3857
3858 sSetDevModelCap .. sSetDevModelCap + $3F0,
3859 sSetDevModelCap - sExecute .. sSetDevModelCap - sExecute + $3F0:
3860 begin
3861{$IFDEF TEXTLOG}CmdInfo := Format('SetDevModelCap P%d', [Player]); {$ENDIF}
3862 if Subject >= nFeature then
3863 Result := eInvalid
3864 else if DevModelTurn[Player] = GTurn then
3865 begin
3866 NewCap := Command shr 4 and $3F; { new value }
3867 with RW[Player].DevModel do
3868 if 1 shl Domain and Feature[Subject].Domains = 0 then
3869 Result := eDomainMismatch
3870 else if not ((Feature[Subject].Preq = preNone) or
3871 (Feature[Subject].Preq = preSun) and
3872 (GWonder[woSun].EffectiveOwner = Player) or
3873 (Feature[Subject].Preq >= 0) and
3874 (RW[Player].Tech[Feature[Subject].Preq] >= tsApplicable)) then
3875 Result := eNoPreq
3876 else
3877 begin
3878 if (Subject in AutoFeature) or (Subject = mcDefense) then
3879 MinCap := 1
3880 else
3881 MinCap := 0; { MinCap - minimum use of feature }
3882 if Subject >= mcFirstNonCap then
3883 MaxCap := 1
3884 else if Subject = mcDefense then
3885 begin
3886 if Domain = dGround then
3887 MaxCap := 2
3888 else
3889 MaxCap := 3;
3890 if RW[Player].Tech[adSteel] >= tsApplicable then
3891 Inc(MaxCap);
3892 end
3893 else
3894 MaxCap := 8; { MaxCap - maximum use of this feature }
3895 if (Domain = dGround) and (Subject = mcDefense) then
3896 CapWeight := 2
3897 else
3898 CapWeight := Feature[Subject].Weight;
3899 if (NewCap < MinCap) or (NewCap > MaxCap) or
3900 (Weight + (NewCap - Cap[Subject]) * CapWeight > MaxWeight) then
3901 Result := eViolation
3902 else if Command >= sExecute then
3903 begin
3904 Cap[Subject] := NewCap;
3905
3906 // mutual feature exclusion
3907 case Subject of
3908 mcSub:
3909 begin
3910 if ServerVersion[Player] >= $010103 then
3911 Cap[mcSeaTrans] := 0;
3912 Cap[mcArtillery] := 0;
3913 Cap[mcCarrier] := 0;
3914 if Cap[mcDefense] > 2 then
3915 Cap[mcDefense] := 2;
3916 end;
3917 mcSeaTrans:
3918 begin
3919 if ServerVersion[Player] >= $010103 then
3920 Cap[mcSub] := 0;
3921 end;
3922 mcCarrier:
3923 Cap[mcSub] := 0;
3924 mcArtillery:
3925 Cap[mcSub] := 0;
3926 mcAlpine:
3927 begin
3928 Cap[mcOver] := 0;
3929 Cap[mcMob] := 0;
3930 end;
3931 mcOver:
3932 Cap[mcAlpine] := 0;
3933 mcMob:
3934 begin
3935 Cap[mcAlpine] := 0;
3936 end;
3937 end;
3938
3939 CalculateModel(RW[Player].DevModel);
3940 end;
3941 end;
3942 end
3943 else
3944 Result := eNoModel;
3945 end;
3946
3947 {
3948 Unit Related Commands
3949 ____________________________________________________________________
3950 }
3951 sRemoveUnit, sRemoveUnit - sExecute:
3952 begin
3953{$IFDEF TEXTLOG}CmdInfo := Format('RemoveUnit P%d Mod%d Loc%d', [Player, RW[Player].Un[Subject].mix, RW[Player].Un[Subject].Loc]); {$ENDIF}
3954 if (Subject >= RW[Player].nUn) or (RW[Player].Un[Subject].Loc < 0) then
3955 Result := eInvalid
3956 else
3957 begin
3958 Result := eRemoved;
3959 Loc0 := RW[Player].Un[Subject].Loc;
3960 if RealMap[Loc0] and fCity <> 0 then { check utilize }
3961 begin
3962 SearchCity(Loc0, Player, cix1);
3963 with RW[Player].City[cix1] do
3964 begin
3965 if (RW[Player].Model[RW[Player].Un[Subject].mix].Kind = mkCaravan)
3966 and ((Project and cpImp = 0) or
3967 (Imp[Project and cpIndex].Kind <> ikShipPart)) or
3968 (Project and cpImp = 0) and
3969 (RW[Player].Model[Project and cpIndex].Kind <> mkCaravan) then
3970 Result := eUtilized;
3971 if Command >= sExecute then
3972 begin
3973 if Result = eUtilized then
3974 begin
3975 with RW[Player].Un[Subject] do
3976 begin
3977 Cost := Integer(RW[Player].Model[mix].Cost) * Health *
3978 BuildCostMod[Difficulty[Player]] div 1200;
3979 if RW[Player].Model[mix].Cap[mcLine] > 0 then
3980 Cost := Cost div 2;
3981 end;
3982 if Project and (cpImp + cpIndex) = cpImp + imTrGoods then
3983 Inc(RW[Player].Money, Cost)
3984 else
3985 begin
3986 Inc(Prod, Cost * 2 div 3);
3987 Project0 := Project0 and not cpCompleted;
3988 if Project0 and not cpAuto <> Project and not cpAuto then
3989 Project0 := Project;
3990 Prod0 := Prod;
3991 end
3992 end;
3993 RemoveUnit_UpdateMap(Player, Subject);
3994 end;
3995 end;
3996 end
3997 else if Command >= sExecute then
3998 RemoveUnit_UpdateMap(Player, Subject);
3999 end
4000 end;
4001
4002 sSetUnitHome, sSetUnitHome - sExecute:
4003 begin
4004{$IFDEF TEXTLOG}CmdInfo := Format('SetUnitHome P%d Mod%d Loc%d', [Player, RW[Player].Un[Subject].mix, RW[Player].Un[Subject].Loc]); {$ENDIF}
4005 if (Subject >= RW[Player].nUn) or (RW[Player].Un[Subject].Loc < 0) then
4006 Result := eInvalid
4007 else
4008 begin
4009 Loc0 := RW[Player].Un[Subject].Loc;
4010 if RealMap[Loc0] and fCity = 0 then
4011 Result := eInvalid
4012 else
4013 begin
4014 SearchCity(Loc0, Player, cix1);
4015 if RW[Player].City[cix1].Flags and chCaptured <> 0 then
4016 Result := eViolation
4017 else if Command >= sExecute then
4018 RW[Player].Un[Subject].Home := cix1;
4019 end;
4020 end;
4021 end;
4022
4023 sSetSpyMission .. sSetSpyMission + (nSpyMission - 1) shl 4,
4024 sSetSpyMission - sExecute .. sSetSpyMission - sExecute +
4025 (nSpyMission - 1) shl 4:
4026 if Command >= sExecute then
4027 SpyMission := Command shr 4 and $F;
4028
4029 sLoadUnit, sLoadUnit - sExecute:
4030 begin
4031{$IFDEF TEXTLOG}CmdInfo := Format('LoadUnit P%d Mod%d Loc%d', [Player, RW[Player].Un[Subject].mix, RW[Player].Un[Subject].Loc]); {$ENDIF}
4032 if (Subject >= RW[Player].nUn) or (RW[Player].Un[Subject].Loc < 0) then
4033 Result := eInvalid
4034 else
4035 Result := LoadUnit(Player, Subject, Command < sExecute);
4036 end;
4037
4038 sUnloadUnit, sUnloadUnit - sExecute:
4039 begin
4040{$IFDEF TEXTLOG}CmdInfo := Format('UnloadUnit P%d Mod%d Loc%d', [Player, RW[Player].Un[Subject].mix, RW[Player].Un[Subject].Loc]); {$ENDIF}
4041 if (Subject >= RW[Player].nUn) or (RW[Player].Un[Subject].Loc < 0) then
4042 Result := eInvalid
4043 else
4044 Result := UnloadUnit(Player, Subject, Command < sExecute);
4045 end;
4046
4047 sSelectTransport, sSelectTransport - sExecute:
4048 if (Subject >= RW[Player].nUn) or (RW[Player].Un[Subject].Loc < 0) then
4049 Result := eInvalid
4050 else
4051 with RW[Player].Model[RW[Player].Un[Subject].mix] do
4052 begin
4053 if Cap[mcSeaTrans] + Cap[mcAirTrans] + Cap[mcCarrier] = 0 then
4054 Result := eInvalid
4055 else if Command >= sExecute then
4056 uixSelectedTransport := Subject;
4057 end;
4058
4059 sCreateUnit .. sCreateUnit + (nPl - 1) shl 4,
4060 sCreateUnit - sExecute .. sCreateUnit - sExecute + (nPl - 1) shl 4:
4061 if (GTestFlags and tfUncover <> 0) or (Difficulty[Player] = 0)
4062 then { supervisor only command }
4063 begin
4064 p1 := Command shr 4 and $F;
4065 Loc1 := Integer(Data);
4066 if (Occupant[Loc1] >= 0) and (p1 <> Occupant[Loc1]) or
4067 (RealMap[Loc1] and fCity <> 0) and
4068 (RealMap[Loc1] shr 27 <> Cardinal(p1)) or
4069 (RW[p1].Model[Subject].Domain < dAir) and
4070 ((RW[p1].Model[Subject].Domain = dSea) <> (RealMap[Integer(Data)] and
4071 fTerrain < fGrass)) then
4072 Result := eViolation
4073 else if Command >= sExecute then
4074 begin
4075 CreateUnit(p1, Subject);
4076 RW[p1].Un[RW[p1].nUn - 1].Loc := Integer(Data);
4077 PlaceUnit(p1, RW[p1].nUn - 1);
4078 UpdateUnitMap(Integer(Data));
4079 end;
4080 end
4081 else
4082 Result := eInvalid;
4083
4084 sMoveUnit + (0 + 6 * 8) * 16, sMoveUnit + (1 + 7 * 8) * 16,
4085 sMoveUnit + (2 + 0 * 8) * 16, sMoveUnit + (1 + 1 * 8) * 16,
4086 sMoveUnit + (0 + 2 * 8) * 16, sMoveUnit + (7 + 1 * 8) * 16,
4087 sMoveUnit + (6 + 0 * 8) * 16, sMoveUnit + (7 + 7 * 8) * 16,
4088 sMoveUnit - sExecute + (0 + 6 * 8) * 16, sMoveUnit - sExecute +
4089 (1 + 7 * 8) * 16, sMoveUnit - sExecute + (2 + 0 * 8) * 16,
4090 sMoveUnit - sExecute + (1 + 1 * 8) * 16, sMoveUnit - sExecute +
4091 (0 + 2 * 8) * 16, sMoveUnit - sExecute + (7 + 1 * 8) * 16,
4092 sMoveUnit - sExecute + (6 + 0 * 8) * 16, sMoveUnit - sExecute +
4093 (7 + 7 * 8) * 16:
4094 begin
4095 dx := (Command shr 4 + 4) and 7 - 4;
4096 dy := (Command shr 7 + 4) and 7 - 4;
4097{$IFDEF TEXTLOG}CmdInfo := Format('MoveUnit P%d I%d Mod%d Loc%d (%d,%d)', [Player, Subject, RW[Player].Un[Subject].mix, RW[Player].Un[Subject].Loc, dx, dy]); {$ENDIF}
4098 if (Subject >= RW[Player].nUn) or (RW[Player].Un[Subject].Loc < 0) then
4099 Result := eInvalid
4100 else
4101 Result := MoveUnit(Player, Subject, dx, dy, Command < sExecute);
4102 end;
4103
4104 {
4105 Settlers Related Commands
4106 ____________________________________________________________________
4107 }
4108 sAddToCity, sAddToCity - sExecute:
4109 begin
4110{$IFDEF TEXTLOG}CmdInfo := Format('AddToCity P%d Mod%d Loc%d', [Player, RW[Player].Un[Subject].mix, RW[Player].Un[Subject].Loc]); {$ENDIF}
4111 if (Subject >= RW[Player].nUn) or (RW[Player].Un[Subject].Loc < 0) then
4112 Result := eInvalid
4113 else if not (RW[Player].Model[RW[Player].Un[Subject].mix].Kind
4114 in [mkSettler, mkSlaves]) and
4115 (RW[Player].Un[Subject].Flags and unConscripts = 0) then
4116 Result := eViolation
4117 else
4118 begin
4119 Loc0 := RW[Player].Un[Subject].Loc;
4120 if RealMap[Loc0] and fCity = 0 then
4121 Result := eInvalid
4122 else
4123 begin
4124 SearchCity(Loc0, Player, cix1);
4125 with RW[Player].City[cix1] do
4126 if not CanCityGrow(Player, cix1) then
4127 Result := eMaxSize
4128 else if Command >= sExecute then
4129 begin { add to city }
4130 if Mode = moPlaying then
4131 SavedTiles[cix1] := 0; // save in every case
4132 if CanCityGrow(Player, cix1) then
4133 CityGrowth(Player, cix1);
4134 if (RW[Player].Model[RW[Player].Un[Subject].mix]
4135 .Kind = mkSettler) and CanCityGrow(Player, cix1) then
4136 CityGrowth(Player, cix1);
4137 RemoveUnit_UpdateMap(Player, Subject);
4138 end;
4139 end;
4140 end;
4141 end;
4142
4143 sStartJob .. sStartJob + $3F0, sStartJob - sExecute .. sStartJob + $3F0
4144 - sExecute:
4145 begin
4146 Loc0 := RW[Player].Un[Subject].Loc;
4147 I := Command shr 4 and $3F; // new job
4148{$IFDEF TEXTLOG}CmdInfo := Format('StartJob P%d Mod%d Loc%d: %d', [Player, RW[Player].Un[Subject].mix, Loc0, I]); {$ENDIF}
4149 if (Subject >= RW[Player].nUn) or (Loc0 < 0) then
4150 Result := eInvalid
4151 else if I >= nJob then
4152 Result := eInvalid
4153 else
4154 begin
4155 Result := StartJob(Player, Subject, I, Command < sExecute);
4156 if Result = eCity then
4157 begin // new city
4158 cix1 := RW[Player].nCity - 1;
4159 AddBestCityTile(Player, cix1);
4160 if Mode = moPlaying then
4161 with RW[Player].City[cix1] do
4162 begin
4163 // SavedResourceWeights[cix1]:=ResourceWeights;
4164 SavedTiles[cix1] := 0; // save in every case
4165 end;
4166 if Mode >= moMovie then { show new city in interface modules }
4167 for p1 := 0 to nPl - 1 do
4168 if (1 shl p1 and GWatching <> 0) and (p1 <> Player) and
4169 (ObserveLevel[Loc0] and (3 shl (2 * p1)) > 0) then
4170 CallPlayer(cShowCityChanged, p1, Loc0);
4171 end;
4172 end;
4173 end;
4174
4175 {
4176 City Related Commands
4177 ____________________________________________________________________
4178 }
4179 sSetCityProject, sSetCityProject - sExecute:
4180 begin
4181 NewProject := Integer(Data) and not cpAuto;
4182{$IFDEF TEXTLOG}CmdInfo := Format('SetCityProject P%d Loc%d: %d', [Player, RW[Player].City[Subject].Loc, NewProject]); {$ENDIF}
4183 if (Subject >= RW[Player].nCity) or (RW[Player].City[Subject].Loc < 0)
4184 then
4185 Result := eInvalid
4186 else
4187 with RW[Player].City[Subject] do
4188 begin
4189 if NewProject = Project then
4190 Result := eNotChanged
4191 else
4192 begin
4193 pt0 := ProjectType(Project0);
4194 pt1 := ProjectType(NewProject);
4195 if NewProject and cpImp = 0 then
4196 begin
4197 if NewProject and cpIndex >= RW[Player].nModel then
4198 Result := eInvalid
4199 else if (NewProject and cpConscripts <> 0) and
4200 not ((RW[Player].Tech[adConscription] >= tsApplicable) and
4201 (RW[Player].Model[NewProject and cpIndex].Domain = dGround)
4202 and (RW[Player].Model[NewProject and cpIndex].Kind < mkScout))
4203 then
4204 Result := eViolation
4205 // else if (RW[Player].Model[NewProject and cpIndex].Kind=mkSlaves)
4206 // and (GWonder[woPyramids].EffectiveOwner<>Player) then
4207 // result:=eNoPreq
4208 end
4209 else if NewProject and cpIndex >= nImp then
4210 Result := eInvalid
4211 else
4212 begin
4213 Preq := Imp[NewProject and cpIndex].Preq;
4214 for I := 0 to nImpReplacement - 1 do
4215 if (ImpReplacement[I].OldImp = NewProject and cpIndex) and
4216 (Built[ImpReplacement[I].NewImp] > 0) then begin
4217 Result := eObsolete;
4218 Break;
4219 end;
4220 if Result = eObsolete then
4221 else if Preq = preNA then
4222 Result := eInvalid
4223 else if (Preq >= 0) and (RW[Player].Tech[Preq] < tsApplicable)
4224 then
4225 Result := eNoPreq
4226 else if Built[NewProject and cpIndex] > 0 then
4227 Result := eInvalid
4228 else if (NewProject and cpIndex < nWonder) and
4229 (GWonder[NewProject and cpIndex].CityID <> WonderNotBuiltYet) then
4230 Result := eViolation // wonder already exists
4231 else if (NewProject and cpIndex = imSpacePort) and
4232 (RW[Player].NatBuilt[imSpacePort] > 0) then
4233 Result := eViolation // space port already exists
4234 else if (NewProject = cpImp + imBank) and (Built[imMarket] = 0)
4235 or (NewProject = cpImp + imUniversity) and
4236 (Built[imLibrary] = 0) or (NewProject = cpImp + imResLab) and
4237 (Built[imUniversity] = 0) or (NewProject = cpImp + imMfgPlant)
4238 and (Built[imFactory] = 0) then
4239 Result := eNoPreq;
4240 case NewProject - cpImp of
4241 woLighthouse, woMagellan, imCoastalFort, imHarbor, imPlatform:
4242 begin { city at ocean? }
4243 Preq := 0;
4244 V8_to_Loc(Loc, Adjacent);
4245 for V8 := 0 to 7 do
4246 begin
4247 Loc1 := Adjacent[V8];
4248 if (Loc1 >= 0) and (Loc1 < MapSize) and
4249 (RealMap[Loc1] and fTerrain = fShore) then
4250 Inc(Preq);
4251 end;
4252 if Preq = 0 then
4253 Result := eNoPreq;
4254 end;
4255 woHoover, imHydro:
4256 begin { city at river or mountains? }
4257 Preq := 0;
4258 V8_to_Loc(Loc, Adjacent);
4259 for V8 := 0 to 7 do
4260 begin
4261 Loc1 := Adjacent[V8];
4262 if (Loc1 >= 0) and (Loc1 < MapSize) and
4263 ((RealMap[Loc1] and fTerrain = fMountains) or
4264 (RealMap[Loc1] and fRiver <> 0)) then
4265 Inc(Preq);
4266 end;
4267 if Preq = 0 then
4268 Result := eNoPreq;
4269 end;
4270 woMIR, imShipComp, imShipPow, imShipHab:
4271 if RW[Player].NatBuilt[imSpacePort] = 0 then
4272 Result := eNoPreq;
4273 end;
4274 if (GTestFlags and tfNoRareNeed = 0) and
4275 (Imp[NewProject and cpIndex].Kind = ikShipPart) then
4276 if RW[Player].Tech[adMassProduction] < tsApplicable then
4277 Result := eNoPreq
4278 else
4279 begin // check for rare resources
4280 if NewProject and cpIndex = imShipComp then
4281 J := 1
4282 else if NewProject and cpIndex = imShipPow then
4283 J := 2
4284 else { if NewProject and cpIndex=imShipHab then }
4285 J := 3;
4286 // j = rare resource required
4287 Preq := 0;
4288 V21_to_Loc(Loc, Radius);
4289 for V21 := 1 to 26 do
4290 begin
4291 Loc1 := Radius[V21];
4292 if (Loc1 >= 0) and (Loc1 < MapSize) and
4293 (RealMap[Loc1] shr 25 and 3 = Cardinal(J)) then
4294 Inc(Preq);
4295 end;
4296 if Preq = 0 then
4297 Result := eNoPreq;
4298 end;
4299 end;
4300
4301 if (Command >= sExecute) and (Result >= rExecuted) then
4302 begin
4303 if pt0 <> ptSelect then
4304 if NewProject and (cpImp or cpIndex) = Project0 and
4305 (cpImp or cpIndex) then
4306 Prod := Prod0
4307 else if (pt1 = ptTrGoods) or (pt1 = ptShip) or (pt1 <> pt0)
4308 and (pt0 <> ptCaravan) then
4309 begin
4310 Inc(RW[Player].Money, Prod0);
4311 Prod := 0;
4312 Prod0 := 0;
4313 Project0 := cpImp + imTrGoods;
4314 end
4315 else
4316 Prod := Prod0 * 2 div 3;
4317 Project := NewProject;
4318 end;
4319 end;
4320 end;
4321 end;
4322
4323 sBuyCityProject, sBuyCityProject - sExecute:
4324 begin
4325{$IFDEF TEXTLOG}CmdInfo := Format('BuyCityProject P%d Loc%d', [Player, RW[Player].City[Subject].Loc]); {$ENDIF}
4326 if (Subject >= RW[Player].nCity) or (RW[Player].City[Subject].Loc < 0)
4327 then
4328 Result := eInvalid
4329 else
4330 with RW[Player].City[Subject] do
4331 if (RW[Player].Government = gAnarchy) or (Flags and chCaptured <> 0)
4332 then
4333 Result := eOutOfControl
4334 else if (Project and cpImp <> 0) and
4335 ((Project and cpIndex = imTrGoods) or
4336 (Imp[Project and cpIndex].Kind = ikShipPart)) then
4337 Result := eInvalid // don't buy colony ship
4338 else
4339 begin
4340 CityReport.HypoTiles := -1;
4341 CityReport.HypoTax := -1;
4342 CityReport.HypoLux := -1;
4343 GetCityReport(Player, Subject, CityReport);
4344 Cost := CityReport.ProdCost;
4345 NextProd := CityReport.ProdRep - CityReport.Support;
4346 if (CityReport.Working - CityReport.Happy > Size shr 1) or
4347 (NextProd < 0) then // !!! change to new style disorder
4348 NextProd := 0;
4349 Cost := Cost - Prod - NextProd;
4350 if (GWonder[woMich].EffectiveOwner = Player) and
4351 (Project and cpImp <> 0) then
4352 Cost := Cost * 2
4353 else
4354 Cost := Cost * 4;
4355 if Cost <= 0 then
4356 Result := eNotChanged
4357 else if Cost > RW[Player].Money then
4358 Result := eViolation
4359 else if Command >= sExecute then
4360 IntServer(sIntBuyMaterial, Player, Subject, Cost);
4361 // need to save material/cost because city tiles are not correct
4362 // when loading
4363 end;
4364 end;
4365
4366 sSellCityProject, sSellCityProject - sExecute:
4367 begin
4368{$IFDEF TEXTLOG}CmdInfo := Format('SellCityProject P%d Loc%d', [Player, RW[Player].City[Subject].Loc]); {$ENDIF}
4369 if (Subject >= RW[Player].nCity) or (RW[Player].City[Subject].Loc < 0)
4370 then
4371 Result := eInvalid
4372 else if Command >= sExecute then
4373 with RW[Player].City[Subject] do
4374 begin
4375 Inc(RW[Player].Money, Prod0);
4376 Prod := 0;
4377 Prod0 := 0;
4378 end;
4379 end;
4380
4381 sSellCityImprovement, sSellCityImprovement - sExecute:
4382 begin
4383{$IFDEF TEXTLOG}CmdInfo := Format('SellCityImprovement P%d Loc%d: %d', [Player, RW[Player].City[Subject].Loc, Integer(Data)]); {$ENDIF}
4384 if (Subject >= RW[Player].nCity) or (RW[Player].City[Subject].Loc < 0)
4385 then
4386 Result := eInvalid
4387 else
4388 with RW[Player].City[Subject] do
4389 if Built[Integer(Data)] = 0 then
4390 Result := eInvalid
4391 else if (RW[Player].Government = gAnarchy) or
4392 (Flags and chCaptured <> 0) then
4393 Result := eOutOfControl
4394 else if Flags and chImprovementSold <> 0 then
4395 Result := eOnlyOnce
4396 else if Command >= sExecute then
4397 begin
4398 Inc(RW[Player].Money, Imp[Integer(Data)].Cost * BuildCostMod
4399 [Difficulty[Player]] div 12);
4400 Built[Integer(Data)] := 0;
4401 if Imp[Integer(Data)].Kind in [ikNatLocal, ikNatGlobal] then
4402 begin
4403 RW[Player].NatBuilt[Integer(Data)] := 0;
4404 case Integer(Data) of
4405 imGrWall:
4406 GrWallContinent[Player] := -1;
4407 imSpacePort:
4408 DestroySpacePort_TellPlayers(Player, -1);
4409 end;
4410 end;
4411 Inc(Flags, chImprovementSold);
4412 end;
4413 end;
4414
4415 sRebuildCityImprovement, sRebuildCityImprovement - sExecute:
4416 begin
4417 OldImp := Integer(Data);
4418{$IFDEF TEXTLOG}CmdInfo := Format('RebuildCityImprovement P%d Loc%d: %d', [Player, RW[Player].City[Subject].Loc, OldImp]); {$ENDIF}
4419 if (Subject >= RW[Player].nCity) or (RW[Player].City[Subject].Loc < 0)
4420 then
4421 Result := eInvalid
4422 else
4423 begin
4424 if (OldImp < 0) or (OldImp >= nImp) or
4425 not (Imp[OldImp].Kind in [ikCommon, ikNatLocal, ikNatGlobal]) then
4426 Result := eInvalid
4427 else
4428 with RW[Player].City[Subject] do
4429 if (Built[OldImp] = 0) or (Project and cpImp = 0) or
4430 not (Imp[Project and cpIndex].Kind in [ikCommon, ikNatLocal,
4431 ikNatGlobal]) then
4432 Result := eInvalid
4433 else if (RW[Player].Government = gAnarchy) or
4434 (Flags and chCaptured <> 0) then
4435 Result := eOutOfControl
4436 else if Flags and chImprovementSold <> 0 then
4437 Result := eOnlyOnce
4438 else if Command >= sExecute then
4439 begin
4440 Inc(Prod, Imp[OldImp].Cost * BuildCostMod[Difficulty[Player]]
4441 div 12 * 2 div 3);
4442 Project0 := Project0 and not cpCompleted;
4443 if Project0 and not cpAuto <> Project and not cpAuto then
4444 Project0 := Project;
4445 Prod0 := Prod;
4446 Built[OldImp] := 0;
4447 if Imp[OldImp].Kind in [ikNatLocal, ikNatGlobal] then
4448 begin // nat. project lost
4449 RW[Player].NatBuilt[OldImp] := 0;
4450 case OldImp of
4451 imGrWall:
4452 GrWallContinent[Player] := -1;
4453 imSpacePort:
4454 DestroySpacePort_TellPlayers(Player, -1);
4455 end;
4456 end;
4457 Inc(Flags, chImprovementSold);
4458 end;
4459 end;
4460 end;
4461
4462 sSetCityTiles, sSetCityTiles - sExecute:
4463 begin
4464{$IFDEF TEXTLOG}CmdInfo := Format('SetCityTiles P%d Loc%d: %x', [Player, RW[Player].City[Subject].Loc, Integer(Data)]); {$ENDIF}
4465 if (Subject >= RW[Player].nCity) or (RW[Player].City[Subject].Loc < 0)
4466 then
4467 Result := eInvalid
4468 else
4469 Result := SetCityTiles(Player, Subject, Integer(Data),
4470 Command < sExecute);
4471 end;
4472
4473 {
4474 Client Exclusive Commands
4475 ____________________________________________________________________
4476 }
4477 else
4478 if Command >= cClientEx then
4479 begin
4480{$IFDEF TEXTLOG}CmdInfo := Format('ClientEx%x P%d', [Command, Player]);
4481 {$ENDIF}
4482 if ProcessClientData[Player] or (Mode = moPlaying) then
4483 CallPlayer(Command, Player, Data)
4484 end
4485 else
4486 Result := eUnknown;
4487 end; { case command }
4488
4489 // do not log invalid and non-relevant commands
4490 if Result = eZOC_EnemySpotted then
4491 begin
4492 Assert(Mode = moPlaying);
4493 CL.State := FormerCLState;
4494 IntServer(sIntDiscoverZOC, Player, 0, ZOCTile);
4495 end
4496 else if Result and rEffective = 0 then
4497 if Mode < moPlaying then
4498 begin
4499{$IFDEF TEXTLOG}CmdInfo := Format('***ERROR (%x) ', [Result]) + CmdInfo;
4500 {$ENDIF}
4501 LoadOK := False;
4502 end
4503 else
4504 begin
4505 if Logged then
4506 CL.State := FormerCLState;
4507 if (Result < rExecuted) and (Command >= sExecute) then
4508 PutMessage(1 shl 16 + 1, Format('INVALID: %d calls %x (%d)',
4509 [Player, Command, Subject]));
4510 end;
4511
4512 if (Command and (cClientEx or sExecute or sctMask) = sExecute or sctEndClient)
4513 and (Result >= rExecuted) then
4514 LastEndClientCommand := Command;
4515{$IFOPT O-}Dec(nHandoverStack, 2); {$ENDIF}
4516end;
4517
4518
4519initialization
4520
4521FindFirst(ParamStr(0), $21, ExeInfo);
4522FindClose(ExeInfo);
4523
4524{$IFOPT O-}nHandoverStack := 0; {$ENDIF}
4525
4526end.
Note: See TracBrowser for help on using the repository browser.