Wednesday, January 30, 2013

Creating external system (BCS) Search content sources and binding carwled properties to metadata properties programmatically

Business Connectivity Services (BCS) is a set of services and features that connect SharePoint-based solutions to sources of external data. It is included in SharePoint Foundation 2010, SharePoint Server 2010, and Office 2010 applications.

Search Service Application (SSA) Allows for the content to be crawled, indexed, and then allows for users to get results through search queries.

Ok, what about combing those two nice features together , The power of BCS to get data from external data sources and the power of SSA to crawl this data, index it and query it in just no time.

Searching you content using SSA can be configured from the central admin like a piece of cake , according to those articles MSDN articles Part 1 & Part 2, and this nice article

In this article I assume that you have gone among all these hassle before, But what about creating  this whole structure in just one click, on feature activation for example.

What about a site scope feature that would create a Business search content source , then creates a search scope associated to this content source ,Perform a full crawl to get the crawled external fields then connect those fields to a managed properties to be easy for advanced search query.

Ok lets start.

I am assume that you have created an external content type with LOB system named "LOBSystemName" and LOB system instance named "LOBSystemInstanceName"

to create the business content source you need to do the following


   1:  using (SPSite site = new SPSite(SiteURL))
   2:              {
   3:                  SearchContext context = SearchContext.GetContext(site);
   4:   
   5:                  Content BSCContent = new Content(context);
   6:   
   7:                  ContentSourceCollection BSCContentSourceCollection = BSCContent.ContentSources;
   8:                  string NewContentSource = "New Content Source Title";
   9:   
  10:                  if (BSCContentSourceCollection.Exists(NewContentSource))
  11:                  {
  12:                      Console.WriteLine("Content Source Already Exsist");
  13:                      
  14:                      return false;
  15:                  }
  16:                  else
  17:                  {
  18:                      try
  19:                      {
  20:                          BusinessDataContentSource BSCContentSource = (BusinessDataContentSource)BSCContentSourceCollection.Create(typeof(BusinessDataContentSource), NewContentSource);
  21:   
  22:                          BSCContentSource.StartAddresses.Add(BusinessDataContentSource.ConstructStartAddress("Default", new Guid("00000000-0000-0000-0000-000000000000"), "LOBSystemName", "LOBSystemInstanceName"));
  23:   
  24:                          BSCContentSource.StartFullCrawl();
  25:   
  26:                          return true;
  27:                      }
  28:                      catch (Exception ex)
  29:                      {
  30:                          Console.WriteLine("Faild to ceate content source");
  31:                          Console.WriteLine(ex.Message);
  32:                          
  33:                          throw new Exception("Faild to ceate content source \n" + ex.Message);
  34:                      }
  35:                  }
  36:   
  37:              }

Now the content source is created you will need to create a search scope to be associated with the newly created content source  the following method will do this task :


/// <summary>
        /// Create new Content Source type Search Scope
        /// </summary>
        /// <param name="site">The new created site </param>
        /// <param name="context">The Search context to create the Search Scope within</param>
        /// <param name="ContentSourceName">The Search content source name to be associated to the new Search Scope</param>
        static private void CreateBCSSearchScope(SPSite site,SearchContext context, string ContentSourceName )
        {
            string scopeName = ContentSourceName ;
            string displayGroupName = "GTS Scopes";
            // remotescopes class retrieves information via search web service so we run this as the search service account
 
            RemoteScopes remoteScopes = new RemoteScopes(SPServiceContext.GetContext(site));
 
 
 
            // see if there is an existing scope
 
            Scope scope = (from s
 
                           in remoteScopes.GetScopesForSite(new Uri(site.Url)).Cast<Scope>()
 
                           where s.Name == scopeName
 
                           select s).FirstOrDefault();
 
 
 
            // only add if the scope doesn't exist already
 
            if (scope == null)
            {
                Schema sspSchema = new Schema(context);
                ManagedPropertyCollection properties = sspSchema.AllManagedProperties;
                scope = remoteScopes.AllScopes.Create(scopeName, "Search Scope for " + scopeName, null, true, "results.aspx", ScopeCompilationType.AlwaysCompile);
                scope.Rules.CreatePropertyQueryRule(ScopeRuleFilterBehavior.Include, properties["ContentSource"], ContentSourceName);
 
            }
 
 
 
            // see if there is an existing display group         
 
            ScopeDisplayGroup displayGroup = (from d
 
                                              in remoteScopes.GetDisplayGroupsForSite(new Uri(site.Url)).Cast<ScopeDisplayGroup>()
 
                                              where d.Name == displayGroupName
 
                                              select d).FirstOrDefault();
 
 
 
            // add if the display group doesn't exist
 
            if (displayGroup == null)
 
                displayGroup = remoteScopes.AllDisplayGroups.Create(displayGroupName, "", new Uri(site.Url), true);
 
 
 
            // add scope to display group if not already added
 
            if (!displayGroup.Contains(scope))
            {
 
                displayGroup.Add(scope);
 
                displayGroup.Default = scope;
 
                displayGroup.Update();
 
            }
 
 
 
            // optionally force a scope compilation so this is available immediately
 
            remoteScopes.StartCompilation();
        }
