@dariyd/react-native-document-scanner
v2.0.19
Published
React Native document scanner using VisionKit (iOS) and ML Kit (Android) with support for both old and new architecture
Maintainers
Keywords
Readme
react-native-document-scanner
Fast, native React Native document scanner for iOS and Android using Apple VisionKit (iOS) and Google ML Kit (Android). Features automatic document detection, edge/perspective correction, multi‑page scanning, configurable image quality, optional Base64, and support for the React Native New Architecture on ios and android.
- iOS: Uses VisionKit framework and VNDocumentCameraViewController
- Android: Uses ML Kit Document Scanner API
Preview
| iOS Demo | Android Demo |
|----------|--------------|
|
|
|
Used in Production Apps
| FileNest AI - Docs Organizer | MyGarage - CarDocs & History |
|:---:|:---:|
|
|
|
Features
- 📱 Cross-platform support (iOS 13+ and Android API 21+)
- 🚀 iOS & Android: Full support for new React Native architecture (Fabric/TurboModules)
- 📸 Automatic document detection and scanning
- 🖼️ Multi-page document scanning
- ⚙️ Configurable image quality
- 📦 Optional base64 encoding
- 🎯 Platform parity - same API for both platforms
Keywords: React Native document scanner, VisionKit document scanner, ML Kit document scanner, scan documents React Native, edge detection, perspective correction, multi‑page scanner
Installation
From npm (Recommended)
npm install @dariyd/react-native-document-scanneror with yarn:
yarn add @dariyd/react-native-document-scannerFrom GitHub (Latest Development)
npm install https://github.com/dariyd/react-native-document-scanner.gitor
yarn add https://github.com/dariyd/react-native-document-scanner.gitiOS Installation
cd ios && pod installAndroid Installation
No additional steps required. The ML Kit dependency will be automatically included.
Post-install Steps
iOS
Add the NSCameraUsageDescription key to your Info.plist:
<key>NSCameraUsageDescription</key>
<string>We need access to your camera to scan documents</string>Android
Add camera permission to your AndroidManifest.xml:
<uses-permission android:name="android.permission.CAMERA" />The module automatically requests camera permission when launching the scanner.
React Native New Architecture
This module requires React Native 0.77.3 or higher and supports the new architecture on iOS, while using the stable old architecture on Android.
iOS: Full support for Fabric and TurboModules - automatically detected and enabled when you enable new architecture in your project.
Android: Uses the stable bridge implementation for maximum compatibility. New architecture support is planned for a future release.
Requirements
- React Native 0.77.3 or higher
- React 18.2.0 or higher
- iOS 13.0 or higher
- Android:
- Minimum SDK: API 21 (Android 5.0)
- Target SDK: API 35 (Android 15) - required by Google Play Store
- Compile SDK: API 35
Enabling New Architecture
✅ iOS: Fully supported - Set RCT_NEW_ARCH_ENABLED=1 in your Podfile or build settings
✅ Android: Fully supported - Keep newArchEnabled=true in your gradle.properties
The iOS implementation will automatically use Fabric/TurboModules when enabled, while Android will continue to use the stable bridge implementation.
Usage
import { launchScanner } from 'react-native-document-scanner';
// Basic usage
const result = await launchScanner();
// With options
const result = await launchScanner({
quality: 0.8,
includeBase64: false,
});
// With EXIF metadata
const result = await launchScanner({
quality: 0.8,
includeExif: true,
});
console.log('EXIF:', result.images[0].exif);
// With EXIF + GPS location
const result = await launchScanner({
quality: 0.8,
includeExif: true,
includeLocationExif: true,
});
console.log('GPS:', result.images[0].exif?.GPSLatitude, result.images[0].exif?.GPSLongitude);
// With callback (optional)
launchScanner({ quality: 0.9 }, (result) => {
if (result.didCancel) {
console.log('User cancelled');
} else if (result.error) {
console.log('Error:', result.errorMessage);
} else {
console.log('Scanned images:', result.images);
}
});API Reference
Methods
import {launchScanner} from 'react-native-document-scanner';launchScanner()
Launch scanner to scan documents.
See Options for further information on options.
The callback will be called with a response object, refer to The Response Object.
Options
| Option | iOS | Android | Description | | ------------------- | --- | ------- | ----------------------------------------------------------------------------------------------------------------------------------------- | | quality | ✅ | ✅ | Number between 0 and 1 for image quality (default: 1). Lower values reduce file size | | includeBase64 | ✅ | ✅ | If true, creates base64 string of the image (Avoid using on large image files due to performance) | | includeExif | ✅ | ✅ | If true, embeds EXIF metadata (timestamps, device info, dimensions) in the image file and returns it in the response (default: false) | | includeLocationExif | ✅ | ✅ | If true, also embeds GPS coordinates in EXIF. Requires location permission — see Location Permission Setup (default: false) |
The Response Object
| key | iOS | Android | Description |
| ------------ | --- | ------- | ------------------------------------------------------------------- |
| didCancel | ✅ | ✅ | true if the user cancelled the process |
| error | ✅ | ✅ | true if error happens |
| errorMessage | ✅ | ✅ | Description of the error, use it for debug purpose only |
| images | ✅ | ✅ | Array of the selected media, refer to Image Object |
Image Object
| key | iOS | Android | Description | | --------- | --- | ------- | -------------------------------------------------- | | base64 | ✅ | ✅ | The base64 string of the image (if includeBase64 is true) | | uri | ✅ | ✅ | The file uri in app specific cache storage | | width | ✅ | ✅ | Image width in pixels | | height | ✅ | ✅ | Image height in pixels | | fileSize | ✅ | ✅ | The file size in bytes | | type | ✅ | ✅ | The file MIME type (e.g., "image/jpeg") | | fileName | ✅ | ✅ | The file name | | exif | ✅ | ✅ | EXIF metadata object (if includeExif is true). See EXIF Object |
EXIF Object
When includeExif is true, each image includes an exif object with the following fields:
| key | iOS | Android | Description |
| -------------------- | --- | ------- | ---------------------------------------------------- |
| DateTimeOriginal | ✅ | ✅ | Scan timestamp (format: yyyy:MM:dd HH:mm:ss) |
| DateTimeDigitized | ✅ | ✅ | Scan timestamp (format: yyyy:MM:dd HH:mm:ss) |
| PixelXDimension | ✅ | ✅ | Image width in pixels |
| PixelYDimension | ✅ | ✅ | Image height in pixels |
| ColorSpace | ✅ | ✅ | Color space (1 = sRGB) |
| Make | ✅ | ✅ | Device manufacturer |
| Model | ✅ | ✅ | Device model |
| Software | ✅ | ✅ | App name |
| GPSLatitude | ✅ | ✅ | Latitude (only if includeLocationExif is true and permission granted) |
| GPSLongitude | ✅ | ✅ | Longitude (only if includeLocationExif is true and permission granted) |
| GPSAltitude | ✅ | ✅ | Altitude in meters |
| GPSHorizontalAccuracy| ✅ | ✅ | GPS accuracy in meters |
| GPSDateTimeUTC | ✅ | ✅ | GPS fix timestamp in UTC |
Location Permission Setup
When using includeLocationExif: true, the package automatically requests location permission. However, you must add the required permission keys to your app:
iOS
Add to your Info.plist:
<key>NSLocationWhenInUseUsageDescription</key>
<string>Your location will be embedded in scanned documents for record-keeping.</string>Android
No additional setup required. The package declares ACCESS_FINE_LOCATION and ACCESS_COARSE_LOCATION permissions in its own manifest, which will be merged automatically.
Permission Behavior
- If permission has not been requested before, the system dialog will appear automatically
- If permission was already granted, no dialog is shown — location is fetched immediately
- If permission was denied, GPS fields are silently skipped and scanning proceeds normally
- Location permission is only requested when
includeLocationExif: true— it has no effect otherwise
Platform Differences
While both platforms provide similar functionality, there are some minor differences:
iOS
- Uses native VisionKit framework
- Requires iOS 13.0 or higher
- Supports PNG format for quality = 1.0, JPEG for quality < 1.0
Android
- Uses Google ML Kit Document Scanner
- Minimum SDK: API level 21 (Android 5.0)
- Target SDK: API level 35 (Android 15) - Google Play Store requirement
- Always outputs JPEG format
- Requires Google Play Services
Troubleshooting
Android: ML Kit not available
If you encounter issues with ML Kit on Android, ensure that:
- Google Play Services is installed on the device/emulator
- Your
compileSdkVersionis 35 or higher - Your
targetSdkVersionis 35 (required by Google Play Store) - Your
minSdkVersionis 21 or higher
iOS: Camera permission denied
Ensure you've added the NSCameraUsageDescription key to your Info.plist.
Example
Check the example/ directory for a complete example app demonstrating the scanner.
Inspired By
- iOS implementation: react-native-image-picker
- Android ML Kit: Google ML Kit Document Scanner
License
MIT
