关于delphi 中的canvas重影问题

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关于delphi 中的canvas重影问题相关的知识,希望对你有一定的参考价值。

大家好,我在delphi中的一个窗体上面用canvas来作图,由于画图的范围超过了窗体的范围,所以在窗体上面用了滚动条,但是当我向右拖动滚动条时,拖出来的部分却出现了重影,请问这个问题有办法来解决吗?

不要在窗体上用滚动条,在窗体上放一个 scrollBox,用scrollBox的canvas画图。 参考技术A 每滚动一下就重画一次。没有其他好方法

Delphi FMX组件重影去除子组件

【中文标题】Delphi FMX组件重影去除子组件【英文标题】:Delphi FMX component ghosting removing subcomponents 【发布时间】:2014-10-20 13:39:24 【问题描述】:

我创建了一个 FMX 组件,它是一个按钮数组。这有一个简单的列和行计数属性。 数组中的每个按钮本身就是一个组件,当行数或列数发生变化时,我会删除所有按钮并重新创建所需的数字。

我遇到的问题是,在删除组件后,它们继续在屏幕上出现鬼影。 例如,默认的列数和行数为 10,但对于手机格式,我在设计时将其更改为 5 x 5。这是结果...

我确信这与“Stored”属性有关,但是我确信我已经尝试在按钮和数组上打开和关闭它,没有任何区别。 组件的源代码(试图删除多余的代码):

type
  TLFArrayButton = class(TStyledControl)

  TLFButtonArray = class(TPanel)
  private
     Private declarations 
    FButtons: Array[0..9, 0..9] of TLFArrayButton;
    Sections: Array[0..3] of TRectangle;
    SectionsText: Array[0..3] of TText;
    FRowCount: integer;
    FColCount: integer;
    FFourUp: boolean;
    FOnArrayButtonClick: TLFButtonArrayButtonClick;
    procedure SetColCount(const Value: integer);
    procedure SetRowCount(const Value: integer);
    procedure CheckButtonArraySize;
    procedure PositionButtons;
    procedure FreeButtons;
    function GetButtonsText(Col, Row: Integer): String;
    procedure SetButtonsText(Col, Row: Integer; const Value: String);
    function GetButtonEnabled(Col, Row: Integer): Boolean;
    procedure SetButtonEnabled(Col, Row: Integer; const Value: Boolean);
    procedure SetFourUp(const Value: boolean);
    procedure OnButtonClick(Sender: TObject);

    procedure OnButtonMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Single);
    procedure SetupSectionHeadders;
    function GetButton(id: integer): TLFArrayButton;
    function GetSectionHeadding(idx: integer): string;
    procedure SetSectionHeadding(idx: integer; const Value: string);
    procedure SetOnArrayButtonClick(const Value: TLFButtonArrayButtonClick);
  protected
     Protected declarations 
    function GetStyleObject: TFmxObject; override;
    procedure Resize; override;
    procedure ApplyTriggers; virtual;
    procedure DoButtonClick(id: integer); virtual;
  public
     Public declarations 
    constructor Create(AOwner: TComponent); override;
    destructor Destroy; override;
    property Button[id: integer]: TLFArrayButton read GetButton;
    property ButtonEnabled[Col, Row: Integer]: Boolean read GetButtonEnabled write SetButtonEnabled;
    property ButtonsText[Col, Row: Integer]: String read GetButtonsText write SetButtonsText;
    property SectionHeadding[idx: integer]: string read GetSectionHeadding write SetSectionHeadding;
  published
     Published declarations 
    property FourUp: boolean read FFourUp write SetFourUp stored true default false;
    property ColCount: integer read FColCount write SetColCount stored true default 10;
    property RowCount: integer read FRowCount write SetRowCount stored true default 10;
    property OnArrayButtonClick: TLFButtonArrayButtonClick read FOnArrayButtonClick write SetOnArrayButtonClick;
 end;

procedure Register;

implementation

$R *.res

procedure Register;
begin
  RegisterComponents('LightFactoryFMX', [TLFButtonArray]);
  RegisterComponents('LightFactoryFMX', [TLFArrayButton]);
end;

 TLFButtonArray 

procedure TLFButtonArray.CheckButtonArraySize;
var
  r, c, i: Integer;
begin
  FreeButtons;
  InvalidateRect(RectF(0,0,width,height));
  i := 1;
  for r := 0 to FRowCount-1 do
  begin
    for c := 0 to FColCount-1 do
    begin
      FButtons[r,c] := TLFArrayButton.Create(Self);
      FButtons[r,c].Parent := Self;
      FButtons[r,c].Stored := false;
      FButtons[r,c].Locked := true;
      FButtons[r,c].ShowID := true;
      FButtons[r,c].ID := i;
      FButtons[r,c].OnClick := OnButtonClick;
      inc(i);
    end;
  end;
