我有一个recusive方法,可以从XML文档创建一个无序列表。要检查我所在的节点,我使用查询字符串来匹配XML文档中的URL。
如果我位于其子节点上,我需要在父节点上添加类'current'。
像这样:
MenuItem 1 MenuItem 2 [class current] MenuItem 3 [class current] (selected node) MenuItem 4
我的XML文档是这样的:
<MenuItem Name="MenuItem 1" Url="MenuItem1.aspx"/>
<MenuItem Name="MenuItem 2" Url="MenuItem2.aspx">
<MenuItem Name="MenuItem 3" Url="MenuItem3.aspx" />
</MenuItem>
<MenuItem Name="MenuItem4" Url="MenuItem4.asp" />
我目前的代码如下:
foreach (XmlNode item in menuitems)
{
if (HttpContext.Current.Request.Url.AbsolutePath.ToLower() == item.Attributes["Url"].Value.ToLower())
{
writer.AddAttribute(HtmlTextWriterAttribute.Class, "current");
}
writer.RenderBeginTag(HtmlTextWriterTag.Li);
// And so on...
}
所以我需要某种函数来找到这个节点并在该节点上放置一个类。
如果您需要更多代码,信息或其他内容,请说明! : - )
谢谢!
答案 0 :(得分:1)
这不是一个特别优雅的解决方案,但如果我正确理解了问题,您可以创建一个方法来检查是否选择了子节点:
private static bool IsChildSelected(XmlNode item)
{
foreach(XmlNode child in item.ChildNodes)
{
if(HttpContext.Current.Request.Url.AbsolutePath.ToLower() == child.Attributes["Url"].Value.ToLower())
{
return true;
}
}
return false;
}
并在迭代节点时检查它:
foreach(XmlNode item in menuitems)
{
if(HttpContext.Current.Request.Url.AbsolutePath.ToLower() == item.Attributes["Url"].Value.ToLower()
|| IsChildSelected(item))
{
writer.AddAttribute(HtmlTextWriterAttribute.Class, "current");
}
writer.RenderBeginTag(HtmlTextWriterTag.Li);
}
答案 1 :(得分:1)
我尝试创建一个小单元测试来展示我的想法。基本上你得到当前选择的节点,然后递归地遍历其父节点。
private bool IsParentOf(XmlNode parentNode, XmlNode node)
{
if (node.ParentNode == null) return false;
return node.ParentNode == parentNode || IsParentOf(parentNode, node.ParentNode);
}
[TestMethod]
public void TestMethod1()
{
string xml =
@"
<MenuItem Name=""MenuItem 1"" Url=""MenuItem1.aspx""/>
<MenuItem Name=""MenuItem 2"" Url=""MenuItem2.aspx"">
<MenuItem Name=""MenuItem 3"" Url=""MenuItem3.aspx"" />
</MenuItem>
<MenuItem Name=""MenuItem4"" Url=""MenuItem4.asp"" />";
string url = "MenuItem3.aspx";
XmlDocument doc = new XmlDocument();
doc.LoadXml("<MenuItems>" + xml + "</MenuItems>");
var xPathFormat = "//MenuItem[@Url='{0}']";
var currentNode = doc.SelectSingleNode(string.Format(xPathFormat, url));
var menuItem1 = doc.SelectSingleNode(string.Format(xPathFormat, "MenuItem1.aspx"));
Assert.IsFalse(IsParentOf(menuItem1, currentNode));
var menuItem2 = doc.SelectSingleNode(string.Format(xPathFormat, "MenuItem2.aspx"));
Assert.IsTrue(IsParentOf(menuItem2, currentNode));
}
答案 2 :(得分:0)
感谢您的回答。但我解决了自己的问题。你们得到了很多努力。
这是我的解决方案:
System.Collections.Hashtable ht = new System.Collections.Hashtable();
ht.Add("Url", HttpContext.Current.Request.Url.AbsolutePath.ToLower());
XmlNode currentpage = Core.FindChildNode(item, "MenuItem", ht);
if (HttpContext.Current.Request.RawUrl.ToLower() == item.Attributes["Url"].Value.ToLower() || currentpage != null)
{
writer.AddAttribute(HtmlTextWriterAttribute.Class, "current");
}
这是我的FindChildNode方法
public static XmlNode FindChildNode(XmlNode parent, string name, System.Collections.Hashtable keyvaluecollection)
{
if (parent.NodeType != XmlNodeType.Element) return null;
XmlElement el = (XmlElement)parent;
XmlNodeList nodes = el.GetElementsByTagName(name);
foreach (XmlNode node in nodes)
{
if (node.NodeType == XmlNodeType.Element)
{
bool found = false;
foreach (string key in keyvaluecollection.Keys)
{
if (node.Attributes[key] != null && node.Attributes[key].Value == (string)keyvaluecollection[key])
{
found = true;
}
else
{
found = false;
break;
}
}
if (found) return node;
}
}
return null;
}
似乎像魅力一样工作! : - )