最近一段时间都在研究SharePoint下的开发,感觉SharePoint中的内容还是比较繁杂的,不知道自己入了门没有?顺手记录下了开发过程中的 一些心得和体会,以加深自己的理解和掌握,如有不对的地方,还请各位不吝指正。今天先讲讲SharePoint下Web Part的开发,这应该算是SharePoint下最常见的开发了吧,今天的重点是关于自定义属性及配置界面的实现。
一、开发环境
(1)Windows 2003 Server
(2)Windows SharePoint Service 3.0(或Microsoft Office SharePoint Server 2007)
(3)Visual Studio 2005
(5)SQL Server 2005
(6)Windows SharePoint Services 3.0 Tools: Visual Studio 2005 Extensions
(7)Windows SharePoint Services SDK 3.0
二、WSS 2.0和WSS 3.0下Web Part开发的不同
众所周知,Windows SharePoint Service 3.0(以下简称WSS 3.0)是基于ASP.NET 2.0进行了重写,因此与WSS 2.0有很多地方存在一些不同。对于Web Part的开发,也是如此。目前网络上的很多资料介绍的Web Part开发,还是基于WSS 2.0的,而非WSS 3.0风格的。
Web Part是首先在WSS 2.0作为仪表板(Dashboard)的替代技术引入的,后来ASP.NET 2.0中也引入了一个新版本的Web Part Framework,该框架可以脱离SharePoint的环境运行。WSS 3.0的Web Part框架也基于该ASP.NET Web Part Framework进行了完全的重构。不过为了兼容性(主要是考虑到以前的解决方案的升级),原WSS 2.0的Web Part框架也得以保留,但对于以后的开发而言,都应该基于WSS 3.0的ASP.NET框架进行。
例如,基于WSS 2.0的Web Part都是从 Microsoft.SharePoint.WebPartPages.WebPart类派生的,而WSS 3.0的Web Part应该从标准的ASP.NET类System.Web.UI.WebControls.WebParts.WebPart继承。下表是基于 ASP.NET Web Part Framework中一些新的属性和类型,以及为了兼容性而保留的对应WSS 2.0模型中的属性和类型,具体的示例将在下面的章节中出现。
ASP.NET Web Parts
SharePoint Backward Compatibility
WebBrowsableAttribute
|
BrowsableAttribute
|
WebDisplayName
|
FriendlyName
|
WebDescription
|
Description
|
Personalizable
|
WebPartStorage
|
PersonalizationScope
|
Storage
|
EditorPart
|
ToolPart
|
EditorPartCollection
|
ToolPart[]
|
CreateEditorParts()
|
GetToolParts()
|
RenderContents()
|
RenderWebPart()
|
SetPersonalizationDirty()
|
SaveProperties
|
三、SharePoint Web Part开发基础
Web Part是一种特殊类型的Web Control(Web控件),它被部署于Web Part Zone控件内。虽然基于ASP.NET的Web Part Framework可以脱离WSS环境运行,但是WSS 3.0中的Web Part Framework提供了一些附加的功能,包括:动态SharePoint站点模型、模板框架、安全框架以及使用WSS Web Part Gallery的Web Part管理。以下二图就是Web Part开发中涉及的基本类,虽然SharePoint针对Web Part Zone和Web PartManager类有特定的实现,但是我们仍将使用ASP.NET Web Part类作为我们开发Web Part应用程序的基类。


WSS 3.0中使用的Web Part Manager类为SPWebPartManager,它是在页面Web Part Zone对象和SharePoint内容数据库之间的桥梁。当在页面中添加一个Web Part时,实际上是在内容数据库中加入了一个Web Part的序列化实例。对于一个最简单的Web Part而言,只需从Web Part 基类System.Web.UI.WebControls.WebParts.WebPart.派生一个具体类,然后重写RenderContents方 法来以HTML方式呈现该Web Part,就像在ASP.NET中开发Web控件一样。一个Web Part的生命周期如下表所示:
方法/事件
描述
OnInit
|
处理控件的初始化
|
OnLoad
|
处理控件的装载
|
CreateChildControls
|
创建子控件
|
EnsureChildControls
|
确保CreateChildControls方法被调用,使用该方法来确保在访问控件的数据之前控件已经存在。
|
OnPreRender
|
处理在呈现可控件之前必须完成的任务,例如数据装载。异步页面任务应该从该方法中被启动。
|
Page.PreRenderComplete
|
在所有的控件都完成其OnPreRender 方法并且页面已经完成了异步任务的执行之后,将会触发该事件。
|
Render
|
呈现整个控件,包括外部标签。
|
RenderContents
|
仅仅呈现控件在外部标签和样式属性内的内容。
|
四、Web Part属性
除了类似输出”Hello World“这样内容的最简单Web Part之外,一般实际的Web Part都是可动态配置的,这是通过Web Part属性来实现的。
在WSS 2.0中,一个Web Part属性是这样子的:

