Skip to content

JavaScript

Derek Jamison edited this page Jun 23, 2024 · 78 revisions

JavaScript

Overview

The latest version of firmware now supports running JavaScript applications, using mjs. This version of JavaScript has some restrictions. Scripts are typically saved to the SD Card/apps/Scripts folder. You launch JS files by starting at the Main Menu on the Flipper Zero and selecting Apps/Scripts/name.js file.

This page is comparing the dev branch of the various firmware, since that is where the updates are happening. If you are on a release branch, some of the features listed here may not yet have been ported over to your firmware.

Just like JavaScript in the browser, Flipper JS has different levels of support depending on which firmware you are running on. 😭 The implemented modules vary by firmware, so check your firmware's applications/system/js_app/modules folder to see what features are supported.

Capabilities

Supported

It depends on your firmware, but this is the current list of capabilities that JavaScript on the Flipper supports: (as of 6/22/2024)

  • BadUSB keyboard support, including different keyboard layouts.
  • BLE Beacon support.
  • Read and write to files on the Flipper.
  • Read and write to serial port (TX/RX or C1/C0 pins).
  • Transmit a Sub-GHz file.
  • See if a Sub-GHz frequency has strong signal on it.
  • Expose a virtual USB drive.
  • GPIO for digital input/output.
  • GPIO for analog input.
  • Dialog to pick a file.
  • Dialog with Left/Right/Center choices.
  • Menu with items to pick.
  • On-screen keyboard for text or hex input.
  • Dynamic textbox (update text contents while app still running).
  • Get the pitch, roll and yaw of the Video Game Module (VGM)
  • Widget support for rending more complex screens.
  • Success/Error notifications.
  • Math functions.
  • Get the battery level of the Flipper.
  • Get the name of the Flipper.
  • Conditional logic, Loops, Number to String, String to Number.

Not supported

Here are some examples of features JavaScript on the Flipper cannot currently do. In some cases, it may be possible to extend the firmware with additional support or to write an FFI (Foreign Function Interface) code that provides access to the needed APIs. If you have ideas of features you would like supported, please reach out on my Discord server.

  • Play tones or vibrate
  • Advanced protocols (SPI, I2C, etc.)
  • Read/Capture Sub-GHz signal (JavaScript can send Sub-GHz)
  • Anything with RFID signals
  • Anything NFC signals
  • Anything with IR signals
  • Anything with iButton signals
  • Anything with U2F
  • Render advanced UI screens
  • d-pad input callbacks
  • Precision timer
  • Callbacks

Common Script Usage

The most common JavaScript pattern seems to be:

  • Use badusb to connect the keyboard to computer
  • Press Windows+R and type powershell
  • Runs the following badusb script (which starts executing after the entire script is entered):
    • tell user to close window when window closes
    • sleep 10 seconds
    • find "Flipper Mass Storage" drive
    • run commands (sometimes from USB drive, sometimes locally)
    • copy output to USB drive
    • delete MRU and history
    • close window
  • Attach USB drive
  • Wait for USB drive to be ejected
  • Stop USB drive

Globals

As of 6/21/2024 the supported modules are as follows:

Feature Description OFW (Official) Momentum Unleashed RogueMaster Xtreme
console Console object for logging
delay Waits num milliseconds
ffi_address Foreign Function Interface (FFI) address
parse_int String to number not implemented
print Print message on screen and logs
require Import module
to_string Number to string
to_hex_string Number to hex string
to_lower_case Convert string to lowercase not implemented
to_upper_case Convert string to uppercase not implemented
__filepath Path of script not implemented
__dirpath Path of script directory not implemented

globals-console

The console object exposes 4 levels of debugging that will get logged to the serial debugger. Each parameter passed will be converted to a string and then concatenated with a space being added between displayed values.

console.error("error message", 7.1, 'c', 'f', 12, false); // "[E][JS] error message 7.100000 c f 12 false"
console.warn("warning"); // Warning
console.log("info", 4.2); // Info
console.debug("debug", 1); // Debug only

globals-delay

The delay function delays for the specified number of milliseconds before continuing to the next statement.

delay(100); // wait 100 milliseconds

globals-ffi-address

The ffi_address function converts a method signature into an address? No examples use this method directly.

globals-parse-int

The parse_int function converts a string into number.

let str = "42";
let num = parse_int(str); // num will be a number with a value of 42

globals-print

The print function displays text on the screen and also in the logs.

The function allows passing multiple parameters of various types.

print("hello");
print(42);
let answer=true;
print("got",answer);

globals-require

The require function loads a module. If the module is unknown it will throw an error.

The parameter is the name of the module.

let badusb = require("badusb");

globals-to-string

The to_string function will convert a number into a string.

The parameter is a number.

let num = 42;
let str = to_string(num); // str is the string "42".

globals-to-hex-string

The to_hex_string function will convert a number into a hex string.

The parameter is a number.

let num = 1337;
let str = to_hex_string(num); // str is the string "539" (because 1337 decimal is 0x539).

globals-to-lower-case

The to_lower_case function converts a string to lowercase.

The parameter is the string to convert.

let str = "Hello World";
print(to_lower_case(str)); // prints: hello world

globals-to-upper-case

The to_upper_case function converts a string to uppercase.

The parameter is the string to convert.

let str = "Hello World";
print(to_upper_case(str)); // prints: HELLO WORLD

globals-filepath

The __filepath is the path to the JS file that is being run.

print("This JS file is",__filepath);

globals-dirpath

The __dirpath is the folder where the JS file is being run from.

let otherFile = __dirpath + "/demo.img"; // path to another file in the same folder as our script.
print(otherFile);

MJS Internals

There are some other concepts that are part of the MJS library, which all firmware seem to include without editing?

// There is a global called 'global'.
global.abc = 123;
print("global.abc =", global.abc); // prints "global.abc = 123"

