Last updated on February 13th, 2025 at 06:20 pm
There’s no shortage of posts regarding updating the background and lock screen images for your endpoints. The question that usually comes up is how to do this on Windows 10/11 Pro. This is because the Intune device restriction policy makes adding background/lock screen images easy, but it’s only compatible with Windows Enterprise/Edu SKUs. If that somehow brought you to this post, go to this link where we walk through how to accomplish this for Windows Pro. This post is slightly different and was inspired by an organization with a unique ask. They had a graphic designer create 52 custom background images related to their business that they wanted to use as the lock screen and background images. The unique part was they wanted these rotated weekly, so a new image was displayed each week.
Luckily, they had Windows Enterprise licensing. So, we can use the device restriction policy in Intune, which makes creating and updating the policy simple. However, assigning someone a manual task to update the policy every week is not a great use of time and is also subject to error or missing the update if they are out of the office. Therefore, we wanted a way to automate this. This post walks through the entire process of how you can automatically rotate your background and lock screen images at a fixed period.
The Intune Policy
First, we need to create our Intune Policy. Navigate to Device Configuration Policies in Intune. Create a new Profile, and select the Device Restrictions Template:
data:image/s3,"s3://crabby-images/6413f/6413f24d1c7f8886d1cc052e3485b7b2a82e9238" alt=""
If you already have the URL for your initial image, you can add it to the lock screen and background fields. If not, leave it blank for now and continue with the policy creation.
data:image/s3,"s3://crabby-images/2dc5f/2dc5ff8e9f93a10c890f9ace5a6836cca150d909" alt=""
Add your assignments and create the policy. After creating the policy, select it and grab the ID from the URL. Copy it and paste it somewhere for now. We will reference this in the automation script.
data:image/s3,"s3://crabby-images/f55d6/f55d6ada3a7737bd3fe48f7a425375dd2a3b2ad7" alt=""
The Azure Storage Account and Image Files
Next, we must create an Azure storage account to hold our background images. Navigate to https://portal.azure.com. Search for storage accounts and create a new one. For something this basic, use the standard SKU and Blob storage as the primary service. Finish creating the storage account—you shouldn’t need to edit anything in the other tabs unless you have specific requirements to do so.
data:image/s3,"s3://crabby-images/37756/377565837b23fd3eac57ad64816376da65b871a6" alt=""
After your SA is created, we need to enable anonymous read access so our endpoints can retrieve images from the container. Under the storage account settings > configuration, select Enabled for Allow Blob anonymous access:
data:image/s3,"s3://crabby-images/4eedb/4eedb3d8285b07a332ed994a6be577e33d497f3e" alt=""
Next, we need to create a container. Under data storage, select Containers and click the + sign. Provide a name, and for the anonymous access level, select anonymous read access for blobs only.
data:image/s3,"s3://crabby-images/f76bc/f76bc6b80c777de233a0d06b0323e2afa73e0137" alt=""
data:image/s3,"s3://crabby-images/04e14/04e14297ed7e660abeac10e28da55c83e56452ba" alt=""
Now we are ready to upload our files. This is important because before uploading them, we must ensure they are correctly named. You don’t need to follow my naming convention if you don’t want to, but you’ll need something to identify the files based on the date. So, I used the naming convention “background-YYYY-MM-DD. Where the date on the file was the date we wanted it implemented. In this case, every Sunday. Our runbook script will run each Sunday and look for a file that starts with “background-” and ends with the date of that Sunday. I also have a default.jpg image uploaded. If the script can’t find a match on the file for that week, it will use the default background image. Here’s how my files look:
data:image/s3,"s3://crabby-images/866bd/866bde2f9b1927786a52c788a4d4d1e4449d553e" alt=""
The Azure Automation Account
Now that we have created our Intune policy and images named and uploaded to our storage account, we need to make the engine that will work for us. For that, we will use an Azure Automation Runbook. From Azure, search for and select Automation Accounts:
data:image/s3,"s3://crabby-images/ce033/ce0333820474bf4e261b066018efcdb037d31b35" alt=""
Create a new Automation Account. Complete the required fields and make sure system-assigned managed identity is selected under the Advanced tab:
data:image/s3,"s3://crabby-images/ecc73/ecc738638124ce8a1628b197d6ebc37a1132e199" alt=""
data:image/s3,"s3://crabby-images/325a6/325a6833745d34e066ab87daafd632eed92ebdb4" alt=""
Next, we must grant our Automation Account identity rights to access our storage account. This is so it can list and read the contents of the container and compare those files to the date in our script. Navigate back to your storage account and select Access Control, then Add a new role assignment. We will grant the automation account identity Reader and Storage Account Key Operator Service roles:
- Add automation account to have reader role permissions to storage account
data:image/s3,"s3://crabby-images/d9eea/d9eeaab6972c21a5c9ee886bd2b06786f264d031" alt=""
data:image/s3,"s3://crabby-images/2e485/2e485907175161c289c9dbdbe64c32bad5a8ba52" alt=""
data:image/s3,"s3://crabby-images/1788b/1788b457ef1eed7fa956de9dd8b4e49193a005fe" alt=""
After adding the role assignments, navigate back to the Automation Account. Under Shared Resources, select Modules and then click Add a module:
data:image/s3,"s3://crabby-images/8ab49/8ab49f76669ba4175d8b745bcd8e69da1582f534" alt=""
Seelct browse from gallery:
data:image/s3,"s3://crabby-images/e364e/e364e010e046949811294fd28092f108fbb32edc" alt=""
Search for and select Microsoft.Graph.Authentication:
data:image/s3,"s3://crabby-images/a53ed/a53edcfc851e6acd942be10877486216ce6eb0dc" alt=""
Click the select button, and when asked which runtime version, select 7.2:
data:image/s3,"s3://crabby-images/f253d/f253d460a4ae0a952d050bb7fc2b56e366b52ac4" alt=""
data:image/s3,"s3://crabby-images/dd8d5/dd8d570a06357eae5bcf835be0d2eca145d5e3de" alt=""
That wraps up the creation of the Automation Account, assigning the correct permissions in Azure, and adding the required module for our automation account. Next, we need to assign our Automation Account identity permissions to Graph API, and upload our Runbook script.
Assigning Graph API Permissions and Adding our Runbook
This step is essential because when an automation account connects to the Graph API, we cannot add scopes to our script and grant consent during authentication. The permission assignments need to be added to the identity before the authentication. For this, we will use PowerShell. Before we look at the script, we need to retrieve our automation account’s ID. The easiest way to find this is under the automation account, Account Settings > Identity:
data:image/s3,"s3://crabby-images/e5e3d/e5e3df6f3e071edae2bd08a0cc0add8fc9f61529" alt=""
Copy your automation account’s ID and use it in the script below for the $serviceprincipalId variable. The rest of the script can be left alone and will do the rest for you. Essentially, this script looks for the permission IDs for each of the objects set in the permissions array, retrieves the permission ID, and assigns it to the automation account’s identity. This short script assumes you already have the Microsoft.Graph.Authentication module installed. The rest of it is requires the Microsoft.Graph.Applications module. Setting these permissions allows the automation account to make the necessary changes to the Intune policy when it runs at its scheduled interval.
Install-Module Microsoft.Graph.Applications -scope currentuser
Connect-MgGraph -Scopes "AppRoleAssignment.ReadWrite.All", "Application.ReadWrite.All"
$serviceprincipalId = f0d61e87-ecce-4068-bee2-cd4554dc1fd0
$permissions= @("Directory.Read.All", "DeviceManagementConfiguration.ReadWrite.All", "DeviceManagementConfiguration.Read.All")
ForEach ($permission in $permissions) {
$GraphResource = Get-MgServicePrincipal -Filter "AppId eq '00000003-0000-0000-c000-000000000000'"
$approle = $GraphResource.AppRoles | Where-Object {$_.value -eq $permission} | Select-Object Value,ID
New-MgServicePrincipalAppRoleAssignment -ServicePrincipalId $ServicePrincipalId -PrincipalId $ServicePrincipalId -AppRoleId $approle.Id -ResourceId $GraphResource.Id
}
To verify that the permissions were correctly set, you can navigate to enterprise applications in Entra, select Managed Identities as the application type:
data:image/s3,"s3://crabby-images/84aa7/84aa78032101171e9e8a96b8e1b29ef742e030bd" alt=""
Select your Automation Account identity and select security > permissions. You’ll see below that we have the Graph Permissions
data:image/s3,"s3://crabby-images/0269d/0269dfc656f733697008dcfdb91e285830ce7bf9" alt=""
Now we can add and test our runbook script. Download the runbook script from Github. Then, go back to your automation account and select Runbooks, and Import a Runbook:
data:image/s3,"s3://crabby-images/b74dc/b74dc6c0d245afa17397fee0a154dce156329335" alt=""
Select to browse for a file and select the runbook script we just downloaded. Provide a name and select runtime version 7.2. Click Import when finished.
data:image/s3,"s3://crabby-images/86196/8619683d6f00a9a4ff5256bc2059678fb016c9d2" alt=""
Now, we need to make a few edits to the runbook. We already briefly discussed this, but the runbook is relatively simple. It will connect to both Azure and MS Graph as the Automation Account identity. It gets the current date in the format of “yyyy-MM-dd” and then generates a variable for the image name using “background-$date”. It lists and reads the names of the images in the storage account container to see if it can find a match. Depending on the results, it uses a PATCH API call to update the Intune policy with the new image URL, or the default fallback image if there was no match. I should also mention that this script currently sets the background and lock screen to the same image. If you want separate images, you’ll need to edit the script and the image names you’re uploading. You’ll need the following items so we can edit the runbook script for your environment:
- Your Intune Policy ID
- Storage account name
- container name
- Resource Group of the storage account
- URL of the blob storage account container where your images were uploaded
Navigate to the runbook in Azure and click Edit at the top of the screen. For quick edits, we can select edit in Portal:
data:image/s3,"s3://crabby-images/5a34e/5a34eec42f5e95eab4e68526cbc5008db653bfa6" alt=""
Add your Intune policy ID as the $policyID variable at the top of the script:
data:image/s3,"s3://crabby-images/85eaf/85eaf42cd3e0a28a79ff8abd8257e3b23437ba52" alt=""
Then add your container name, storage account name, and the resource group of your storage account under the respective variables:
data:image/s3,"s3://crabby-images/b98e9/b98e9c376c76322189818797923189d86270f573" alt=""
Lastly, add the URL for your storage account container under the If and else statements. You’ll see the else statement uses the default.jpg image as the fallback method if there is no match for an image. Click the save button at the top when you’re finished.
data:image/s3,"s3://crabby-images/664ce/664ce5706ac335d11efeb38e6352b26cf2b6a348" alt=""
After you’ve made your edits, you can navigate to the test pane to test the runbook:
data:image/s3,"s3://crabby-images/8f6a2/8f6a260ab0498b0a5014ddc1fc85088f82aecaee" alt=""
Successful should show completed.
data:image/s3,"s3://crabby-images/10728/10728557e2439c5f66f9c77570ccee8ab859ddbe" alt=""
Your Intune policy should reflect your image based on the date (if one exists). Otherwise, it will default to the default image.
data:image/s3,"s3://crabby-images/30586/30586dd871f95ff4fdb6887dbeac53d4f4c8a88c" alt=""
Lastly, we need to publish the runbook and create a schedule. Back in the edit runbook page, there is a button to Publish the runbook. Click the button so the runbook is published with your saved changes:
data:image/s3,"s3://crabby-images/30603/30603b11214c79faff062d79ac56ec9620153af1" alt=""
For our schedule, this should run at the same interval you want your images rotated with how they’re named. For example, if you want the images rotated every Sunday so they’re new on the first workday of the week, you’ll want the automation runbook to run on Sunday and your image files to be named as “Background-2025-01-26” then “background-2025-02-02”, and so on to match the date of each Sunday (this is how our example images are named). From the Runbook, click Resources > Schedules, and add a new schedule:
data:image/s3,"s3://crabby-images/30a79/30a79530eb8a6d33975b0c92953de479bf2fcc1a" alt=""
Set your recurring schedule:
data:image/s3,"s3://crabby-images/914a8/914a85e87f9daacb964d54c607d06a8f831b8700" alt=""
As the jobs run, you can track if they were successful under the Jobs tab for the automation runbook:
data:image/s3,"s3://crabby-images/6bd77/6bd777e72b7461acc6f758c29d6a6dce99109fcf" alt=""
Now, you can sit back and let the automation rotate the images for you 👍