The CLI will guide you through the process of bootstrapping one of our starter templates.
If this is your first One app, we recommend taking this route, as you can choose between bare-bones example all the way up to a full stack with a user system and authentication. We will be launching more starters as the project matures.
While we recommend using a preset to get started, it's helpful to walk through creating a new One app from scratch.
For those of you who prefer to learn by doing, we've put together a 5-minute tutorial that will have you building a basic app in no time.
Let's begin with our package.json:
package.json
{
"name":"one-app",
"version":"0.0.0",
"type":"module",
"scripts":{
"dev":"one dev",
"build":"one build",
"serve":"one serve"
},
"dependencies":{
"expo":"~51.0.28",
"one":"latest",
"react":"^18.3.1",
"react-native":"^0.74.1",
"react-native-web":"^0.19.12"
},
"devDependencies":{
"vite":"6.0.0-beta.1"
}
}
It is strongly recommended to pin the one version for production stability:
npm install one@latest --save-exact
This ensures consistent behavior across different environments and deployments.
If using pnpm or a package manager that symlinks dependencies, you'll want to disable that setting.
One has (just a few) patches that fix dependencies. In part this is to enable switching between React 19 on web and React 18 on native, in part this is because the React Native ecosystem has many, many weirdly published packages - think JSX or Flowtype published in .js files, or non-standard legacy babel experimental transforms, etc. We are working with package authors to improve this, but we've found patches to be both simpler and require less mysterious behavior/slowdowns than the alternative - making Vite handle everything internally with magic.
One will create a TSConfig for you when you first start the dev server, with some defaults that work well. Of course you can create your own if you'd prefer.
One expects your file system routes to be placed in an app directory.
app/_layout.tsx serves as the top-level wrapper for all routes, ideal for:
Consistent page structure
Global styles or themes
Persistent navigation
App-wide state management
One uses the <Slot /> component to indicate where child components should render, allowing for nested layouts and fine-grained control over your UI structure.
Let's create a root layout app/_layout.tsx:
app/_layout.tsx
import{ Slot }from'one'
exportdefaultfunctionLayout(){
return<Slot/>
}
And then create a simple index page app/index.tsx:
If the port isn't taken, you should be able to load http://localhost:8081 and to see your website (use --port to change the port).
On native, you can use an Expo Go development app and load your app with exp://0.0.0.0:8081. Press the key combination q, r in your terminal which is running the dev server, and scan the QR code with your iPhone's camera if your computer and iPhone are connected to the same WiFi network.
Note that for web-only projects you don't need react-native, and further note that if you want something that has a much broader styling featureset and much lighter bundle-size on web, you can use Tamagui, which has no dependency on React Native.