长期的BUG:这是我在2003年Borland新闻组第一次提出的问题:
> Accepted fix for WindowState = wsMaximized?
然后在2006年再次:
> wsMaximized breaks it,NOT caused by Position=poScreenCenter,reproducible dfm
然后在2008年再次:
> Forms not starting maximized
有人在2012年的Embarcadero论坛上问过:
> Thread: Application not starting with maximized window
现在是时候把这个18岁的BUG移植到Stackoverflow。也许有人最终想出了一个解决方法。
重现步骤:
我的帖子包含六个故障模式,但最简单的是:
>在表单上拖放标签和编辑:
>为TEdit添加OnEnter事件:
procedure TForm1.Edit1Enter(Sender: TObject);begin Label1.Font.Style := Label1.Font.Style + [fsBold];end;
>并设置形式:
> windowstate to wsMaximized
> autoScroll为False
和bazinga,失败。
从2008年的帖子开始的另一组步骤之一:
Create a new app and a form. Set the form to maximized (windowstate = wsMaximized) at design time. Drop a ListVIEw control on the formDuring OnShow,add 20 empty items to the List vIEw:
06001
Set the form’s autoScroll property to false (autoScroll = False) at design time
当然,我不是在“固定在RadStudio的版本n,只是使用”。我正在寻找一个实际的修复(如果有一个);这可能包括引用对VCL源的相关更改,当CodeGear终于修复它。 (如果它甚至固定)。
注意:将位置从podesigned更改为其他位置不会解决它。
解决方法
一个可怕的,丑陋的,可怕的,令人厌恶的,解决方法我一直在使用是在OnShow期间启动一个定时器,然后当定时器触发时,最大化形式:
procedure TForm1.tmrVclMaximizeHackTimer(Sender: TObject);begin Self.windowstate := wsMaximized;end;
我后来改进了这个黑客在OnShow期间发布消息;其基本上与定时器消息相同,而不必使用定时器:
const WM_MaximizeWindow = WM_APP + ;procedure TForm1.FormShow(Sender: TObject);begin if (Self.windowstate = wsMaximized) then begin Self.windowstate := wsnormal; PostMessage(Self.Handle,WM_MaximizeWindow,0); end;end;private procedure WMMaximizeWindow(var Message: TMessage); message WM_MaximizeWindow;procedure TForm1.WMMaximizeWindow(var Message: TMessage);begin Self.windowstate := wsMaximized;end;
有时我发明Delphi从来没有做过的OnAfterShow事件:
const WM_AfterShow = WM_APP + ;procedure TForm1.FormShow(Sender: TObject);begin PostMessage(Self.Handle,WM_AfterShow,0); if (Self.windowstate = wsMaximized) then begin Self.windowstate := wsnormal; FMaximiZeneeded := True; end;end;private procedure WMAfterShow(var Message: TMessage); message WM_AfterShow;procedure TForm1.WMAfterShow(var Message: TMessage);begin if FMaximiZeneeded then begin FMaximiZeneeded := False; Self.windowstate := wsMaximized; end;end;
但没有黑客比黑客更好。
解决方法 我只测试了第一个再现情况(与D7,D2007,XE2),并能够复制D7和D2007但不是与XE2的问题。问题,我看到,是标签,其字体更改,请求其父重新对齐自身。这最终导致在窗体(在TWinControl.AdjustSize)中具有恢复的宽度/高度的SetwindowPos调用,即使窗体已经最大化 – 这导致奇怪的,行为上最大化但是没有在视觉上最大化,形成坐在屏幕上。
我追踪了D2007和XE2中的代码,以便能够想出什么是不同的。 TWinControl.AlignControls中的代码在两个版本之间不同。最重要的是最后一句话。
D2007:
procedure TWinControl.AlignControls(AControl: TControl; var Rect: TRect); .. { Apply any constraints } if Showing then AdjustSize;end;
XE2:
procedure TWinControl.AlignControls(AControl: TControl; var Rect: TRect); .. // Apply any constraints if FautoSize and Showing then DoAdjustSize;end;
我希望这可以,不知何故,帮助你设计/决定使用什么解决方法。
解决方法我可以建议(虽然我没有测试它throughly)是强制显示形式最大化早:
procedure TForm1.FormCreate(Sender: TObject);var wplc: twindowPlacement;begin if not autoScroll and (windowstate = wsMaximized) then begin wplc.length := SizeOf(wplc); GetwindowPlacement(Handle,@wplc); wplc.rcnormalposition.Right := wplc.rcnormalposition.left + WIDth; wplc.rcnormalposition.Bottom := wplc.rcnormalposition.top + Height; wplc.showCmd := SW_MAXIMIZE; SetwindowPlacement(Handle,@wplc); end;end;
上面的工作原理是因为它在VCL为表单设置可见标志之前强制将焦点设置为编辑控件(OnEnter事件)。反过来,标签的对齐请求不会导致表单大小调整。另外,由于在VCL调用ShowWindow时窗体的窗口已经可见,它不会导致窗体在任何阶段以恢复的状态显示。
但是,我不知道它是否会帮助不同的复制场景。
最后,虽然我可以看到,在新的Delphi版本中的行为被纠正,我不会认为这是一个错误在VCL。在我看来,用户代码应该负责不导致窗口调整,而窗口显示状态正在改变。我将采取的具体情况的行动的过程将推迟修改标签的字体,直到VCL完成显示窗体。
总结以上是内存溢出为你收集整理的delphi – wsMaximized形式不会出现最大化全部内容,希望文章能够帮你解决delphi – wsMaximized形式不会出现最大化所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)