首发,转载自少数派:https://sspai.com/post/66129

正如快捷指令,Tasker 之于手机端,Windows 上的任务计划程序也能够让不同软件协同工作,方便地完成各种具有固定流程的任务,但这方面的资料比之数不胜数的「快捷指令库」和「Tasker 教程」说是九牛一毛也不为过。

本文旨在为这「一毛」添砖加瓦,用三个简单例子阐释如何使用任务计划程序创建在固定时间运行的任务、周期运行的任务以及随固定事件触发的任务。任务计划程序本身并非实现自动化任务的唯一的要点,在脚本中使用合适的命令行工具也是,因此,文中每一节的内容将包括脚本编写和任务计划程序配置两个方面。

想要任务计划程序正确工作须确保 Task Scheduler 服务启动类型为「自动」。示例脚本运行环境为 Windows 10 和 PowerShell 7.1.3,不保证适用于低版本 Windows。

定时任务:Windows 深色/浅色模式切换

强光下将系统和程序切换为浅色模式让显示更清晰,暗光下将系统和程序切换成深色能减少对眼睛的刺激。实现以上效果的思路是在一天的早晨和晚上的固定时间执行一系列固定任务。两种方法,让 Windows 10 也能自动切换深/浅色主题 一文虽然已经给出方法但并不完美。

切换系统和程序的深/浅模式需要完成三个步骤:切换系统显示模式,更换壁纸,切换不跟随系统样式的应用程序显示模式。

切换系统显示模式可以用命令行修改注册表。修改位于HKCU\SOFTWARE\Microsoft\Windows\CurrentVersion\Themes\Personalize 的 AppsUseLightTheme 和 SystemUsesLightTheme 值,为 0 时是深色模式,为 1 时为浅色模式。

更改 Windows 壁纸则有打包好的 Wallpaper-CLI 程序,安装 Node.js 后输入npm install --global wallpaper-cli安装此应用。之后,使用 wallpaper [imagepath]就能更换壁纸。

像是 Windows Terminal、Rime 等应用程序的配色由自己的配置文件控制,我们需要找到并修改它们的配置文件。以 Rime 输入法为例,修改 weasel.custom.yaml 文件的style/color_scheme值并重新部署即可切换皮肤。我们既可以使用-replace的方式,也可以准备两份配置文件用Copy-Item覆盖掉原文件。

综上所述,创建名为Toggle-Night.ps1的文件以切换深色模式,写入如下命令:

reg.exe add HKCU\SOFTWARE\Microsoft\Windows\CurrentVersion\Themes\Personalize /v AppsUseLightTheme /t REG_DWORD /d 0 /f
reg.exe add HKCU\SOFTWARE\Microsoft\Windows\CurrentVersion\Themes\Personalize /v SystemUsesLightTheme /t REG_DWORD /d 0 /f

wallpaper "D:\blank.jpg" ## 更换壁纸为 black.jpg

## 下面是 Rime 输入法的修改
(Get-Content "C:\...\Rime\weasel.custom.yaml") -replace("ink", "dark_temple") | Out-File "C:\...Rime\weasel.custom.yaml" ## 修改主题皮肤为 dark_temple
cd "C:\...\Rime\weasel-0.14.3\";./WeaselDeployer.exe /deploy ## 重新部署

脚本编写完成后,进入任务计划程序,创建名为「深色模式」的任务,浅色模式同理:

  • 新建触发器。选择「按预定计划」,在开始一栏中写入我们希望系统切换为深色模式的时间。勾选「每天」,设置每隔「1」天发生一次,注意不要勾选「重复任务间隔」。
  • 新建操作。选择「启动程序」,在「程序和脚本」栏中写入「pwsh.exe」,「添加参数」中写入「-nop -w hidden -file "D:...\Toggle-Night.ps1"」。-w hidden会让脚本执行时终端只闪烁出现一次立刻消失。如果使用默认的 PowerShell 5 的话,需将「pwsh.exe」改为「powershell.exe」。
  • 不要勾选「条件」栏中的「只有计算机使用交流电源时才启动此任务」。在「设置」栏中,勾选「允许按需运行任务」和「如果过了计划时间,立即启动任务」。

效果

我们可以考虑将一些需要定时运行的任务都写入Toggle-Night.ps1中,不必为他们单独创建任务。例如我就将 Rime 输入法的词库同步、Chocolatey 升级都添加进了脚本。

周期任务:备份频繁变动的文件

Word 等现代编辑器都有每隔一定时间备份正在编辑的文件的功能,我们希望能够使用脚本实现类似的效果,备份诸如游戏本地存档、本地笔记等频繁变动的文件,方便回溯历史。不同于间隔时间很长的定时任务,这类脚本重复的间隔需要是几小时乃至几分钟。

