When designing an Sharepoint implementation I started out with OOTB Sharepoint behaviour. I added a LookupField to the list and pointed it to another list. But OOTB all listitems from the referenced list would show up in the dropdown as options. I wanted to be able to control which options would be shown.
I've found an open source fieldcontrol at http://extendedlookup.codeplex.com/ which would be able to do just that by using a view. Unfortunatly it didn't work right-away for me, probably due to some updates from SP-hotfixes. After making some minor changes, i was able to get it working again.
When fiddling with the control I noticed that when a selected item wasn't included in the view it would revert to the first/default-value. Once again i made a minor change, which would make it able to add the selected-item as an option.
Update: just noticed the archived comment at http://extendedlookup.codeplex.com/:
See other projects it looks similar: http://www.codeplex.com/CustomFieldControls http://www.codeplex.com/iLoveSharePoint/Release/ProjectReleases.aspx?ReleaseId=15633
Tuesday, March 10, 2009
Monday, March 9, 2009
Web Part Audience Targeting RoleProvider-roles through SharePoint Groups
I've managed to spend some days to understand the Audience Targeting functionality of SharePoint Server 2007 (MOSS). I wish to share my conclusions:
* Global Audiences (rule-based, SSP) is OOTB only possible when using SharePoint UserProfiles. I wanted to bind it to an ASP.NET Profile Provider, but that's not possible.
* The Global Audiences (rule-based) functionality is realized directly in-database via stored-procedures.
* It is possible to target (changing visibility) Web Parts to SharePoint Groups.
* If you target a Web Part to a "ASP.NET RoleProvider"-role by adding the role to a Sharepoint Group does not work. (Forms Based Authentication)
* It is possible to override the "TargetTo"-FieldType by replacing the 12\Template\xml\fldtypes_TargetTo.xml file with your own version. This way you are able to hook your own FieldControl into the Audience-functionality.
* Most code which is related to Targeting is hardwired to use "Microsoft.Office.Server.Audience.AudienceManager". It is thus not easy replaceable or extendable
* The ContentByQuery-WebPart is hardwired to use field with ID "{61cbb965-1e04-4273-b658-eedaa662f48d}" as Audience-field.
* The AuthorizationFilter.Editor of Web Parts is buggy. When removing Sharepoint Groups as Target, many strange bugs become apparent.
* It is not possible to manually AudienceFilter the ContentByQuery-WebPart by selecting the "Audience"-field as filter. It gave me an error because it was trying to compare an ntext- with an nvarchar-column.
I wrote the following class to be able to correctly use RoleProvider-roles as Sharepoint-groups for targeting Web Parts. You should register the class as ResourceFilter in the web.config of your SharePoint-site.
* Global Audiences (rule-based, SSP) is OOTB only possible when using SharePoint UserProfiles. I wanted to bind it to an ASP.NET Profile Provider, but that's not possible.
* The Global Audiences (rule-based) functionality is realized directly in-database via stored-procedures.
* It is possible to target (changing visibility) Web Parts to SharePoint Groups.
* If you target a Web Part to a "ASP.NET RoleProvider"-role by adding the role to a Sharepoint Group does not work. (Forms Based Authentication)
* It is possible to override the "TargetTo"-FieldType by replacing the 12\Template\xml\fldtypes_TargetTo.xml file with your own version. This way you are able to hook your own FieldControl into the Audience-functionality.
* Most code which is related to Targeting is hardwired to use "Microsoft.Office.Server.Audience.AudienceManager". It is thus not easy replaceable or extendable
* The ContentByQuery-WebPart is hardwired to use field with ID "{61cbb965-1e04-4273-b658-eedaa662f48d}" as Audience-field.
* The AuthorizationFilter.Editor of Web Parts is buggy. When removing Sharepoint Groups as Target, many strange bugs become apparent.
* It is not possible to manually AudienceFilter the ContentByQuery-WebPart by selecting the "Audience"-field as filter. It gave me an error because it was trying to compare an ntext- with an nvarchar-column.
I wrote the following class to be able to correctly use RoleProvider-roles as Sharepoint-groups for targeting Web Parts. You should register the class as ResourceFilter in the web.config of your SharePoint-site.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using Microsoft.Office.Server.Audience;
using System.Collections;
using System.Globalization;
using Microsoft.SharePoint.WebPartPages;
using Microsoft.SharePoint;
namespace AudienceManagerUI
{
/// <summary>
/// Replace <RuntimeFilter Assembly="Microsoft.Office.Server, Version=12.0.0.0, Culture=neutral, PublicKeyToken=71e9bce111e9429c" Class="Microsoft.Office.Server.Audience.AudienceManager" BuilderURL="audience_chooser.aspx" />
/// With <RuntimeFilter Assembly="AudienceManagerUI, Version=1.0.0.0, Culture=neutral, PublicKeyToken=6cb89fff80058439" Class="AudienceManagerUI.PatchedResourceManager" BuilderURL="audience_chooser.aspx" />
/// </summary>
public class PatchedResourceManager : Microsoft.SharePoint.WebPartPages.IRuntimeFilter2
{
private Microsoft.Office.Server.Audience.AudienceManager audienceManager = new Microsoft.Office.Server.Audience.AudienceManager();
public ArrayList InitializeStrings(CultureInfo cultureInfO)
{
return audienceManager.InitializeStrings(cultureInfO);
}
public IToolPaneControl GetToolPaneControl()
{
return audienceManager.GetToolPaneControl();
}
public bool IsFilterActive()
{
return audienceManager.IsFilterActive();
}
/// <summary>
/// Checks that the values in the IsIncludedFilter property are valid before they are saved
/// </summary>
/// <param name="PersistedString"></param>
/// <returns></returns>
public string ValidateIsIncludedFilter(string PersistedString)
{
return audienceManager.ValidateIsIncludedFilter(PersistedString);
}
public bool CheckRuntimeRender(string isIncludedFilter)
{
bool result = audienceManager.CheckRuntimeRender(isIncludedFilter);
if (result) return result;
string[] targets = isIncludedFilter.Split(new string[] {";;" }, StringSplitOptions.None);
if (targets.Length == 3)
{
string[] groups = targets[2].Split(new char[] {';'}, StringSplitOptions.RemoveEmptyEntries);
foreach (string target in groups)
{
try
{
SPGroup group = SPContext.Current.Site.RootWeb.SiteGroups[target];
if (group.ContainsCurrentUser) return true;
}
catch (SPException)
{
}
}
}
return false;
}
}
}
Subscribe to:
Posts (Atom)