/usr/src/castle-game-engine-6.4/window/castleuistate.pas is in castle-game-engine-src 6.4+dfsg1-2.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 | {
Copyright 2015-2017 Michalis Kamburelis.
This file is part of "Castle Game Engine".
"Castle Game Engine" is free software; see the file COPYING.txt,
included in this distribution, for details about the copyright.
"Castle Game Engine" is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
----------------------------------------------------------------------------
}
{ UI state (TUIState). }
unit CastleUIState;
{$I castleconf.inc}
interface
uses Classes, Generics.Collections,
CastleConfig, CastleKeysMouse, CastleImages, CastleUIControls,
CastleGLImages, CastleVectors, CastleRectangles;
type
TUIStateList = class;
{ UI state, to manage the state of your game UI.
See also
https://castle-engine.sourceforge.io/manual_2d_user_interface.php#section_ui_state
for an overview of using TUIState.
In simple cases, only one state is @italic(current) at a given time,
and it can be get or set using the @link(TUIState.Current) property.
In more complex cases, you can use @link(TUIState.Push) and @link(TUIState.Pop)
to build a stack of states, and in effect multiple states are active at the same time.
All of the states on stack are @italic(started), but only the top-most is @italic(resumed).
Each state has @link(Start) and @link(Stop)
methods that you can override to perform work when state becomes
part of the current state stack, or stops being part of it.
You can also override @link(Resume) and @link(Pause) methods,
to perform work when the state becomes the top-most state or is no longer
the top-most state. The distinction becomes important once you play
around with pushing/popping states.
The names are deliberaly similar to Android lifecycle callback names.
You can add/remove state-specific UI controls in various ways.
You can add them in the constructor of this state (and then free in destructor),
or add them in @link(Start), free in @link(Stop).
@orderedList(
@item(It's simplest and best to add/keep children controls as real
children of the current state, so add them
using methods @link(TUIControl.InsertFront) and
@link(TUIControl.InsertBack).)
@item(Eventually, for special tricks, you can add controls that are
conceptually the state "children" directly to the
@code(StateContainer.Controls) list.
This allows to keep some children on the @code(StateContainer.Controls)
list for a longer
time (not only when this state is active), which may be useful for optimization,
to not reinitialize GL resources too often.
To do this, add controls using
@code(StateContainer.Controls.InsertFront(...)), remove them by
@code(StateContainer.Controls.Remove(...)),
and make sure to override @link(InsertAtPosition) method such that state instance
is inserted in @code(StateContainer.Controls) right behind your UI.)
)
Current state is also placed on the list of container controls.
This way state is notified
about UI events, and can react to them. Since state-specific UI
should always be at the front of us, or our children,
so in case of events that can be "handled"
(like TUIControl.Press, TUIControl.Release events)
the state-specific UI controls will handle them @italic(before)
the state itself (if you override TUIControl.Press or such in state,
be sure to call @code(inherited) first, to make sure it really
happens).
This way state can
@unorderedList(
@item(catch press/release and similar events, when no other
state-specific control handled them,)
@item(catch update, GL context open/close and other useful events,)
@item(can have it's own render function, to directly draw UI.)
)
See the TUIControl class for a lot of useful methods that you can
override in your state descendants to capture various events. }
TUIState = class(TUIControl)
private
FStartContainer: TUIContainer;
FInterceptInput: boolean;
FFreeAtStop: TComponent;
procedure InternalStart;
procedure InternalStop;
class var FStateStack: TUIStateList;
class function GetCurrent: TUIState; static;
class procedure SetCurrent(const Value: TUIState); static;
class function GetCurrentTop: TUIState; static;
class function GetStateStack(const Index: Integer): TUIState; static;
protected
{ Container on which state works. By default, this is Application.MainWindow.
When the state is current, then @link(Container) property (from
ancestor, see TUIControl.Container) is equal to this. }
function StateContainer: TUIContainer; virtual;
{ Position on @code(StateContainer.Controls) where we insert this state.
By default, state is inserted as the front-most control, so position is equal
to @code(StateContainer.Controls.Count). }
function InsertAtPosition: Integer; virtual;
{ Assign this component as owner for your controls,
to make them freed during nearest @link(Stop). }
function FreeAtStop: TComponent;
public
{ When @true, state operations will send a log to CastleLog. }
class var Log: boolean;
{ Current state. In case multiple states are active (only possible
if you used @link(Push) method), this is the bottom state
(use @link(CurrentTop) to get top state).
Setting this resets whole state stack. }
class property Current: TUIState read GetCurrent write SetCurrent;
class property CurrentTop: TUIState read GetCurrentTop;
{ Pushing the state adds it at the top of the state stack.
The state known as @link(Current) is conceptually at the bottom of state stack, always.
When it is nil, then pushing new state sets the @link(Current) state.
Otherwise @link(Current) state is left as-it-is, new state is added on top. }
class procedure Push(const NewState: TUIState);
{ Pop the current top-most state, whatever it is. }
class procedure Pop;
{ Pop the top-most state, checking it is as expected.
Makes a warning, and does nothing, if the current top-most state
is different than indicated. This is usually a safer (more chance
to easily catch bugs) version of Pop than the parameter-less version. }
class procedure Pop(const CurrentTopMostState: TUIState);
class function StateStackCount: Integer;
class property StateStack [const Index: Integer]: TUIState read GetStateStack;
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
{ State becomes active, it's now part of the state stack.
Started state is part of the StateStack, and will soon become
running (top-most on the stack). When the state is set to be current,
by @code(TUIState.Current := MyState), this happens:
@orderedList(
@item(MyStart is pushed as the top-most state on state stack.)
@item(MyStart.Start is called.)
@item(MyStart is added to the @code(StateContainer.Controls) list,
so the state methods GLContextOpen and Resize are called
(as for all normal TUIControl instances).)
@item(MyStar.Resume is called.)
) }
procedure Start; virtual;
{ State is no longer active, no longer part of state stack.
When the state stops becoming active, this happens:
@orderedList(
@item(MyStart.Pause is called.)
@item(MyStart is removed from the
@code(StateContainer.Controls) list.
So the state method GLContextClose is called
(as for all normal TUIControl instances).)
@item(MyStart.Stop is called.)
@item(MyStart is removed from the on state stack.)
)
This is always called to finalize the started state.
When the state is destroyed, it's @link(Pause) and @link(Stop)
are called too, so you can use this method to reliably finalize whatever
you initialized in @link(Start). }
procedure Stop; virtual;
{ State is now the top-most state. See @link(Start) and @link(Stop)
docs about state lifecycle methods.
This is called after @link(Start), it is also called
when you pop another state, making this state the top-most. }
procedure Resume; virtual;
{ State is no longer the top-most state. See @link(Start) and @link(Stop)
docs about state lifecycle methods.
This is called before @link(Stop), it is also called
when another state is pushed over this state, so this stops
being the the top-most state. }
procedure Pause; virtual;
procedure Finish; virtual; deprecated 'use Stop';
function Rect: TRectangle; override;
{ State is right now part of the state stack, which means
it's between @link(Start) and @link(Stop) calls.
The state is added to the stack before the @link(Start) call,
and removed after the @link(Stop) call, so this returns @true
during all the methods --- @link(Start), @link(Resume), @link(Pause), @link(Stop). }
function Active: boolean;
{ Prevents passing mouse/keyboard events to the UI states underneath.
More precisely, when this property is @true, then the
@link(Press), @link(Release) and @link(Motion) events are marked as
"handled" in this UI state. This means that they will not be processed
further, by UI controls under this state, in particular by UI states
that are underneath this state in @italic(state stack) (created
by @link(Push) method). They will also not be passed to final container
(TCastleWindowCustom, TCastleControlCustom) callbacks like
TCastleWindowCustom.OnPress (as these callbacks are always used at the end,
when nothing else handled the event). }
property InterceptInput: boolean read FInterceptInput write FInterceptInput
default false;
function Press(const Event: TInputPressRelease): boolean; override;
function Release(const Event: TInputPressRelease): boolean; override;
function Motion(const Event: TInputMotion): boolean; override;
procedure Update(const SecondsPassed: Single;
var HandleInput: boolean); override;
end;
TUIStateList = class(specialize TObjectList<TUIState>);
implementation
uses SysUtils,
CastleWindow, CastleFilesUtils, CastleUtils, CastleTimeUtils, CastleLog;
{ TUIState --------------------------------------------------------------------- }
class function TUIState.GetCurrent: TUIState;
begin
if (FStateStack = nil) or
(FStateStack.Count = 0) then
Result := nil else
Result := FStateStack[0];
end;
class function TUIState.GetCurrentTop: TUIState;
begin
if (FStateStack = nil) or
(FStateStack.Count = 0) then
Result := nil else
Result := FStateStack[FStateStack.Count - 1];
end;
class procedure TUIState.SetCurrent(const Value: TUIState);
begin
{ exit early if there's nothing to do }
if (StateStackCount = 0) and (Value = nil) then
Exit;
if (StateStackCount = 1) and (FStateStack[0] = Value) then
Exit;
{ Remove and finish topmost state.
The loop is written to work even when some state Stop method
changes states. }
while StateStackCount <> 0 do
Pop;
{ deallocate empty FStateStack }
if Value = nil then
FreeAndNil(FStateStack);
Push(Value);
end;
class procedure TUIState.Push(const NewState: TUIState);
begin
if NewState <> nil then
begin
{ pause previous top-most state }
if (FStateStack <> nil) and
(FStateStack.Count <> 0) then
FStateStack.Last.Pause;
{ create FStateStack on demand now }
if FStateStack = nil then
FStateStack := TUIStateList.Create(false);
FStateStack.Add(NewState);
NewState.InternalStart;
NewState.Resume;
end;
end;
class procedure TUIState.Pop;
var
TopState: TUIState;
begin
TopState := FStateStack.Last;
TopState.Pause;
TopState.InternalStop;
if TopState = FStateStack.Last then
FStateStack.Delete(FStateStack.Count - 1) else
WritelnWarning('State', 'Topmost state is no longer topmost after its Stop method. Do not change state stack from state Stop methods.');
{ resume new top-most state }
if (FStateStack <> nil) and
(FStateStack.Count <> 0) then
FStateStack.Last.Resume;
end;
class procedure TUIState.Pop(const CurrentTopMostState: TUIState);
begin
if (FStateStack = nil) or (FStateStack.Count = 0) then
begin
WritelnWarning('State', 'Cannot pop UI state, that stack is empty');
Exit;
end;
if FStateStack.Last <> CurrentTopMostState then
begin
WritelnWarning('State', 'Cannot pop UI state, top-most state is expected to be ' + CurrentTopMostState.ClassName + ', but is ' + FStateStack.Last.ClassName);
Exit;
end;
Pop;
end;
class function TUIState.StateStackCount: Integer;
begin
if FStateStack = nil then
Result := 0 else
Result := FStateStack.Count;
end;
class function TUIState.GetStateStack(const Index: Integer): TUIState;
begin
if FStateStack = nil then
raise EInternalError.CreateFmt('TUIState.GetStateStack: state stack is empty, cannot get state index %d',
[Index]);
Result := FStateStack[Index];
end;
function TUIState.InsertAtPosition: Integer;
begin
Result := StateContainer.Controls.Count;
end;
function TUIState.FreeAtStop: TComponent;
begin
if FFreeAtStop = nil then
FFreeAtStop := TComponent.Create(Self);
Result := FFreeAtStop;
end;
procedure TUIState.InternalStart;
begin
{ typically, the Start method will initialize some stuff,
making the 1st SecondsPassed non-representatively large. }
StateContainer.Fps.ZeroNextSecondsPassed;
if CastleLog.Log and Log then
WritelnLog('UIState', 'Starting state ' + Name + ':' + ClassName);
Start;
{ actually insert, this will also call GLContextOpen and Resize.
However, check first that we're still the current state,
to safeguard from the fact that Start changed state
(like the loading state, that changes to play state immediately in start). }
if FStateStack.IndexOf(Self) <> -1 then
StateContainer.Controls.Insert(InsertAtPosition, Self);
end;
procedure TUIState.InternalStop;
begin
StateContainer.Controls.Remove(Self);
Stop;
if CastleLog.Log and Log then
WritelnLog('UIState', 'Stopped state ' + Name + ':' + ClassName);
end;
function TUIState.StateContainer: TUIContainer;
begin
if FStartContainer <> nil then
{ between Start and Stop, be sure to return the same thing
from StateContainer method. Also makes it working when Application
is nil when destroying state from CastleWindow finalization. }
Result := FStartContainer
else
begin
if Application.MainWindow = nil then
raise Exception.Create('Assign Application.MainWindow before starting TUIState');
Result := Application.MainWindow.Container;
end;
end;
constructor TUIState.Create(AOwner: TComponent);
begin
inherited;
end;
destructor TUIState.Destroy;
begin
{ finish yourself and remove from FStateStack, if present there }
if (FStateStack <> nil) and
(FStateStack.IndexOf(Self) <> -1) then
begin
if FStateStack.Last = Self then
Pause;
InternalStop;
FStateStack.Remove(Self);
{ deallocate empty FStateStack. Doing this here allows to deallocate
FStateStack only once all states finished gracefully. }
if FStateStack.Count = 0 then
FreeAndNil(FStateStack);
end;
inherited;
end;
procedure TUIState.Resume;
begin
if CastleLog.Log and Log then
WritelnLog('UIState', 'Resuming state ' + Name + ':' + ClassName);
end;
procedure TUIState.Pause;
begin
if CastleLog.Log and Log then
WritelnLog('UIState', 'Paused state ' + Name + ':' + ClassName);
end;
procedure TUIState.Start;
begin
FStartContainer := StateContainer;
end;
procedure TUIState.Stop;
begin
{$warnings off}
Finish;
{$warnings on}
FStartContainer := nil;
FreeAndNil(FFreeAtStop);
end;
procedure TUIState.Finish;
begin
end;
function TUIState.Rect: TRectangle;
begin
{ 1. always capture events on whole container
2. make child controls (anchored to us) behave like anchored to whole window. }
Result := ParentRect;
end;
function TUIState.Active: boolean;
begin
Result := (FStateStack <> nil) and
(FStateStack.IndexOf(Self) <> -1);
end;
function TUIState.Press(const Event: TInputPressRelease): boolean;
begin
Result := inherited;
Result := Result or InterceptInput;
end;
function TUIState.Release(const Event: TInputPressRelease): boolean;
begin
Result := inherited;
Result := Result or InterceptInput;
end;
function TUIState.Motion(const Event: TInputMotion): boolean;
begin
Result := inherited;
Result := Result or InterceptInput;
end;
procedure TUIState.Update(const SecondsPassed: Single;
var HandleInput: boolean);
begin
{ do not allow controls underneath to handle input }
if InterceptInput then
HandleInput := false;
end;
end.
|