A boilerplate for creating remote-updatable Scriptable widgets. Includes setup, components, utils and examples to develop in the comfort of TypeScript.
These widget examples are included in the boilerplate.
Just a simple sticky widget
Page views widget
Coronavirus stats in a country
Demoing included components
This is not required and can also be filled into the Widget Setting after adding the widget
// Variables used by Scriptable.
// These must be at the very top of the file. Do not edit.
// icon-color: yellow; icon-glyph: sticky-note;
const RequestWithTimeout = (url, timeoutSeconds = 5) => {
const request = new Request(url);
request.timeoutInterval = timeoutSeconds;
return request;
};
const ROOT_MODULE_PATH = "widget-loader";
const widgetModuleDownloadConfig = {
moduleName: "stickyWidgetModule",
rootUrl: "https://scriptable-ts-boilerplate.vercel.app/compiled-widgets/widget-modules/",
defaultWidgetParameter: "",
downloadQueryString: "__downloadQueryString__",
};
async function getOrCreateWidgetModule({ moduleName, rootUrl, downloadQueryString }, forceDownload = false) {
const fm = FileManager.local();
const rootModuleDir = fm.joinPath(fm.libraryDirectory(), ROOT_MODULE_PATH);
enforceDir(fm, rootModuleDir);
const widgetModuleDir = fm.joinPath(rootModuleDir, moduleName);
enforceDir(fm, widgetModuleDir);
const widgetModuleFilename = `${moduleName}.js`;
const widgetModuleEtag = `${moduleName}.etag`;
const widgetModulePath = fm.joinPath(widgetModuleDir, widgetModuleFilename);
const widgetModuleEtagPath = fm.joinPath(widgetModuleDir, widgetModuleEtag);
const widgetModuleDownloadUrl = rootUrl + widgetModuleFilename + (downloadQueryString.startsWith("?") ? downloadQueryString : "");
try {
// Check if an etag was saved for this file
if (fm.fileExists(widgetModuleEtagPath) && !forceDownload) {
const lastEtag = fm.readString(widgetModuleEtagPath);
const headerReq = RequestWithTimeout(widgetModuleDownloadUrl);
headerReq.method = "HEAD";
await headerReq.load();
const etag = getResponseHeader(headerReq, "Etag");
if (lastEtag && etag && lastEtag === etag) {
console.log(`ETag is same, return cached file for ${widgetModuleDownloadUrl}`);
return widgetModulePath;
}
}
console.log("Downloading library file '" + widgetModuleDownloadUrl + "' to '" + widgetModulePath + "'");
const req = RequestWithTimeout(widgetModuleDownloadUrl);
const libraryFile = await req.load();
const etag = getResponseHeader(req, "Etag");
if (etag) {
fm.writeString(widgetModuleEtagPath, etag);
}
fm.write(widgetModulePath, libraryFile);
}
catch (error) {
console.error("Downloading module failed, return existing module");
console.error(error);
}
return widgetModulePath;
}
const getResponseHeader = (request, header) => {
if (!request.response) {
return undefined;
}
const key = Object.keys(request.response["headers"])
.find(key => key.toLowerCase() === header.toLowerCase());
return key ? request.response["headers"][key] : undefined;
};
const enforceDir = (fm, path) => {
if (fm.fileExists(path) && !fm.isDirectory(path)) {
fm.remove(path);
}
if (!fm.fileExists(path)) {
fm.createDirectory(path);
}
};
const DEBUG = false;
const FORCE_DOWNLOAD = false;
const widgetModulePath = await getOrCreateWidgetModule(widgetModuleDownloadConfig, FORCE_DOWNLOAD);
const widgetModule = importModule(widgetModulePath);
const widget = await widgetModule.createWidget({
widgetParameter: args.widgetParameter || widgetModuleDownloadConfig.defaultWidgetParameter,
debug: DEBUG
});
// preview the widget if in app
if (!config.runsInWidget) {
await widget.presentSmall();
}
Script.setWidget(widget);
Script.complete();
Intrigued by the possibilities offered by the Scriptable App to create custom iOS Widgets in Javascript, I wondered whether this would also be useful for prototyping product-services requiring real widget interactions. The other route, publishing a actual native iOS app to TestFlight, just felt way to convoluted.
Of course, there are also some drawbacks. Regular widgets for instance can be informed by their related app that they should update. For widgets created in Scriptable, this only happens periodically. But besides such minor points there's just a lot you can do with Scriptable!