Wednesday, August 29, 2012

Lookup field dropdown position issue

I was trying to create a custom webpart that will display a form to submit data in a list. The list had few lookup fields.
Whenever, I click on that image it displays us with a drop-down containing all the values, but the funniest part is you can select a particular item either by double click or by selecting it once and clicking it else where on the page or by tabbing system of the keyboard. But I want it work like a normal drop-down selection in other words just with a single click and also the control should be closer to other controls from usability perspective.


To fix this issue. you need to set InDesign ="true";

 Ex.BaseFieldControl webControl = field.FieldRenderingControl;

webControl.ID =
string.Format("ctrl_{0}_{1}", field.InternalName, pplCount);


webControl.FieldName = field.Title;

webControl.ItemContext =
SPContext.GetContext(HttpContext.Current, lstReqData.DefaultView.ID, lstReqData.ID, lstReqData.ParentWeb);

webControl.RenderContext = SPContext.GetContext(HttpContext.Current, lstReqData.DefaultView.ID, lstReqData.ID, lstReqData.ParentWeb);

webControl.ControlMode = SPControlMode.New;

webControl.ListId = lstReqData.ID;

webControl.InDesign =
true;

 If you are using form field you can set it by following changes in aspx page

<SharePoint:FormField runat="server" InDesign="true" ID="myFieldId" ControlMode="New" FieldName="myFieldName" />

Wednesday, June 6, 2012

Use Metadata Navigation in Wiki Pages (Sharepoint 2010)

If you want to use Metadata navigation tree in Wiki Page, you have to do the following:
  • Activate Metadata Navigation and Filtering feature on site level
image
  • Navigate to Library settings page of your “Pages” library and select Metadata navigation settings
  • From “Configure Navigation Hierarchies” select “Wiki Categories” and click “Add” and “OK”
image
  • Navigate to Pages library again
http://<Your Wiki Site Url>/Pages/Forms/AllItems.aspx

But when you click on any of the pages on the right panel, the navigation tree will disappear.

 

You have to change the master page or better to create and deploy a new one (via Sharepoint Designer or wsp)
You have to include a reference to MetadataNavTree control and put it in some placeholder (QuickLaunchNav) on the master page file.
  • Put this markup in the beginning of the page
<%@ Register TagPrefix="wssuc" TagName="MetadataNavTree" src="~/_controltemplates/MetadataNavTree.ascx" %>
  • Put this code in <asp:ContentPlaceHolder id="QuickLaunchNav" runat="server"> tag
<asp:ContentPlaceHolder id="QuickLaunchNav" runat="server">
<wssuc:MetadataNavTree id="mdnt" runat="server" />

The result is:

Monday, May 21, 2012

Calling Javascript / Jquery / ECMA script from an UpdatePanel in SharePoint

You've probably used ClientScript.RegisterStartupScript in your SharePoint application page and it works just fine.
for example:-
Page.ClientScript.RegisterClientScriptBlock(this.GetType(), "Close", "<script type=\"text/javascript\">SP.UI.ModalDialog.commonModalDialogClose(0);</script>");
 However, if you try it within an UpdatePanel control, you would notice that it simply doesn't execute.

The solution to this problem is to use ScriptManager.RegisterStartupScript instead. You can call it like so:
ScriptManager.RegisterClientScriptBlock(UpdatePanelSites, UpdatePanelSites.GetType(), "Close", "SP.UI.ModalDialog.commonModalDialogClose(0);", true);

Thursday, January 12, 2012

Select Distinct or Unique values from SharePoint List



I was looking around for the simple way of getting unique values from SharePoint list without affecting performance. I have seen several ways of doing it. For example, we can get all the items and then check if item exists in the array or some other collection. Another example, you can load your data into DataTable and then select unique values using DataTable method, following is snippet of code below on how to use it

GetUniqueColumnData("List Name", "ColumnName");


private void GetUniqueColumnData(string ListName, string ColumnName)
{
try
{
using (SPSite site = SPContext.Current.Site)
{
using (SPWeb web = site.OpenWeb())
{
DataTable dtList = new DataTable("TableName");
SPList list = web.Lists[ListName];
if (list != null)
{
SPQuery query = new SPQuery();
query.Query = "<OrderBy><FieldRef Name='" + ColumnName + "'/></Order By><FieldRef Name='" + ColumnName + "'/>";
dtList = list.GetItems(query).GetDataTable();
dtList = new DataView(dtList).ToTable(true, new string[] { ColumnName });
}
}
}
}
catch (System.Exception ex)
{
LoggingService.LogErrorInULS(ex.Message + "Stack Trace: "+ ex.StackTrace,TraceSeverity.High);
}
}