end;

constructor TLFButtonArray.Create(AOwner: TComponent);
begin
  inherited;
  Stored := false;
  StyleLookup := 'panelstyle';
  FFourUp := false;
  FColCount := 10;
  FRowCount := 10;
  CheckButtonArraySize;
  PositionButtons;
end;

procedure TLFButtonArray.FreeButtons;
var
  r, c: Integer;
begin
  for r := 0 to FRowCount-1 do
  begin
    for c := 0 to FColCount-1 do
    begin
      if assigned(FButtons[r, c]) then
      begin
        FButtons[r, c].Free;
        FButtons[r, c] := NIL;
      end;
    end;
  end;
end;

procedure TLFButtonArray.PositionButtons;
var
  r, c, s, b, loc_col, loc_row, num_per_sec: Integer;
  x, y, dw, dh: single;
  tmpBtn: TLFArrayButton;
begin
  if FFourUp then
  begin
    loc_col := ColCount div 2;
    loc_row := RowCount div 2;
    num_per_sec := loc_col * loc_row;
    //
    dw := (Width - INTER_SECTION_SPACING - ((FColCount+1) * BUTTON_SPACING)) / FColCount;
    dh := (Height - INTER_SECTION_SPACING - (Sections[0].Height * 2) - ((FRowCount+1) * BUTTON_SPACING)) / FRowCount;
    x := BUTTON_SPACING;
    y := BUTTON_SPACING;

    for s := 0 to 3 do
    begin
      x := Sections[s].Position.X;
      y := Sections[s].Position.Y + Sections[s].Height + BUTTON_SPACING;
      for b := 1 to num_per_sec do
      begin
        tmpBtn := GetButton((s*num_per_sec) + b);
        if assigned(tmpBtn) then
        begin
          tmpBtn.Position.X := x;
          tmpBtn.Position.Y := y;
          tmpBtn.Width := dw;
          tmpBtn.Height := dh;
          x := x + dw + BUTTON_SPACING;
          if x > (Sections[s].Position.X + Sections[s].Width) then
          begin
            x := Sections[s].Position.X;
            y := y + dh + BUTTON_SPACING;
          end;
        end;
      end;
    end;
  end
  else
  begin
    dw := (Width - ((FColCount+1) * BUTTON_SPACING)) / FColCount;
    dh := (Height - ((FRowCount+1) * BUTTON_SPACING)) / FRowCount;
    x := BUTTON_SPACING;
    y := BUTTON_SPACING;
    for r := 0 to FRowCount-1 do
    begin
      for c := 0 to FColCount-1 do
      begin
        FButtons[r,c].Position.X := x;
        FButtons[r,c].Position.Y := y;
        FButtons[r,c].Width := dw;
        FButtons[r,c].Height := dh;
        x := x + dw + BUTTON_SPACING;
      end;
      x := BUTTON_SPACING;
      y := y + dh + BUTTON_SPACING;
    end;
  end;
end;

procedure TLFButtonArray.Resize;
begin
  inherited;
  if FourUp then
    SetupSectionHeadders;
  PositionButtons;
end;

destructor TLFButtonArray.Destroy;
begin
  FreeButtons;
  inherited;
end;

procedure TLFButtonArray.SetColCount(const Value: integer);
begin
  if FColCount <> Value then
  begin
    if (Value > 0) and (Value < 11) then
    begin
      FColCount := Value;
      CheckButtonArraySize;
      PositionButtons;
    end;
  end;
end;

procedure TLFButtonArray.SetRowCount(const Value: integer);
begin
  if FRowCount <> Value then
  begin
    if (Value > 0) and (Value < 11) then
    begin
      FRowCount := Value;
      CheckButtonArraySize;
      PositionButtons;
    end;
  end;
end;

【问题讨论】:

相关***.com/q/27818697/960757. 【参考方案1】:

由于适用于 iOS/Android 的 Delphi 中的 ARC(自动引用计数),表单仍然引用按钮。因此FButtons[r, c].Free; 应该是FButtons[r, c].DisposeOf;

【讨论】:

ARC 的陷阱。您始终需要记住谁都在引用您的类/组件。为什么。因为调用 SomeComponent.Free 不再调用析构函数类,而只是将“SomeComponent”变量设置为 nil。只有当该类/组件的引用计数为 0 时,才会调用类/组件的析构函数。

以上是关于关于delphi 中的canvas重影问题的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Linux 中的 NodeJs Azure Functions 上修复“错误:/home/site/wwwroot/node_modules/canvas/build/Release/canv

在Canvas上放置.png文件的好方法?

新年第一个目标一张表盘串讲所有canves的知识点

canves基础自整理

canvas学习-----画直线

一个在h5的canvas元素中画扑克牌jquery插件实现