After receiving many comments from two previous posts based on Vacation Trackers in SharePoint, one such area of interest is around automating the calculation of time off that includes factoring in weekends and holidays. This is what we are going to tackle in this post.
The two previous vacation tracker posts are below:
SharePoint Team Vacation Tracker
5 Big Enhancements to a SharePoint Vacation Tracker in Microsoft Lists
The End Result: Vacation Tracker SharePoint List
The vacation tracker will have 4 columns used to track time off that are updated by Power Automate.
- The Total Days Off Column is the difference between the Start Date and End Date
- The Workdays column is the total number of weekdays between the Start Date and End Date not including weekends.
- The Holidays column is the total number of holidays between the Start Date and End Date
- The Net Days Off column is the difference between the Workdays column and Holidays column.
The image below shows the All Items view of the finished product.
data:image/s3,"s3://crabby-images/9a1bc/9a1bc63116a0ec1c372a23fee70c84831a7a1bd5" alt="SharePoint List All Items View"
The image below shows the Hours view of the finished product.
data:image/s3,"s3://crabby-images/c5833/c5833cc55eb15d5043f2c1026603b21691f8f1a5" alt="SharePoint List Hours View"
Catch my video on this blob post below:
Stop Infinite Loops When Updating the SharePoint List
This flow will create an infinite repeat loop when the first change is made to the list but there are steps that can be taken in SharePoint to prevent this from happening. I came across the solution from Scott Shearer’s blog which you can access here for reference. These next steps are based off of his blog and adapted for this List.
Step 1: Turn on list versioning:
Go to List Settings, then click on Versioning Settings, then select “Yes” to “Create a version each time you edit an item in this list?”, then click “Ok”.
data:image/s3,"s3://crabby-images/d8996/d8996bb612b52ba5a0d6d25a295985278ce4508b" alt="Set Up List Versioning"
data:image/s3,"s3://crabby-images/39862/398624524ceadefac7429c42b7b503bc40f3da86" alt="Select Versioning Settings"
data:image/s3,"s3://crabby-images/d8092/d80922dcaaedc0dbc38134d392e51249dac1cc0e" alt="Select Create a Version Each Time"
Next, we are to create a new column, from the SharePoint Settings page go to “Create Column”. This column is our Append Text column.
data:image/s3,"s3://crabby-images/0000f/0000f8a8581d9b748cc324948e91c2bd57e77704" alt="Create New Column Called StopRun"
The steps below are for the creation of the StopRun column:
- Enter “StopRun” for the column name under the “Column Name” field.
- The “Type of Information in this column” is “Multiple Lines of Text”
- “Specify the Type of Text to Allow” as “Plain Text”
- Make sure “Append Changes to Existing Text is set to “Yes”.
- Then click on “Ok to save the column.
data:image/s3,"s3://crabby-images/38b9b/38b9b63e26b3e2bd184c176a0301d56aebb9952e" alt="Setup New Column Details"
Then from the Settings page, select “Advanced Settings”.
data:image/s3,"s3://crabby-images/e2b43/e2b432a0c3d68848f426856b5f1fae80a0da8d47" alt="Select Advanced Settings"
Make sure to change “Allow Management of Content Types” to “Yes”, then select “OK”.
data:image/s3,"s3://crabby-images/ea946/ea946c545a5d40dd840ab619fa8791f7c211977a" alt="Select Allow Management of Content Types"
Now from the Settings page, click on “Item”.
data:image/s3,"s3://crabby-images/f99e5/f99e5a362716bff8dd6ff726bd7389d368b1f15d" alt="Select Item"
Then click on “StopRun”.
data:image/s3,"s3://crabby-images/7c7c7/7c7c7983b63a33a9182773bf1861285228b22d54" alt="Select StopRun"
Under “This Column Is:”, select “Hidden” and then select “Ok”.
data:image/s3,"s3://crabby-images/45be6/45be66562731d7aa5554ec4b58abc4727bdf3d0d" alt="Select StopRun as Hidden Column"
To summarize what we did here, we created a hidden column in SharePoint that is updated by the flow with the value “StopRun” on a new entry or changed entry. We setup the flow in the next step and its trigger will only start if the value in StopRun is null. This will prevent the infinite loop.
Develop Power Automate Flow
The first action, the trigger, is “When an Item is Created or Modified in SharePoint”. Select the SharePoint site address and the name of the SharePoint List to use for this flow. I’ve updated the recurrence of how often to check for changes to the list to every 30 seconds but it’s entirely up to you and your requirements.
data:image/s3,"s3://crabby-images/d4b55/d4b552f50f5689a4e0baf2efd32141f8b7d0e001" alt="Trigger: When an Item is Created or Modified in SharePoint"
To build on the work in the first part of the blog to prevent repeat loops, update the settings in the trigger by adding the condition below. This means the flow will only run if the StopRun column in the SharePoint List has no value.
@equals(triggerBody()?[‘StopRun’],null)
data:image/s3,"s3://crabby-images/24b8a/24b8a0ad800e7e6408ed3498983ba2f698040169" alt="Update Trigger Settings to Prevent Infinite Loops"
The next three actions create the main variables for this flow. The first two are coming from the SharePoint List, the vacation Start Date and End Date. The third action is creating an array of holidays to track.
The action Initialize Variable-Start Date creates the first variable based on the vacation Start Date. To do this, find the Initialize Variable action, name it varStartDate, for the Type select “String”, and for the value select the Start Date from the trigger action under dynamic content.
data:image/s3,"s3://crabby-images/7fb43/7fb43440d7fb8393de9dadf882fe75d626272e5d" alt="Intialize Var Start Date"
The action Initialize Variable-End Date creates the variable based on the vacation End Date. To do this, find the Initialize Variable action, name it varEndDate, for the Type select “String”, and for the value select the End Date from the trigger action under dynamic content.
data:image/s3,"s3://crabby-images/939ec/939ec67204abcd795016e80427a99b0aa45d5f54" alt="Intialize Var End Date"
The next action 2-Initialize Variable-Holidays creates another variable but this time it’s an array. To do this, find the Initialize Variable action, name it varHolidaysArray, and for the Type select “Array”. The values start with a hard left bracket, with date values in double quotation marks and separated by a column and ending with a hard right bracket. The date values need to be in the format YYYY-MM-DD.
For simplicity I added a few US Federal holidays for the end of 2024 and the first half of 2025. Someone will need to manually update these values each year for this flow to remain operational. It would be my recommendation to create another SharePoint List where you could maintain the list of federal holidays and retrieve them in the flow.
data:image/s3,"s3://crabby-images/0c5af/0c5afa72f9bdfb36cd082ce0cf99ee5faacd6ed8" alt="Intialize Var Holidays"
The next action is a compose action and its expression below converts the StartDate field into the number of ticks. Ticks represents the number of 100-nanosecond intervals that have elapsed since 12:00am, January 1, 0001 which is the start of the Gregorian calendar. By doing this we can perform date calculations similar to what we do in Excel by converting the date into a number value starting at 1/1/1900.
ticks(triggerBody()?[‘StartDate’])
data:image/s3,"s3://crabby-images/1b446/1b446f796ac25dcc645b47247fe0bc69130c9239" alt="Compose StartDateTicks"
The next action is similar to above but for the End Date. See it’s expression below.
ticks(triggerBody()?[‘EndDate’])
data:image/s3,"s3://crabby-images/71994/71994f748f1cd801e16e47d8fdf76bf54881843e" alt="Compose EndDateTicks"
For the next action add an Initialize Variable action, name the action Full Days, name the variable varFullDays, the type is an Integer, and its value is an expression detailed below. The expression retrieves the tick values from the start and end date, subtracts the end date from the start date then divides that value by 864,000,000,000 (which is the number ticks in a day) to convert the value in days and then add 1 day to value so the Start Date is included as day in the calculation.
add(div(sub(outputs(‘4-Compose-EndDateTicks’), outputs(‘3-Compose-StartDateTicks’)),864000000000),1)
data:image/s3,"s3://crabby-images/fa462/fa4626465dcde2d57d38930537b45f3d3f27d205" alt="Initialize Variables Full Days"
The next action is another Initialize Variable action to be called WorkDays. The variable name is varWorkDays, its type is an integer and it’s value is varFullDays which is the variable from the previous action. This variable will be used to calculate the number of workdays between the Start Date and End Date in the upcoming actions.
data:image/s3,"s3://crabby-images/54a4e/54a4ed7b08f29357b2d66dca4094bdc0ac2aff87" alt="Initialize Variables Work Days"
The next action is a Decrement Variable action which is an action that decreases the value of the variable. This action is called Workdays by 2 for each full week. The name of the variable is varWorkDays. The value is an expression detailed below, and it divides the variable by 7 (converting its value into full weeks) and then multiplying that value by 2.
mul(div(variables(‘varFullDays’),7),2)
data:image/s3,"s3://crabby-images/dfd12/dfd123032bc9e0317ff88e64ad78e958785bfa2a" alt="Initialize Variables Work Days by 2"
The next action is called “WorkDays by 1 if Start and End are the same day on a weekend”, it is another Decrement action that is using varWorkDays. It’s value is an expression which is shown below. The formula is checking if the Start Date and End Date fall on the same day of the week and if that day is a weekend. If both conditions are true it returns a 1 else it returns a 0. This means the variable will either decrease by one or not at all.
if(equals(dayOfWeek(variables(‘varStartDate’)),dayOfWeek(variables(‘varEndDate’))),
if(or(equals(dayOfWeek(variables(‘varStartDate’)),0),equals(dayOfWeek(variables(‘varStartDate’)),6)),1,0),0)
data:image/s3,"s3://crabby-images/fa45b/fa45bbc9a32980518ebe6b930ea07db7928668ff" alt="Decrement Variable Work Days by 1"
The next action is another Decrement Variable, it’s name is “WorkDays by 2 if Start Sat and End Midweek”. It’s working with varWorkDays and it’s expression is below. The results of this expression returns a 2 if the Start Date is a Saturday and the End Date is a week day, if so it will decrease the variable by 2, else it will return a 0 and not affect the variable.
if(and(equals(dayOfWeek(variables(‘varStartDate’)),6), not(or(equals(dayOfWeek(variables(‘varEndDate’)),0), equals(dayOfWeek(variables(‘varEndDate’)),6)))),2,0)
data:image/s3,"s3://crabby-images/858d0/858d00f578d8aa99ce5fef473d505b1d28c64406" alt="Decrement Variable Work Days by 2"
The next action is another Decrement Variable action, it’s name is “WorkDays by 1 if Start Sun and End Midweek” It’s working with varWorkDays and it’s expression is below. The results of this expression returns a 1 if the Start Date is a Sunday and the End Date is a weekday, else it returns a 0.
if(and(equals(dayofweek(triggerBody()[‘StartDate’]),0),not(or(equals(dayofweek(triggerBody()[‘EndDate’]),0),equals(dayofweek(triggerBody()[‘EndDate’]),6)))),1,0)
data:image/s3,"s3://crabby-images/41bc2/41bc2e8d1fd9cefbebb5a76e5beab0274b3f3e7a" alt="Decrement Variable Work Days by 1"
The next action is another Decrement Variable action, it’s name is “WorkDays by 1 if Start Midweek and End Sat” It’s working with varWorkDays and it’s expression is below. The results of this expression returns a 1 if the Start Date is a weekday and the End Date is a Satruday, else it returns a 0.
if(and(not(or(equals(dayofweek(triggerBody()[‘StartDate’]),0),equals(dayofweek(triggerBody()[‘StartDate’]),6))),equals(dayofweek(triggerBody()[‘EndDate’]),6)),1,0)
data:image/s3,"s3://crabby-images/02261/02261294c02c13a0d78a7383cc99e328b0a45105" alt="Decrement Variable Work Days by 1 Midweek"
The next action is another Decrement Variable action, it’s name is “WorkDays by 2 if Start Midweek and End Sun” It’s working with varWorkDays and it’s expression is below. The results of this expression returns a 2 if the Start Date is a weekday and the End Date is a Sunday, else it returns a 0.
if(and(not(or(equals(dayofweek(triggerBody()[‘StartDate’]),0),equals(dayofweek(triggerBody()[‘StartDate’]),6))),equals(dayofweek(triggerBody()[‘EndDate’]),0)),2,0)
data:image/s3,"s3://crabby-images/41553/41553afe1bfb9e49769de2df60d5e7104da70eae" alt="Decrement Variable Work Days by 2 Midweek"
The next action is an Initialize Variable action to create a new variable called “HolidaystoRemove”. It’s an integer and starts with no value.
data:image/s3,"s3://crabby-images/74e3f/74e3ff9d74114402856f3dfe2ddfc886a9486e48" alt="Initialize Holidays to Remove"
The next action is an Apply to Each action and is cycling through the variable varHolidaysArray.
data:image/s3,"s3://crabby-images/ff6b7/ff6b7aa0e77f9a23af8e496ef29a945d93079b9e" alt="Apply to Each"
The next action is a Condition and evaluates each date in the array to see if it is between the Start Date and End Date. For each false value nothing will happen.
data:image/s3,"s3://crabby-images/a11cd/a11cd14151120bafaaf191add90d3ede08b39dd4" alt="Condition"
For each true condition increase the variable HolidaystoRemove by 1.
data:image/s3,"s3://crabby-images/a0a7c/a0a7c9042b7909542a9a7fc597874a39d1d7e330" alt="Increment Variable"
The full view of the Apply to Each action is below:
data:image/s3,"s3://crabby-images/538e7/538e70632e93f791074b4a7258c342bb078620de" alt="Apply to Each Loop"
The next action is another Decrement Variable action called “WorkDays by HolidaystoRemove” and it decreases the number of varWorkDays by the number of HolidaystoRemove variable.
data:image/s3,"s3://crabby-images/91f4e/91f4ee7bde754d9a5eefde93d6f03386222c89c5" alt="Decrement Workdays from Holidays"
The last action is a SharePoint List Update Item action. Select the Site Address, List Name and ID from the same site as entered in trigger.
Next we are going to update 5 columns in our SharePoint List.
- Total Days Off: Expression, see below.
- Work Days: Expression, see below.
- Holidays: use HolidaystoRemove variable
- Net Days Off: use varWorkDays
- StopRun: Enter “NoRun” directly in the field.
data:image/s3,"s3://crabby-images/31297/3129725755129f6ce44d800acf0c2ae4731b280c" alt="Update Item"
The Total Days Off column has an expression which calculates the difference between the Start Date and End Date and adds 1 to it.
data:image/s3,"s3://crabby-images/aafd5/aafd526b3f1d7b9305655bdb479bee61cb93b718" alt="Total Days Off"
The Workdays column is an expression which calculates the difference between all workdays, less weekends and holidays.
data:image/s3,"s3://crabby-images/009f7/009f71c13b6d22c1b063940dac5d4f6d9412da0a" alt="Workdays"
This flow can be updated based on the columns created in the SharePoint List. I choose to have Total Days Off, WorkDays, Holidays, and Net Days Off. It may make more sense to show to Total Days Off, Weekends and Holidays or other ways to mix and match. The point is you can update it based on your requirements.