简介
想从头开始构建一个使用 Next.js 的移动应用吗?本指南将带您完成创建一个从一开始就配置为移动的 Next.js 15 项目,然后将其打包为原生 iOS 和 Android 应用 Capacitor 8.
通过本教程,您将拥有一个可以在模拟器上运行的工作移动应用,可以继续开发并最终发布到 App Store 和 Google Play。
所需时间: ~30 分钟
您将构建:
- 一个新的 Next.js 15 项目,带有 App 路由
- 静态导出配置为移动
- Capacitor 8,带有必需的插件
- 原生 iOS 和 Android 应用
- 实时重载开发设置
已经有一个 Next.js 应用程序?检查出 将您的 Next.js 应用程序转换为移动应用 反之亦然。
先决条件
确保您安装了这些:
- Node.js 18+ (请与
node --version) - Bun 包管理器(
curl -fsSL https://bun.sh/install | bash) - Xcode (仅限 macOS,用于 iOS 开发)
- Android Studio (for Android development)
步骤 1: 创建一个新 Next.js 项目
首先创建一个新的 Next.js 15 项目:
bunx create-next-app@latest my-mobile-app
当提示时,请选择这些选项:
- TypeScript: 是(推荐)
- ESLint: 是
- Tailwind CSS: 是(推荐用于移动端样式)
src/目录: 是- App 路由器: 是 (推荐)
- 导入别名: 默认 (
@/*)
导航到您的项目:
cd my-mobile-app
步骤 2: 为静态导出配置 Next.js
Capacitor 需要静态 HTML/JS/CSS 文件。为静态导出配置 Next.js,更新 next.config.ts:
import type { NextConfig } from 'next';
const nextConfig: NextConfig = {
output: 'export',
images: {
unoptimized: true,
},
// Ensure trailing slashes for proper routing in Capacitor
trailingSlash: true,
};
export default nextConfig;
为什么这些设置?
output: 'export'— 生成静态 HTML 而不是需要 Node.js 服务器images: { unoptimized: true }— 禁用 Next.js 图像优化 (需要服务器)trailingSlash: true— 确保在原生 WebView 中正确路由
步骤 3: 添加移动脚本
更新你的 package.json 与移动开发脚本:
{
"scripts": {
"dev": "next dev",
"build": "next build",
"start": "next start",
"lint": "next lint",
"mobile": "bun run build && bunx cap sync",
"mobile:ios": "bun run mobile && bunx cap open ios",
"mobile:android": "bun run mobile && bunx cap open android"
}
}
测试构建:
bun run build
你应该看到一个 out 目录中包含你的静态文件。
步骤 4: 安装 Capacitor 8
安装 Capacitor 核心包:
bun add @capacitor/core
bun add -D @capacitor/cli
安装大多数移动应用程序需要的必备插件:
bun add @capacitor/app @capacitor/keyboard @capacitor/splash-screen @capacitor/status-bar @capacitor/preferences
这些插件的作用:
- @capacitor/app — 前台/后台应用程序生命周期事件(深度链接)
- @capacitor/keyboard — 控制键盘行为
- @capacitor/splash-screen — 原生启动屏幕控制
- @capacitor/status-bar — 样式设备状态栏
- @capacitor/preferences — 键值存储(类似 localStorage 但原生)
步骤 5:初始化 Capacitor
初始化 Capacitor 以及您的项目详细信息:
bunx cap init "My Mobile App" com.example.mymobileapp --web-dir out
替换:
"My Mobile App"用您的应用程序显示名称com.example.mymobileapp用您的应用程序 ID(反向域名表示法)
创建 capacitor.config.ts. 更新插件配置:
import type { CapacitorConfig } from '@capacitor/cli';
const config: CapacitorConfig = {
appId: 'com.example.mymobileapp',
appName: 'My Mobile App',
webDir: 'out',
plugins: {
SplashScreen: {
launchShowDuration: 2000,
launchAutoHide: true,
androidScaleType: 'CENTER_CROP',
splashFullScreen: true,
splashImmersive: true,
},
Keyboard: {
resize: 'body',
resizeOnFullScreen: true,
},
StatusBar: {
style: 'light',
},
},
};
export default config;
步骤 6: 添加本机平台
安装平台包:
bun add @capacitor/ios @capacitor/android
生成本机项目:
bunx cap add ios
bunx cap add android
创建 ios 和 android 包含本机项目的目录。
步骤 7: 构建和运行
构建您的项目并同步本机平台:
bun run mobile
在 iOS 模拟器中打开:
bun run mobile:ios
或 Android 模拟器:
bun run mobile:android
在 Xcode (iOS) 中:
- 从设备下拉菜单中选择一个模拟器
- 点击播放按钮或按
Cmd + R
在 Android Studio 中:
- 等待 Gradle 完成同步
- 从设备下拉菜单中选择一个模拟器
- 点击运行按钮或按
Shift + F10
步骤 8:设置实时重载
为了更快的开发,启用实时重载,修改后立即在设备上显示
- 找到你的本地 IP 地址:
# macOS
ipconfig getifaddr en0
# Windows
ipconfig
- 创建一个开发 Capacitor 配置。添加到
capacitor.config.ts:
import type { CapacitorConfig } from '@capacitor/cli';
const devConfig: CapacitorConfig = {
appId: 'com.example.mymobileapp',
appName: 'My Mobile App',
webDir: 'out',
server: {
url: 'http://YOUR_IP_ADDRESS:3000',
cleartext: true,
},
plugins: {
// ... same plugin config
},
};
const prodConfig: CapacitorConfig = {
appId: 'com.example.mymobileapp',
appName: 'My Mobile App',
webDir: 'out',
plugins: {
// ... same plugin config
},
};
const config = process.env.NODE_ENV === 'development' ? devConfig : prodConfig;
export default config;
- 启动开发服务器并将配置复制到本机:
bun run dev &
NODE_ENV=development bunx cap copy
- 在 Xcode/Android Studio 中重建
现在,您的 Next.js code 的编辑将在设备上热重载。
步骤 9:创建第一个移动屏幕
让我们创建一个简单的移动友好的主屏幕。更新 src/app/page.tsx:
'use client';
import { useEffect, useState } from 'react';
import { App } from '@capacitor/app';
import { Keyboard } from '@capacitor/keyboard';
export default function Home() {
const [appInfo, setAppInfo] = useState<{ name: string; version: string } | null>(null);
useEffect(() => {
// Get app info on mount
App.getInfo().then(setAppInfo).catch(console.error);
// Handle back button on Android
const backHandler = App.addListener('backButton', ({ canGoBack }) => {
if (!canGoBack) {
App.exitApp();
} else {
window.history.back();
}
});
// Hide keyboard when tapping outside inputs
const keyboardHandler = Keyboard.addListener('keyboardWillShow', () => {
document.body.classList.add('keyboard-open');
});
return () => {
backHandler.then(h => h.remove());
keyboardHandler.then(h => h.remove());
};
}, []);
return (
<main className="min-h-screen bg-linear-to-b from-blue-500 to-blue-700 flex flex-col items-center justify-center p-6 text-white">
<h1 className="text-4xl font-bold mb-4">My Mobile App</h1>
<p className="text-xl mb-8 text-center opacity-90">
Built with Next.js 15 + Capacitor 8
</p>
{appInfo && (
<div className="bg-white/20 rounded-lg p-4 backdrop-blur-sm">
<p className="text-sm">
{appInfo.name} v{appInfo.version}
</p>
</div>
)}
<div className="mt-12 space-y-4 w-full max-w-sm">
<button className="w-full py-4 px-6 bg-white text-blue-600 rounded-xl font-semibold text-lg shadow-lg active:scale-95 transition-transform">
Get Started
</button>
<button className="w-full py-4 px-6 bg-white/20 text-white rounded-xl font-semibold text-lg backdrop-blur-sm active:scale-95 transition-transform">
Learn More
</button>
</div>
</main>
);
}
步骤 10:添加安全区域处理
移动设备有凹口、主屏幕指示器和状态栏。使用 Tailwind 添加安全区域处理。
更新 src/app/globals.css:
@tailwind base;
@tailwind components;
@tailwind utilities;
:root {
--sat: env(safe-area-inset-top);
--sar: env(safe-area-inset-right);
--sab: env(safe-area-inset-bottom);
--sal: env(safe-area-inset-left);
}
body {
padding-top: var(--sat);
padding-right: var(--sar);
padding-bottom: var(--sab);
padding-left: var(--sal);
}
/* Prevent text selection on mobile */
* {
-webkit-user-select: none;
user-select: none;
-webkit-tap-highlight-color: transparent;
}
/* Allow text selection in inputs */
input, textarea {
-webkit-user-select: auto;
user-select: auto;
}
/* Keyboard handling */
.keyboard-open {
--sab: 0px;
}
项目结构
您的项目现在应该像这样:
my-mobile-app/
├── android/ # Android native project
├── ios/ # iOS native project
├── out/ # Static build output
├── src/
│ ├── app/
│ │ ├── globals.css
│ │ ├── layout.tsx
│ │ └── page.tsx
│ └── ...
├── capacitor.config.ts # Capacitor configuration
├── next.config.ts # Next.js configuration
├── package.json
└── ...
下一步
您现在有一个工作的 Next.js 移动应用。接下来要做的事情是:
必备设置
- App 图标: 替换默认图标在
ios/App/App/Assets.xcassets和android/app/src/main/res - 启动屏幕: 在原生项目中自定义或使用
@capacitor/splash-screen配置 - 深度链接: 配置 URL 方案
添加更多功能
- 相机:
bun add @capacitor/camera - 地理位置:
bun add @capacitor/geolocation - 推送通知:
bun add @capacitor/push-notifications - 文件系统:
bun add @capacitor/filesystem
原生UI和过渡
使用Capgo插件代替Konsta UI实现原生移动体验:
- @capgo/capacitor-native-navigation — Liquid Glass标签栏和原生导航栏
- @capgo/capacitor-transitions — 原生感觉的页面过渡
bun add @capgo/capacitor-native-navigation @capgo/capacitor-transitions
bunx cap sync
为Tailwind安全区域添加 @capgo/tailwind-capacitor:
bun add -D tailwind-capacitor
查看 使用@capgo/capacitor-native-navigation, 使用@capgo/capacitor-transitions, 和 tailwind-capacitor 仓库 为 Next.js 专用设置。
修复 iOS 布局问题 (视口、安全区域和水平溢出)
如果内容在 iOS 上看起来被裁切、偏移或水平滚动,仅添加或调整视口标签通常无法解决问题。按照以下顺序检查这些问题。 overflow-x: hidden 确保视口元标签已正确应用
App Router
从 (app/Pages Router viewport 将视口元标签放在 app/layout.tsx:
import type { Viewport } from 'next';
export const viewport: Viewport = {
width: 'device-width',
initialScale: 1,
viewportFit: 'cover',
};
export (pages/from pages/_app.tsx, 不要 _document.tsx.
只需从一个根包装器中处理 iOS 安全区域
创建一个单独的应用程序外壳并在那里应用安全区域填充 — 不在多个嵌套组件中:
html,
body,
#__next {
width: 100%;
min-height: 100%;
margin: 0;
padding: 0;
overflow-x: hidden;
}
* {
box-sizing: border-box;
}
.app-shell {
min-height: 100dvh;
width: 100%;
padding-top: env(safe-area-inset-top);
padding-right: env(safe-area-inset-right);
padding-bottom: env(safe-area-inset-bottom);
padding-left: env(safe-area-inset-left);
}
将所有页面内容包装在 .app-shell在头部、模态对话框和布局包装器中重复的安全区域填充通常会使 UI 看起来被裁剪或过大。
使用 @capgo/tailwind-capacitor, 你可以用类似于 pt-safe pb-safe px-safe 的工具表达相同的填充。
将 Capacitor iOS contentInset 设置为 never 第一个
In __CAPGO_KEEP_0__ 中,优先使用原生 inset 并让 CSS (或 Native Navigation 的) capacitor.config.ts禁用安全区域: contentInsetMode: 'css'混合 __CAPGO_KEEP_0__ 自动内容 inset 与 CSS padding 是双倍间距的常见原因:
const config: CapacitorConfig = {
appId: 'com.example.myapp',
appName: 'my-app',
webDir: 'out',
ios: {
contentInset: 'never',
},
};
Mixing Capacitor’s automatic content inset with CSS env(safe-area-inset-*) 通常的罪魁祸首是使用
, Tailwind
, 固定像素宽度或 100vw在 Safari Web Inspector 中运行: w-screen使用 Tailwind,替换 min-width.
为
[...document.querySelectorAll('*')]
.filter(el => el.scrollWidth > document.documentElement.clientWidth)
.map(el => ({
el,
tag: el.tagName,
class: el.className,
scrollWidth: el.scrollWidth,
clientWidth: document.documentElement.clientWidth,
}));
With Tailwind, replace w-screen with w-full 尽可能时。许多水平溢出问题来自 100vw / w-screen,重复的安全区域填充或固定宽度容器 — 而不是视口元标签本身。
无线更新
设置 Capgo 推送更新而无需重新提交应用商店:
bunx @capgo/cli init
故障排除
构建失败时出现“找不到模块”
运行 bun install 并再试一次。
iOS:“找不到签名身份” 打开Xcode,转到签名和能力,选择您的开发团队。
Android: “SDK位置未找到”
创建 android/local.properties 与 sdk.dir=/path/to/android/sdk
设备上未显示的更改
确保你已经运行 bun run mobile 在live reload中,验证IP地址是否正确并且开发服务器正在运行
资源
- Capacitor 8 文档
- Next.js 15 文档
- Capgo - 实时更新
- @capgo/capacitor-native-navigation
- @capgo/capacitor-transitions
- @capgo/tailwind-capacitor
准备好将应用程序运输?了解Capgo如何帮助您更快地交付更新 — 注册免费帐户 今天。
继续 Build a Next.js Mobile App from Scratch with Capacitor 8
如果您正在使用 Build a Next.js Mobile App from Scratch with Capacitor 8 来规划CI/CD自动化,连接它与 Capgo CI/CD 为Capgo CI/CD中的产品工作流程 Capgo Native Builds 为Capgo Native Builds中的产品工作流程, Capgo 集成 为产品工作流程在 Capgo 集成中 CI/CD 集成 为 CI/CD 集成的实现细节 GitHub 动作集成 为 GitHub 动作集成的实现细节