The site is the current site you are creating this whole topology for and the Search context is already defined in the previous method

OK, Recap , Content Source Created DONE , Search Scope created and associated DONE, Now we need to perform a full crawl to get Crawled properties "Database table fields  in our case" to be able to map those properties to a managed metadata properties.

to do this step we need to make a while loop to just start the crawl and waits till its done then call the create managed property method... the following code snippet is the key :


                        BSCContentSource.StartFullCrawl();
 
                        Console.WriteLine("Carwling wil start in 10 secounds");
                        Thread.Sleep(10 * 1000);
                        Console.WriteLine("Carwling Started");
 
                        do
                        {
                            Thread.Sleep(10 * 1000);
                            Console.WriteLine("Waiting the content source to finish crawling..");
                        } while (BSCContentSource.CrawlStatus != CrawlStatus.Idle);
 
                        Console.WriteLine("Crawling has been done successfully !");
                        PMFLogger.Instance.LogInformation("Crawling has been done successfully !");
 
                        //Start creating/mapping the new Metadata Properties
                        CreateBCSMetadataProperties(context, ModelName);
Now the data is crawled, indexed ....ok lets create the managed properties and map the crawled fields to those properties , In my case i had the following Columns in the SQL database -External data source -

ID, Title, StartDate, EndDate, Entity, Type, OrganizationUnit, Owner, Alias and StrategyPlan.


        /// <summary>
        /// Creates new Business Search Metadata Properties to be used in the search
        /// </summary>
        /// <param name="context">The Search context to create the Business Search Metadata Properties within</param>
        /// <param name="ModelName">The newly created BCS Model name to aquire and map columns from</param>
        static private void CreateBCSMetadataProperties(SearchContext context,string ModelName)
        {
            Schema sspSchema = new Schema(context);
            ManagedPropertyCollection properties = sspSchema.AllManagedProperties;
 
            //Create the content properties if they does not exist else get the already created ones for the mappings
            ManagedProperty ectID;
            ManagedProperty ectTitle;
            ManagedProperty ectStartDate;
            ManagedProperty ectEndDate;
            ManagedProperty ectEntity;
            ManagedProperty ectType;
            ManagedProperty ectOrganizationUnit;
            ManagedProperty ectOwner;
            ManagedProperty ectAlias;
            ManagedProperty ectStrategyPlan;
            
 
            if (!properties.Contains("ectID"))
            {
                ectID = properties.Create("ectID", ManagedDataType.Text);
                ectID.EnabledForScoping = true;
            }
            else
                ectID = properties["ectID"];
 
            if (!properties.Contains("ectTitle"))
            {
                ectTitle = properties.Create("ectTitle", ManagedDataType.Text);
                ectTitle.EnabledForScoping = true;
            }
            else
                ectTitle = properties["ectTitle"];
 
            if (!properties.Contains("ectStartDate"))
            {
                ectStartDate = properties.Create("ectStartDate", ManagedDataType.DateTime);
                ectStartDate.EnabledForScoping = true;
                ectStartDate.HasMultipleValues = false;
            }
            else
                ectStartDate = properties["ectStartDate"];
 
            if (!properties.Contains("ectEndDate"))
            {
                ectEndDate = properties.Create("ectEndDate", ManagedDataType.DateTime);
                ectEndDate.EnabledForScoping = true;
                ectEndDate.HasMultipleValues = false;
            }
            else
                ectEndDate = properties["ectEndDate"];
            
            if (!properties.Contains("ectEntity"))
            {
                ectEntity = properties.Create("ectEntity", ManagedDataType.Text);
                ectEntity.EnabledForScoping = true;
                ectEntity.HasMultipleValues = false;
            }
            else
                ectEntity = properties["ectEntity"];
 
            if (!properties.Contains("ectType"))
            {
                ectType = properties.Create("ectType", ManagedDataType.Text);
                ectType.EnabledForScoping = true;
                ectType.HasMultipleValues = false;
            }
            else
                ectType = properties["ectType"];
 
            if (!properties.Contains("ectOrganizationUnit"))
            {
                ectOrganizationUnit = properties.Create("ectOrganizationUnit", ManagedDataType.Text);
                ectOrganizationUnit.EnabledForScoping = true;
                ectOrganizationUnit.HasMultipleValues = false;
            }
            else
                ectOrganizationUnit = properties["ectOrganizationUnit"];
 
            if (!properties.Contains("ectOwner"))
            {
                ectOwner = properties.Create("ectOwner", ManagedDataType.Text);
                ectOwner.EnabledForScoping = true;
                ectOwner.HasMultipleValues = false;
            }
            else
                ectOwner = properties["ectOwner"];
 
            if (!properties.Contains("ectAlias"))
            {
                ectAlias = properties.Create("ectAlias", ManagedDataType.Text);
                ectAlias.EnabledForScoping = true;
                ectAlias.HasMultipleValues = false;
                ectAlias.MaxCharactersInPropertyStoreIndex = 450;
            }
            else
                ectAlias = properties["ectAlias"];
 
 
            if (!properties.Contains("ectStrategyPlan"))
            {
                ectStrategyPlan = properties.Create("ectStrategyPlan", ManagedDataType.Text);
                ectStrategyPlan.EnabledForScoping = true;
                ectStrategyPlan.HasMultipleValues = false;
            }
            else
                ectStrategyPlan = properties["ectStrategyPlan"];
            
 
            //Map the Query Crawled Properties to the Managed Property
            MaptoManagedProperty(context, ectID, ModelName + " Items.ID", ManagedDataType.Text);
            MaptoManagedProperty(context, ectTitle, ModelName + " Items.Name", ManagedDataType.Text);
            MaptoManagedProperty(context, ectStartDate, ModelName + " Items.StartDate", ManagedDataType.DateTime);
            MaptoManagedProperty(context, ectEndDate, ModelName + " Items.EndDate", ManagedDataType.DateTime);
            MaptoManagedProperty(context, ectEntity, ModelName + " Items.Entity", ManagedDataType.Text);
            MaptoManagedProperty(context, ectType, ModelName + " Items.Type", ManagedDataType.Text);
            MaptoManagedProperty(context, ectOrganizationUnit, ModelName + " Items.OrganizationUnit", ManagedDataType.Text);
            MaptoManagedProperty(context, ectOwner, ModelName + " Items.Owner", ManagedDataType.Text);
            MaptoManagedProperty(context, ectAlias, ModelName + " Items.Alias", ManagedDataType.Text);
            MaptoManagedProperty(context, ectStrategyPlan, ModelName + " Items.StrategyPlan", ManagedDataType.Text);
        }
 
        /// <summary>
        /// Maps the external content type columns to a specific Managed Property
        /// </summary>
        /// <param name="context">The Search context to map the the external content type columns within</param>
        /// <param name="managedProperty">The managed property to map the column to</param>
        /// <param name="crawledPropertyName">The crawled property "Column" Name</param>
        /// <param name="DataType"></param>
        private static void MaptoManagedProperty(SearchContext context, ManagedProperty managedProperty, string crawledPropertyName, ManagedDataType DataType)
        {
            SPSecurity.RunWithElevatedPrivileges(() =>
            {
                Schema schema = new Schema(context);
 
                try
                {
                    Category category = schema.AllCategories["Business Data"];
                    var crawledProps = category.QueryCrawledProperties(crawledPropertyName, 1, Guid.NewGuid(), String.Empty, true).Cast<CrawledProperty>();
                    var crawledProp = crawledProps.FirstOrDefault();
                    if (crawledProp != null)
                    {
                        var mappings = managedProperty.GetMappings();
                        mappings.Add(new Mapping(crawledProp.Propset, crawledProp.Name, crawledProp.VariantType, managedProperty.PID));
                        managedProperty.SetMappings(mappings);
                        managedProperty.Update();
                    }
                    else
                    {
                        Console.WriteLine("Query Crawled Property " + crawledPropertyName + " was not found - Mapping faild.");
                    }
                }
                catch (Exception ex)
                {
                    throw new Exception("Faild to map field to Crawled Property \n" + ex.Message);
                }
            });
 
        }