[Description("DataSourceName"),
Category("Miscellaneous"),
DefaultValue(""),
WebPartStorage(Storage.Shared),
FriendlyName("DataSourceName"),
Browsable(false)]
publicstringDataSourceName

...{

get...{returnthis.dataSourceName;}

set...{this.dataSourceName=value;}
}
当然,这不是我们推荐的用法,而应该采用如下新的语法:
[WebDescription("DataSourceName"),
Category("Miscellaneous"),
DefaultValue(""),
Personalizable(PersonalizationScope.Shared),
WebDisplayName("DataSourceName"),
WebBrowsable(false)]
publicstringDataSourceName

...{

get...{returnthis.dataSourceName;}

set...{this.dataSourceName=value;}
}
上述两个声明是完全等价的,大家可以参考第二节中的对应表来一一比较。注意到上述属性中WebBrowsable的属性为false,如果该属性为 true的话,就是使用默认的属性编辑界面,也就是一个文本输入框,如果想定制该属性的输入界面,则应该将WebBrowsable属性置为false, 然后提供定制属性界面。
在WSS 2.0中,自定义属性配置界面是通过重写Web Part的GetToolParts方法实现的,例如:


/**////<summary>
///配置界面。
///</summary>
///<returns></returns>
publicoverrideToolPart[]GetToolParts()

...{
ToolPart[]tools=newToolPart[3];
tools[0]=newMicrosoft.SharePoint.WebPartPages.WebPartToolPart();
tools[1]=newMicrosoft.SharePoint.WebPartPages.CustomPropertyToolPart();
tools[2]=new MyToolPart();
tools[2].Title="My Properties";
returntools;
}
其中,
Microsoft.SharePoint.WebPartPages.WebPartToolPart类提供标准的属性配置界面,如布局,大小等;而Microsoft.SharePoint.WebPartPages.CustomPropertyToolPart则提供默认的自定义属性配置界面,这是利用反射实现的,如上述属性WebBrowsable为false,则意味着使用该类来提供默认的配置界面;而第三项
MyToolPart则是为自己实现的定制配置界面。
因为这是WSS 2.0中已经过时的用法,所以在此不再详述,我们重点关注一下WSS 3.0中的对应的实现。在WSS 3.0(ASP.NET 2.0)中,自定义属性配置界面是通过重写Web Part的CreateEditorParts方法实现的:

/**////<summary>
///配置界面。
///</summary>
///<returns></returns>
publicoverrideEditorPartCollectionCreateEditorParts()