// Unsigned and Signed arrays of various sizes:
// Uint8Array, Int8Array, Uint16Array, Int16Array, Uint32Array, Int32Array.
// NOTE: You will get TYPE_ERROR for out of bounds!
let b1 = Uint16Array(6); // 6 elements
b1[0] = 1;
b1[1] = 200;
b1[2] = 2000;
b1[3] = 42000;
b1[4] = 4000;
b1[5] = 1000;
let b2 = Int16Array(b1.buffer); // Create using the ArrayBuffer.
print(b1[1],b2[1]);
print(b1.buffer.byteLength); // 12 bytes
print(b1.length); // 6 elements
let b3 = b1.buffer.slice(2,4); // The buffer is Bytes! [start index, stop index = not inclusive)
let b4 = Uint16Array(b3);
print(b4[0]); // 200
let data = [1,13,45,66];
data[2] = data[2] - 3;
let b5 = Int8Array(data);
data[2] = 55;
data = data.splice(0,2); // elements 0 and 1 only.
print(data.length, b5[2]); // 2 42

print("12345".slice(2,4)); // "34"
print("531234543215415412".indexOf("54", 7)); // 11
let c1 = "123ABC".charCodeAt(3);
let c2 = chr(c1);
print (c1, c2); // 65 A

let x = Object.create({y:42});
x.z = 10;
print (x.y, x.z); // 42 10

let n = 42/0;
print (n);
print (isNaN(n)); // true
let n2 = NaN;
print (isNaN(n2)); // true
print (typeof n); // prints the text: 'number'
// NOTE: (typeof foo) "gives [foo] in not defined error".

function calc(a,b) {
 if (a===b) { // Always use === instead of ==
  return (a*b)+20;
 }
 if (a!==b) { // Always use !== instead of !=
  return (a*b)+10;
 }
}

print (calc(3,3)); // 29
print (calc(2,3)); // 16

// Example of calling native API calls...
// Not great, because we might not call release() if user aborts scripts.
let acquire = ffi("int furi_hal_speaker_acquire(int)");
let start = ffi("void furi_hal_speaker_start(float, float)");
let stop = ffi("void furi_hal_speaker_stop()");
let release = ffi("void furi_hal_speaker_release()");
let speaker = acquire(1000);
if (speaker) {
  start(1000,1.0);
  delay(100);
  stop();
  release();
}

die("Crashing"); // Exits with error message.

// What about: ffi_cb_free, mkstr, getMJS, s2o

Supported Modules

As of 6/21/2024 the supported modules are as follows:

Feature Description OFW (Official) Momentum Unleashed RogueMaster Xtreme
badusb Send keystrokes as hid missing: quit, altPrint, altPrintln, num0-num9 missing: layout_path parameter in setup
blebeacon Send BLE events not implemented
dialog Show dialog, select File missing: pickFile
flipper Device info
gpio Input/Output hardware pins not implemented missing readAnalog, startAnalog, stopAnalog missing readAnalog, startAnalog, stopAnalog
keyboard Text and Byte input not implemented missing feature: add_illegal_symbols missing feature: add_illegal_symbols missing feature: add_illegal_symbols
math Math functions
notification LED sequences
serial Serial port missing: end, readAny
storage Flipper SD Card files not implemented not implemented missing: append, copy, move, mkdir
subghz Sub-GHz Radio not implemented
submenu Submenu
textbox Textbox emptyText instead of clearText
usbdisk USB disk images not implemented
vgm Video Game Module sensor not implemented not implemented
widget UI drawing not implemented missing: addIcon missing: addIcon missing: addIcon

badusb

let badusb = require("badusb");

badusb is used to simulate an HID (Human Interface Device), like a USB keyboard, sending various keystrokes to the attached computer.

  • setup({ vid: number, pid: number, mfr_name: string, prod_name: string, layout: string }): undefined | error
  • quit(): undefined | error
  • isConnected(): bool | error
  • press(...keyAndModifiers: string | number): undefined | error
  • hold(...keyAndModifiers: string | number): undefined | error
  • release(...keyAndModifiers: string | number | undefined): undefined | error
  • print(text: string, delay: number | undefined): undefined | error
  • println(text: string, delay: number | undefined): undefined | error
  • altPrint(text: string, delay: number | undefined): undefined | error
  • altPrintln(text: string, delay: number | undefined): undefined | error

badusb-setup

The setup function takes an configuration object with the vendor id (vid), the product id (pid), the manufacturer name (mfr_name) and the product name (prod_name). This function will change the USB port from the typical Flipper port into a device with the specified vid/pid, acting as a USB keyboard. When you are done using the bad usb, you should call quit. You cannot call this method a second time without first calling quit. The layout parameter (layout) should be a fully qualified path to a .kl keyboard layout.

badusb.setup({ vid: 0xAAAA, pid: 0xBBBB, mfr_name: "Flipper", prod_name: "Zero", layout: "/ext/badusb/assets/layouts/en-US.kl" });

badusb-quit

The quit function will change the USB port back to the typical Flipper port. This will also happen when the script exits.

badusb.quit();

badusb-is-connected

The isConnected function will return true if the Flipper Zero is connected to a computer.

let connected = badusb.isConnected();

badusb-press

The press function will press and release a key combination. See badusb keys for the list of additional supported key names.

badusb.press("CTRL", "c");

badusb-hold

The hold function will hold down a key combination. Use release to release the keys. See badusb keys for the list of additional supported key names.

badusb.hold("ENTER");
delay(2000);
badusb.release("ENTER");

badusb-release

The release function will release a key combination that was pressed with hold. See badusb keys for the list of additional supported key names.

badusb.hold("ENTER");
delay(2000);
badusb.release("ENTER");

badusb-print

The print function will type the string specified. It takes an optional 2nd parameter with a delay in milliseconds (up to 60 seconds) to wait before returning to the caller.

