@au5ton Scriptable Widgets

A boilerplate for creating remote-updatable Scriptable widgets. Includes setup, components, utils and examples to develop in the comfort of TypeScript.

Widgets available

1. Pick a widget

Select one of the widgets below.

Custom Countdown

Add a countdown with many customizations

Minecraft Server

View MOTD, players online, and other info.

Plex Streams

View the number of active streams on your Plex Media Server.

Uniswap Position

View information about your Uniswap V3 positions.

More examples coming soon!
2. Provide its default input

This is not required and can also be filled into the Widget Setting after adding the widget

3. Copy the snippet
// Variables used by Scriptable.
// These must be at the very top of the file. Do not edit.
// icon-color: purple; icon-glyph: hashtag;

const RequestWithTimeout = (url, timeoutSeconds = 5) => {
    const request = new Request(url);
    request.timeoutInterval = timeoutSeconds;
    return request;
};

const ROOT_MODULE_PATH = "widget-loader";
const widgetModuleDownloadConfig = {
    moduleName: "CustomCountdown.WidgetModule",
    rootUrl: "https://scriptable.austinj.net/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();
4. Paste the snippet
Open the Scriptable App
And paste the snippet in a new Script.
Now it's ready to be added as a widget. Just go in wiggle mode and add it to your homescreen!

Try the boilerplate

Visit the Github repo or the official Scriptable Documentation. Follow any updates on Twitter @jasperhartong.