跳过主要内容
教程

使用 Capacitor 8 从零开始构建一个 Next.js 移动应用

一步一步地创建一个新的 Next.js 15 项目,并使用 Capacitor 8 将其转换为原生 iOS 和 Android 移动应用。适合从零开始进行移动优先开发。

马丁·多纳迪厄

马丁·多纳迪厄

内容营销人员

使用 Capacitor 8 从零开始构建一个 Next.js 移动应用

简介

想从头开始构建一个使用 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

创建 iosandroid 包含本机项目的目录。

步骤 7: 构建和运行

构建您的项目并同步本机平台:

bun run mobile

在 iOS 模拟器中打开:

bun run mobile:ios

或 Android 模拟器:

bun run mobile:android

在 Xcode (iOS) 中:

  1. 从设备下拉菜单中选择一个模拟器
  2. 点击播放按钮或按 Cmd + R

在 Android Studio 中:

  1. 等待 Gradle 完成同步
  2. 从设备下拉菜单中选择一个模拟器
  3. 点击运行按钮或按 Shift + F10

步骤 8:设置实时重载

为了更快的开发,启用实时重载,修改后立即在设备上显示

  1. 找到你的本地 IP 地址:
# macOS
ipconfig getifaddr en0

# Windows
ipconfig
  1. 创建一个开发 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;
  1. 启动开发服务器并将配置复制到本机:
bun run dev &
NODE_ENV=development bunx cap copy
  1. 在 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.xcassetsandroid/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实现原生移动体验:

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.propertiessdk.dir=/path/to/android/sdk

设备上未显示的更改 确保你已经运行 bun run mobile 在live reload中,验证IP地址是否正确并且开发服务器正在运行

资源

准备好将应用程序运输?了解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 动作集成的实现细节

Capacitor 实时更新

当 web 层面的 bug 活跃时,通过 Capgo 将修复推送给用户,而不是等待几天的 app store 审核。用户在后台接收更新,而原生变化仍然在正常的审查路径中。

立即开始

博客最新文章

Capgo 为您提供了创建真正专业的移动应用所需的最佳见解。