badusb.print("Hello", 1000); // Wait 1 second before typing next part...
badusb.print(" world!");

badusb-println

The println function will type the string specified and then press Enter. It takes an optional 2nd parameter with a delay in milliseconds (up to 60 seconds) to wait before returning to the caller.

badusb.println("Hello", 1000); // Wait 1 second before typing next part...
badusb.println("world!");

badusb-alt-print

The altPrint function will type the string specified using the numeric keypad. It takes an optional 2nd parameter with a delay in milliseconds (up to 60 seconds) to wait before returning to the caller.

badusb.altPrint("Hello", 1000); // Wait 1 second before typing next part...
badusb.alpPrint(" world!");

badusb-alt-println

The altPrintln function will type the string specified using the numeric keypad and then press Enter. It takes an optional 2nd parameter with a delay in milliseconds (up to 60 seconds) to wait before returning to the caller.

badusb.altPrintln("Hello", 1000); // Wait 1 second before typing next part...
badusb.altPrintln("world!");

badusb keys

    {"CTRL", KEY_MOD_LEFT_CTRL},
    {"SHIFT", KEY_MOD_LEFT_SHIFT},
    {"ALT", KEY_MOD_LEFT_ALT},
    {"GUI", KEY_MOD_LEFT_GUI},

    {"DOWN", HID_KEYBOARD_DOWN_ARROW},
    {"LEFT", HID_KEYBOARD_LEFT_ARROW},
    {"RIGHT", HID_KEYBOARD_RIGHT_ARROW},
    {"UP", HID_KEYBOARD_UP_ARROW},

    {"ENTER", HID_KEYBOARD_RETURN},
    {"PAUSE", HID_KEYBOARD_PAUSE},
    {"CAPSLOCK", HID_KEYBOARD_CAPS_LOCK},
    {"DELETE", HID_KEYBOARD_DELETE_FORWARD},
    {"BACKSPACE", HID_KEYBOARD_DELETE},
    {"END", HID_KEYBOARD_END},
    {"ESC", HID_KEYBOARD_ESCAPE},
    {"HOME", HID_KEYBOARD_HOME},
    {"INSERT", HID_KEYBOARD_INSERT},
    {"NUMLOCK", HID_KEYPAD_NUMLOCK},
    {"PAGEUP", HID_KEYBOARD_PAGE_UP},
    {"PAGEDOWN", HID_KEYBOARD_PAGE_DOWN},
    {"PRINTSCREEN", HID_KEYBOARD_PRINT_SCREEN},
    {"SCROLLLOCK", HID_KEYBOARD_SCROLL_LOCK},
    {"SPACE", HID_KEYBOARD_SPACEBAR},
    {"TAB", HID_KEYBOARD_TAB},
    {"MENU", HID_KEYBOARD_APPLICATION},

    {"F1", HID_KEYBOARD_F1},
    {"F2", HID_KEYBOARD_F2},
    {"F3", HID_KEYBOARD_F3},
    {"F4", HID_KEYBOARD_F4},
    {"F5", HID_KEYBOARD_F5},
    {"F6", HID_KEYBOARD_F6},
    {"F7", HID_KEYBOARD_F7},
    {"F8", HID_KEYBOARD_F8},
    {"F9", HID_KEYBOARD_F9},
    {"F10", HID_KEYBOARD_F10},
    {"F11", HID_KEYBOARD_F11},
    {"F12", HID_KEYBOARD_F12},
    {"F13", HID_KEYBOARD_F13},
    {"F14", HID_KEYBOARD_F14},
    {"F15", HID_KEYBOARD_F15},
    {"F16", HID_KEYBOARD_F16},
    {"F17", HID_KEYBOARD_F17},
    {"F18", HID_KEYBOARD_F18},
    {"F19", HID_KEYBOARD_F19},
    {"F20", HID_KEYBOARD_F20},
    {"F21", HID_KEYBOARD_F21},
    {"F22", HID_KEYBOARD_F22},
    {"F23", HID_KEYBOARD_F23},
    {"F24", HID_KEYBOARD_F24},

    {"NUM0", HID_KEYPAD_0}, // Not in OFW as of 3/26/2024
    {"NUM1", HID_KEYPAD_1}, // Not in OFW as of 3/26/2024
    {"NUM2", HID_KEYPAD_2}, // Not in OFW as of 3/26/2024
    {"NUM3", HID_KEYPAD_3}, // Not in OFW as of 3/26/2024
    {"NUM4", HID_KEYPAD_4}, // Not in OFW as of 3/26/2024
    {"NUM5", HID_KEYPAD_5}, // Not in OFW as of 3/26/2024
    {"NUM6", HID_KEYPAD_6}, // Not in OFW as of 3/26/2024
    {"NUM7", HID_KEYPAD_7}, // Not in OFW as of 3/26/2024
    {"NUM8", HID_KEYPAD_8}, // Not in OFW as of 3/26/2024
    {"NUM9", HID_KEYPAD_9}, // Not in OFW as of 3/26/2024

blebeacon

let blebeacon = require("blebeacon");

blebeacon is used to send BLE (Bluetooth Low Energy) beacon messages.

  • isActive(): bool | error
  • setConfig(mac: Uint8Array, power: number | undefined, intvMin: number | undefined, intvMax: number | undefined): undefined | error
  • setData(data: Uint8Array): undefined | error
  • start(): undefined | error
  • stop(): undefined | error
  • keepAlive(keep: boolean): undefined | error

blebeacon-is-active

The isActive function returns true if a BLE beacon is currently running, otherwise it returns false. You can call stop to stop the running beacon.

let bool_active = blebeacon.isActive();

blebeacon-keep-alive

Pass a true to the keepAlive function if you want the BLE beacon to continue running when your script exits. The default configuration is false, which will stop sending the BLE beacon when the script exits.

blebeacon.keepAlive(false);

