AppleScript初体验,让你的mac实现自动化UI操作

embedded/2024/12/22 9:02:43/

AppleScript 简介

AppleScript是苹果公司开发的一种脚本语言,用于操作MacOS及其应用程序,在实现MacOS自动化工作方面非常给力。
我们可以使用AppleScript用来完成一些重复琐碎的工作,AppleScript具有简单自然的语法,另外系统也提供了语法查询字典,可以很方便的查询语法

官方说明文档:在 Mac 上使用 AppleScript 和“终端”自动执行任务 - 官方 Apple 支持 (中国)

概念图:

AppleScript 能做什么

1.能与用户交互,如响应用户输入、通知中心发送通知等。

2.能控制其他 app ,如果其他 app 提供了相关 AS 接口可以直接控制,如果没有提供相关接口则需要通过 AS 语法来获取界面上的按钮等操作元素进行操作。

3.能执行自动化任务如发送邮件、执行定时脚本、打开音乐播放器、锁屏、响应自定义的 URL Scheme 后执行脚本等。

基本语法

每行的前面加上 ---(三个短横线)或者 # 表示注释本行。

用 (*xxx*) (括号+星号)包裹的 xxx 表述块注释掉 xxx 的内容。

他的基本跟口头预类似,不过如果你对英文语法不自信,比如不清楚 besides against over 等在具体语境中的意思,建议还是老老实实使用基本语法。

AppleScript 的形式

AppleScript 常用的有两种形式,Script(脚本)、Application(应用),这里只介绍前两种,Script bundle 和 Text 我没用这里不展开。

注:这里需要说明一点的是,即使是相同的代码,保存成脚本运行和保存成应用运行,界面是不一样的。

脚本

脚本顾名思义就是像 js 一样的可执行文件,不同的是 as 是在 Mac 桌面环境执行,而 js 是在 Chrome 中执行的。双击即可运行。下图是脚本 process 显示进度的界面,通过点击脚本右上角的「运行」按钮运行:

应用

保存成 Application 的 AS 的后缀名跟普通的应用如 Safari、Chrome、微信等应用一样,会以 .app 结尾,也可以查看包内容。有多种运行方式如 URL Scheme 调用、双击、拖拽内容到 app 图标上,你需要写相关的事件函数来响应相关用户操作。下图是 应用process 显示进度的界面,通过双击 「xxx.app」运行:

其他

除了上述两者,其他的不常用,不过如果你需要将 script 保存成服务以在状态栏或者任意界面的右键的服务中能够调用该脚本,这需要上述两者之外的形式:

你可以选择将脚本保存成系统 Service,此时上述 process 代码运行的时候,进度条会显示在顶部状态栏:

常用操作代码

以下操作基于「应用」类型的 AS,下列的语法在终端中使用 osascript -e '语法内容' 可以快速验证。需要注意的是,类似于多行的表达如 process 进度条的显示,osascript 是没办法做到的,虽然你可以在很多多行命令如 tell 中使用多个 -e 参数来串联如 osascript -e ‘tell application “Finder”’ -e ‘end tell’ ,但是此语法对 process 进度条无效。

用户交互

在通知中心显示信息

display notification "通知内容" with title "通知 title" subtitle "通知副标题"

弹窗

display dialog "这是个通知"

获取用户输入

display dialog "What's your name?" default answer "" with icon note buttons {"取消", "确认"} default button "确认"

注:AS 没有办法生成类似表单一样的组件,只能生成上面这种对话框,类似网页中的 prompt

播放给定文本

say "What is your name?" using "Alex" speaking rate 140 pitch 42 modulation 60

其他

让用户选择文件夹、文件、颜色、从列表中选择一项等,暂时省略。

执行命令行

do shell script "echo $PATH"

注意,命令行中的 PATH 环境变量为 /usr/bin:/bin:/usr/sbin:/sbin ,因此无法在其中执行诸如 node 、nvm 等后来安装的命令,这些命令执行的时候需要手动指定命令所在位置。

执行 node 脚本: 