现于 D 盘有一 Obsidian 的笔记库 Note 文件夹,需要每 1 小时为它创建一次备份。文件夹中是数百个 KB 大小的 Markdown 文件,通过复制 Copy-Item 到其他文件夹的方式备份会产生大量小文件,不方便管理。因此,我使用了 7zip 的命令行工具7z。使用7z将所有文件以压缩包的形式备份到 OneDrive 文件夹,并排除 .obsidian 等基本不变的文件夹,其命令为7z a "$env:OneDrive/Data/Note/Obsidian.zip" "D:/Note/*" -xr!'.obsidian'

当然,我们不能让备份文件这样一直累加上去,需要配以清理脚本——到备份文件到达一定数目时,删除最老的备份。最终的脚本 Backup-Note.ps1 如下:

## 以时间戳的方式命名备份的压缩包
$date = Get-Date -Format 'yyyy-MM-dd-mm'
7z a "$env:OneDrive/Data/Note/Obsidian-$date.zip" "D:/Note/*" -xr!'.obsidian'
## 当备份文件超过 5 个时,删除最老的备份
$Number = [System.IO.Directory]::GetFiles("$env:OneDrive\Data\Note").Count
if ($Number -ge 5){
 cd $env:OneDrive\Data\Note
 Get-Childitem | Sort-Object -Property LastWriteTime -Top 1 | Remove-Item ##注意,Powershell 5 不支持 -top
}

然而不同于第一节中的定时任务的是,如果还使用pwsh.exe -w hidden的方式来隐藏终端的话,我们将会每隔 2 小时看到终端闪烁一下,体验可想可知。为了彻底隐藏脚本执行时的终端,一个思路是使用 JS 脚本来执行 PowerShell 脚本 —— 创建文件 Backup-Note.js,写入

var wshShell = new ActiveXObject("WScript.Shell");
wshShell.Run('pwsh.exe -nop -file "D:/.../Backup-Note.ps1"', 0, false);

双击该 JS 脚本绑定 wscript 的打开方式,可以看到执行时不会有任何窗口。

接下来在任务计划程序中创建名为「备份 Note」的任务:

  • 新建触发器,选择「按预定计划」,将开始栏的时间调整为过去的任意时间,并勾选「重复任务间隔」,数值设置为「1 小时」,持续时间为「无限期」;
  • 新建操作,选择「启动程序」,在程序和脚本栏中写入 Backup-Note.js 的路径;
  • 其他设置参考第一节。

事件任务:网络状态更改时,启/停应用

任务计划程序提供了诸如「登录时」、「空闲状态」等常见的事件,可以用于配置开机自启软件等任务,这些任务配置起来相对简单。下面展示复杂一点的,需要配合 Windows 事件日志设置的任务 —— 网络状态更改时,启动/停止某程序。

诸如公司代理等软件,只有在连接网络时运行才有意义。在 PowerShell 中启/停应用比较简单,Start-ProcessStop-Process就能做到。比较麻烦的是找出网络状态更改时的事件日志。

通过搜索打开 Windows 事件查看器,定位到应用程序和服务日志 - Microsoft - Windows - NetworkProfile - Operational,窗口右侧会显示出所有已经记录的日志。为了准确地找到网络状态更改的日志,首先选择「清除日志」,然后手动断开网络再连接,这时候刷新事件查看器,会多出几行网络日志。

查看这些多出来的日志,可以发现连接网络时会产生一个 ID 为 10000 的日志,断开网络时会生成 ID 为 10001 的日志。

创建名为「启动代理软件」的任务,「停止软件」任务的配置同理。:

  • 新建触发器,选择「发生事件时」,在下拉菜单中找到 Microsoft-Windows-NetworkProfile/Operational,选择源为 NetworkProfile,填写事件 ID 为上面我们记录的 10000;

  • 新建操作,选择启动程序,填入代理软件的路径;
  • 在设置栏中,勾选「如果任务失败,按以下频率重新启动」。不要勾选「如果任务运行时间超过……」一项,因为在这个任务中,只要网络是连接状态,程序会一直保持「正在运行」状态。其他设置随意。

尾巴

笔者通过以上三个简单例子,展示了不同的触发式任务在任务计划程序中的配置方式,同时穿插讲述了隐藏脚本运行时的终端窗口的两种思路以及查找固定事件日志的方法。但自动化程序能做的远远不只于此,脚本的许多细节也可以优化。

读者可以根据自己的想象力和需求配置出更棒的自动化任务。