blebeacon-set-config

The setConfig function sets the mac address for the BLE beacon. You can optionally specify the power level, min time (milliseconds) and max time (milliseconds) parameters.

let mac = Uint8Array([0x42,0x42,0x42,0x42,0x42,0x42]);
blebeacon.setConfig(mac, 0x1F, 50, 150); // power level: 0x1F, minTime: 50ms, maxTime: 150ms

blebeacon-set-data

The setData function sets the data for the BLE beacon.

let packet = [14, 0xFF, 0x75, 0x00, 0x01, 0x00, 0x02, 0x00, 0x01, 0x01, 0xFF, 0x00, 0x00, 0x43, 0x05];
blebeacon.setData(Uint8Array(packet));

blebeacon-start

The start function will start sending BLE beacons. You should call setConfig and setData before calling start.

blebeacon.start();

blebeacon-stop

The stop function will stop sending a BLE beacon if it is currently running. If it is already stopped, the function call will be ignored.

blebeacon.stop();

dialog

let dialog = require("dialog");

dialog is used to show a message (with Left/Center/Right options) or allow user to select a file.

  • message({ header: string, text: string, button_left: string | undefined, button_center: string | undefined, button_right: string | undefined }): string | error
  • custom(header: string, text: string): boolean | error
  • pickFile(path: string, ext: string): string | error

dialog-custom

The custom function will display a dialog with header, text, and LCR button choices. If back is pressed it will return empty string. You can set a button to undefined if you do not want it to be selectable.

let dialog_params = ({
    header: "Test header",
    text: "Test text",
    button_left: "Left",
    button_right: "Right",
    button_center: "OK"
});
let result = dialog.custom(dialog_params);

// Check result
if (result === "") { print("Back pressed."); }
else if (result === dialog_params.button_left) { print("Left pressed."); }
else if (result === dialog_params.button_right) { print("Right pressed."); }
else if (result === dialog_params.button_center) { print("OK pressed."); }

dialog-message

The message function will display a dialog with header, text, and OK button. If back is pressed it will return false, otherwise true.