...{
EditorPartCollectionbaseParts=base.CreateEditorParts();
List<EditorPart>editorParts=newList<EditorPart>(1);
EditorPartpart=newMyEditorPart();
part.ID=this.ID+"_MyEditor";
editorParts.Add(part);
returnnewEditorPartCollection(baseParts,editorParts);
}
要实现一个EditorPart,最简单的是重写三个方法:
(1)CreateChildControls:负责创建EditorPart的子控件
(2)ApplyChanges:保存配置到Web Part的属性
(3)SyncChanges:从Web Part属性初始化EditorPart
以下是一个具体的WebPart的代码:
usingSystem;
usingSystem.Runtime.InteropServices;
usingSystem.Web.UI;
usingSystem.Web.UI.WebControls.WebParts;
usingSystem.Xml.Serialization;
usingMicrosoft.SharePoint;
usingMicrosoft.SharePoint.Utilities;
usingSystem.ComponentModel;
usingSystem.Collections.Generic;
namespaceKingFactoryTagValueWebPart2
{
[Guid("3fd90ae1-9ff9-4b5e-8262-ac2649be9ebe")]
publicclassKingFactoryTagValueWebPart2:System.Web.UI.WebControls.WebParts.WebPart
{
privatestringdataSourceName;
privatestringrtTagName;
privatestringrefreshInterval;
privatestringwebUrl;
///<summary>
///构造函数。
///</summary>
publicKingFactoryTagValueWebPart2()
{
this.ExportMode=WebPartExportMode.All;
this.refreshInterval="5000";
}
///<summary>
///输出错误信息。
///</summary>
///<paramname="writer"></param>
///<paramname="errorInfo"></param>
privatevoidRenderError(HtmlTextWriterwriter,stringerrorInfo)
{
stringcontent="<span><tableclass="ms-WPBody"style="padding:0px;width:100%;">"+
"< tr><tdvalign="top"style="padding-left:4px;padding-right:4px;"> <imgsrc="/_layouts/1033/images/error.gif"alt="错误信息"/>< /td>"+
"<tdwidth="100%"style="padding-left:4px;padding-right:4px;">"+
errorInfo+
"</td></tr></table></span>";
writer.Write(content);
}
///<summary>
///输出结果
///</summary>
///<paramname="writer"></param>
protectedoverridevoidRender(HtmlTextWriterwriter)
{
//检查网址
if(this.webUrl==null||this.webUrl==string.Empty)
{
RenderError(writer,"错误:请选择网站");
return;
}
//检查数据源
if(this.dataSourceName==null||this.dataSourceName==string.Empty)
{
RenderError(writer,"错误:请选择数据源");
return;
}
//检查变量
if(this.rtTagName==null||this.rtTagName==string.Empty)
{
RenderError(writer,"错误:请选择变量");
return;
}
//刷新间隔
if(this.refreshInterval==null||this.refreshInterval==string.Empty)
this.refreshInterval="5000";
//生成输出
SPWebweb=SPContext.Current.Web;
stringcontent=string.Format("<iframesrc="{0}?UserName={1}&DataSourceName={2}&TagName={3}&RefreshInterval={4}"width="100%"height="100%"></iframe>",
this.webUrl,
SPEncode.UrlEncode(web.CurrentUser.LoginName),
SPEncode.UrlEncode(this.dataSourceName),
SPEncode.UrlEncode(this.rtTagName),
this.RefreshInterval);
writer.Write(content);
}
///<summary>
///获得编辑界面。
///</summary>
///<returns></returns>
publicoverrideEditorPartCollectionCreateEditorParts()
{
EditorPartCollectionbaseParts=base.CreateEditorParts();
List<EditorPart>editorParts=newList<EditorPart>(1);
EditorPartpart=newKingFactoryTagValueEditorPart2();
part.ID=this.ID+"_tagValueEditor";
part.Title = "KingFactory变量选择";
editorParts.Add(part);
returnnewEditorPartCollection(baseParts,editorParts);
}
[WebDescription("WebURL"),
Category("Miscellaneous"),
DefaultValue(""),
Personalizable(PersonalizationScope.Shared),
WebDisplayName("WebURL"),
WebBrowsable(false)]
publicstringWebURL
{
get{returnthis.webUrl;}
set{this.webUrl=value;}
}
[WebDescription("DataSourceName"),
Category("Miscellaneous"),
DefaultValue(""),
Personalizable(PersonalizationScope.Shared),
WebDisplayName("DataSourceName"),
WebBrowsable(false)]
publicstringDataSourceName
{
get{returnthis.dataSourceName;}
set{this.dataSourceName=value;}
}
[WebDescription("TagName"),
Category("Miscellaneous"),
DefaultValue(""),
Personalizable(PersonalizationScope.Shared),
WebDisplayName("TagName"),
WebBrowsable(false)]
publicstringRtTagName
{
get{returnthis.rtTagName;}
set{this.rtTagName=value;}
}
[WebDescription("RefreshInterval"),
Category("Miscellaneous"),
DefaultValue(""),
Personalizable(PersonalizationScope.Shared),
WebDisplayName("RefreshInterval"),
WebBrowsable(false)]
publicstringRefreshInterval
{
get{returnthis.refreshInterval;}
set{this.refreshInterval=value;}
}
}
}
以下为对应的EditorPart的代码:
usingSystem;
usingSystem.Collections.Generic;
usingSystem.Text;
usingSystem.Web;
usingSystem.Web.UI;
usingSystem.Web.UI.WebControls;
usingSystem.Web.UI.WebControls.WebParts;
usingMicrosoft.SharePoint;
usingMicrosoft.SharePoint.Administration;
namespaceKingFactoryTagValueWebPart2
{
publicclassKingFactoryTagValueEditorPart2:EditorPart
{
protectedLabelLabel1;
protectedLabelLabel2;
protectedLabelLabel3;
protectedLabelLabel4;
protectedLabelLabel5;
protectedTextBoxrefreshInterval;
protectedTextBoxselectedTagName;
protectedTreeViewTagNameView;
protectedDropDownListwebSiteList;
protectedDropDownListdataSourceList;
///<summary>
///创建子控件。
///</summary>
protectedoverridevoidCreateChildControls()
{
base.CreateChildControls();
this.Label1=newLabel();
this.Label1.Text="选择网站:";
this.Label1.Width=80;
this.Label2=newLabel();
this.Label2.Text="选择数据源:";
this.Label3=newLabel();
this.Label3.Text="刷新间隔:";
this.Label4=newLabel();
this.Label4.Text="选择变量:";
this.refreshInterval=newTextBox();
this.selectedTagName=newTextBox();
this.TagNameView=newTreeView();
this.webSiteList=newDropDownList();
this.dataSourceList=newDropDownList();
this.dataSourceList.SelectedIndex=-1;
this.webSiteList.SelectedIndex=-1;
this.webSiteList.AutoPostBack=true;
this.webSiteList.SelectedIndexChanged+=newEventHandler(webSiteList_SelectedIndexChanged);
this.dataSourceList.AutoPostBack=true;
this.dataSourceList.SelectedIndexChanged+=newEventHandler(dataSourceList_SelectedIndexChanged);
this.TagNameView.ImageSet=TreeViewImageSet.XPFileExplorer;
this<span sty
分享到:
相关推荐
然而,对于希望在 SharePoint 环境中快速开发定制化 WebPart 的开发者来说,理解 SharePoint WebPart 的底层机制可能会是一项挑战。此时,一种名为“SharePoint WebPart 用户控件包装器”的工具应运而生。 #### 二...
《亮剑.NET:SharePoint Server 2007开发实战》是针对Microsoft SharePoint Server 2007这一企业级协作平台的开发指南。本书详细介绍了如何利用.NET Framework和SharePoint Server 2007的强大功能进行高效开发。下面...
2. **基于WebPart的开发方式**:另一种可能的方式是通过创建自定义WebPart来集成SmartQuery。WebPart是SharePoint中可重用的、用户可配置的组件。开发者可以在WebPart中集成SmartQuery的相关功能,允许用户在页面上...
少儿编程scratch项目源代码文件案例素材-直升机飞行.zip
wanjunshe_Python-Tensorflow_12888_1745868924470
健康监测_Android开发_BLE蓝牙通信_心率数据采集与存储_基于小米手环2的实时心率监测应用_支持后台长时间运行的心率记录工具_可导出SQLite数据库的心率数据分析系统_适
少儿编程scratch项目源代码文件案例素材-种花模拟器.zip
嵌入式系统开发_FreeRTOS实时操作系统_STM32F103C8T6微控制器_OLED显示屏_DHT11温湿度传感器_多任务调度_多级菜单设计_万年历算法_电子闹钟功能_参数配
基于python实现的粒子群的VRP(车辆配送路径规划)问题建模求解+源码+项目文档+算法解析,适合毕业设计、课程设计、项目开发。项目源码已经过严格测试,可以放心参考并在此基础上延申使用,详情见md文档 算法设计的关键在于如何向表现较好的个体学习,标准粒子群算法引入惯性因子w、自我认知因子c1、社会认知因子c2分别作为自身、当代最优解和历史最优解的权重,指导粒子速度和位置的更新,这在求解函数极值问题时比较容易实现,而在VRP问题上,速度位置的更新则难以直接采用加权的方式进行,一个常见的方法是采用基于遗传算法交叉算子的混合型粒子群算法进行求解,这里采用顺序交叉算子,对惯性因子w、自我认知因子c1、社会认知因子c2则以w/(w+c1+c2),c1/(w+c1+c2),c2/(w+c1+c2)的概率接受粒子本身、当前最优解、全局最优解交叉的父代之一(即按概率选择其中一个作为父代,不加权)。 算法设计的关键在于如何向表现较好的个体学习,标准粒子群算法引入惯性因子w、自我认知因子c1、社会认知因子c2分别作为自身、当代最优解和历史最优解的权重,指导粒子速度和位置的更新,这在求解函数极值问题时比较容易实现,而在VRP问题上,速度位置的更新则难以直接采用加权的方式进行,一个常见的方法是采用基于遗传算法交叉算子的混合型粒子群算法进行求解,这里采用顺序交叉算子,对惯性因子w、自我认知因子c1、社会认知因子c2则以w/(w+c1+c2),c1/(w+c1+c2),c2/(w+c1+c2)的概率接受粒子本身、当前最优解、全局最优解交叉的父代之一(即按概率选择其中一个作为父代,不加权)。
scratch少儿编程逻辑思维游戏源码-猫猫粉碎.zip
scratch少儿编程逻辑思维游戏源码-蓝胡子.zip
scratch少儿编程逻辑思维游戏源码-美食大亨.zip
scratch少儿编程逻辑思维游戏源码-洛克人.zip
scratch少儿编程逻辑思维游戏源码-龙冲刺.zip
思幻个人引导页V2.2版本11月29日更新.zip
scratch少儿编程逻辑思维游戏源码-骑士风斩法.zip
移动应用开发_H5CSS3ionicng-cordovaMVVM模式_基于HTML5和CSS3技术实现多页面布局ionic指令数据绑定ui-route单页跳转调用手机
少儿编程scratch项目源代码文件案例素材-植物大战僵尸创造版 Ver. 1.0.3.zip
scratch少儿编程逻辑思维游戏源码-日落(2).zip
动态星空背景个人主页(带后台).zip