/usr/src/castle-game-engine-5.2.0/x3d/x3d_time.inc is in castle-game-engine-src 5.2.0-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 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 | {
Copyright 2008-2014 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.
----------------------------------------------------------------------------
}
{$ifdef read_interface}
TTimeDependentNodeHandler = class;
{ }
IAbstractTimeDependentNode = interface(IAbstractChildNode)
['{19D68914-F2BA-4CFA-90A1-F561DB264678}']
function GetTimeDependentNodeHandler: TTimeDependentNodeHandler;
property TimeDependentNodeHandler: TTimeDependentNodeHandler
read GetTimeDependentNodeHandler;
end;
TTimeFunction = function: TFloatTime of object;
{ Common helper for all X3DTimeDependentNode descendants.
This includes things descending from interface IAbstractTimeDependentNode,
in particular (but not only) descending from class
TAbstractTimeDependentNode.
It would be cleaner to have Node declared as IAbstractTimeDependentNode,
and have IAbstractTimeDependentNode contain common fields.
Then a lot of fields of this class would not be needed, as they
would be accessible as IAbstractTimeDependentNode fields.
TODO: maybe in the future. }
TTimeDependentNodeHandler = class
private
FIsActive: boolean;
FIsPaused: boolean;
FElapsedTime: TFloatTime;
FElapsedTimeInCycle: TFloatTime;
procedure SetIsActive(const Value: boolean);
procedure SetIsPaused(const Value: boolean);
procedure SetElapsedTime(const Value: TFloatTime);
public
Node: TX3DNode;
CurrentTime: TX3DTime;
{ These describe current state of this TimeDependentNode,
see X3D specification about "Time" component.
Their setting causes appropriate events to be generated
(with Time = CurrentTime, so be sure to update it before changing
these properties. SetTime automatically does this for you.).
@groupBegin }
property IsActive: boolean read FIsActive write SetIsActive;
property IsPaused: boolean read FIsPaused write SetIsPaused;
property ElapsedTime: TFloatTime read FElapsedTime write SetElapsedTime;
{ @groupEnd }
public
{ Cycle interval for this time-dependent node. }
OnCycleInterval: TTimeFunction;
function CycleInterval: TFloatTime;
public
Fdloop: TSFBool;
FdpauseTime: TSFTime;
FdresumeTime: TSFTime;
FdstartTime: TSFTime;
FdstopTime: TSFTime;
{ May be @nil if node doesn't have an "enabled" field. }
Fdenabled: TSFBool;
EventelapsedTime: TSFTimeEvent;
EventisActive: TSFBoolEvent;
EventisPaused: TSFBoolEvent;
{ May be @nil if node doesn't have a "cycleTime" event. }
EventcycleTime: TSFTimeEvent;
{ Like ElapsedTime, but spent in the current cycle.
When cycleInterval = 0, this is always 0.
When cycleInterval <> 0, this is always >= 0 and < cycleInterval . }
property ElapsedTimeInCycle: TFloatTime read FElapsedTimeInCycle;
{ Call this when world time increases.
This is the most important method of this class, that basically
implements time-dependent nodes operations.
OldValue, NewValue and TimeIncrease must match, as produced
by TCastleSceneCore.SetTime and friends.
When ResetTime = true, this means that "TimeIncrease value is unknown"
(you @italic(must) pass TimeIncrease = 0 in this case).
This can happen only when were called by ResetTime.
In other circumstances, TimeIncrease must be >= 0.
(It's allowed to pass TimeIncrease = 0 and ResetTime = false, this doesn't
advance the clock, but is a useful trick to force some update,
see TVRMScene.ChangedField implementation.)
References: see X3D specification "Time" component,
8.2 ("concepts") for logic behind all those start/stop/pause/resumeTime,
cycleInterval, loop properties.
@returns(If some state of time-dependent node changed.) }
function SetTime(const OldValue, NewValue: TX3DTime;
const TimeIncrease: TFloatTime; const ResetTime: boolean): boolean;
end;
TAbstractTimeDependentNode = class(TAbstractChildNode, IAbstractTimeDependentNode)
private
FTimeDependentNodeHandler: TTimeDependentNodeHandler;
{ To satify IAbstractTimeDependentNode }
function GetTimeDependentNodeHandler: TTimeDependentNodeHandler;
public
procedure CreateNode; override;
destructor Destroy; override;
private FFdLoop: TSFBool;
public property FdLoop: TSFBool read FFdLoop;
private FFdPauseTime: TSFTime;
public property FdPauseTime: TSFTime read FFdPauseTime;
private FFdResumeTime: TSFTime;
public property FdResumeTime: TSFTime read FFdResumeTime;
private FFdStartTime: TSFTime;
public property FdStartTime: TSFTime read FFdStartTime;
private FFdStopTime: TSFTime;
public property FdStopTime: TSFTime read FFdStopTime;
{ Event out } { }
private FEventElapsedTime: TSFTimeEvent;
public property EventElapsedTime: TSFTimeEvent read FEventElapsedTime;
{ Event out } { }
private FEventIsActive: TSFBoolEvent;
public property EventIsActive: TSFBoolEvent read FEventIsActive;
{ Event out } { }
private FEventIsPaused: TSFBoolEvent;
public property EventIsPaused: TSFBoolEvent read FEventIsPaused;
function CycleInterval: TFloatTime; virtual; abstract;
property TimeDependentNodeHandler: TTimeDependentNodeHandler
read FTimeDependentNodeHandler;
end;
TTimeSensorNode = class(TAbstractTimeDependentNode, IAbstractSensorNode)
private
procedure EventElapsedTimeReceive(
Event: TX3DEvent; Value: TX3DField; const Time: TX3DTime);
public
procedure CreateNode; override;
class function ClassNodeTypeName: string; override;
class function URNMatching(const URN: string): boolean; override;
private FFdCycleInterval: TSFTime;
public property FdCycleInterval: TSFTime read FFdCycleInterval;
{ Event out } { }
private FEventCycleTime: TSFTimeEvent;
public property EventCycleTime: TSFTimeEvent read FEventCycleTime;
{ Event out } { }
private FEventFraction_changed: TSFFloatEvent;
public property EventFraction_changed: TSFFloatEvent read FEventFraction_changed;
{ Event out } { }
private FEventTime: TSFTimeEvent;
public property EventTime: TSFTimeEvent read FEventTime;
private FFdEnabled: TSFBool;
public property FdEnabled: TSFBool read FFdEnabled;
function CycleInterval: TFloatTime; override;
{ Send TimeSensor output events, without actually activating the TimeSensor.
This is useful in situations when you want the X3D scene state to reflect
given time, but you do not want to activate sensor and generally you do
not want to initialize anything that would continue animating on it's own.
Using TimeSensor this way is contrary to the VRML/X3D specified behavior.
But it's useful when we have VRML/X3D scene inside T3DResource,
that is shared by many T3D instances (like creatures and items) that want to
simultaneusly display different time moments of the same scene within
a single frame. In other words, this is useful when a single scene is shared
(if you have a 100 creatures in your game using the same 3D model,
you don't want to create 100 copies of this 3D model in memory).
We ignore TimeSensor.loop (FdLoop) field,
instead we follow our own Loop parameter.
We take into account TimeSensor.cycleInterval (FdCycleInterval) and
TimeSensor.enabled (FdEnabled) fields, just like normal TimeSensor.
We send out fraction_changed (EventFraction_changed) and time (EventTime)
output events. }
procedure FakeTime(const Time: TFloatTime; const Loop: boolean);
end;
{$endif read_interface}
{$ifdef read_implementation}
{ Field types for startTime/stopTime ----------------------------------------- }
type
{ SFTime that ignores it's input event when parent time-dependent node
is active. }
TSFTimeIgnoreWhenActive = class(TSFTime)
protected
class function ExposedEventsFieldClass: TX3DFieldClass; override;
procedure ExposedEventReceive(Event: TX3DEvent; AValue: TX3DField;
const Time: TX3DTime); override;
end;
TSFStopTime = class(TSFTime)
protected
class function ExposedEventsFieldClass: TX3DFieldClass; override;
procedure ExposedEventReceive(Event: TX3DEvent; AValue: TX3DField;
const Time: TX3DTime); override;
end;
class function TSFTimeIgnoreWhenActive.ExposedEventsFieldClass: TX3DFieldClass;
begin
Result := TSFTime;
end;
procedure TSFTimeIgnoreWhenActive.ExposedEventReceive(
Event: TX3DEvent; AValue: TX3DField; const Time: TX3DTime);
{ Is parent node an active X3DTimeDependentNode.
Doing this inside local procedure to limit lifetime of IAbstractTimeDependentNode,
to secure against http://bugs.freepascal.org/view.php?id=10374 }
function Active: boolean;
begin
Result := (ParentNode <> nil) and
Supports(ParentNode, IAbstractTimeDependentNode) and
(ParentNode as IAbstractTimeDependentNode).TimeDependentNodeHandler.IsActive;
end;
begin
if Active then Exit;
inherited;
end;
class function TSFStopTime.ExposedEventsFieldClass: TX3DFieldClass;
begin
Result := TSFTime;
end;
procedure TSFStopTime.ExposedEventReceive(Event: TX3DEvent; AValue: TX3DField;
const Time: TX3DTime);
{ Is parent node an active X3DTimeDependentNode, and AValue <= startTime.
Doing this inside local procedure to limit lifetime of IAbstractTimeDependentNode,
to secure against http://bugs.freepascal.org/view.php?id=10374 }
function Active: boolean;
var
H: TTimeDependentNodeHandler;
begin
Result := (ParentNode <> nil) and
Supports(ParentNode, IAbstractTimeDependentNode);
if Result then
begin
H := (ParentNode as IAbstractTimeDependentNode).TimeDependentNodeHandler;
Result := H.IsActive and
( (AValue as TSFTime).Value <= H.FdStartTime.Value );
end;
end;
begin
if Active then Exit;
inherited;
end;
{ TTimeDependentNodeHandler -------------------------------------------------- }
procedure TTimeDependentNodeHandler.SetIsActive(const Value: boolean);
begin
if Value <> FIsActive then
begin
FIsActive := Value;
EventIsActive.Send(Value, CurrentTime);
end;
end;
procedure TTimeDependentNodeHandler.SetIsPaused(const Value: boolean);
begin
if Value <> FIsPaused then
begin
FIsPaused := Value;
EventIsPaused.Send(Value, CurrentTime);
end;
end;
procedure TTimeDependentNodeHandler.SetElapsedTime(const Value: TFloatTime);
begin
if Value <> FElapsedTime then
begin
FElapsedTime := Value;
EventElapsedTime.Send(Value, CurrentTime);
end;
end;
function TTimeDependentNodeHandler.CycleInterval: TFloatTime;
begin
Assert(Assigned(OnCycleInterval));
Result := OnCycleInterval();
end;
function TTimeDependentNodeHandler.SetTime(
const OldValue, NewValue: TX3DTime;
const TimeIncrease: TFloatTime; const ResetTime: boolean): boolean;
var
NewIsActive: boolean;
NewIsPaused: boolean;
NewElapsedTime: TFloatTime;
CycleTimeSend: boolean;
CycleTime: TFloatTime;
{ $define LOG_TIME_DEPENDENT_NODES}
{ Increase NewElapsedTime and ElapsedTimeInCycle, taking care of
CycleInterval and looping.
StopOnNonLoopedEnd says what to do if NewElapsedTime passed
CycleInterval and not looping.
May indicate that CycleTime should be send (by setting CycleTimeSend to true
and CycleTime value) if the *new* cycle started. This means
that new ElapsedTime reached the non-zero CycleInterval
and loop = TRUE. }
procedure IncreaseElapsedTime(const Increase: TFloatTime;
StopOnNonLoopedEnd: boolean);
begin
NewElapsedTime := NewElapsedTime + Increase;
FElapsedTimeInCycle := FElapsedTimeInCycle + Increase;
if FElapsedTimeInCycle > CycleInterval then
begin
if CycleInterval <> 0 then
begin
if FdLoop.Value then
begin
FElapsedTimeInCycle := FloatModulo(FElapsedTimeInCycle, CycleInterval);
{ Send the time value when the cycle started, which was
a little earlier than CurrentTime: earlier by ElapsedTimeInCycle. }
CycleTimeSend := true;
CycleTime := CurrentTime.Seconds - ElapsedTimeInCycle;
end else
begin
if StopOnNonLoopedEnd then
NewIsActive := false;
end;
end else
begin
{ for cycleInterval = 0 this always remains 0 }
FElapsedTimeInCycle := 0;
if (not FdLoop.Value) and StopOnNonLoopedEnd then
NewIsActive := false;
end;
end;
end;
begin
{ if ResetTime, then TimeIncrease must be always exactly 0 }
Assert((not ResetTime) or (TimeIncrease = 0));
{$ifdef LOG_TIME_DEPENDENT_NODES}
if Log then
WritelnLog('TimeDependentNodes', Format('%s: time changes from %f (by +%f) to %f. Cycle interval: %f. Before state: active %s, paused %s, loop %s',
[ Node.NodeTypeName,
OldValue.Seconds, TimeIncrease, NewValue.Seconds,
CycleInterval,
BoolToStr[IsActive], BoolToStr[IsPaused], BoolToStr[FdLoop.Value]
]));
{$endif}
CurrentTime := NewValue;
if (Fdenabled <> nil) and (not Fdenabled.Value) then
begin
IsActive := false;
Exit;
end;
{ Note that each set of IsActive, IsPaused, ElapsedTime may generate events.
So we cannot carelessly set them many times in this method,
as double events are bad (besides possible unneeded overhead with
propagating them, route ignore events at the same timestamp,
since they may indicate loops in routes).
Solution: below we will operate on local copies of these variables,
like NewIsActive, NewIsPaused etc.
Only at the end of this method we will actually set the properties,
causing events (if their values changed). }
{ For ResetTime, set time-dependent node properties to default
(like after TTimeDependentNodeHandler creation) at the beginning. }
if ResetTime then
begin
NewIsActive := false;
NewIsPaused := false;
NewElapsedTime := 0;
FElapsedTimeInCycle := 0;
end else
begin
NewIsActive := IsActive;
NewIsPaused := IsPaused;
NewElapsedTime := ElapsedTime;
{ Leave ElapsedTimeInCycle as it was }
end;
CycleTimeSend := false;
if not NewIsActive then
begin
if (NewValue.Seconds >= FdStartTime.Value) and
( (NewValue.Seconds < FdStopTime.Value) or
{ stopTime is ignored if it's <= startTime }
(FdStopTime.Value <= FdStartTime.Value) ) and
{ avoid starting the movie if it should be stopped according
to loop and cycleInterval }
not ( (NewValue.Seconds - FdStartTime.Value >
CycleInterval) and
(not FdLoop.Value) ) then
begin
NewIsActive := true;
NewIsPaused := false;
NewElapsedTime := 0;
FElapsedTimeInCycle := 0;
{ Do not advance by TimeIncrease (time from last Time),
advance only by the time passed since startTime. }
IncreaseElapsedTime(NewValue.Seconds - FdStartTime.Value, true);
if not CycleTimeSend then
begin
{ Then we still have the initial cycleTime event to generate
(IncreaseElapsedTime didn't do it for us).
This should be the "time at the beginning of the current cycle".
Since IncreaseElapsedTime didn't detect a new cycle,
so NewElapsedTime = NewValue.Seconds - FdStartTime.Value fits
(is < ) within the CycleInterval. So startTime is the beginning
of our cycle.
Or StartedNewCycle = false may mean that CycleInterval is zero
or loop = FALSE. We will check later (before actually sending
cycleTime) that sensor is active, and if it's active ->
we still should make the initial cycleTime.
So in both cases, proper cycleTime is startTime. }
CycleTimeSend := true;
CycleTime := FdStartTime.Value;
end;
Result := true;
end;
end else
if NewIsPaused then
begin
if (NewValue.Seconds >= FdResumeTime.Value) and
(FdResumeTime.Value > FdPauseTime.Value) then
begin
NewIsPaused := false;
{ Advance only by the time passed since resumeTime. }
IncreaseElapsedTime(NewValue.Seconds - FdResumeTime.Value, true);
Result := true;
end;
end else
begin
Result := true;
if (NewValue.Seconds >= FdStopTime.Value) and
{ stopTime is ignored if it's <= startTime }
(FdStopTime.Value > FdStartTime.Value) then
begin
NewIsActive := false;
{ advance only to the stopTime }
if TimeIncrease <> 0 then
IncreaseElapsedTime(TimeIncrease -
(NewValue.Seconds - FdStopTime.Value), false);
end else
if (NewValue.Seconds >= FdPauseTime.Value) and
(FdPauseTime.Value > FdResumeTime.Value) then
begin
NewIsPaused := true;
{ advance only to the pauseTime }
if TimeIncrease <> 0 then
IncreaseElapsedTime(TimeIncrease -
(NewValue.Seconds - FdPauseTime.Value), false);
end else
begin
{ active and not paused movie }
if ResetTime then
begin
NewElapsedTime := 0;
FElapsedTimeInCycle := 0;
end else
IncreaseElapsedTime(TimeIncrease, true);
end;
end;
{ now set actual IsActive, IsPaused, ElapsedTime properties from
their NewXxx counterparts. We take care to set them in proper
order, to send events in proper order:
if you just activated the movie, then isActive should be sent first,
before elapsedTime.
If the movie was deactivated, then last elapsedTime should be sent last.
Send cycleTime only if NewIsActive, and after sending isActive = TRUE. }
if NewIsActive then
begin
IsActive := NewIsActive;
if not NewIsPaused then
begin
IsPaused := NewIsPaused;
ElapsedTime := NewElapsedTime;
end else
begin
ElapsedTime := NewElapsedTime;
IsPaused := NewIsPaused;
end;
if CycleTimeSend and (EventCycleTime <> nil) then
EventCycleTime.Send(CycleTime, CurrentTime);
end else
begin
if not NewIsPaused then
begin
IsPaused := NewIsPaused;
ElapsedTime := NewElapsedTime;
end else
begin
ElapsedTime := NewElapsedTime;
IsPaused := NewIsPaused;
end;
IsActive := NewIsActive;
end;
{ This will be true in most usual situations, but in some complicated
setups sending isActive/isPaused/elapsedTime (and sending elapsedTime
causes sending other events for TimeSensor) may cause sending another
event to the same node, thus calling SetTime recursively,
and changing values at the end. Example: rrtankticks when often
clicking on firing the cannon. So these assertions do not have to be
true in complicated scenes.
Assert(IsActive = NewIsActive);
Assert(IsPaused = NewIsPaused);
Assert(ElapsedTime = NewElapsedTime);
}
{$ifdef LOG_TIME_DEPENDENT_NODES}
if Log then
WritelnLog('TimeDependentNodes', Format('%s: after: active %s, paused %s',
[ Node.NodeTypeName,
BoolToStr[IsActive],
BoolToStr[IsPaused]]));
{$endif}
end;
{ VRML/X3D nodes ------------------------------------------------------------- }
procedure TAbstractTimeDependentNode.CreateNode;
begin
inherited;
FFdLoop := TSFBool.Create(Self, 'loop', false);
Fields.Add(FFdLoop);
FFdPauseTime := TSFTime.Create(Self, 'pauseTime', 0);
FdPauseTime.ChangesAlways := [chTimeStopStart];
Fields.Add(FFdPauseTime);
{ X3D specification comment: (-Inf,Inf) }
FFdResumeTime := TSFTime.Create(Self, 'resumeTime', 0);
FdResumeTime.ChangesAlways := [chTimeStopStart];
Fields.Add(FFdResumeTime);
{ X3D specification comment: (-Inf,Inf) }
FFdStartTime := TSFTimeIgnoreWhenActive.Create(Self, 'startTime', 0);
FdStartTime.ChangesAlways := [chTimeStopStart];
Fields.Add(FFdStartTime);
{ X3D specification comment: (-Inf,Inf) }
FFdStopTime := TSFStopTime.Create(Self, 'stopTime', 0);
FdStopTime.ChangesAlways := [chTimeStopStart];
Fields.Add(FFdStopTime);
{ X3D specification comment: (-Inf,Inf) }
FEventElapsedTime := TSFTimeEvent.Create(Self, 'elapsedTime', false);
Events.Add(FEventElapsedTime);
FEventIsActive := TSFBoolEvent.Create(Self, 'isActive', false);
Events.Add(FEventIsActive);
FEventIsPaused := TSFBoolEvent.Create(Self, 'isPaused', false);
Events.Add(FEventIsPaused);
DefaultContainerField := 'children';
FTimeDependentNodeHandler := TTimeDependentNodeHandler.Create;
FTimeDependentNodeHandler.Node := Self;
FTimeDependentNodeHandler.Fdloop := FdLoop;
FTimeDependentNodeHandler.FdpauseTime := FdPauseTime;
FTimeDependentNodeHandler.FdresumeTime := FdResumeTime;
FTimeDependentNodeHandler.FdstartTime := FdStartTime;
FTimeDependentNodeHandler.FdstopTime := FdStopTime;
{ descendants shoud set FTimeDependentNodeHandler.Fdenabled }
FTimeDependentNodeHandler.EventisActive:= EventisActive;
FTimeDependentNodeHandler.EventisPaused := EventisPaused;
FTimeDependentNodeHandler.EventelapsedTime := EventelapsedTime;
{ descendants shoud set FTimeDependentNodeHandler.EventCycleTime }
{ descendants shoud override CycleInterval }
FTimeDependentNodeHandler.OnCycleInterval := @CycleInterval;
end;
destructor TAbstractTimeDependentNode.Destroy;
begin
FreeAndNil(FTimeDependentNodeHandler);
inherited;
end;
function TAbstractTimeDependentNode.GetTimeDependentNodeHandler: TTimeDependentNodeHandler;
begin
Result := FTimeDependentNodeHandler;
end;
procedure TTimeSensorNode.CreateNode;
begin
inherited;
FFdCycleInterval := TSFTimeIgnoreWhenActive.Create(Self, 'cycleInterval', 1);
Fields.Add(FFdCycleInterval);
{ X3D specification comment: (0,Inf) }
FEventCycleTime := TSFTimeEvent.Create(Self, 'cycleTime', false);
Events.Add(FEventCycleTime);
{ cycleTime_changed name is used e.g. by
www.web3d.org/x3d/content/examples/Basic/StudentProjects/WallClock.x3d }
FEventCycleTime.AddAlternativeName('cycleTime_changed', 0);
FEventFraction_changed := TSFFloatEvent.Create(Self, 'fraction_changed', false);
Events.Add(FEventFraction_changed);
FEventTime := TSFTimeEvent.Create(Self, 'time', false);
Events.Add(FEventTime);
FFdEnabled := TSFBool.Create(Self, 'enabled', true);
Fields.Add(FFdEnabled);
DefaultContainerField := 'children';
{ set TimeDependentNodeHandler }
TimeDependentNodeHandler.Fdenabled := FdEnabled;
TimeDependentNodeHandler.EventCycleTime := EventCycleTime;
{ On receiving new elapsedTime, we send other continous events. }
EventelapsedTime.OnReceive.Add(@EventElapsedTimeReceive);
end;
function TTimeSensorNode.CycleInterval: TFloatTime;
begin
Result := FdCycleInterval.Value;
end;
class function TTimeSensorNode.ClassNodeTypeName: string;
begin
Result := 'TimeSensor';
end;
class function TTimeSensorNode.URNMatching(const URN: string): boolean;
begin
Result := (inherited URNMatching(URN)) or
(URN = URNVRML97Nodes + ClassNodeTypeName) or
(URN = URNX3DNodes + ClassNodeTypeName);
end;
procedure TTimeSensorNode.EventElapsedTimeReceive(
Event: TX3DEvent; Value: TX3DField; const Time: TX3DTime);
var
Fraction: Single;
begin
if FdEnabled.Value then
begin
Fraction := TimeDependentNodeHandler.ElapsedTimeInCycle / CycleInterval;
if (Fraction = 0) and (Time.Seconds > FdStartTime.Value) then Fraction := 1;
Eventfraction_changed.Send(Fraction, Time);
EventTime.Send(Time.Seconds, Time);
end;
end;
var
FakeX3DTime: TX3DTime;
procedure TTimeSensorNode.FakeTime(const Time: TFloatTime; const Loop: boolean);
begin
if FdEnabled.Value then
begin
{ This method may be called with any Time values sequence, e.g. decreasing.
So we have to avoid the normal VRML/X3D mechanism that prevents
passing ROUTEs with an older timestamp (see TX3DRoute.EventReceive).
We do this by simply using a fake timestamp that is always growing. }
if FakeX3DTime.PlusTicks <> High(FakeX3DTime.PlusTicks) then
Inc(FakeX3DTime.PlusTicks) else
begin
FakeX3DTime.Seconds += 1;
FakeX3DTime.PlusTicks := 0;
end;
if Loop then
EventFraction_changed.Send(Frac(Time / CycleInterval), FakeX3DTime) else
EventFraction_changed.Send(Clamped(Time / CycleInterval, 0, 1), FakeX3DTime);
EventTime.Send(Time, FakeX3DTime);
end;
end;
procedure RegisterTimeNodes;
begin
NodesManager.RegisterNodeClasses([
TTimeSensorNode
]);
end;
{$endif read_implementation}
|