let result = dialog.message("Dialog header", "Press OK button");
if (result) { print("OK pressed.); } else { print("Back pressed."); }

dialog-pick-file

The pickFile function will prompt the user to select a file. The first parameter is the directory to start in. The second parameter is the file extension.

let result = dialog.pickFile("/ext/subghz", ".sub"); // Use "*" for second parameter to allow all files.
print("File path:"+result);

flipper

let flipper = require("flipper");

flipper is used to access information about the Flipper Zero device.

flipper-get-model

The getModel function will return the model of the device (like "flipper Zero")

let result = flipper.getModel();
print("Model:"+result); // prints "Model: flipper Zero"

flipper-get-name

The getName function will return the unique name of this Flipper.

let result = flipper.getName();
print("Name:"+result); // prints "Name: Rim"

flipper-get-battery-charge

The getBatteryCharge function will return the current battery level of this Flipper.

let result = flipper.getBatteryCharge();
print("Battery:"+result); // prints "Battery: 98"

gpio

let gpio = require("gpio");

gpio is used to access the General Purpose Input/Output (GPIO) pins of the Flipper Zero.

  • init(pin: string, mode: string, pull: string) : undefined | error
  • read(pin: string) : boolean | error
  • write(pin: string, data: boolean) : undefined | error
  • startAnalog([scale : double])
  • stopAnalog()
  • readAnalog(pin: string) : double | error

gpio-init

The init function configures the pin for the specified mode.

The first parameter is the name of the pin:

  • "PA7", "PA6", "PA4", "PB3", "PB2", "PC3", "PA14" (pin 10), "PA13" (pin 12), "PB6" (pin 13), "PB7" (pin 14), "PC1", "PC0", "PB14" (pin 17).

The second parameter supports the following:

  • "input", "outputPushPull", "outputOpenDrain", "analog", it also supports these values (but lacks APIs to leverage the advanced state):
  • "altFunctionPushPull", "altFunctionOpenDrain", "interruptRise", "interruptFall", "interruptRiseFall", "eventRise", "eventFall", "eventRiseFall"

The third parameter supports the following:

  • "no", "up" (applies an internal pull-up resistor to 3V3), "down" (applies an internal pull-down resistor to GND)
gpio.init("PA7", "outputPushPull", "no"); // Configure pin 2 (Pin A7) as output with no pull-up resistor.
gpio.init("PA6", "outputOpenDrain", "no"); // Configure pin 3 (Pin A6) as output (open-drain) with no pull-up resistor.
gpio.init("PA4", "input", "up"); // Configure pin 4 (Pin A4) as input with a pull-up resistor.

gpio-read

The read function reads the status of a pin. The pin should first be init as "input".

let result = gpio.read("PA4"); // returns false when pin PA4 is low, otherwise true.
print(result); 

gpio-write

The write function changes the status of a pin. The pin should first be init as "outputPushPull" or "outputOpenDrain".

If the pin is "outputPushPull", setting false will set the pin to 0 volts and true will set the pin to 3.3 volts. If the pin is "outputOpenDrain", setting false will set the pin to 0 volts and true will set the pin to floating.

gpio.write("PA7", true); // Set the pin to 3.3 volts
gpio.write("PA6", false); // Set the pin to 0 volts

gpio-start-analog

The startAnalog function configures the Flipper Zero for reading analog signals. You must call this prior to using readAnalog. You must call stopAnalog() prior to calling this function a second time. You can pass an optional parameter with the reference voltage range to use (either 2048 or 2500). Analog pins should not exceed this reference voltage.

gpio.startAnalog(2048); // 2.048 volt reference [pins can be 0-2048mV].

gpio-stop-analog

The stopAnalog function configures the Flipper Zero to stop reading analog signals. You must already have called startAnalog prior to calling this function.

gpio.stopAnalog();
gpio.startAnalog(2500); // Switch to a 2.5 Volt reference range.

gpio-read-analog

The readAnalog function reads the analog voltage on a GPIO pin. Only PA7, PA6, PA4, PC3, PC1 and PC0 are supported. The voltage on the pin should be between 0 volts and the reference voltage that was specified when startAnalog was called. The measured circuit should have a 10K or less impedance for the value to be accurate.

gpio.init("PA7", "analog", "no");
gpio.startAnalog(2048); // 0 - 2.048 volts on analog pins.
let millivolts = gpio.readAnalog("PA7");
print(to_string(millivolts) + "mV");

keyboard

let keyboard = require("keyboard");

keyboard is used to input text or hex data.

  • setHeader(header: string) : undefined | error
  • text(allocSize: number, defaultText: string, selected: boolean) : string | undefined | error
  • byte(numBytes: number, data: Uint8Array) : Uint8Array | undefined | error

keyboard-set-header

The setHeader function sets the header text that will be displayed by the byte and text functions.

keyboard.setHeader("Example Input");

keyboard-text

The text function prompts the user to enter some text and then returns the inputted text. If the user presses the back button then undefined is returned.

The first parameter is the number of bytes of memory to allocate for the response, which includes a byte for the null-terminator. So if you want to allow entering up to 8 characters, you would pass a value of 8+1 to the function.

The second parameter is the default text to display.

The third parameter is true if the text should be selected otherwise false. Having the text selected allows the user to quickly replace the text with something else.

let result = keyboard.text(8+1, "demo", true);
print("Text is:"+result);

keyboard-byte

The byte function prompts the user to enter a hex value and then returns the value. If the user presses the back button then undefined is returned.

The first parameter is the number of bytes for the hex value.

The second parameter is the initial hex value to display.

let result = keyboard.byte(6, Uint8Array([1, 2, 3, 4, 5, 6]));
if (result !== undefined) {
    let data = Uint8Array(result);
    result = "0x";
    for (let i = 0; i < data.byteLength; i++) {
        if (data[i] < 0x10) result += "0";
        result += to_hex_string(data[i]);
    }
}

math

let math = require("math");

math is used to perform various math operations on numbers.

  • abs(x : double) : double | undefined
  • acos(x : double) : double | nan
  • acosh(x : double) : double | undefined | error
  • asin(x : double) : double | error
  • asinh(x : double) : double | error
  • atan(x : double) : double | error
  • atan2(y : double, x : double) : double | error
  • atanh(x : double) : double | error
  • cbrt(x : double) : double | error
  • ceil(x : double) : double | error
  • clz32(x : double) : double | error
  • cos(x : double) : double | error
  • exp(x : double) : double | error
  • floor(x : double) : double | error
  • is_equal(x : double, y : double, delta : double) : boolean | error
  • log(x : double) : double | error
  • max(x : double, y : double) : double | error
  • min(x : double, y : double) : double | error
  • pow(base : double, exponent : double) : double | error
  • random() : double | error
  • sign(x : double) : double | error
  • sin(x : double) : double | error
  • sqrt(x : double) : double | error
  • trunc(x : double) : double | error
  • PI : double
  • E : double
  • EPSILON : double

math-abs

The abs function returns the absolute value of the input parameter. Negative numbers become positive and positive numbers are unchanged.

print(math.abs(-4)); // 4

math-acos

math-acosh

math-asin

math-asinh

math-atan

math-atab2

math-atanh

math-cbrt

math-ciel

math-clz32

math-cos

math-exp

math-floor

math-is-equal

The isEqual function compares two values for equality (within specified delta).

print(math.isEqual(3.0/9.0, 0.3333, 0.0001)); // true

math-log

math-max

The max function returns the larger of the two parameters.

print(math.max(15,4)); // 15

math-min

The min function returns the smaller of the two parameters.

print(math.max(15,4)); // 4

math-pos

math-random

The random function returns a random number between 0 and 2 (exclusive).

print(math.random()); // 1.082337

math-sign

math-sin

math-sqrt

math-trunc

The trunc function removes the fractional part of a number.

print(trunc(-3.432)); // -3.0

math-pi

The PI constant is a double with value 3.14159265358979323846.

var pi = math.PI;

math-e

The E constant is a double with value 2.7182818284590452354.

var e = math.E;

#math-epsilon The EPSILON constant is a double with a value that represents the smallest positive double value that is greater than zero.

var really_small_positive = math.EPSILON;

notification

let notify = require("notification");

notification is used to blink the status LED, play tones, and vibrate the Flipper Zero.

  • success() : undefined
  • error() : undefined
  • blink(color: string, type: string) : undefined | error

notification-success

The success function is called to indicate a successful action to the user. Status light blinks green, vibrates, plays happy "do-de-da-lat" sound.

notify.success();

notification-error

The error function is called to indicate an error to the user. Status light blinks red, vibrates, plays mad "do-do" sound.

notify.error();

notification-blink

The blink function is called to cause the status LED to blink. Often this is called in a loop with a delay to have the LED blink multiple times. NOTE: If the USB cable is plugged in the LED is turned to green and the blink may not be seen.

The first parameter is the name of the color. Supported colors are: "blue", "red", "green", "yellow", "cyan", "magenta".

The second parameter is the type of notification. Supported types are: "short", "long".

for (let i = 0; i < 10; i++) {
    notify.blink("red", "short");
    delay(500);
}

serial

let serial = require("serial");

serial is used for USART (pins 13, 14) and LPUART (pins 15, 16) communication on the Flipper Zero.

  • setup(port: string, baudrate: number) : undefined | error
  • end() : undefined | error
  • write(string | number | Uint8Array | [...number]) : undefined | error
  • read(readlen: number [, timeout: number] ) : string | undefined | error
  • readln( [timeout: number] ) : string | undefined | error
  • readBytes(readlen: number [, timeout: number] ) : array | undefined | error
  • readAny( [timeout: number] ) : string | undefined | error
  • expect(array | string [, timeout: number] ) : number | undefined | error

serial-setup

The setup function is used to initialize the serial port. Once it is initialized, it cannot be initialized again until end is called.

The first parameter is the port. The valid values are "usart" (for pins 13 & 14) or "lpuart" (for pins 15 & 16).

The second parameter is the baud rate.

serial.setup("lpuart", 115200);

serial-end

The end function is used to deinitialize the serial port. It should only be called after the setup function is called.

serial.end();

serial-write

The write function sends data on the serial port. The port must already be initialized. The parameter can be a string, number or array. The numeric values must all be in the range of (0x00..0xFF).

serial.write([0x0a]);
serial.write("uci\n");

storage

let storage = require("storage");

storage is used for accessing the SD card in the Flipper Zero.

  • read(path: string) : string | error
  • read v2(path: string, size: number, offset: number) : string | error
  • write(path: string, data: string) : boolean | error
  • write v2(path: string, data: array | string) : boolean | error
  • append(path: string, data: string) : boolean | error
  • copy(sourcePath: string, targetPath: string) : boolean
  • move(originalPath: string, newPath: string) : boolean
  • exists(path: string) : boolean | error
  • mkdir(path: string) : boolean
  • remove(path: string) : boolean | error
  • virtualInit(path: string) : boolean | error
  • virtualMount() : undefined | error
  • virtualQuit() : undefined | error

storage-read

In Version 1.0 (still used by Xtreme) the read function reads a file up to 128KB in size and returns the contents as a string.

The first parameter is the path of the file.

let data = storage.read("/ext/demo.txt");
print(data);

In Version 2.0 (Momentum and Unleashed) the read function reads a file up to 128KB in size, starting at the specified offset and returns the contents as a binary array.

The first parameter is the path of the file. The second parameter is the size to read. The third parameter is the offset.

function arraybuf_to_string(arraybuf) {
    let string = "";
    let data_view = Uint8Array(arraybuf);
    for (let i = 0; i < data_view.length; i++) {
        string += chr(data_view[i]);
    }
    return string;
}
let data = storage.read("/ext/demo.txt", 4096, 20480); // Read up to 4096 bytes, starting at offset 20480.
print(arraybuf_to_string(data));

storage-write

The write function writes a string to a file, returning true if successful. Overwrites existing file.

The first parameter is the path of the file.

The second parameter is the contents to write. (In version 2, Momentum/RogueMaster you can also write binary data)

let result = storage.write("/ext/demo.txt", "Hello world");
if (!result) { print("Failed to write file."); }

storage-append

The append function appends a string to a file, returning true if successful.

The first parameter is the path of the file.

The second parameter is the contents to append.

let result = storage.append("/ext/demo.txt", "Line 2.");
if (!result) { print("Failed to append to file."); }

storage-copy

The copy function copies a file from one location to another, returning undefined if successful (otherwise returning the error description).

The first parameter is the path of the source file.

The second parameter is the path of the target file.

let result = storage.copy("/ext/demo.txt", "/ext/demo2.txt");
if (result !== undefined) {
  print("Failed to copy file", result);
}

storage-move

The move function moves a file from one location to another, returning undefined if successful (otherwise returning the error description).

The first parameter is the path of the source file.

The second parameter is the new path.

let result = storage.move("/ext/demo.txt", "/ext/demo2.txt");
if (result !== undefined) {
  print("Failed to move file", result);
}

storage-exists

The exists function returns true if a file exists at the specified path, otherwise false.

The parameter is the path of the file.

let result = storage.exists("/ext/demo.txt");
if (result) { print("File exists"); } else { print("File does not exist"); }

storage-mkdir

The mkdir function create a new directory. It returns true if successful, otherwise false.

The parameter is the path of the directory.

if (!storage.exists("/ext/demo")) {
  let result = storage.mkdir("/ext/demo");
  if (!result) {
    print("Failed to create directory");
  }
}

storage-remove

The remove function deletes a file at the specified path, returning true if successful.

The parameter is the path of the file.

let result = storage.remove("/ext/demo.txt");
if (!result) { print("Failed to remove file."); }

storage-virtual-init

The virtualInit function maps virtual storage to a file.

The first parameter is the path to the file to mount. The file should be created with usbdisk.createImage.

let usb = require("usbdisk");
let path = __dirpath + "4MB.img";
usbdisk.createImage(path, 4*1024*1024); // Create the image using usbdisk.createImage.
storage.virtualInit(path);

storage-virtual-mount

The virtualMount function attaches virtual storage to the /mnt/ folder.

let result = storage.virtualMount();
if (result) {
  storage.write("/mnt/demo.txt", "Hello World");
} else {
  print("failed to mount");
}

storage-virtual-quit

The virtualQuit function detaches virtual storage from the /mnt/ folder.

storage.virtualQuit();

subghz

let subghz = require("subghz");

subghz is used for accessing the subghz radio (CC1101) on the Flipper Zero.

subghz-get-frequency

The getFrequency function returns the current frequency for the received or last sent signal.

let rssi = subghz.getRssi();
print("rssi: ", rssi);

subghz-get-rssi

The getRssi function returns the current RSSI (Receive Signal Signal Indicator) for the received signal. The subghz must be in receive mode.

let rssi = subghz.getRssi();
print("rssi: ", rssi);

subghz-get-state

The getState function returns the current state of the radio. Valid states are "RX", "TX", "IDLE" and "".

let state = subghz.getState();
print("state: ", state);

subghz-is-external

The isExternal function returns true if an external CC1101 radio is being used, otherwise it returns false.

let result = subghz.isExternal();
if (result) { print("external radio"); } else { print("internal radio"); }

subghz-set-frequency

The setFrequency function is used to set the frequency of the radio. The state must be "IDLE" before setting the frequency.

The first parameter is the frequency in hertz. The frequency must be a frequency allowed by the firmware.

subghz.setFrequency(433920000);

subghz-set-idle

The setIdle function is used to set the state of the radio to "IDLE".

subghz.setIdle();

subghz-set-rx

The setRx function is used to set the state of the radio to "RX".

subghz.setRx();

subghz-setup

The setup function initializes the subghz radio. It configures to use the external radio based on firmware settings. It sets the radio to "IDLE" and changes the default frequency to 433.92MHz.

subghz.setup();

subghz-transmit-file

The transmitFile function sets a subghz file. The '.sub' file can use a supported protocol or it can use RAW format. The file is sent on the frequency specified in the file, and the getFrequency value is updated. If the file is successful sent the function will return true.

let result = subghz.transmitFile("/ext/subghz/demo.sub");
if (result) { print("sent"); } else { print("failed to send"); }

submenu

let submenu = require("submenu");

submenu is used for creating a scrollable menu of items and returning the selected one.

  • addItem(label: string, id: number) : undefined | error
  • setHeader(label: string) : undefined | error
  • show() : number : undefined

submenu-add-item

The addItem function adds a new entry to the menu.

The first parameter is the name of the entry.

The second parameter is the id associated with the entry, which will be returned when the user selects the item.

submenu.addItem("Calculate tip", 0);

submenu-set-header

The setHeader function updates the header that is displayed at the top of the submenu.

submenu.setHeader("Choose an item");

submenu-show

The show function displays the submenu and returns the id of the item selected. If the user presses the Back button, then undefined is returned.

let result = submenu.show();
if (result === undefined) {
  print("User pressed back");
} else if (result === 0) {
  print("User selected id 0.");
}

textbox

let textbox = require("textbox");

textbox is used for creating a dynamic display of text on the Flipper Zero. Contents are displayed in a non-blocking way, allowing the script to continue running and update the UI. The user can scroll the textbox contents, however whenever it is updated, focus is reset.

textbox-add-text

The addText function appends the specified text to the end of the textbox.

The parameter is the contents to append.

textbox.addText("Hello world");

textbox-close

The close function closes the textbox.

textbox.close();

textbox-clear-text

The clearText function clears the contents of the textbox. NOTE: On Xtreme firmware, this is still called emptyText.

textbox.clearText();

textbox-is-open

The isOpen function returns true if the textbox is currently visible. After a show command the textbox is visible until it is closed by either calling close or by the user pressing the Back button.

let result = textbox.isOpen();
if (!result) { print("textbox is closed"); } else { print("textbox is open"); }

textbox-set-config

The setConfig function configures the textbox.

The first parameter is where focus should be set when it is updated. Valid values are "start" (set focus to beginning of text) and "end" (set focus to the end of the text).

The second parameter is the font to use. Valid values are "text" and "hex".

textbox.setConfig("end", "text");

textbox-show

The show function displays the textbox. It will be visible until it is closed by either calling close or by the user pressing the Back button.

textbox.show();

usbdisk

let usbdisk = require("usbdisk");

usbdisk is used for exposing a special file on the SD Card as a USB thumb drive via the Flipper Zero's USB port. This is helpful for copying files between the host computer and the Flipper Zero without needing any special software installed.

  • createImage(path: string, capacity: number) : undefined | error
  • start(path: string) : undefined | error
  • stop() : undefined | error
  • wasEjected() : boolean | error

usbdisk-create-image

The createImage function is used to create a special file that can be used as the USB drive. Note: This file can also be used by storage.virtualInit to read and write files to it.

The first parameter is the path to the file.

The second parameter is the number of bytes to reserve for the image.

usbdisk.createImage("/ext/apps_data/mass_storage/4MB.img", 4*1024*1024);

usbdisk-start

The start function will expose the file as a USB drive.

The parameter is the path to the file.

usbdisk.start("/ext/apps_data/mass_storage/4MB.img");

usbdisk-stop

The stop function will detach the USB drive.

usbdisk.stop();

usbdisk-was-ejected

The wasEjected function will return true if the USB drive has been ejected from the host computer.

// Wait for the disk to be ejected.
while (usbdisk.wasEjected()) {
  delay(1000);
}

vgm

let vgm = require("vgm");

vgm is used to access the sensors of the Video Game Module. See this project for more details on pitch, roll and yaw. VGM pitch-roll-yaw

#vgm-get-pitch The getPitch function returns the pitch of the VGM in degrees (-90 to 90).

let pitch = vgm.getPitch();
print("Pitch", pitch);

#vgm-get-roll The getRoll function returns the roll of the VGM in degrees (-180 to 180).

let roll = vgm.getRoll();
print("Roll", roll);

#vgm-get-yaw The getYaw function returns the yaw of the VGM in degrees (-180 to 180).

let yaw = vgm.getYaw();
print("Yaw", yaw);

#vgm-delta-yaw The delyaYaw function returns the amount of change in yaw of the VGM in degrees if it exceeds the threshold, otherwise 0.

The first parameter is the threshold of yaw change.

The second parameter is the timeout in milliseconds.

let yaw = vgm.deltaYaw(45, 3000); // Wait for Flipper to be rotated at least 45 degrees in the next 3 seconds.
if (yaw===0) {
  print("Flipper was not rotated.");
} if (yaw>0) {
  print("Flipper was rotated clockwise");
} else {
  print("Flipper was rotated counter-clockwise");
}

widget

let widget = require("widget");

widget is used for displaying non-blocking UI elements.

  • addBox(x: number, y: number, w: number, h: number): number
  • addCircle(x: number, y: number, r: number): number
  • addDisc(x: number, y: number, r: number): number
  • addDot(x: number, y: number): number
  • addFrame(x: number, y: number, w: number, h: number): number
  • addGlyph(x: number, y: number, ch: number): number
  • addIcon(x: number, y: number, icon: string): number
  • addLine(x1: number, y1: number, x2: number, y2: number): number
  • addRbox(x: number, y: number, w: number, h: number, r: number): number
  • addRframe(x: number, y: number, w: number, h: number, r: number): number
  • addText(x: number, y: number, font: string, text: string): number
  • addXbm(x: number, y: number, index: number): number
  • close(): undefined
  • loadImageXbm(path: string): number
  • remove(id: number): boolean
  • isOpen(): boolean
  • show(): undefined

widget-add-box

The addBox function will add a filled in box at (x,y) with a width of (w,h).

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the width.

The fourth parameter is the height.

let x=10; // over
let y=15; // down
let w=20; // 20 pixels wide
let h=10; // 10 pixels high
let id = widget.addBox(x,y,w,h); // You can remove later with: remove(id);

widget-add-circle

The addCircle function will add a circle centered at (x,y) with a radius of (r).

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the radius.

let x=10; // over
let y=15; // down
let r=5; // 5 pixel radius
let id = widget.addCircle(x,y,r); // You can remove later with: remove(id);

widget-add-disc

The addCircle function will add a filled in disc centered at (x,y) with a radius of (r).

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the radius.

let x=10; // over
let y=15; // down
let r=5; // 5 pixel radius
let id = widget.addDisc(x,y,r); // You can remove later with: remove(id);

widget-add-dot

The addDot function will set the pixel at (x,y).

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

let x=10; // over
let y=15; // down
let id = widget.addDot(x,y); // You can remove later with: remove(id);

widget-add-frame

The addFrame function will add an outlined rectangle at (x,y) with a width of (w,h).

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the width.

The fourth parameter is the height.

let x=10; // over
let y=15; // down
let w=20; // 20 pixels wide
let h=10; // 10 pixels high
let id = widget.addFrame(x,y,w,h); // You can remove later with: remove(id);

widget-add-glyph

The addGlyph function will add a glyph at (x,y) using the character specified.

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the code of the glyph.

let x=10; // over
let y=15; // down
let ch=35; // typically ascii code of glyph.
let id = widget.addGlyph(x, y, ch); // You can remove later with: remove(id);

widget-add-icon

The addIcon function will add an icon at (x,y) using the icon specified. addIcon requires the firmware to know the name of the icons, which takes additional memory and may not be possible to implement in all firmware. For cross-firmware compatibility your script can use addXbm with local .xbm files instead.

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the name of the icon.

let x=10; // over
let y=15; // down
let icon="ButtonUp_7x4";
let id = widget.addIcon(x, y, icon); // You can remove later with: remove(id);

widget-add-line

The addLine function will draw a line from (x1,y1) to (x2,y2).

The first parameter is the x coordinate of the first point. 0 is left most. 127 is right most.

The second parameter is the y coordinate of the first point. 0 is top most. 63 is bottom most.

The third parameter is the x coordinate of the second point. 0 is left most. 127 is right most.

The fourth parameter is the y coordinate of the second point. 0 is top most. 63 is bottom most.

let x1=10; // over
let y1=15; // down
let x2=100; // over
let y2=35; // down
let id = widget.addLine(x1, y1, x2, y2); // You can remove later with: remove(id);

widget-add-rbox

The addRbox function will add a filled in box at (x,y) with a width of (w,h) with rounded corners of radius (r).

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the width.

The fourth parameter is the height.

The firth parameter is the radius.

let x=10; // over
let y=15; // down
let w=20; // 20 pixels wide
let h=10; // 10 pixels high
let r=3; // 3 pixel radius
let id = widget.addBox(x,y,w,h,r); // You can remove later with: remove(id);

widget-add-rframe

The addRbox function will add an outlined rectangle at (x,y) with a width of (w,h) with rounded corners of radius (r).

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the width.

The fourth parameter is the height.

The firth parameter is the radius.

let x=10; // over
let y=15; // down
let w=20; // 20 pixels wide
let h=10; // 10 pixels high
let r=3; // 3 pixel radius
let id = widget.addRbox(x,y,w,h,r); // You can remove later with: remove(id);

widget-add-text

The addText function will add text at (x,y) with the specified font.

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the font. Valid values are: "Primary", "Secondary".

The fourth parameter is the height.

The firth parameter is the radius.

let x=10; // over
let y=15; // down
let font="Primary"; // Large font
let text="Hello!";
let id = widget.addText(x,y,font,text); // You can remove later with: remove(id);

widget-add-xbm

The addXbm function will render a XBM image at (x,y) with the specified image.

The first parameter is the x coordinate. 0 is left most. 127 is right most.

The second parameter is the y coordinate. 0 is top most. 63 is bottom most.

The third parameter is the image identifier. Use loadImageXbm to get the identifier.

let x=10; // over
let y=15; // down
let identifier= widget.loadImageXbm(__dirpath + "/demo.xbm");
let id = widget.addText(x,y,identifier); // You can remove later with: remove(id);

widget-close

The close function will stop displaying the widget.

widget-load-image-xbm

The loadImageXbm function is used to load an image. You can then pass the returned value to addXbm.

The parameter is the path to the xbm file.

let identifier= widget.loadImageXbm(__dirpath + "/demo.xbm");

 

widget-remove

The remove function is used to remove a previously added component.

The parameter is the id that was returned from a previous add... method.

let line = widget.addLine(10, 10, 100, 60);
widget.show();
delay(1000);
widget.remove(line); // Remove the previously added component.
delay(1000);
widget.close();

widget-is-open

The isOpen function is used to tell if the widget is being displayed. When a user presses the Back button the widget will be dismissed and this function will return false.

widget.show();
while (widget.isOpen()) {
  delay(100);
}

widget-show

The show function is used to display the widget. The widget can be dismissed by pressing the Back button or by calling close.

widget.show();
Clone this wiki locally