基于 react-native-vision-camera 5.x 构建的 React Native 相机库,提供模态化相机界面,支持单拍 / 连拍 / 视频录制 / 捏合变焦 / 镜头切换 / 点击对焦。
yarn add @unif/react-native-camera \
react-native-vision-camera \
react-native-vision-camera-worklets \
react-native-nitro-modules \
react-native-nitro-image \
react-native-reanimated \
react-native-worklets \
react-native-reanimated-carousel \
react-native-gesture-handler \
react-native-safe-area-contextiOS 还需 pod install。
vision-camera 5.x 把 Frame Processor / 多线程能力拆到了同伴包 react-native-vision-camera-worklets,并在内部通过懒 require 引用它。即使本库不使用任何 Frame Processor,消费端打包器(Metro 等)在静态解析阶段仍会解析 vision-camera 内部那处 require——缺失该包会直接报错:打包期 Unable to resolve module react-native-vision-camera-worklets,或运行时 Cannot use Frame Processors - react-native-vision-camera-worklets is not installed。
因此它是必装的同伴包,版本与 react-native-vision-camera 对齐(同为 ^5.x)。vision-camera 自身未将其声明为 peer(视作可选),本库已在 peerDependencies 中显式声明,以提醒消费者一并安装。
iOS Info.plist:
<key>NSCameraUsageDescription</key><string>...</string>
<key>NSMicrophoneUsageDescription</key><string>...</string>
<key>NSPhotoLibraryAddUsageDescription</key><string>...</string>Android AndroidManifest.xml:
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<uses-permission android:name="android.permission.READ_MEDIA_IMAGES" />import { useCamera } from '@unif/react-native-camera';
const App = () => {
const [api, holder] = useCamera();
return (
<View>
<Text onPress={async () => {
const res = await api.open({
cameraMode: [
{ mode: 'single', quality: 0.9 },
{ mode: 'continuous' },
],
dataRetainedMode: 'clear',
});
// res.code: 200=ok, 0=cancelled, 403=no_permission, 404=no_device, 500=capture_failed, 503=video_failed
}}>打开相机</Text>
{holder}
</View>
);
};返回 [CameraApi, React.ReactElement],把 holder 渲染进 React 树,调用 api.open(config) 打开相机。
open(config: OpenConfig): Promise<CameraResult>close(): void
| 字段 | 类型 | 说明 |
|---|---|---|
cameraMode |
CameraMode[] |
至少一项;多项时底部出现模式 tab |
dataRetainedMode |
'clear' | 'retain' |
模式切换时是否保留已拍照片 |
| 字段 | 类型 | 默认 | 说明 |
|---|---|---|---|
type |
'back' | 'front' |
back |
初始前/后摄 |
flashMode |
'auto' | 'on' | 'off' |
— | 初始闪光(保留作兼容;闪光实际由相机内 UI 控制) |
mode |
'single' | 'continuous' | 'video' |
— | 拍摄模式 |
quality |
number |
0.9 |
JPEG 压缩 0~1 |
recTime |
number |
— | 录制时长上限(秒,video) |
| 字段 | 类型 | 说明 |
|---|---|---|
code |
0 | 200 | 403 | 404 | 500 | 503 |
状态码 |
data |
CustomPhotoFile[] |
拍摄的文件列表 |
message |
string |
描述信息 |
| 字段 | 类型 | 说明 |
|---|---|---|
id |
string |
唯一 id |
cameraType |
'back' | 'front' |
拍摄时的前/后摄 |
cameraMode |
'single' | 'continuous' | 'video' |
模式(原版字段名,= mode) |
path |
string |
本地文件路径 |
uri |
string |
文件 uri(file://) |
width |
number |
宽(px) |
height |
number |
高(px) |
mime |
'image/jpeg' | 'video/mp4' |
MIME 类型 |
mode |
'single' | 'continuous' | 'video' |
模式(2.x 字段名,= cameraMode) |
duration? |
number |
时长(秒,video) |
本库依赖 react-native-vision-camera 等 native 模块,jest 环境无法直接加载。消费者在测试里 mock 本库:
jest.mock('@unif/react-native-camera', () =>
require('@unif/react-native-camera/mock')
);mock 后 useCamera() 返回 [api, null],api.open / api.close 是 jest.fn,open 默认 resolve { code: 0, data: [], message: 'cancelled' }。按需覆盖单次返回:
const [api] = useCamera();
(api.open as jest.Mock).mockResolvedValueOnce({
code: 200,
data: [{ id: '1700000000000-0', cameraType: 'back', cameraMode: 'single', path: '/x.jpg', uri: 'file:///x.jpg', width: 1, height: 1, mime: 'image/jpeg', mode: 'single' }],
message: 'ok',
});工具函数(toFileUri / buildPhotoFile 等)与所有类型在 mock 中保留真实实现。
v2.0.0 的破坏性变更:
- 移除
cameraMode[i].photoResolution/videoResolution→ 改用quality(0~1)控制 JPEG 压缩 - 移除
watermark配置项(将在 v2.1.x 重新加入) - 从顶层
@unif/react-native-camera直接导入类型(不再走/lib/typescript/src/utilsdeep path)
MIT