9

背景

我使用一些 FireMonkey 控件创建了一个 GUI。

  • 一些控件是动画的,它们的外观会自动更新。
  • 某些控件仅响应用户交互(滑块等)而更新。

问题

与用户控件的交互阻止了对动画控件的更新,从而导致动画不连续。

故障动画的视频

上面视频中的动画控件由 TTimer 组件驱动。使用 FireMonkey 的动画组件时问题仍然存在。

调查

滑块控件在调整时调用 Repaint()。平滑调整滑块将生成密集的 Repaint() 调用流,这些调用会阻止其他控件的更新。

该怎么办?

在一个控件不断更新时冻结动画不适合我的应用程序。我的第一个想法是将 Repaint() 调用换成类似于 VCL Invalidate() 方法的东西,但 FireMonkey 没有任何可比的 AFAIK。

这个问题有很好的解决方法吗?

4

1 回答 1

4

正如 Arnaud Bouchez 在上面的评论中建议的那样,我创建了一个基于计时器的重绘方法。到目前为止,它似乎工作。

代码

unit FmxInvalidateHack;

interface

uses
  Fmx.Types;

procedure InvalidateControl(aControl : TControl);


implementation

uses
  Contnrs;

type
  TInvalidator = class
  private
  protected
    Timer : TTimer;
    List  : TObjectList;
    procedure Step(Sender : TObject);
  public
    constructor Create;
    destructor Destroy; override;

    procedure AddToQueue(aControl : TControl);
  end;

var
  GlobalInvalidator : TInvalidator;

procedure InvalidateControl(aControl : TControl);
begin
  if not assigned(GlobalInvalidator) then
  begin
    GlobalInvalidator := TInvalidator.Create;
  end;
  GlobalInvalidator.AddToQueue(aControl);
end;


{ TInvalidator }

constructor TInvalidator.Create;
const
  FrameRate = 30;
begin
  List  := TObjectList.Create;
  List.OwnsObjects := false;

  Timer := TTimer.Create(nil);
  Timer.OnTimer  := Step;
  Timer.Interval := round(1000 / FrameRate);
  Timer.Enabled  := true;
end;

destructor TInvalidator.Destroy;
begin
  Timer.Free;
  List.Free;
  inherited;
end;

procedure TInvalidator.AddToQueue(aControl: TControl);
begin
  if List.IndexOf(aControl) = -1 then
  begin
    List.Add(aControl);
  end;
end;

procedure TInvalidator.Step(Sender: TObject);
var
  c1: Integer;
begin
  for c1 := 0 to List.Count-1 do
  begin
    (List[c1] as TControl).Repaint;
  end;
  List.Clear;
end;


initialization

finalization
  if assigned(GlobalInvalidator) then GlobalInvalidator.Free;

end.

==

用法

可以通过调用重新绘制控件:

InvalidateControl(MyControl);

InvalidateControl() 过程不会立即重新绘制控件。相反,它将控件添加到列表中。全局计时器稍后检查列表,调用 Repaint() 并从列表中删除控件。使用此方法,可以根据需要使控件无效,但不会像快速 Repaint() 调用那样阻止其他控件的更新。

于 2011-12-08T00:42:09.007 回答