Ok you will find some objects that needs to be described:
The "ModelName" object is the External content type model name

Why is "EnabledForScoping" set to true .... that is easy to able the new Managed properties to be exposed to search scopes

Why is some properties "HasMultipleValues" set to false ..... ok if your Managed property is a datetime or int the return data in the search result will be System.DateTime[] not the actual crawled value by setting this property to false issue solved :)

Why is "MaxCharactersInPropertyStoreIndex" set to 450 , this to Reduce storage requirements for text properties by using a hash for comparison, or simply to be able to order search results by this metadata property.

The Whole code in one block .... Have a nice day :)



using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
 
using Microsoft.SharePoint;
using Microsoft.SharePoint.Client;
using Microsoft.SharePoint.Administration;
using Microsoft.Office.Server;
using Microsoft.Office.Server.Search.Administration;
using System.Threading;
 
 
namespace BCSModelGeneration
{      
    public class ContentSourceGenerator
    {
        /// <summary>
        /// Create Bussiness Connectivity Service LOB SharePoint Content Source
        /// </summary>
        /// <param name="InitialCatalog">Rdb Connection tenant Initial Catalog</param>
        /// <param name="SiteURL">The new created site - Entity</param>
        /// <param name="ModelName">The new created Model Name </param>
        /// <returns>The status of the LOB SharePoint Content Source creation</returns>
        internal static bool CreateBCSContentSource(string InitialCatalog, string SiteURL, string ModelName)
        {
            using (SPSite site = new SPSite(SiteURL))
            {
                SearchContext context = SearchContext.GetContext(site);
 
                Content BSCContent = new Content(context);
 
                ContentSourceCollection BSCContentSourceCollection = BSCContent.ContentSources;
                string NewContentSource = InitialCatalog;
 
                if (BSCContentSourceCollection.Exists(NewContentSource))
                {
                    Console.WriteLine("Content Source Already Exsist");
                    return false;
                }
                else
                {
                    try
                    {
                        BusinessDataContentSource BSCContentSource = (BusinessDataContentSource)BSCContentSourceCollection.Create(typeof(BusinessDataContentSource), NewContentSource);
 
                        BSCContentSource.StartAddresses.Add(BusinessDataContentSource.ConstructStartAddress("Default", new Guid("00000000-0000-0000-0000-000000000000"), InitialCatalog, InitialCatalog));
 
                        WeeklySchedule Weekly = CreatWeeklySchedule(context, 2);
 
                        BSCContentSource.FullCrawlSchedule = Weekly;
 
                        DailySchedule Daily = CreateDailySchedule(context, 2);
                        BSCContentSource.IncrementalCrawlSchedule = Daily;
 
                        BSCContentSource.Update();
 
                        BSCContentSource.StartFullCrawl();
 
                        Console.WriteLine("Carwling wil start in 10 secounds");
                        Thread.Sleep(10 * 1000);
                        Console.WriteLine("Carwling Started");
                        //Start creating the new Search Scope
                        CreateBCSSearchScope(site, context, BSCContentSource.Name);
 
                        do
                        {
                            Thread.Sleep(10 * 1000);
                            Console.WriteLine("Waiting the content source to finish crawling..");
                        } while (BSCContentSource.CrawlStatus != CrawlStatus.Idle);
 
                        Console.WriteLine("Crawling has been done successfully !");
 
                        //Start creating/mapping the new Metadata Properties
                        CreateBCSMetadataProperties(context, ModelName);
 
                        Console.WriteLine("Content Source Created");
 
                        Console.WriteLine("Starting a new Crawling process to the content source to fill the new mapped Metadata Properties");
 
                        BSCContentSource.StartFullCrawl();
 
                        return true;
                    }
                    catch (Exception ex)
                    {
                        Console.WriteLine("Faild to ceate content source");
                        Console.WriteLine(ex.Message);
                        throw new Exception("Faild to ceate content source \n" + ex.Message);
                    }
                }
 
            }
        }
 