set node to "/Users/x/.nvm/versions/node/v14.19.1/bin/node"set appPath to "/Applications/Xhelper.app/Contents/Resources/Scripts/"do shell script node & " " & appPath & "index.js"

获取上一个语句的输出结果

直接在上一句语句后立即使用 result 表示结果即可:

do shell script "echo $PATH"display dialog result with title "通知"

显示调试信息

直接使用 log 语句即可,然后在 Script Editor 下方的 Replies 中获取输出信息,可以在这里看到 shell script 命令执行的过程中,脚本向控制台输出的内容,如执行 js 脚本的时候的 console.log 等。

log do shell script "echo $PATH"

显示进度

显示进度用到 progress ,语句相对来说比较复杂,因为需要显示不同的状态进度,需要多行:

set progress total steps to 3set progress completed steps to 0set progress description to "处理中..."delay 1set progress completed steps to 1delay 1set progress completed steps to 2delay 1set progress completed steps to 3

相关的截图可以在上面看到。

延迟一段时间后执行

delay n

n 表示秒数,支持小数。一般用来模拟用户操作的时候的延迟操作,否则可能因为脚本操作过快页面还未来得及显示等。一些奇怪的问题也可以通过延迟执行来解决,此语句可以类比于 js 中的 setTimeout 大法。

调用其他应用

注意,此处会有系统警告,需要你确认才能执行:

确认后可以在设置-安全与隐私-隐私-自动化中确认:

可以调用其他应用来执行相应的操作,下面是唤起 Terminal 应用后,激活然后执行命令:

tell application "Terminal"if not (exists window 1) then reopenactivatedo script "echo $PATH" in window 1end tell

此语句做了一个判断,如果第一个打开的窗口存在,则激活之,然后在其内执行命令(do script );如果不存在的话则会新建一个窗口然后在其内执行命令。因为每个 do script 都会打开一个新的 Terminal 这个判断可以保证复用窗口。不过需要注意的是,如果 window 1 内已经有一个已经存在的进程(如 server 未中断),那么这个命令是不会执行的,还需要额外的判断,此处不展开。

响应 URL Scheme

有些时候你不想先找到 AS 脚本,然后双击运行;或者有时候你需要从另一个应用中调起你写的 AS 应用,这个时候就用到了 URL Scheme。

首先你需要修改 Info.plist 文件,可以在 Script Editor 的右侧,右键 「显示在 Finder」 中:

然后向上一级,可以看到该文件:

内容是一个类似 XML 的东西编辑之,在其内加入如下字段:

<array><dict><key>CFBundleURLName</key><string>Open File</string><key>CFBundleURLSchemes</key><array><string>xhelper</string></array></dict></array>

然后确认一下,用 XCode 打开该文件可以看到类似的条目:

表示添加成功。

这个操作的意思是,将该 AS 应用响应以 xhelper (可以任意修改成你想要的开头,不要与已有的重复即可)开头的 URL。

设置完成后,需要在 Script Editor 中保存一下,或者 cmd + L 一下或者运行一次(或者都做一遍),重新编译,然后才能生效。

然后,为了让 AS 响应该 URL,需要设置一个事件监听:

on open location this_URLdisplay dialog this_URLend open location

上述代码中,this_URL 即为以你 xhelper 打开的链接。

此时在浏览器中(截图是 Chrome)测试一个链接:

然后 AS 应用就会响应然后弹窗显示 xhelper://你好! ,注意这里是不用 encode 编码的。

我的使用

在这里放出我自己的 Craft build 代码,用的是上述的 URL Scheme,其中的 node 执行的 js 都是

​从之前的工作流改造的,没有什么新逻辑。另外 AS 的语法比较口语化,直接看就能看懂每句话是什么意思,这里就不一一解释了。

