SharePoint is Flowers and Rainbows and Unicorns
13963 단어 SharePoint
PRINT THIS
EMAIL THIS
REPORT THIS
Join Now!
SharePoint is Flowers and Rainbows and Unicorns
SPGridView, SPMenuField, Grouping, Postback
Using the SPGridView and SPMenuField objects go a long way to encapsulate the functionality and look & feel of SharePoint into your custom controls, pages and web parts. This post describes a specific desired functionality, the obstacles encountered, and the way to get these controls to do what you want. For a detailed description of the basics of getting the SPGridView and SPMenuField working, check out these two posts by POWLO, I highly recommend them.
Goal: Create an SPGridView with an SPMenuField that allows grouping, with one of the menu items posting back to the page on which the SPGridView is instantiated.
That sounds like a lot so let’s take it step by step and work through each piece, gradually adding the next layer of complexity. To flesh out the scenario imagine you are a large organization, with several hundred sub-units and occasionally these sub-units become inactive and you want to archive them. Call our sub-units stores. Also imagine that a record of these stores is kept in a table with the following schema:
CREATE TABLE [dbo].[blog_post_1](
[ID] [int] IDENTITY(1,1) NOT NULL,
[StoreName] [nvarchar](50) NOT NULL,
[StateName] [nvarchar](25) NOT NULL,
[IsArchived] [bit] NOT NULL)
The first thing we want to do is just display all of our stores in a list, with the store name link to an editor page and a menu field also linking to the same editor page. Do not enable grouping yet, we get to that later. <asp:SqlDataSource ID="sqlData" runat="Server" ConnectionString="Server=127.0.0.1;Database=dev;Uid=;Pwd=;"
SelectCommand="SELECT * FROM blog_post_1" SelectCommandType="Text" />
<SharePointWebControls:MenuTemplate ID="spmStoreMenu" runat="server">
<SharePointWebControls:MenuItemTemplate Text="Edit Store" ID="miEditStore" runat="server"
Description="Manage this store" ImageUrl="/_layouts/images/actionseditpage.gif"
Sequence="1" ClientOnClickNavigateUrl="StoreManager.aspx?ID=%ID%" />
</SharePointWebControls:MenuTemplate>
<table width="300">
<tr>
<td>
<SharePointWebControls:SPGridView ID="spgvGrid" runat="server" DataKeyNames="ID"
DataSourceID="sqlData" AllowGrouping="false" AutoGenerateColumns="false" GroupField="StateName"
AllowGroupCollapse="true" DisplayGroupFieldName="false">
<Columns>
<SharePointWebControls:SPMenuField NavigateUrlFields="ID" NavigateUrlFormat="StoreManager.aspx?ID={0}"
TextFields="StoreName" MenuTemplateId="spmStoreMenu" HeaderText="Store" TokenNameAndValueFields="ID=ID" />
<SharePointWebControls:SPBoundField DataField="IsArchived" HeaderText="Status" />
</Columns>
</SharePointWebControls:SPGridView>
</td>
</tr>
</table>
Done. We now have a spiffy looking grid and our field links to an edit page so each store can be edited and we have a menu item. Now let’s get the menu setup with a new item performing a post back to the instantiating page to set the store to archived. The page needs to implement System.Web.UI.IPostBackEventHandler and you will need to come up with a scheme for your post back event args, but that would be a topic for another post.
Add the following to your basic page:
<asp:SqlDataSource ID="sqlData" runat="Server" ConnectionString="Server=127.0.0.1;Database=dev;Uid=;Pwd=;"
SelectCommand="SELECT * FROM blog_post_1" SelectCommandType="Text" />
<SharePointWebControls:MenuTemplate ID="spmStoreMenu" runat="server">
<SharePointWebControls:MenuItemTemplate Text="Edit Store" ID="miEditStore" runat="server"
Description="Manage this store" ImageUrl="/_layouts/images/actionseditpage.gif"
Sequence="1" ClientOnClickNavigateUrl="StoreManager.aspx?ID=%ID%" />
<SharePointWebControls:MenuItemTemplate Text="Archive" ID="miArchiveStore" runat="server"
Description="Mark this store as archived." ImageUrl="/_layouts/images/plicon.png"
Sequence="2" ClientOnClickUsingPostBackEvent="__page,ARCHIVE::%ID%" />
</SharePointWebControls:MenuTemplate>
<table width="300">
<tr>
<td>
<SharePointWebControls:SPGridView ID="spgvGrid" runat="server" DataKeyNames="ID"
DataSourceID="sqlData" AllowGrouping="true" AutoGenerateColumns="false" GroupField="StateName"
AllowGroupCollapse="true" DisplayGroupFieldName="false">
<Columns>
<SharePointWebControls:SPMenuField NavigateUrlFields="ID" NavigateUrlFormat="StoreManager.aspx?ID={0}"
TextFields="StoreName" MenuTemplateId="spmStoreMenu" HeaderText="Store" TokenNameAndValueFields="ID=ID" />
<SharePointWebControls:SPBoundField DataField="IsArchived" HeaderText="Status" />
</Columns>
</SharePointWebControls:SPGridView>
</td>
</tr>
</table>
A couple of interesting things are happening in the code we just added. The key piece here is the ClientOnClickUsingPostBackEvent attribute and the “__page,_ACTION_;%ID%” value. If you look at the ClientOnClickUsingPostBackEvent method in reflector you can see that it interprets this as being the containing page, and directs the post back there. You can actually send the post back event through any control that implements IPostBackEventHandler, which SPGridView does, opening up the possiblities for some elgant solutions. Anything after the “,” is passed as the eventArgument string to the RaisePostBackEvent. Also note that the TokenNameAndValueFields attribute of the menu field determines the substitution vars (ex: %ID%) available in the menu items.
If you got everything wired up correctly then your IPostBackEventHandler.RaisePostBackEvent method should be getting fired whenever you click on any of the “Archive” menu items and you should be getting as part of the eventArgument the id of each store. Beauty. Now we just have to enable grouping in the SPGridView and test the postback and we are all set.
However after we try a post back with grouping enabled we get an nice, friendly error:
Flying Monkey Buckets, now it renders correctly, but fails (and how) on post back. The only thing we changed from the working version was that we enabled grouping. So what gives? If you look at the error and spend some more time in reflector you will find that the args.Row.DataItem is null when the page tries to load the state, specifically in the GridView.OnRowCreated method. This is not an easy error to get at from the page level, so it turns out the easiest thing to do is create our own SPGridView class and fix it there; and it only takes a few lines of code – so if you ever feel like buying me a beer…. You will need to expose an event, override the LoadControlState method in your custom grid class, and add an event handler to you page.
namespace UserControls { public class CustomGrid : SPGridView { protected override void LoadControlState(object savedState) { base.LoadControlState(savedState); if (this.DataSource == null) { this.InvokeRequiresDataSource(); } } public event EventHandler RequiresDataSource; protected void InvokeRequiresDataSource() { EventHandler handler = this.RequiresDataSource; if (handler != null) { handler(this, new EventArgs()); } } } }
And the entire contents of the test page we ended up with: <%@ Page Language="C#" Debug="true" MasterPageFile="~masterurl/default.master" Title="Post 1 Example"
meta:progid="SharePoint.WebPartPage.Document" %>
<%@ Assembly Name="Microsoft.SharePoint, Version=12.0.0.0, Culture=neutral,PublicKeyToken=71e9bce111e9429c" %>
<%@ Assembly Name="System.Web, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a" %>
<%@ Import Namespace="Microsoft.SharePoint" %>
<%@ Import Namespace="System.IO" %>
<%@ Import Namespace="System.Web.UI" %>
<%@ Import Namespace="System.Data" %>
<%@ Import Namespace="System.Collections.Generic" %>
<%@ Import Namespace="System.Data.SqlClient" %>
<%@ Register Assembly="Microsoft.SharePoint, Version=12.0.0.0, Culture=neutral,PublicKeyToken=71e9bce111e9429c"
TagPrefix="SharePointWebControls" Namespace="Microsoft.SharePoint.WebControls" %>
<%@ Implements Interface="System.Web.UI.IPostBackEventHandler" %>
<%@ Register Assembly="CustomGrid, Version=1.0.0.0, Culture=neutral, PublicKeyToken=76ce4e6c43aad137" Namespace="UserControls" TagPrefix="uc" %>
<script runat="server">
void IPostBackEventHandler.RaisePostBackEvent(string eventArgument)
{
//parse the event argument and mark the store as archived.
spgvGrid.DataBind(); //need to rebind the grid.
}
protected void spgvGrid_RequiresDataSource(object sender, EventArgs e)
{
spgvGrid.DataBind();
}
</script>
<asp:Content ID="Content5" ContentPlaceHolderID="PlaceHolderMain" runat="server">
<asp:SqlDataSource ID="sqlData" runat="Server" ConnectionString="Server=127.0.0.1;Database=dev;Uid=;Pwd=;"
SelectCommand="SELECT * FROM blog_post_1" SelectCommandType="Text" />
<SharePointWebControls:MenuTemplate ID="spmStoreMenu" runat="server">
<SharePointWebControls:MenuItemTemplate Text="Edit Store" ID="miEditStore" runat="server"
Description="Manage this store" ImageUrl="/_layouts/images/actionseditpage.gif"
Sequence="1" ClientOnClickNavigateUrl="StoreManager.aspx?ID=%ID%" />
<SharePointWebControls:MenuItemTemplate Text="Archive" ID="miArchiveStore" runat="server"
Description="Mark this store as archived." ImageUrl="/_layouts/images/plicon.png"
Sequence="2" ClientOnClickUsingPostBackEvent="__page,ARCHIVE::%ID%" />
</SharePointWebControls:MenuTemplate>
<table width="300">
<tr>
<td>
<uc:CustomGrid ID="spgvGrid" runat="server" DataKeyNames="ID" DataSourceID="sqlData"
AllowGrouping="true" AutoGenerateColumns="false" GroupField="StateName" AllowGroupCollapse="true"
DisplayGroupFieldName="false" OnRequiresDataSource="spgvGrid_RequiresDataSource">
<Columns>
<SharePointWebControls:SPMenuField NavigateUrlFields="ID" NavigateUrlFormat="StoreManager.aspx?ID={0}"
TextFields="StoreName" MenuTemplateId="spmStoreMenu" HeaderText="Store" TokenNameAndValueFields="ID=ID" />
<SharePointWebControls:SPBoundField DataField="IsArchived" HeaderText="Status" />
</Columns>
</uc:CustomGrid>
</td>
</tr>
</table>
</asp:Content>
Now give it a shot and your page should postback without problem. You will need to do the same trick if you are creating a callback menu item, but that is another post.
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
Excel에서 사이트의 목록 목록을 손쉽게 만들기
Excel의 OData 연동 기능을 사용하면 비 프로그래밍 방식으로 SharePoint Online 목록을 쉽게 만들 수 있습니다.
※ SharePoint Server 2013, 2016에서도 갈 수 있습니다.
먼저...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.
<asp:SqlDataSource ID="sqlData" runat="Server" ConnectionString="Server=127.0.0.1;Database=dev;Uid=;Pwd=;"
SelectCommand="SELECT * FROM blog_post_1" SelectCommandType="Text" />
<SharePointWebControls:MenuTemplate ID="spmStoreMenu" runat="server">
<SharePointWebControls:MenuItemTemplate Text="Edit Store" ID="miEditStore" runat="server"
Description="Manage this store" ImageUrl="/_layouts/images/actionseditpage.gif"
Sequence="1" ClientOnClickNavigateUrl="StoreManager.aspx?ID=%ID%" />
</SharePointWebControls:MenuTemplate>
<table width="300">
<tr>
<td>
<SharePointWebControls:SPGridView ID="spgvGrid" runat="server" DataKeyNames="ID"
DataSourceID="sqlData" AllowGrouping="false" AutoGenerateColumns="false" GroupField="StateName"
AllowGroupCollapse="true" DisplayGroupFieldName="false">
<Columns>
<SharePointWebControls:SPMenuField NavigateUrlFields="ID" NavigateUrlFormat="StoreManager.aspx?ID={0}"
TextFields="StoreName" MenuTemplateId="spmStoreMenu" HeaderText="Store" TokenNameAndValueFields="ID=ID" />
<SharePointWebControls:SPBoundField DataField="IsArchived" HeaderText="Status" />
</Columns>
</SharePointWebControls:SPGridView>
</td>
</tr>
</table>
<asp:SqlDataSource ID="sqlData" runat="Server" ConnectionString="Server=127.0.0.1;Database=dev;Uid=;Pwd=;"
SelectCommand="SELECT * FROM blog_post_1" SelectCommandType="Text" />
<SharePointWebControls:MenuTemplate ID="spmStoreMenu" runat="server">
<SharePointWebControls:MenuItemTemplate Text="Edit Store" ID="miEditStore" runat="server"
Description="Manage this store" ImageUrl="/_layouts/images/actionseditpage.gif"
Sequence="1" ClientOnClickNavigateUrl="StoreManager.aspx?ID=%ID%" />
<SharePointWebControls:MenuItemTemplate Text="Archive" ID="miArchiveStore" runat="server"
Description="Mark this store as archived." ImageUrl="/_layouts/images/plicon.png"
Sequence="2" ClientOnClickUsingPostBackEvent="__page,ARCHIVE::%ID%" />
</SharePointWebControls:MenuTemplate>
<table width="300">
<tr>
<td>
<SharePointWebControls:SPGridView ID="spgvGrid" runat="server" DataKeyNames="ID"
DataSourceID="sqlData" AllowGrouping="true" AutoGenerateColumns="false" GroupField="StateName"
AllowGroupCollapse="true" DisplayGroupFieldName="false">
<Columns>
<SharePointWebControls:SPMenuField NavigateUrlFields="ID" NavigateUrlFormat="StoreManager.aspx?ID={0}"
TextFields="StoreName" MenuTemplateId="spmStoreMenu" HeaderText="Store" TokenNameAndValueFields="ID=ID" />
<SharePointWebControls:SPBoundField DataField="IsArchived" HeaderText="Status" />
</Columns>
</SharePointWebControls:SPGridView>
</td>
</tr>
</table>
namespace UserControls { public class CustomGrid : SPGridView { protected override void LoadControlState(object savedState) { base.LoadControlState(savedState); if (this.DataSource == null) { this.InvokeRequiresDataSource(); } } public event EventHandler RequiresDataSource; protected void InvokeRequiresDataSource() { EventHandler handler = this.RequiresDataSource; if (handler != null) { handler(this, new EventArgs()); } } } }
<%@ Page Language="C#" Debug="true" MasterPageFile="~masterurl/default.master" Title="Post 1 Example"
meta:progid="SharePoint.WebPartPage.Document" %>
<%@ Assembly Name="Microsoft.SharePoint, Version=12.0.0.0, Culture=neutral,PublicKeyToken=71e9bce111e9429c" %>
<%@ Assembly Name="System.Web, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a" %>
<%@ Import Namespace="Microsoft.SharePoint" %>
<%@ Import Namespace="System.IO" %>
<%@ Import Namespace="System.Web.UI" %>
<%@ Import Namespace="System.Data" %>
<%@ Import Namespace="System.Collections.Generic" %>
<%@ Import Namespace="System.Data.SqlClient" %>
<%@ Register Assembly="Microsoft.SharePoint, Version=12.0.0.0, Culture=neutral,PublicKeyToken=71e9bce111e9429c"
TagPrefix="SharePointWebControls" Namespace="Microsoft.SharePoint.WebControls" %>
<%@ Implements Interface="System.Web.UI.IPostBackEventHandler" %>
<%@ Register Assembly="CustomGrid, Version=1.0.0.0, Culture=neutral, PublicKeyToken=76ce4e6c43aad137" Namespace="UserControls" TagPrefix="uc" %>
<script runat="server">
void IPostBackEventHandler.RaisePostBackEvent(string eventArgument)
{
//parse the event argument and mark the store as archived.
spgvGrid.DataBind(); //need to rebind the grid.
}
protected void spgvGrid_RequiresDataSource(object sender, EventArgs e)
{
spgvGrid.DataBind();
}
</script>
<asp:Content ID="Content5" ContentPlaceHolderID="PlaceHolderMain" runat="server">
<asp:SqlDataSource ID="sqlData" runat="Server" ConnectionString="Server=127.0.0.1;Database=dev;Uid=;Pwd=;"
SelectCommand="SELECT * FROM blog_post_1" SelectCommandType="Text" />
<SharePointWebControls:MenuTemplate ID="spmStoreMenu" runat="server">
<SharePointWebControls:MenuItemTemplate Text="Edit Store" ID="miEditStore" runat="server"
Description="Manage this store" ImageUrl="/_layouts/images/actionseditpage.gif"
Sequence="1" ClientOnClickNavigateUrl="StoreManager.aspx?ID=%ID%" />
<SharePointWebControls:MenuItemTemplate Text="Archive" ID="miArchiveStore" runat="server"
Description="Mark this store as archived." ImageUrl="/_layouts/images/plicon.png"
Sequence="2" ClientOnClickUsingPostBackEvent="__page,ARCHIVE::%ID%" />
</SharePointWebControls:MenuTemplate>
<table width="300">
<tr>
<td>
<uc:CustomGrid ID="spgvGrid" runat="server" DataKeyNames="ID" DataSourceID="sqlData"
AllowGrouping="true" AutoGenerateColumns="false" GroupField="StateName" AllowGroupCollapse="true"
DisplayGroupFieldName="false" OnRequiresDataSource="spgvGrid_RequiresDataSource">
<Columns>
<SharePointWebControls:SPMenuField NavigateUrlFields="ID" NavigateUrlFormat="StoreManager.aspx?ID={0}"
TextFields="StoreName" MenuTemplateId="spmStoreMenu" HeaderText="Store" TokenNameAndValueFields="ID=ID" />
<SharePointWebControls:SPBoundField DataField="IsArchived" HeaderText="Status" />
</Columns>
</uc:CustomGrid>
</td>
</tr>
</table>
</asp:Content>
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
Excel에서 사이트의 목록 목록을 손쉽게 만들기Excel의 OData 연동 기능을 사용하면 비 프로그래밍 방식으로 SharePoint Online 목록을 쉽게 만들 수 있습니다. ※ SharePoint Server 2013, 2016에서도 갈 수 있습니다. 먼저...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.