        /// <summary>
        /// Creats a Weekly Schedule for the search content source
        /// </summary>
        /// <param name="context">The Search context to create the schedule within</param>
        /// <param name="WeeksInterval">Indicates that the content should be crawled every "WeeksInterval" number of weeks</param>
        /// <returns></returns>
        private static WeeklySchedule CreatWeeklySchedule(SearchContext context,int WeeksInterval)
        {
            WeeklySchedule Weekly = new WeeklySchedule(context);
 
            Weekly.BeginDay = DateTime.Now.Day;
            Weekly.BeginMonth = DateTime.Now.Month;
            Weekly.BeginYear = DateTime.Now.Year;
            //Starts at 1:00 AM
            Weekly.StartHour = 1;
            Weekly.StartMinute = 00;
            //Indicates that the content should be crawled every WeeksInterval weeks.
            Weekly.WeeksInterval = WeeksInterval;
            return Weekly;
        }
 
        /// <summary>
        /// Creats a Daily Schedule for the search content source
        /// </summary>
        /// <param name="context">The Search context to create the schedule within</param>
        /// <param name="DaysInterval">Indicates that the content should be crawled every "DaysInterval" number of days.</param>
        /// <returns></returns>
        private static DailySchedule CreateDailySchedule(SearchContext context,int DaysInterval)
        {
            DailySchedule Daily = new DailySchedule(context);
            Daily.BeginDay = DateTime.Now.Day;
            Daily.BeginMonth = DateTime.Now.Month;
            Daily.BeginYear = DateTime.Now.Year;
            //Starts at 1:00 AM
            Daily.StartHour = 1;
            Daily.StartMinute = 00;
            //Indicates that the content should be crawled every DaysInterval days.
            Daily.DaysInterval = DaysInterval;
            //Adjusting the daily schedule to run every hour
            //Hourly.RepeatInterval = 60;
            //Hourly.RepeatDuration = 1440;
            return Daily;
        }
 