but I discovered one very simple method for selecting Distinct Values in SharePoint called GetDistinctFieldValues. Please see snippet of code below on how to use it


using (SPSite site = SPContext.Current.Site)
{
using (SPWeb web = site.OpenWeb())
{
SPList objList = web.Lists["List Name"];
SPField field = objList.Fields.GetField("Field Name");

Object[,] values;
uint numberValues = objList.GetDistinctFieldValues(field, out values);

for (int i = 0; i < numberValues; i++)
comboBox1.Items.Add(values.GetValue(0, i).ToString());

}
}

Tuesday, October 18, 2011

Remove Event Receiver in Sharepoint List

I encountered a strange behavior in SharePoint; this phenomenon happens when I created an event handler and attach it to a list template using a web-site level feature. I saved the list as template, the event handler reference is saved inside the list template (STP file) and when I created an instance using the template it caused the following problems:
1.     The event handler activated automatically on a web site where the feature is not enabled.
2.     If the event feature is enabled in the web site the event will pop twice (In my case it was sending mail twice on Item Added event).
I wrote a small console application to remove attached event handlers.

        static void Main(string[] args)
        {
            try
            {
                string strAppUrl = ConfigurationManager.AppSettings["applicationUrl"].ToString();
                string[] strDelimit = { "/" };
                string[] strSites = strAppUrl.Split(strDelimit, StringSplitOptions.RemoveEmptyEntries);
                string strSiteURL = strSites[0] + "//" + strSites[1];
                string strWebRelUrl = strAppUrl.Substring(strSiteURL.Length);
                using (SPSite site = new SPSite(strSiteURL))
                using (SPWeb web = site.OpenWeb(strWebRelUrl))
                {
                    string listName = ConfigurationManager.AppSettings["ListName"].ToString();
                    SPList list = web.Lists[listName];

                    if (list != null)
                    {
                        int countVal = list.EventReceivers.Count;
                        if (countVal > 0)
                        {
                            for (int i = 0; i < countVal; i++)
                            {
                                Console.WriteLine(list.EventReceivers[0].Name.ToString() + "Deleting");
                                list.EventReceivers[0].Delete();
                            }
                            //Update the list.
                            list.Update();
                            Console.WriteLine("Operation completed succesfully");

                            Console.ReadLine();
                        }
                        else
                        {
                            Console.WriteLine("No event handlers are attached to the list : " + list.Title);
                            Console.ReadLine();
                        }
                    }
                }
            }
            catch (Exception ex)
            {
                Console.Write(ex.Message);
                Console.ReadLine();
            }
        }
        

Wednesday, October 12, 2011

InfoPath 2010 migration with user profile and sharepoint user group services



As we know one of the exciting features in InfoPath 2010 is the ability to extend or enhance the forms used by SharePoint lists for creating, editing, or showing list items. Today, you can modify list forms layouts, set validation rules, or create additional views using little or no code. When you are finished modifying the list forms, reflecting your changes back to SharePoint is just a matter of using the one-click publishing capability that comes out of the box with the list form.
In this post, I will show how to use Infopath data connection to build rules for hiding/disabling fields based on the SharePoint group membership of current user. In my case, I would like to fill First Name, Last Name, E-Mail Address and Employee Status based on PID (Employee ID) from user profile and Hide the Save and Cancel buttons as shown below for any users who are not members of  SharePoint group “Site Owners”.

