Getting Started
This guide walks you through creating your first SkyScript app using the Hello World example as a reference. By the end you will have a working app that reads X-Plane datarefs and executes commands from JavaScript.
Prerequisites
- Node.js 18 or later
- A text editor
- SkyScript plugin installed in X-Plane 12 (see Installation)
Scaffold the Project
Create a new folder inside apps/ in your SkyScript plugin directory (or in the repo if you're developing from source):
mkdir apps/my-app && cd apps/my-appCreate the manifest
Every app needs a manifest.yaml at its root. Create apps/my-app/manifest.yaml:
name: My App
hide_addressbar: true
framerate: 20See the Manifest Reference for all available fields.
Bootstrap with Create React App
The Hello World example uses React with TypeScript. You can use any framework — or plain HTML — but React is a good starting point:
npx create-react-app . --template typescriptSet the homepage field in package.json so the built assets use relative paths:
{
"homepage": "."
}Project structure
After scaffolding, your app should look like this:
apps/my-app/
├── manifest.yaml # Required — app metadata
├── package.json
├── public/
│ └── index.html
├── src/
│ ├── index.tsx
│ ├── App.tsx
│ └── ...
└── tsconfig.jsonAdd TypeScript Declarations
Create src/react-app-env.d.ts (or add to your existing one) so TypeScript knows about the SkyScript API:
/// <reference types="react-scripts" />
interface SkyscriptXplm {
getDataref(ref: string): Promise<number | string | number[]>;
setDataref(ref: string, value: number | string, valueType?: string): Promise<void>;
executeCommand(command: string): Promise<void>;
}
interface Window {
skyscript: {
xplm: SkyscriptXplm;
};
}Write Your App
Replace src/App.tsx with a simple component that reads a dataref:
import { useState } from 'react';
function App() {
const [zuluTime, setZuluTime] = useState('--');
async function readTime() {
try {
const value = await window.skyscript.xplm.getDataref(
'sim/time/zulu_time_sec'
);
setZuluTime(Number(value).toFixed(1));
} catch (err: any) {
setZuluTime(`Error: ${err.message}`);
}
}
return (
<div style={{ padding: 40, textAlign: 'center', color: '#e0e0e0' }}>
<h1>My App</h1>
<p>Zulu time: {zuluTime}</p>
<button onClick={readTime}>Read Dataref</button>
</div>
);
}
export default App;Build
npm run buildThis produces a build/ folder containing index.html and static assets. SkyScript serves this folder directly.
Deploy to X-Plane
If you're developing outside the plugin directory, copy the built output:
cp -r build/ /path/to/X-Plane/Resources/plugins/SkyScript/apps/my-app/
cp manifest.yaml /path/to/X-Plane/Resources/plugins/SkyScript/apps/my-app/If you're developing inside the repo's apps/ folder, the build script handles this automatically:
# From the repo root
./build_platforms.sh macThis compiles the plugin, builds all apps, and copies everything to the X-Plane plugins folder.
Test
- In X-Plane, go to Plugins > SkyScript > Reload configuration.
- Open Plugins > SkyScript > My App.
- Click Read Dataref — you should see the current Zulu time.
Next Steps
- Read the full JavaScript API Reference to learn about
setDatarefandexecuteCommand. - Check the Manifest Reference for options like
user_agent,framerate, andaudio_muted. - Look at the Hello World source for a more complete example with polling, custom dataref input, and command execution.