        /// <summary>
        /// Create new Content Source type Search Scope
        /// </summary>
        /// <param name="site">The new created site - Entity</param>
        /// <param name="context">The Search context to create the Search Scope within</param>
        /// <param name="ContentSourceName">The Search content source name to be associated to the new Search Scope</param>
        static private void CreateBCSSearchScope(SPSite site,SearchContext context, string ContentSourceName )
        {
            string scopeName = ContentSourceName ;
            string displayGroupName = "GTS Scopes";
            // remotescopes class retrieves information via search web service so we run this as the search service account
 
            RemoteScopes remoteScopes = new RemoteScopes(SPServiceContext.GetContext(site));
 
 
 
            // see if there is an existing scope
 
            Scope scope = (from s
 
                           in remoteScopes.GetScopesForSite(new Uri(site.Url)).Cast<Scope>()
 
                           where s.Name == scopeName
 
                           select s).FirstOrDefault();
 
 
 
            // only add if the scope doesn't exist already
 
            if (scope == null)
            {
                Schema sspSchema = new Schema(context);
                ManagedPropertyCollection properties = sspSchema.AllManagedProperties;
                scope = remoteScopes.AllScopes.Create(scopeName, "Search Scope for " + scopeName, null, true, "results.aspx", ScopeCompilationType.AlwaysCompile);
                scope.Rules.CreatePropertyQueryRule(ScopeRuleFilterBehavior.Include, properties["ContentSource"], ContentSourceName);
 
            }
 
 
 
            // see if there is an existing display group         
 
            ScopeDisplayGroup displayGroup = (from d
 
                                              in remoteScopes.GetDisplayGroupsForSite(new Uri(site.Url)).Cast<ScopeDisplayGroup>()
 
                                              where d.Name == displayGroupName
 
                                              select d).FirstOrDefault();
 
 
 
            // add if the display group doesn't exist
 
            if (displayGroup == null)
 
                displayGroup = remoteScopes.AllDisplayGroups.Create(displayGroupName, "", new Uri(site.Url), true);
 
 
 
            // add scope to display group if not already added
 
            if (!displayGroup.Contains(scope))
            {
 
                displayGroup.Add(scope);
 
                displayGroup.Default = scope;
 
                displayGroup.Update();
 
            }
 
 
 
            // optionally force a scope compilation so this is available immediately
 
            remoteScopes.StartCompilation();
        }
 
