React Native相机拍照报错:Failed to take picture - Preview is paused
解决React Native相机拍照时「Preview is paused」错误
这个报错我之前也碰到过,本质就是相机预览还没处于活跃状态就触发了拍照操作,下面给你几个靠谱的修复方案:
方案1:等相机完全初始化后再允许拍照
相机组件需要一点时间完成初始化和预览启动,我们可以加个状态跟踪相机是否准备就绪,只有就绪后才允许点击拍照:
首先更新你的state,新增相机就绪状态:
constructor(props) { super(props); this.state = { takingPic: false, cameraReady: false // 新增:标记相机是否初始化完成 }; }
然后给RNCamera添加onCameraReady回调,相机准备好时更新状态:
<RNCamera ref={ref => { this.camera = ref; }} captureAudio={false} style={{flex: 1}} type={RNCamera.Constants.Type.back} onCameraReady={() => this.setState({cameraReady: true})} // 相机就绪时触发 androidCameraPermissionOptions={{ title: 'Permission to use camera', message: 'We need your permission to use your camera', buttonPositive: 'Ok', buttonNegative: 'Cancel', }} > <Text onPress={this.takePicture}>Snap</Text> </RNCamera>
最后修改takePicture方法,增加相机就绪的检查:
takePicture = async () => { // 新增cameraReady的判断,确保相机已经准备好 if (this.camera && !this.state.takingPic && this.state.cameraReady) { let options = { quality: 0.85, base64: true, }; this.setState({takingPic: true}); try { const data = await this.camera.takePictureAsync(options); Alert.alert('Success', JSON.stringify(data)); } catch (err) { Alert.alert('Error', 'Failed to take picture: ' + (err.message || err)); return; } finally { this.setState({takingPic: false}); } } }
方案2:拍照前手动恢复预览(方案1无效时尝试)
如果偶尔还是出现预览暂停的情况,可以在拍照前主动调用resumePreview方法恢复预览:
takePicture = async () => { if (this.camera && !this.state.takingPic) { let options = { quality: 0.85, base64: true, }; this.setState({takingPic: true}); try { // 先强制恢复预览,再执行拍照 await this.camera.resumePreview(); const data = await this.camera.takePictureAsync(options); Alert.alert('Success', JSON.stringify(data)); } catch (err) { Alert.alert('Error', 'Failed to take picture: ' + (err.message || err)); return; } finally { this.setState({takingPic: false}); } } }
额外注意点
- 确保相机权限已经正常获取:虽然你配置了权限弹窗,但可以在组件挂载时主动检查权限状态,避免权限问题导致预览异常。
- 避免相机组件频繁重渲染:如果父组件的state频繁更新导致
RNCamera重新挂载,也可能触发预览暂停,可以用React.memo包裹相机组件做性能优化。
内容的提问来源于stack exchange,提问作者Moon Food




