[英]How to reduce this IF-Else ladder in c#
这是我创建的IF -Else梯形图,用于将第一个可视控件集中在我的窗体上。根据要求,任何控件都可以隐藏在窗体上。所以我必须找到第一个可见的控件并将其聚焦。
if (ddlTranscriptionMethod.Visible)
{
ddlTranscriptionMethod.Focus();
}
else if (ddlSpeechRecognition.Visible)
{
ddlSpeechRecognition.Focus();
}
else if (!SliderControl1.SliderDisable)
{
SliderControl1.Focus();
}
else if (ddlESignature.Visible)
{
ddlESignature.Focus();
}
else
{
if (tblDistributionMethods.Visible)
{
if (chkViaFax.Visible)
{
chkViaFax.Focus();
}
else if (chkViaInterface.Visible)
{
chkViaInterface.Focus();
}
else if (chkViaPrint.Visible)
{
chkViaPrint.Focus();
}
else
{
chkViaSelfService.Focus();
}
}
}
有没有其他方法可以做到这一点。 我认为使用LINQ会占用性能,因为我必须遍历整个页面集合。 我在页面很深,有主页。请建议。
我觉得你的树很好。 这当然看起来像一个可以简化的逻辑树,你有一个良好的嗅觉怀疑它。 但是,似乎逻辑树反映了您的需求。 这个逻辑确实很复杂 ,这是C#为您提供处理这种情况的条件框架。 我不认为它可以改进。
如果您有一个应该具有焦点的简单控件列表,并且您希望将焦点放在列表中的第一个可见控件上,则可以执行以下操作:
(From c in ListOfControls
Where c.visible = true
Select c).First.Focus();
但是,看起来你有一些额外的标准,所以这是行不通的。
两种方法:
如果您只是尝试将第一个可见控件聚焦在窗体上,那么我将用一个循环替换整个梯形图:
foreach (Control c in Controls)
{
if (c.Visible)
{
c.Focus();
break;
}
}
如果需要聚焦内部控件,请使用递归方法:
bool FocusFirst(ControlCollection controls)
{
foreach (Control c in controls)
{
if (c.Visible)
{
c.Focus();
FocusFirst(c.Controls);
break;
}
}
}
您可以在符合标准后返回,例如:
if (ddlTranscriptionMethod.Visible)
{
ddlTranscriptionMethod.Focus();
return;
}
if (ddlSpeechRecognition.Visible)
{
ddlSpeechRecognition.Focus();
return;
}
等等..
您可以迭代控件并设置焦点(如果可见)。 但我建议您使用状态模式以获得更好的代码可读性。
您所做的就是设置焦点,即客户端功能。 我个人会在javascript(使用jQuery)中这样做。 未设置为可见的ASP.NET控件不会在HTML中呈现,因此您可以查找这些元素的存在,或者如果您只是在寻找第一个可见的启用控件,则可能会有更简单的方法。
怎么样跳转似乎你可以在这里使用它。
当要评估的项目列表很大(有时它非常大)时,我尝试将评估顺序与条件逻辑分开,如下所示:
List<WebControl> wcBasics = new List<WebControl>();
wcBasics.Add(ddlTranscriptionMethod);
wcBasics.Add(ddlSpeechRecognition);
wcBasics.Add(ddlESignature);
List<CheckBox> checks = new List<CheckBox>();
checks.Add(chkViaFax);
checks.Add(chkViaInterface);
checks.Add(chkViaPrint);
private void Focus()
{
foreach (WebControl c in wcBasics)
if (c.Visible) {
c.Focus();
return;
}
if (!tblDistributionMethods.Visible) return;
foreach (CheckBox chk in checks)
if (chk.Visible) {
chk.Focus();
return;
}
}
chkViaSelfService.Focus();
}
这是一款经典的状态机。 通过设置机器,它可以使代码更容易理解和维护, 尽管它可能会增加总线数 。
我不会详细介绍您的代码。 通过确定用户正在操作的状态,您可以以可理解的特定状态方式以编程方式更改不同的值。 (下面的伪代码)
enum FormStates
{
Initial_View
Working_View
Edit_View
Shutdown_View
};
{ // Somewhere in code
switch (theCurrentState)
{
case Initial_View :
Control1.Enabled = true;
Control2.Enabled = true;
theCurrentState = Working_View;
break;
case Working_View
if (string.empty(Contro1.Text) == false)
{
Control2.Enabled = false;
Speachcontrol.Focus();
theCurrentState = Edit_view;
}
else // Control 2 is operational
{
Control1.Enabled = false;
SliderControl.Focus();
}
case Edit_View:
...
break;
break;
}
通过按逻辑步骤组织代码,可以更轻松地添加更多状态,而不会危及巨大的if / else。
这是对问题略有不同的看法。 首先,定义一个接口来表示可能会或可能不会聚焦的控件:
public interface IFormControl
{
bool Focus();
}
然后创建一个处理简单案例的实现:
public class FormControl : IFormControl
{
private readonly Control _control;
public FormControl(Control control)
{
_control = control;
}
public bool Focus()
{
if(_control.Visible)
{
_control.Focus();
}
return _control.Visible;
}
}
并创建另一个处理更困难的案例:
public class DependentFormControl : IFormControl
{
private readonly Control _control;
private readonly Func<bool> _prerequisite;
public DependentFormControl(Control control, Func<bool> prerequisite)
{
_control = control;
_prerequisite = prerequisite;
}
public bool Focus()
{
var focused = _prerequisite() && _control.Visible;
if(focused)
{
_control.Focus();
}
return focused;
}
}
然后,创建一个扩展方法,将焦点设置在序列中的第一个控件上:
public static void FocusFirst(this IEnumerable<IFormControl> formControls)
{
var focused = false;
foreach(var formControl in formControls)
{
if(formControl.Focus())
{
break;
}
}
}
最后,创建要集中的控件集:
var controls = new FormControl[]
{
new FormControl(ddlTranscriptionMethod),
new FormControl(ddlSpeechRecognition),
new DependentFormControl(SliderControl1, () => !SliderControl1.SliderDisable),
new FormControl(ddlESignature),
new DependentFormControl(chkViaFax, () => tblDistributionMethods.Visible),
new DependentFormControl(chkViaInterface, () => tblDistributionMethods.Visible),
new DependentFormControl(chkViaPrint, () => tblDistributionMethods.Visible),
new DependentFormControl(chkViaSelfService, () => tblDistributionMethods.Visible)
};
controls.FocusFirst();
您可以在页面加载时创建控件集,并在必要时调用.FocusFirst()
。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.