Last Update: Sep 04, 2024 | Published: Dec 05, 2014
In an earlier article I guided you through the process of creating a PowerShell based tool to keep track of battery power. Even if you didn’t need the end result, the process I followed and techniques can be helpful in your own projects. In this article I want to continue exploring what I can do with my previous work. I’m always looking to see how far I can push my PowerShell projects. Sometimes PowerShell, or Windows, pushes back but even that can be educational.
With my battery check tool, one limiting factor is that I have to remember to periodically run the function in my PowerShell session. But then there’s that whole typing-thing. Oh, the agony. To alleviate my pain and suffering, I first need something that will give me a more visual clue about battery life.
I decided I wanted an old-fashioned popup box. Being an old-school VBScript guy I always liked the Popup() method from the Wscript.Shell object. This is even easier to use in PowerShell.
$shell = new-object -ComObject wscript.shell
$shell.Popup("Hey! What are you looking at?")
The Popup() method has additional parameters to control the title, the button selection and icons. What I especially like about this method is that unlike the VBScript MessageBox, I can set a timeout value so that if I don’t click on a button, the popup will still automatically dismiss after a specified number of seconds.
$shell.Popup("Hey! What are you looking at?",10,"Popup",0+48)
The second parameter, 10, is the number of seconds to display the popup. If you set it to -1 the popup will remain until the user clicks a button. The last parameter is a combination of buttons and icons. You can learn more about these values by reading the official MSDN documentation for the popup method.
For my needs, I only need to present a popup if the battery level reaches a certain level and is not connected to power. Here’s the brief monitoring script I came up with that I call Watch-Battery.ps1.
#requires -version 3.0 <# Watch-Battery.ps1 Test battery and display popup when it gets too low Limit is the percent remaining battery power. Timeout is the number of seconds to display the popup message. This script is meant to be run locally. #> [cmdletbinding()] Param( [int]$Limit=50, [int]$Timeout = 10 ) Write-Verbose "$(Get-Date) Testing battery" #get battery data $battery = Get-CimInstance -ClassName Win32_Battery Write-Verbose ($battery | out-string) $charge = $battery.EstimatedChargeRemaining #run if charge is <= the limit and battery is not charging if ($charge -le $Limit -AND $battery.batteryStatus -ne 2) { [string]$Runtime = New-TimeSpan -minutes $battery.EstimatedRunTime #define a message here string which must be left justified $msg = @" The battery is at $charge% with a remaining runtime of $runtime. You might consider plugging in some power. "@ Write-Verbose $msg #display the message using a Wscript Popup $wscript = New-Object -ComObject Wscript.shell $wscript.Popup($msg,$Timeout,"Battery Alert",48) | out-Null } #close if Write-Verbose "$(Get-Date) Finished testing battery." #end of script
The script checks the Win32_Battery class and compares the EstimatedChargeRemaining property to the parameterized Limit. If the value is less than or equal than the limit, then the script creates and displays the popup. The TimeOut parameter controls how long the popup is displayed in seconds.
I use a here string to define the message, inserting values from variables. When using a here string in a script file, it needs to be left justified. Technically not all of it does but I find it just as easy to left justify the entire thing.
Here’s an example of the script in action. At the prompt I can type:
C:scriptswatch-battery.ps1 -Limit 75
And get this:
Of course I still have to manually type the command. I first tried creating a PowerShell scheduled job.
$options = New-ScheduledJobOption -ContinueIfGoingOnBattery -StartIfOnBattery #check every 15 minutes $trigger = New-JobTrigger -RepeatIndefinitely -RepetitionInterval (New-Timespan -Minutes 15) -Once -at "12:00AM" $paramHash = @{ FilePath = "C:scriptswatch-battery.ps1" Name = "BatteryCheck" Trigger = $Trigger MaxResultCount = 8 ArgumentList = 75 ScheduledJobOption = $options RunNow = $True } Register-ScheduledJob @paramHash
While the job technically ran, I never saw the popup on my screen. PowerShell scheduled jobs run in separate PowerShell instances that are not interactive when launched by the Task Scheduler.
Although I suppose instead of displaying the popup I could have the script email me or send me a text message on my phone. But the interactive popup appears to be a non-starter with scheduled jobs. I did try using a Scheduled Task which worked slightly better.
#requires Windows 8 and later $paramHash = @{ Once = $True at = "12:00AM" RepetitionInterval = (New-Timespan -Minutes 15) RepetitionDuration = (New-Timespan -Hours 24) } $Trigger = New-ScheduledTaskTrigger @paramHash $paramHash = @{ AllowStartIfOnBatteries = $True DontStopIfGoingOnBatteries = $True } $settings = New-ScheduledTaskSettingsSet @paramHash #use a threshhold of 30 $paramHash = @{ Execute = "PowerShell.exe" Argument = "-nologo -noprofile -windowstyle Hidden -file c:scriptswatch-battery.ps1 30" } $action = New-ScheduledTaskAction @paramHash $paramHash = @{ TaskName = "BatteryCheck" Trigger = $trigger Action = $action Settings = $settings } Register-ScheduledTask @paramHash
I got the popup, but I end up with a flashing PowerShell window. So what to do? I’ll cover that in a separate article.
In the meantime, look at your PowerShell tools and see where else you can take them. It can make for an exciting journey.