Hi, I’m trying some basic scripting of OmniFocus on MacOS with javascript. I’m using an actual script, not the console. It appears that the context is hugely different between the two.
However, I’m not clear why my simple basic scripting does not work. Can someone kindly tell me why this script throws an error:
#!/usr/bin/osascript -l JavaScript
var OmniFocus = new Application("/Applications/OmniFocus.app");
OmniFocus.activate();
var inbox = OmniFocus.inbox;
console.log("inbox length: "+ inbox.length)
console.log("Flattened projects length: "+ OmniFocus.flattenedProjects.length)
Is there a javascript tutorial for Omnifocus out there? I just don’t understand why my shell script cannot do the very basic stuff that I can do easily from the Console.
Thanks for any help
Two completely different programming interfaces, and two quite different JavaScript interpreter instances. One system-wide (osascript), one embedded in the Omni app itself.
The shell can only access the JavaScript for Automation interface, the objects of which are shared by the AppleScript interface (two different language options for osascript).
The console in Omni apps gives access to a completely different instance of a JavaScript interpreter (private to the particular app, unlike the system-wide osascript -l JavaScript instance) and a completely different programming interface, with different objects, methods and properties.
Current development focuses entirely on the JSContext to which the console (and Omni Automation plugins) give access.
The old osascript interface is quite different, and in legacy / sunset mode.
(Because it uses a JSContext external to the application, it also has more interface traffic overheads, and executes more slowly)
FWIW if you wanted to do that kind of thing in Script Editor (JS makes use of $ and various other things which don’t play well in the shell, even with a unix heredoc), you could trying writing something like the snippet below.
You will notice that it involves a different object structure (API) from the currently maintained in-app OmniJS JSContext that you can explore with the Console and omnijs plugins.
In Script Editor, with the language selector at top-left set to JavaScript:
The omni-automation site is certainly the fullest resource on the API, but it’s not very pedagogic, or even very self-confident or reliable, on JavaScript.
Probably worth looking at the first 5 chapters of “Eloquent JavaScript” for a more solid and graduated introduction to the basics, including things like:
why not to use var
why to use === and not ==
preferring [] and {} to new Array() and new Object()
Yes but it’s all fluff and no business. Sample code is few and far between for how to do this. I need examples personally, not treatises and class diagrams. I am not good enough at this to infer from the class heirarchy the exact syntax to do what I want in a fairly new programming language. Looking for sample code really.
OK, well in that case, omni-automation is no help (it deals with the omniJS interface – the object model available through the Console and plugins) what you need for shell scripts is the listing of the osascript objects and methods which you can get from Script Editor.
In Script Editor set the language selector at top left to JavaScript
In the Script Editor menu system: File > Open Dictionary...
Thanks! You have helped me tremendously with just your short snippet. If I may, how did you come to know how to do those simple things? Did you attend a secret meeting? Intuition from years of experience? Do you work for Omni? :-) Wondering how I might more easily improve. Thanks
I have looked in the Dictionary but it’s difficult for me to map that onto javascript as it’s all in Applescript. Mapping that to javascript has been a head-scratcher so far for me. Any advice there?
On the contrary, the site has an extensive collection of sample code for all kinds of uses of the API, most illustrated and commented around either a ready-to-install plug-in or start-to-finish code. Look in the hamburger menu and drill down into every section — the site navigation is a bit unusual but everything’s there. This is then adaptable to your workflow or preferred JS coding style.
I see. I assumed Omni Automation was being referred to, because the ‘treatises and class diagrams’ (which I found quite useful!) are in that section, not in the minimal osascript coverage.
Just an idea, which I haven’t tried myself: instead of using the legacy osascript API, would the following work?
shell script calls a ‘shortcut’ (in macOS Monterey)
the shortcut contains an OmniFocus ‘Omni Automation Script’ action (which runs a script)
that action contains the desired script.
Advantage is that the new API is being used. The script in the shortcut action doesn’t even have to be hard-coded, it can be retrieved from somewhere else beforehand in the shortcut. I see also that the shortcuts run shell command can pass input to the shortcut with the –input-path flag, so could the entire script to execute be passed?
I think we would have to understand more about the context in which the OP needs to use a shell script.
It’s not technically impossible to launch omniJS code from the shell – you can pass a code string to the osascript interface’s .evaluateJavascript method:
but adding a third layer of string within string (omniJS string within osascript string within shell script string) risks wasting time fiddling with the layered escaping of special characters.
The simplest example of such a shell script – the main function is evaluated by the shell in the osascript JSContext, which converts the omniJS function to a string and evaluates that code string in the omniJS JSContext, getting back a string result which, in turn, it passes back to the shell.
(Not that all dollars and backticks have to be escaped to stop the shell from applying its default interpretations to those characters)