Поделиться через


Programming navigation in WSS 3.0

Modifying navigation elements through code is quite tricky in WSS 3.0. Here are few scenarios, I’ve tried to cover that I think will be helpful in playing around with WSS 3.0 navigation elements. I’ve used the top navigation bar in these examples. But the same code should work for quick launch as well.

BTW: Being in Microsoft SharePoint Developer support, I figure quick launch customization is quite popular J

Add simple navigation nodes

Here’s my previous post on this scenario. Below is how you can add top navigation nodes for non-SharePoint sites.

using (SPSite site = new SPSite("<sharepoint site url>"))

{

using (SPWeb web = site.OpenWeb())

{

SPNavigationNodeCollection nodes = web.Navigation.TopNavigationBar;

SPNavigationNode newNode1 = new SPNavigationNode("MS TechNet", "https://technet.microsoft.com", true);

SPNavigationNode newNode2 = new SPNavigationNode("SharePoint SDK Online", "https://msdn.microsoft.com/sharepoint", true);

nodes.AddAsFirst(newNode2);

nodes.AddAsLast(newNode1);

MessageBox.Show("Nodes added");

}

}

This adds very simple top navigation nodes, one at the start and the other at the end.

Note: If the node you are adding is not a sub-site within that particular site collection, you must instantiate the SPNavigationNode object as highlighted above. The URL attribute must begin with “https://” and isExternal attribute must be set to true.

If isExternal attribute is not set or set to false, the OM at runtime adds a “/” before the URL attribute you provide. So, if you provide "https://technet.microsoft.com" it becomes “/https://technet.microsoft.com” and will error out.

If the URL attribute is provided like “www.microsoft.com”, SharePoint injects the root site collection URL and the result will become “https://<sharepoint site url>/www.microsoft.com” – which is wrong!

A bit tricky navigation node addition/modification

To add node “between” those that are already present, use a code like the following:

using (SPSite site = new SPSite("<sharepoint site url>"))

{

using (SPWeb web = site.OpenWeb())

{

SPNavigationNodeCollection nodes = web.Navigation.TopNavigationBar;

SPNavigationNode newNode = new SPNavigationNode("SilverLight 2.0", "https://silverlight.net",true); // test

SPNavigationNode preNode1 = nodes.Navigation.GetNodeByUrl("/Sub1/default.aspx");

foreach (SPNavigationNode preNode2 in nodes)

{

if (preNode2.Url == "https://msdn.microsoft.com/sharepoint")

{

nodes.Add(newNode, preNode2);

break;

}

}

nodes.Add(newNode, preNode1);

MessageBox.Show("Node adjusted");

}

}

Note: The one highlight in green shows how to add a node after a node that’s NOT a node pointing to a sub-site within that site collection. The one highlighted in pink shows how to add a node after a node that’s a node pointing to a sub-site within that site collection.

Rearrange navigation nodes

Check out my previous post on sorting quick launch. You can do rearrangement that way. I have used alphabetical order, but that can be done using any measurable parameter that we can use as a reference.

A customer sent me the below code:

using (SPSite site = new SPSite("<sharepoint site url>"))

{

using (SPWeb web = site.OpenWeb())

{

SPNavigationNodeCollection nodes = web.Navigation.TopNavigationBar;

SPNavigationNode delNode = nodes.Navigation.GetNodeByUrl("/Sub4/default.aspx");

nodes.Delete(delNode);

SPNavigationNode newNode = new SPNavigationNode("Sub4", "/Sub4/default.aspx");

SPNavigationNode preNode = nodes.Navigation.GetNodeByUrl("/Sub1/default.aspx");

nodes.Add(newNode, preNode);

MessageBox.Show("Rearranged");

}

}

While the above code looks good, the SPNavigationNode object provides three variations of “Move” methods as SPNavigationNodeCollection does for “Add”. Below is a sample:

using (SPSite site = new SPSite("<sharepoint site url>"))

{

using (SPWeb web = site.OpenWeb())

{

SPNavigationNodeCollection nodes = web.Navigation.TopNavigationBar;

SPNavigationNode nodeToMove = nodes.Navigation.GetNodeByUrl("/Sub4/default.aspx");

SPNavigationNode preNode = nodes.Navigation.GetNodeByUrl("/Sub1/default.aspx");

nodeToMove.Move(nodes, preNode);

MessageBox.Show("Rearranged");

}

}