on open location this_URLtryset startOffset to offset of "://" in this_URLset the content to text from (startOffset + 3) to -1 of this_URLset node to "/Users/x/.nvm/versions/node/v14.19.1/bin/node"set appPath to "/Applications/Xhelper.app/Contents/Resources/Scripts/"--- 如果有两个参数表示需要发布到 wechat,如果只有一个参数则参数即为 base64 编码的内容set AppleScript's text item delimiters to "&"set arguments to every text item of the contentset AppleScript's text item delimiters to ""set listLength to the length of argumentsif listLength = 2 thenset realContent to item 2 of argumentsdo shell script "echo " & realContent & " > " & appPath & "content.base64.txt"tell application "Terminal"if not (exists window 1) then reopenactivateset alive to do script node & " " & appPath & "index.js" in window 1--- 等待上一个脚本执行完毕后再执行下一个脚本repeatdelay 0.1if not busy of alive then exit repeatend repeatbeepdo script node & " " & appPath & "wechat.js" in window 1end tellelsedo shell script "echo " & content & " > " & appPath & "content.base64.txt"tell application "Terminal"if not (exists window 1) then reopenactivatedo script node & " " & appPath & "index.js" in window 1end tellend ifon error error_messagedisplay dialog error_message buttons {"Cancel"} default button 1

效果:


http://www.ppmy.cn/embedded/10882.html

相关文章

【C语言】每日一题,快速提升(10)!

&#x1f525;博客主页&#x1f525;&#xff1a;【 坊钰_CSDN博客 】 欢迎各位点赞&#x1f44d;评论✍收藏⭐ 题目&#xff1a;圣诞树 输入&#xff1a; 1输出&#xff1a; * * * * * **说明&#xff1a; 输入&#xff1a; 2输出&#xff1a; * * * * * * * …

isListEqual方法比较

这个方法有改进空间吗&#xff1f; private static boolean isListEqual(List<String> l0, List<String> l1) {if (l0 null && l1 null)return true;if (l0 l1)return true;if (l0 null || l1 null)return false;if (l0.size() ! l1.size())return f…

Rust Tracing 入门

Tracing 是一个强大的工具&#xff0c;开发人员可以使用它来了解代码的行为、识别性能瓶颈和调试问题。 Rust 是一种以其性能和安全保证而闻名的语言&#xff0c;在它的世界中&#xff0c;跟踪在确保应用程序平稳高效运行方面发挥着至关重要的作用。 在本文中探讨Tracing 的概…

【行为型模型】迭代器模式

一、迭代器模式概述 迭代器模式定义&#xff1a;提供一种方法顺序访问一个聚合对象中的各个元素&#xff0c;而又不暴露其内部的表示。把游走的任务放在送代器上&#xff0c;而不是聚合上。这样简化了聚含的接口和实现,也让责任各得其所。(对象行为型) 迭代器模式的优缺点&…

算法题解记录18+++搜索二维矩阵Ⅱ

本题可以说是运用二分查找的典例&#xff0c;即使是面对矩阵&#xff0c;只要是它保持“排序好”这样的结构&#xff0c;就一定能采用二分查找法。【你知道的&#xff0c;对于排序好的数组&#xff0c;二分查找几乎是最优秀的算法】 当然&#xff0c;答案提供的是“Z字形查找法…

mysql索引详解

一、索引类型 简介 MySQL支持多种类型的索引&#xff0c;每种索引类型都适用于特定的场景和用途。下面是MySQL中常见的索引类型&#xff1a; 注意事项 引擎支持&#xff1a;在创建索引时&#xff0c;需要注意不同的索引类型可能需要特定的存储引擎支持。例如&#xff0c;全文…

剑指Offer题目笔记31(图的搜索)

面试题105&#xff1a; 解决方案&#xff1a; ​ 逐一扫描矩阵的每一个格子&#xff0c;如果遇到一个值为1的格子并且它不在已知的岛屿上&#xff0c;那么就到达了一个新的岛屿&#xff0c;于是搜索这个岛屿并且计算它的面积&#xff0c;在比较所有岛屿的面积之后就可以知道最…

Dubbo如何支持透明化的远程方法调用及其调用过程中的优势

作为资深的架构师&#xff0c;在微服务架构的设计与实施中&#xff0c;RPC&#xff08;远程过程调用&#xff09;技术是不可或缺的一环。在众多RPC框架中&#xff0c;Dubbo以其出色的性能和易用性赢得了众多开发者的青睐。Dubbo能够实现透明化的远程方法调用&#xff0c;为微服…