        /// <summary>
        /// Creates new Business Search Metadata Properties to be used in the search
        /// </summary>
        /// <param name="context">The Search context to create the Business Search Metadata Properties within</param>
        /// <param name="ModelName">The newly created BCS Model name to aquire and map columns from</param>
        static private void CreateBCSMetadataProperties(SearchContext context,string ModelName)
        {
            Schema sspSchema = new Schema(context);
            ManagedPropertyCollection properties = sspSchema.AllManagedProperties;
 
            //TODO: ADD THE NEW FIELDS ADDED TO THE VIEW AFTER FINALIZATION
            //Create the content properties if they does not exist else get the already created ones for the mappings
            ManagedProperty ectID;
            ManagedProperty ectTitle;
            ManagedProperty ectStartDate;
            ManagedProperty ectEndDate;
            ManagedProperty ectEntity;
            ManagedProperty ectType;
            ManagedProperty ectOrganizationUnit;
            ManagedProperty ectOwner;
            ManagedProperty ectAlias;
            ManagedProperty ectStrategyPlan;
            
 
            if (!properties.Contains("ectID"))
            {
                ectID = properties.Create("ectID", ManagedDataType.Text);
                ectID.EnabledForScoping = true;
            }
            else
                ectID = properties["ectID"];
 
            if (!properties.Contains("ectTitle"))
            {
                ectTitle = properties.Create("ectTitle", ManagedDataType.Text);
                ectTitle.EnabledForScoping = true;
            }
            else
                ectTitle = properties["ectTitle"];
 
            if (!properties.Contains("ectStartDate"))
            {
                ectStartDate = properties.Create("ectStartDate", ManagedDataType.DateTime);
                ectStartDate.EnabledForScoping = true;
                ectStartDate.HasMultipleValues = false;
            }
            else
                ectStartDate = properties["ectStartDate"];
 
            if (!properties.Contains("ectEndDate"))
            {
                ectEndDate = properties.Create("ectEndDate", ManagedDataType.DateTime);
                ectEndDate.EnabledForScoping = true;
                ectEndDate.HasMultipleValues = false;
            }
            else
                ectEndDate = properties["ectEndDate"];
            
            if (!properties.Contains("ectEntity"))
            {
                ectEntity = properties.Create("ectEntity", ManagedDataType.Text);
                ectEntity.EnabledForScoping = true;
                ectEntity.HasMultipleValues = false;
            }
            else
                ectEntity = properties["ectEntity"];
 
            if (!properties.Contains("ectType"))
            {
                ectType = properties.Create("ectType", ManagedDataType.Text);
                ectType.EnabledForScoping = true;
                ectType.HasMultipleValues = false;
            }
            else
                ectType = properties["ectType"];
 
            if (!properties.Contains("ectOrganizationUnit"))
            {
                ectOrganizationUnit = properties.Create("ectOrganizationUnit", ManagedDataType.Text);
                ectOrganizationUnit.EnabledForScoping = true;
                ectOrganizationUnit.HasMultipleValues = false;
            }
            else
                ectOrganizationUnit = properties["ectOrganizationUnit"];
 
            if (!properties.Contains("ectOwner"))
            {
                ectOwner = properties.Create("ectOwner", ManagedDataType.Text);
                ectOwner.EnabledForScoping = true;
                ectOwner.HasMultipleValues = false;
            }
            else
                ectOwner = properties["ectOwner"];
 
            if (!properties.Contains("ectAlias"))
            {
                ectAlias = properties.Create("ectAlias", ManagedDataType.Text);
                ectAlias.EnabledForScoping = true;
                ectAlias.HasMultipleValues = false;
                ectAlias.MaxCharactersInPropertyStoreIndex = 450;
            }
            else
                ectAlias = properties["ectAlias"];
 
 
            if (!properties.Contains("ectStrategyPlan"))
            {
                ectStrategyPlan = properties.Create("ectStrategyPlan", ManagedDataType.Text);
                ectStrategyPlan.EnabledForScoping = true;
                ectStrategyPlan.HasMultipleValues = false;
            }
            else
                ectStrategyPlan = properties["ectStrategyPlan"];
            
 
            //TODO: ADD THE NEW FIELDS ADDED TO THE VIEW AFTER FINALIZATION
            //Map the Query Crawled Properties to the Managed Property
            MaptoManagedProperty(context, ectID, ModelName + " Items.ID", ManagedDataType.Text);
            MaptoManagedProperty(context, ectTitle, ModelName + " Items.Name", ManagedDataType.Text);
            MaptoManagedProperty(context, ectStartDate, ModelName + " Items.StartDate", ManagedDataType.DateTime);
            MaptoManagedProperty(context, ectEndDate, ModelName + " Items.EndDate", ManagedDataType.DateTime);
            MaptoManagedProperty(context, ectEntity, ModelName + " Items.Entity", ManagedDataType.Text);
            MaptoManagedProperty(context, ectType, ModelName + " Items.Type", ManagedDataType.Text);
            MaptoManagedProperty(context, ectOrganizationUnit, ModelName + " Items.OrganizationUnit", ManagedDataType.Text);
            MaptoManagedProperty(context, ectOwner, ModelName + " Items.Owner", ManagedDataType.Text);
            MaptoManagedProperty(context, ectAlias, ModelName + " Items.Alias", ManagedDataType.Text);
            MaptoManagedProperty(context, ectStrategyPlan, ModelName + " Items.StrategyPlan", ManagedDataType.Text);
        }
 
