Power Query: Update Your Calendar
21 October 2020
Welcome to our Power Query blog. This week, I look at calendar creation.
It’s been a while since I have looked at how to create a calendar, and Power Query has moved on since then. It’s time to bring my calendar creation approach up to date.
I will start by creating a list of dates, since that is what a calendar essentially is. I create a Blank Query from the ‘Other Sources’ option on the ‘New Query’ section of the ‘Get & Transform’ section of the Data tab.
data:image/s3,"s3://crabby-images/bf875/bf875fbca8b28871042c0d9fc47149cc0051df51" alt=""
I call my new query ‘Calendar’.
data:image/s3,"s3://crabby-images/0d0a8/0d0a8a148ca37d1520870656a52d9d5518dcd6d6" alt=""
I will use the M function List.Dates(). If I just input the function for the step, Power Query will show the parameters required.
data:image/s3,"s3://crabby-images/59389/593897bd837ba66d8eb46058a88b4b6d5d2c0e29" alt=""
I start by using the dates for this year, incrementing by one (1).
data:image/s3,"s3://crabby-images/bbc63/bbc63c9b52daf25b7803a7034ae53930be1df812" alt=""
When I choose to invoke, my list of dates is created:
data:image/s3,"s3://crabby-images/e41cd/e41cd2bcb170a8dec98cbc252ce0cda824785bf3" alt=""
The M code used is:
= Calendar(#date(2020, 1, 1), 366, #duration(1, 0, 0, 0))
Because I invoked a function, I now have two queries, ‘Calendar’ and ‘Invoked Function’. I only want to keep the ‘Calendar’ query, so I copy the M code generated in the source step of ‘Invoked Function’ into my ‘Calendar’ query. I don’t enter Calendar(#date… since that would create circular logic; I need to amend the M code to:
= List.Dates(#date(2020, 1, 1), 366, #duration(1, 0, 0, 0))
data:image/s3,"s3://crabby-images/414db/414dbba4380e66dd611c01237b08e32afe55fa33" alt=""
When I enter this, my ‘Calendar’ query will change from a function to a list.
data:image/s3,"s3://crabby-images/7459b/7459bb5badec0968c05e39686fd9a741ef5cdb64" alt=""
I delete the ‘Invoked Function’ query (which I just broke by changing ‘Calendar’). Then, I transform my list to a table:
data:image/s3,"s3://crabby-images/95181/95181b9333a6be8fd7763f18b466bed771b23a0a" alt=""
I accept the default options.
data:image/s3,"s3://crabby-images/0c061/0c0613aa892e80efb52bdf0510241d1b96a0ae53" alt=""
I have also changed the column name to Date and transformed the data type to Date.
I need to make some changes to my source step so that the calendar is more flexible. My source step is currently:
= List.Dates(#date(2020, 1, 1), 366, #duration(1, 0, 0, 0))
The start and end dates are fixed; I want these to be easily changed in my query instead. I can add a step to my code which will contain the start date.
data:image/s3,"s3://crabby-images/58e98/58e98a6a7108cd3544565272283f9bbbc3f40cbf" alt=""
When I enter this, I get the date I have specified.
data:image/s3,"s3://crabby-images/237c5/237c54d9c788c9628f36d25f5e6a7295d5ba416d" alt=""
I rename this step ‘StartDate’. I can now reference this step in my Source step.
data:image/s3,"s3://crabby-images/0b841/0b84197518b010f0c6fe9b75974970aa813e96c1" alt=""
The M code I am now using for my Source is:
= List.Dates(StartDate, 366, #duration(1, 0, 0, 0)
There is no end date parameter as such, just the number of dates generated. I can choose to always create the calendar ending today, in which case I need to find out now many days until today. I start by finding out today’s date in M.
data:image/s3,"s3://crabby-images/f3d9e/f3d9ef5c0f73c2dd64858214b5952a75b174f137" alt=""
The M code I have used is,
= DateTime.LocalNow()
which gives me the exact moment. I only need the date, so I can use DateTime.Date() to extract the data I want.
data:image/s3,"s3://crabby-images/6bfa0/6bfa0c15ba6ef134e5ad935deba91db081cdbbad" alt=""
The M code I have used is:
= DateTime.Date(DateTime.LocalNow())
I rename this step ‘Today’ and add a step to calculate the interval between today and my start date.
data:image/s3,"s3://crabby-images/e28b2/e28b213e763e8e422f119a4a50e99f8689c965af" alt=""
I have too much information again, so I need to use Duration.Days() to get the section I want.
data:image/s3,"s3://crabby-images/0229c/0229c26855d9e27a3d7a58dfa0b29809f537c83b" alt=""
The M code I have used is:
= Duration.Days(Today - StartDate)
I can rename my step Calendar_Length. I need to change the Source step to use Calendar_Length.
data:image/s3,"s3://crabby-images/254e6/254e6e3c74f8fd2543352eb1a35e325caef5117b" alt=""
The M code is now:
= List.Dates(StartDate, Calendar_Length, #duration(1, 0, 0, 0))
I have the basis for my calendar, and I can easily identify and change the start date. Next time, I’ll expand the calendar to make it more useful. I’ll also point out the deliberate mistake in the Calendar_Length.
Come back next time for more ways to use Power Query!