Here are my steps to achieve the goal:
1.       The first very step obviously is clicking the Customize Form on the ribbon to edit the form in InfoPath Designer.
2.       Adding two SOAP Web Services data connections to the forms, there are a lot of articles about this topic, so I wouldn’t go into any detail such as this one.
§  GetUserProfileByName in http://<server-name>/_vti_bin/UserProfileService.asmx. Check ‘Automatically retrieve data checkbox when the form opened’. So, the first action that you will create is a query for that connection. If you do not pass a parameter to the User Profile Service, the profile information that is returned is for the current user, which is the outcome that you want for a new request.
§  GetGroupCollectionFromUser in http://server-name/_vti_bin/UserGroup.asmx (put a sample value, of an existing account such as domain\account_name). Clear ‘automatically retrieve data when the form opened’ checkbox.
3.       Now the problem is that the generated data connection doesn’t display the returned fields. As shown in the picture, the dataFields children elements only contain userLoginName which is not correct.
4.        I found an approach from Sumit’s SharePoint blog, he explains how to correct the xsd in ‘Correcting the xsd for the Return Fields’ section in his post.
5.       Save the form as source files as we are going to modify the xml schema of the GetGroupCollectionFromUser data connection. To save the form as sources, go to Files -> Publish as shown below and then click Export Source Files on the next window. The exported files would look like the below screenshot.
6.       It consists of infopath definition file (manifest.xsf), xml schemas of all data connections (.xsd files), some xml files that contain sample data of the data connections and an xsl file (view.xsl) used for form rendering. In our case, we only focus on GetGroupCollectionFormUser.xsd (the xml schema for GetGroupCollectionFromUser data connection).
7.       Modify the GetGroupCollectionFromUser1.xsd, and make sure to close the InfoPath Designer before editing. To summarize his approach basically, we need to add this below type definition

<s:complexType name="GetGroupCollectionFromUserType">
    <s:sequence>
      <s:element minOccurs="0" maxOccurs="1" name="userLoginName" type="s:string"/>
      <s:element minOccurs="0" maxOccurs="1" name="Groups">
        <s:complexType>
          <s:sequence>
            <s:element maxOccurs="unbounded" name="Group" >
              <s:complexType>
                <s:attribute name="ID" type="s:unsignedShort"></s:attribute>
                <s:attribute name="Name" type="s:string"></s:attribute>
                <s:attribute name="Description" type="s:string"></s:attribute>
                <s:attribute name="OwnerID" type="s:unsignedByte"></s:attribute>
                <s:attribute name="OwnerIsUser" type="s:string"></s:attribute>
              </s:complexType>
            </s:element>
          </s:sequence>
        </s:complexType>
      </s:element>
    </s:sequence>
  </s:complexType>

8.       below this line (at the 2nd line of the file)
<s:import namespace="http://www.w3.org/2001/XMLSchema"></s:import>
9.       Then find this below:
<s:element name="GetGroupCollectionFromUser">
   <s:complexType>
     <s:sequence>
  <s:element minOccurs="0" maxOccurs="1" name="userLoginName" type="s:string">
         </s:element>
</s:sequence>
    </s:complexType>
  </s:element>

10.   Replace it with this:
<s:element name="GetGroupCollectionFromUser" type="tns:GetGroupCollectionFromUserType">
</s:element>

11.   Save GetGroupCollectionFromUser1.xsd and open manifest.xsf in design mode. You can see ID, Name, Description, OwnerID and OwnerIsUser fields under Group as displayed in the picture.

12.   Now my connections are configured. Next step is to create a rule to invoke GetUserProfileByName web service to fill details from user profile and invoke GetGroupCollectionFromUser web services method to hide buttons.
13.   As the method is not invoked when the form is opened, we need to invoke the method after the GetUserProfileByName invocation is finished. To do this, Choose GetUserProfileByName data connection and apply rules on the Get User Details button.
14.   Set rule type is Action. The rule actions are:
                               I.            Set field’s value : AccountName from the GetUserProfileByName to PID
                              II.           Query GetUserProfileByName data connection to get user details for given PID
                           III.            Set FirstName field form user profile. This blog shows how to set the property value.

                             IV.            Using step III. you can fill Last Name, Email and Employee Status fields.
                               V.            Now the next step is to query GetGroupCollectionFromUser to get SharePoint user group for this user.
                            VI.            Set field’s value: the userLoginName of the GetGroupCollectionFromUser’s queryFields to the Property     value AccountName from the GetUserProfileByName.  




             VII.     Query for data: GetGroupCollectionFromUser.
            VIII.      Create a formatting rule for the field (in my case approval field) that we want to disable/hide if current user is not member of a SharePoint Group. Put a condition as below, where The “Site Owners” is the SharePoint Group Name.
     


    And finally when you publish the form, it would look like the below screen-shot:
     
Once you click, “Get User Details”, you will get the below screen which contains “Submit” button for users who are members of “Site Owners”.
NOTE: “Submit” button will not be seen below “Child Details” section for users who are not members of “Site Owners”.