        /// <summary>
        /// Maps the external content type columns to a specific Managed Property
        /// </summary>
        /// <param name="context">The Search context to map the the external content type columns within</param>
        /// <param name="managedProperty">The managed property to map the column to</param>
        /// <param name="crawledPropertyName">The crawled property "Column" Name</param>
        /// <param name="DataType"></param>
        private static void MaptoManagedProperty(SearchContext context, ManagedProperty managedProperty, string crawledPropertyName, ManagedDataType DataType)
        {
            SPSecurity.RunWithElevatedPrivileges(() =>
            {
                Schema schema = new Schema(context);
 
                try
                {
                    Category category = schema.AllCategories["Business Data"];
                    var crawledProps = category.QueryCrawledProperties(crawledPropertyName, 1, Guid.NewGuid(), String.Empty, true).Cast<CrawledProperty>();
                    var crawledProp = crawledProps.FirstOrDefault();
                    if (crawledProp != null)
                    {
                        var mappings = managedProperty.GetMappings();
                        mappings.Add(new Mapping(crawledProp.Propset, crawledProp.Name, crawledProp.VariantType, managedProperty.PID));
                        managedProperty.SetMappings(mappings);
                        managedProperty.Update();
                    }
                    else
                    {
                        Console.WriteLine("Query Crawled Property " + crawledPropertyName + " was not found - Mapping faild.");
                    }
                }
                catch (Exception ex)
                {
                    throw new Exception("Faild to map field to Crawled Property \n" + ex.Message);
                }
            });
 
        }
    }
}

Tuesday, January 29, 2013

Disable (Read-only) for sharepoint 2010 People Picker (PeopleEditor)

Using the SharePoint 2010 People Picker (People Editor) user control in your custom visual web part is very powerful tool (which by the way makes you web part looks nice "JUST KIDDING IT IS SHIT" ) .

Any way .... you may be depressed and forced by the business needs to use this shit control, also by business needs you may want to disable this control programmatically to and use it only for display "Read Only". What we have learned in the ASP.NET is that any web control should have the property "Enabled" which if you ever set it to false the control is not enabled "LOGIC" , but hell no , The people picker do not offer you the easy way.
So lets have the following scenario, You want to auto fill this control in the page load with a specific user then disable it.

Assuming you have the following people editor control on your page or web part
 <SharePoint:PeopleEditor ID="PeoplePickerOwner" runat="server" Width="350" SelectionSet="User" MultiSelect="false" AllowEmpty="true" ValidationGroup="AdvancedSearchVld" />  

