Lars Nielsen's Discoveries

January 5, 2011

Page layout to show Modified By metadata and version

Filed under: Publishing,SharePoint — Lars Nielsen @ 9:23 pm
Tags: ,

A few times now I have created web sites and Intranet using the publishing features in SharePoint.  I don’t use the Publishing Portal site definition – I just start with a blank site collection and turn on the features that I need (Publishing Infrastructure on the site collection has to be turned on of course).  I also prefer to create my own information architecture even if the built-in content types, site columns and page layouts are adequate.  I create a special content type based on the Article page content type, and then create one or more page layouts to use it.  I keep cutomised elements separate and give them a name prefix so I can easily see what has been customized.

For a page layout I needed to show, automatically on every page, the last person who edited the page, the date it was changed and the version number.  You can do this by including special controls on the page layout file.  Here’s how to do it.

Go to the root site of the site collection.  In Site Settings go to the Master Page gallery.  To make things easier temporarily, you can turn off (in the Document Library settings) content approval, workflows, and require checkout.  That’s assuming the site is not live and you’re the only one working on it.  Just remember to turn all those things back on again before the site goes live.

Open the root of the site collection in SharePoint Designer.  Navigate to the Master Page and Page Layouts gallery (_catalogs/masterpage) and either create a new page layout or edit one you have already created.  Here’s a good description of how to create page layouts in SharePoint.  Open up the .aspx page for the page layout.

In your layout you can add the following markup to show the “Created By” field as the person’s name – and clicking on the name takes you to their profile page:

<SharePoint:FormField ID="FormField1" ControlMode="Display" FieldName="Author" DisableInputFieldLabel="true" runat="server"/>

You can use a similar syntax to extract other data about the page, for example:

Last modified: <SharePoint:FormField ID="FormField2" ControlMode="Display" FieldName="Modified" DisableInputFieldLabel="true" runat="server"/>

Version: <SharePoint:FormField ID="FormField3" ControlMode="Display" FieldName="Version" DisableInputFieldLabel="true" runat="server"/>

You also need to make sure that you have the directive at the top of the layout page:

<%@Register TagPrefix="SharePoint" Assembly="Microsoft.SharePoint, Version=, Culture=neutral, PublicKeyToken=71e9bce111e9429c" namespace="Microsoft.SharePoint.WebControls"%>


December 28, 2010

Update the approval status in a SharePoint Designer workflow

Filed under: Publishing,SharePoint — Lars Nielsen @ 8:49 pm
Tags: ,

I recently had to build a workflow in SharePoint Designer to model a fairly lengthy approval process for policy documents involving multiple stages of review, each of which has to be passed before the draft document moves to the next review stage. The workflow itself was quite long but essentially boiled down to a series of repeated steps. Each step involved assigning a task to a reviewer (or reviewers) and then updating the status of the document to show what stage it was at. The Policies document library has enforced check in/out, version control, and content approval all turned on. I also needed a field called Review Stage to show what stage the document was at – that was a simple choice field with 7 different stages.

Version History

The obvious thing was to add the Review Stage field to the document library.  After some playing around with the workflow, however, I decided to do things a more complex way.  I created a new list called Document Review Stages with two fields, one was the choice field Review Stage and the other was a number field called DocumentID.  For each policy document in the document library, there is a corresponding entry in the Policy Review Stages list.  The DocumentID holds the ID (the built-in column that SharePoint populates automatically) of the policy document.  In other words, I effectively created a one-to-one relationship between the Policies document library and the Policy Review Stages list.  Of course, this being SharePoint and not a full-blown database, this isn’t a proper relationship – there’s no enforced referential integrity, no cascading updates, etc.  But the basic principle is the same.

So why did I do it in this convoluted way?  Well because the Policies document library has enforced check-in/out and major/minor version control.  So suppose I had the Review Stage column on the Policies document library itself.  That would mean any change to the document, even just changing its metadata, would create a new version.   And that means that the workflow, if it wants to progress the document to the next review stage, would have to first check out the document, then set the Review Stage field, then check the document back in again.  And you’d then have 2 versions of that document, both exactly the same in content, but with different Review Stage metadata values.  This would result in a confusing version history for each document, with some genuinely different versions of the document (edited by different reviewers) but also many identical copies of each document created automatically by the workflow as it moves the document through the review stages.  By moving the  Review Stage column to a different list, the workflow can update the Review Stage field without creating a fresh copy of the document in the version history. That makes the version history much cleaner and more readable.

Approval Catch-22

This being a SharePoint Designer workflow, not a Visual Studio one, I had to attach it to the Policies document library.  I got the workflow to find the Policy Review Stages list item related to its corresponding policy document, and update the Review Stage column on the list.  So far so good, but then right at the end of the workflow, the last step was to make the document go live by setting its approval status to be “Approved”.  There’s a special workflow action to do just that, so it seems easy enough.  But there’s a catch.  If you want to make the document Approved status, you have to change it, and the workflow complains if you try to change it without first checking it out.  OK, so you change your workflow to check the document out first, and then approve it.  But now the workflow complains because you can’t approve a document that is checked out.  Catch-22!

So how do you get round this?  Well you can use the Start Another Workflow action to start the normal Approval workflow on the document.  The built-in Approval workflow can work with a document that’s not checked out.  So you end your workflow early, before the final approval task is allocated, and hand off control to a standard Approval workflow to manage the final approval step.

Create a free website or blog at