All the other rearrangement can be carried out in the same manner.

Deleting a node

We have to carefully use the SPNavigationNodeCollection for this. As any other collection in SharePoint, if we directly loop through this collection using a foreach loop, we’ll run into code failures because of change in the collection that’s being enumerated. So, use a for loop to run through the total number of SPNavigationNode in the collection. And then, based on some checks, delete the nodes. Sample below:

using (SPSite site = new SPSite("<sharepoint site url>"))

{

using (SPWeb web = site.OpenWeb())

{

SPNavigationNodeCollection nodes = web.Navigation.TopNavigationBar;

for (int i = nodes.Count-1; i >= 0; i--)

{

if (nodes[i].Url.Equals("https://silverlight.net"))

{

nodes[i].Delete();

}

}

MessageBox.Show("Node(s) deleted");

}

}

To Summarize

Navigation in WSS 3.0 is far more extensible and customizable than it was in WSS 2.0. The sample show above should work for quick launch navigation without problems. MOSS 2007 sites follow a different navigation approach. More details in my previous post.

Comments

  • Anonymous
    July 04, 2008
    Hi, Can I program the navigation bar to be dynamic ? Meaning based on Roles it updates itself.... .. or is this something Global to the site ? Cheers, /Sf

  • Anonymous
    July 04, 2008
    Hi, Can I program the navigation bar to be dynamic ? Meaning based on Roles it updates itself.... .. or is this something Global to the site ? Cheers, /Sf

  • Anonymous
    July 11, 2008
    Is is possible to do fly-outs on the quick launch?

  • Anonymous
    July 17, 2008
    Will you please provide us screen shots for the same. i.e for navigation menu and submenus Thanks.

  • Anonymous
    July 23, 2008
    Hi Could you pls tell me how to get all sites name from navigation? I have this code but it is returning all the sites and pages also SPNavigationNodeCollection quickLaunchNodes = CURRENT_WEB.Navigation.QuickLaunch;   foreach (SPNavigationNode node in quickLaunchNodes)            { return node.title.tostring(); }

  • Anonymous
    October 13, 2008
    hi all, I wanted to create topnavigationbar in sharepoint following structure : Level 1   level  1.1         level 1.1.1                  level 1.1.1.1         level 1.1.2  level  1.2 I wrote code below : using (SPSite site = new SPSite(this.Context.Request.Url.ToString()))        {        using (SPWeb web = site.OpenWeb())               {               SPNavigationNodeCollection node = web.Navigation.TopNavigationBar;               SPNavigationNode level_1 = new SPNavigationNode("level 1", "");               node.AddAsFirst(level_1);               SPNavigationNode level_11 = new SPNavigationNode("level 1.1", "");               level_1.Children.AddAsFirst(level_11);              SPNavigationNode level_111 = new SPNavigationNode("level 1.1.1", "");              level_11.Children.AddAsFirst(level_111);              SPNavigationNode level_1111 = new SPNavigationNode("level 1.1.1.1", "");              level_111.Children.AddAsFirst(level_1111);             SPNavigationNode level_112 = new SPNavigationNode("level 1.1.2", "");            level_11.Children.AddAsFirst(level_112);           SPNavigationNode level_12 = new SPNavigationNode("level 1.2", "");          level_1.Children.AddAsFirst(level_12);          web.Update();            }   } But it only showed : Level 1     level 1.1     level 1.2 Can you help me, please . Thanks.

  • Anonymous
    October 16, 2008
    Hi Cuong, You need to set "StaticDisplayLevels" & "MaximumDynamicDisplayLevels" to include more levels depending on your requirement. Refer this article: http://blogs.msdn.com/sharepoint/archive/2007/04/26/customizing-the-quick-launch-menu-adding-fly-out-menus-to-sharepoint-navigation.aspx for more information. Cheers, Sridhar

  • Anonymous
    March 05, 2009
    Hi Sridhar, How can i add submenu in existing Top Navigation Node? I did a lot but it ends up creating one additional Top Node instead of sub-menu.? Thanks