To auto fill and disable this control use the following method



   1:  private void SetOwnerUser()
   2:          {
   3:              if (CurrentUser != null)
   4:              {
   5:                  PeoplePickerOwner.Entities.Clear();
   6:                  PickerEntity UserEntity = new PickerEntity();
   7:                  UserEntity.DisplayText = CurrentUser.Name;
   8:                  UserEntity.Key = CurrentUser.LoginName;
   9:                  PeoplePickerOwner.Entities.Add(PeoplePickerOwner.ValidateEntity(UserEntity));
  10:                  PeoplePickerOwner.ShowButtons = false;
  11:                  PeoplePickerOwner.AllowTypeIn = false;
  12:              }
  13:          }




UPDATE: Danial Martins asked "I select the name press backspace it deletes it. Is there a workaround?", Thanks to Hoshang Akeryi, Below is the solution:




1
2
3
4
5
6
7
8
9
ppl.ShowButtons = false;
ppl.AllowTypeIn = false;
than add the script to your page:
<script type="text/javascript">    
   $(document).ready(function () 
   {
        $(this).find("div[Title='People Picker']").attr("contentEditable",false);
   }
</script>



Monday, January 21, 2013

Reset-Set-validate-Resolve Sharepoint people picker By javascript


SharePoint 2010 people picker is a very useful control to be used on your custom forms or custom visual webparts to select people within your corporate .

Some of the clients requirements would specify to preload  this field with some values by default as the current logged in user , ok what about doing this in the client side , Now everyone is going towards the client side operations , we do not want to exhaust the server those operations.

People picker could be handled by javascript methods:

To Reset  the people picker value you could use:

   1:  function ResetPP() {
   2:          var identifier = '<%= PeoplePicker.ClientID%>';
   3:          var value = '';
   4:          var tags = document.getElementsByTagName('DIV');
   5:   
   6:          for (var i = 0; i < tags.length; i++) {
   7:              var tempString = tags[i].id;
   8:              if ((tempString.startsWith(identifier)) && (tempString.indexOf('_upLevelDiv') > 0)) {
   9:                  tags[i].innerHTML = value;
  10:                  break;
  11:              }
  12:          }
  13:      }


To Set the People Picker with a certain value:
to use it something like :

   1:  SetPP('<%=PeoplePicker.ClientID%>', 'sps\administrator');
 
The Method:

   1:  function SetPP(identifier ,value) {
   2:   
   3:          var tags = document.getElementsByTagName('DIV');
   4:   
   5:          for (var i = 0; i < tags.length; i++) {
   6:              var tempString = tags[i].id;
   7:              if ((tempString.startsWith(identifier)) && (tempString.indexOf('_upLevelDiv') > 0)) {
   8:                  tags[i].innerHTML = value;
   9:                  break;
  10:              }
  11:          }
  12:      }
 
Find if the People Picker is resolved:

   1:  function IsPeoplePickerValueResolved() {
   2:          var eEntityData = $("div[id='divEntityData']");
   3:          if (eEntityData.length > 0) {
   4:              var isResolved = eEntityData.attr("isresolved");
   5:              return (isResolved == "True");
   6:          }
   7:          return false;
   8:      }
 
Validate the People Picker:

   1:  function ValidateOwner(PeoplePickerOwner, PeoplePickerContextName) {
   2:          if (!ValidatePickerControl(PeoplePickerOwner.id)) {
   3:              ShowValidationError();
   4:              return false;
   5:          }
   6:          var arg = getUplevel(PeoplePickerOwner.id);
   7:          var ctx = PeoplePickerOwner.id;
   8:          EntityEditorSetWaitCursor(ctx);
   9:          WebForm_DoCallback(PeoplePickerContextName, arg, EntityEditorHandleCheckNameResult, ctx, EntityEditorHandleCheckNameError, true);
  10:          return false;
  11:      }
 
To use this method you can use the following:


   1:  var PeoplePickerOwner = document.getElementById('<%= PeoplePicker.ClientID %>');
   2:  var PeoplePickerOwnerHiddenSpanData = document.getElementById('<%= PeoplePickerOwner.ClientID + "_hiddenSpanData"%>');
   3:  var PeoplePickerContextName = $("#" + PeoplePickerOwnerHiddenSpanData.id).attr("name").replace("$hiddenSpanData", "");
   4:   
   5:  ValidateOwner(PeoplePickerOwner, PeoplePickerContextName);
 
To call the "Click" the check names button:

   1:  function ClickCheckName() {
   2:          document.getElementById('<%=PeoplePicker.ClientID%>' + "_checkNames").click();
   3:      }