mirror of
https://github.com/zoriya/react-native-background-downloader.git
synced 2025-12-06 06:56:10 +00:00
first commit
This commit is contained in:
1
.gitattributes
vendored
Normal file
1
.gitattributes
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
*.pbxproj -text
|
||||||
46
.gitignore
vendored
Normal file
46
.gitignore
vendored
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
|
||||||
|
# OSX
|
||||||
|
#
|
||||||
|
.DS_Store
|
||||||
|
|
||||||
|
# node.js
|
||||||
|
#
|
||||||
|
node_modules/
|
||||||
|
npm-debug.log
|
||||||
|
yarn-error.log
|
||||||
|
|
||||||
|
|
||||||
|
# Xcode
|
||||||
|
#
|
||||||
|
build/
|
||||||
|
*.pbxuser
|
||||||
|
!default.pbxuser
|
||||||
|
*.mode1v3
|
||||||
|
!default.mode1v3
|
||||||
|
*.mode2v3
|
||||||
|
!default.mode2v3
|
||||||
|
*.perspectivev3
|
||||||
|
!default.perspectivev3
|
||||||
|
xcuserdata
|
||||||
|
*.xccheckout
|
||||||
|
*.moved-aside
|
||||||
|
DerivedData
|
||||||
|
*.hmap
|
||||||
|
*.ipa
|
||||||
|
*.xcuserstate
|
||||||
|
project.xcworkspace
|
||||||
|
|
||||||
|
|
||||||
|
# Android/IntelliJ
|
||||||
|
#
|
||||||
|
build/
|
||||||
|
.idea
|
||||||
|
.gradle
|
||||||
|
local.properties
|
||||||
|
*.iml
|
||||||
|
|
||||||
|
# BUCK
|
||||||
|
buck-out/
|
||||||
|
\.buckd/
|
||||||
|
*.keystore
|
||||||
|
|
||||||
45
README.md
Normal file
45
README.md
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
|
||||||
|
# react-native-background-download
|
||||||
|
|
||||||
|
## Getting started
|
||||||
|
|
||||||
|
`$ npm install react-native-background-download --save`
|
||||||
|
|
||||||
|
### Mostly automatic installation
|
||||||
|
|
||||||
|
`$ react-native link react-native-background-download`
|
||||||
|
|
||||||
|
### Manual installation
|
||||||
|
|
||||||
|
|
||||||
|
#### iOS
|
||||||
|
|
||||||
|
1. In XCode, in the project navigator, right click `Libraries` ➜ `Add Files to [your project's name]`
|
||||||
|
2. Go to `node_modules` ➜ `react-native-background-download` and add `RNBackgroundDownload.xcodeproj`
|
||||||
|
3. In XCode, in the project navigator, select your project. Add `libRNBackgroundDownload.a` to your project's `Build Phases` ➜ `Link Binary With Libraries`
|
||||||
|
4. Run your project (`Cmd+R`)<
|
||||||
|
|
||||||
|
#### Android
|
||||||
|
|
||||||
|
1. Open up `android/app/src/main/java/[...]/MainActivity.java`
|
||||||
|
- Add `import com.eko.RNBackgroundDownloadPackage;` to the imports at the top of the file
|
||||||
|
- Add `new RNBackgroundDownloadPackage()` to the list returned by the `getPackages()` method
|
||||||
|
2. Append the following lines to `android/settings.gradle`:
|
||||||
|
```
|
||||||
|
include ':react-native-background-download'
|
||||||
|
project(':react-native-background-download').projectDir = new File(rootProject.projectDir, '../node_modules/react-native-background-download/android')
|
||||||
|
```
|
||||||
|
3. Insert the following lines inside the dependencies block in `android/app/build.gradle`:
|
||||||
|
```
|
||||||
|
compile project(':react-native-background-download')
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
```javascript
|
||||||
|
import RNBackgroundDownload from 'react-native-background-download';
|
||||||
|
|
||||||
|
// TODO: What to do with the module?
|
||||||
|
RNBackgroundDownload;
|
||||||
|
```
|
||||||
|
|
||||||
37
android/build.gradle
Normal file
37
android/build.gradle
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
|
||||||
|
buildscript {
|
||||||
|
repositories {
|
||||||
|
jcenter()
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
classpath 'com.android.tools.build:gradle:1.3.1'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
apply plugin: 'com.android.library'
|
||||||
|
|
||||||
|
android {
|
||||||
|
compileSdkVersion 23
|
||||||
|
buildToolsVersion "23.0.1"
|
||||||
|
|
||||||
|
defaultConfig {
|
||||||
|
minSdkVersion 16
|
||||||
|
targetSdkVersion 22
|
||||||
|
versionCode 1
|
||||||
|
versionName "1.0"
|
||||||
|
}
|
||||||
|
lintOptions {
|
||||||
|
abortOnError false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
mavenCentral()
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
compile "com.tonyodev.fetch2:fetch2:2.0.0-RC12"
|
||||||
|
compile 'com.facebook.react:react-native:+'
|
||||||
|
}
|
||||||
|
|
||||||
BIN
android/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
BIN
android/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
Binary file not shown.
6
android/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
6
android/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
#Tue Apr 24 11:44:55 IDT 2018
|
||||||
|
distributionBase=GRADLE_USER_HOME
|
||||||
|
distributionPath=wrapper/dists
|
||||||
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
|
zipStorePath=wrapper/dists
|
||||||
|
distributionUrl=https\://services.gradle.org/distributions/gradle-2.2-all.zip
|
||||||
172
android/gradlew
vendored
Normal file
172
android/gradlew
vendored
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
#!/usr/bin/env sh
|
||||||
|
|
||||||
|
##############################################################################
|
||||||
|
##
|
||||||
|
## Gradle start up script for UN*X
|
||||||
|
##
|
||||||
|
##############################################################################
|
||||||
|
|
||||||
|
# Attempt to set APP_HOME
|
||||||
|
# Resolve links: $0 may be a link
|
||||||
|
PRG="$0"
|
||||||
|
# Need this for relative symlinks.
|
||||||
|
while [ -h "$PRG" ] ; do
|
||||||
|
ls=`ls -ld "$PRG"`
|
||||||
|
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||||
|
if expr "$link" : '/.*' > /dev/null; then
|
||||||
|
PRG="$link"
|
||||||
|
else
|
||||||
|
PRG=`dirname "$PRG"`"/$link"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
SAVED="`pwd`"
|
||||||
|
cd "`dirname \"$PRG\"`/" >/dev/null
|
||||||
|
APP_HOME="`pwd -P`"
|
||||||
|
cd "$SAVED" >/dev/null
|
||||||
|
|
||||||
|
APP_NAME="Gradle"
|
||||||
|
APP_BASE_NAME=`basename "$0"`
|
||||||
|
|
||||||
|
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||||
|
DEFAULT_JVM_OPTS=""
|
||||||
|
|
||||||
|
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||||
|
MAX_FD="maximum"
|
||||||
|
|
||||||
|
warn () {
|
||||||
|
echo "$*"
|
||||||
|
}
|
||||||
|
|
||||||
|
die () {
|
||||||
|
echo
|
||||||
|
echo "$*"
|
||||||
|
echo
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
# OS specific support (must be 'true' or 'false').
|
||||||
|
cygwin=false
|
||||||
|
msys=false
|
||||||
|
darwin=false
|
||||||
|
nonstop=false
|
||||||
|
case "`uname`" in
|
||||||
|
CYGWIN* )
|
||||||
|
cygwin=true
|
||||||
|
;;
|
||||||
|
Darwin* )
|
||||||
|
darwin=true
|
||||||
|
;;
|
||||||
|
MINGW* )
|
||||||
|
msys=true
|
||||||
|
;;
|
||||||
|
NONSTOP* )
|
||||||
|
nonstop=true
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
||||||
|
|
||||||
|
# Determine the Java command to use to start the JVM.
|
||||||
|
if [ -n "$JAVA_HOME" ] ; then
|
||||||
|
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||||
|
# IBM's JDK on AIX uses strange locations for the executables
|
||||||
|
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||||
|
else
|
||||||
|
JAVACMD="$JAVA_HOME/bin/java"
|
||||||
|
fi
|
||||||
|
if [ ! -x "$JAVACMD" ] ; then
|
||||||
|
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
location of your Java installation."
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
JAVACMD="java"
|
||||||
|
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
location of your Java installation."
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Increase the maximum file descriptors if we can.
|
||||||
|
if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
|
||||||
|
MAX_FD_LIMIT=`ulimit -H -n`
|
||||||
|
if [ $? -eq 0 ] ; then
|
||||||
|
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
|
||||||
|
MAX_FD="$MAX_FD_LIMIT"
|
||||||
|
fi
|
||||||
|
ulimit -n $MAX_FD
|
||||||
|
if [ $? -ne 0 ] ; then
|
||||||
|
warn "Could not set maximum file descriptor limit: $MAX_FD"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Darwin, add options to specify how the application appears in the dock
|
||||||
|
if $darwin; then
|
||||||
|
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Cygwin, switch paths to Windows format before running java
|
||||||
|
if $cygwin ; then
|
||||||
|
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
|
||||||
|
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
|
||||||
|
JAVACMD=`cygpath --unix "$JAVACMD"`
|
||||||
|
|
||||||
|
# We build the pattern for arguments to be converted via cygpath
|
||||||
|
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
|
||||||
|
SEP=""
|
||||||
|
for dir in $ROOTDIRSRAW ; do
|
||||||
|
ROOTDIRS="$ROOTDIRS$SEP$dir"
|
||||||
|
SEP="|"
|
||||||
|
done
|
||||||
|
OURCYGPATTERN="(^($ROOTDIRS))"
|
||||||
|
# Add a user-defined pattern to the cygpath arguments
|
||||||
|
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
|
||||||
|
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
|
||||||
|
fi
|
||||||
|
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
||||||
|
i=0
|
||||||
|
for arg in "$@" ; do
|
||||||
|
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
|
||||||
|
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
|
||||||
|
|
||||||
|
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
|
||||||
|
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
|
||||||
|
else
|
||||||
|
eval `echo args$i`="\"$arg\""
|
||||||
|
fi
|
||||||
|
i=$((i+1))
|
||||||
|
done
|
||||||
|
case $i in
|
||||||
|
(0) set -- ;;
|
||||||
|
(1) set -- "$args0" ;;
|
||||||
|
(2) set -- "$args0" "$args1" ;;
|
||||||
|
(3) set -- "$args0" "$args1" "$args2" ;;
|
||||||
|
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
|
||||||
|
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
|
||||||
|
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
|
||||||
|
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
|
||||||
|
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
|
||||||
|
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
|
||||||
|
esac
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Escape application args
|
||||||
|
save () {
|
||||||
|
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
|
||||||
|
echo " "
|
||||||
|
}
|
||||||
|
APP_ARGS=$(save "$@")
|
||||||
|
|
||||||
|
# Collect all arguments for the java command, following the shell quoting and substitution rules
|
||||||
|
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
|
||||||
|
|
||||||
|
# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong
|
||||||
|
if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then
|
||||||
|
cd "$(dirname "$0")"
|
||||||
|
fi
|
||||||
|
|
||||||
|
exec "$JAVACMD" "$@"
|
||||||
84
android/gradlew.bat
vendored
Normal file
84
android/gradlew.bat
vendored
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
@if "%DEBUG%" == "" @echo off
|
||||||
|
@rem ##########################################################################
|
||||||
|
@rem
|
||||||
|
@rem Gradle startup script for Windows
|
||||||
|
@rem
|
||||||
|
@rem ##########################################################################
|
||||||
|
|
||||||
|
@rem Set local scope for the variables with windows NT shell
|
||||||
|
if "%OS%"=="Windows_NT" setlocal
|
||||||
|
|
||||||
|
set DIRNAME=%~dp0
|
||||||
|
if "%DIRNAME%" == "" set DIRNAME=.
|
||||||
|
set APP_BASE_NAME=%~n0
|
||||||
|
set APP_HOME=%DIRNAME%
|
||||||
|
|
||||||
|
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||||
|
set DEFAULT_JVM_OPTS=
|
||||||
|
|
||||||
|
@rem Find java.exe
|
||||||
|
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||||
|
|
||||||
|
set JAVA_EXE=java.exe
|
||||||
|
%JAVA_EXE% -version >NUL 2>&1
|
||||||
|
if "%ERRORLEVEL%" == "0" goto init
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||||
|
echo.
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
echo location of your Java installation.
|
||||||
|
|
||||||
|
goto fail
|
||||||
|
|
||||||
|
:findJavaFromJavaHome
|
||||||
|
set JAVA_HOME=%JAVA_HOME:"=%
|
||||||
|
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
||||||
|
|
||||||
|
if exist "%JAVA_EXE%" goto init
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
||||||
|
echo.
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
echo location of your Java installation.
|
||||||
|
|
||||||
|
goto fail
|
||||||
|
|
||||||
|
:init
|
||||||
|
@rem Get command-line arguments, handling Windows variants
|
||||||
|
|
||||||
|
if not "%OS%" == "Windows_NT" goto win9xME_args
|
||||||
|
|
||||||
|
:win9xME_args
|
||||||
|
@rem Slurp the command line arguments.
|
||||||
|
set CMD_LINE_ARGS=
|
||||||
|
set _SKIP=2
|
||||||
|
|
||||||
|
:win9xME_args_slurp
|
||||||
|
if "x%~1" == "x" goto execute
|
||||||
|
|
||||||
|
set CMD_LINE_ARGS=%*
|
||||||
|
|
||||||
|
:execute
|
||||||
|
@rem Setup the command line
|
||||||
|
|
||||||
|
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
||||||
|
|
||||||
|
@rem Execute Gradle
|
||||||
|
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
|
||||||
|
|
||||||
|
:end
|
||||||
|
@rem End local scope for the variables with windows NT shell
|
||||||
|
if "%ERRORLEVEL%"=="0" goto mainEnd
|
||||||
|
|
||||||
|
:fail
|
||||||
|
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||||
|
rem the _cmd.exe /c_ return code!
|
||||||
|
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
||||||
|
exit /b 1
|
||||||
|
|
||||||
|
:mainEnd
|
||||||
|
if "%OS%"=="Windows_NT" endlocal
|
||||||
|
|
||||||
|
:omega
|
||||||
6
android/src/main/AndroidManifest.xml
Normal file
6
android/src/main/AndroidManifest.xml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
|
||||||
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
package="com.eko">
|
||||||
|
|
||||||
|
</manifest>
|
||||||
|
|
||||||
309
android/src/main/java/com/eko/RNBackgroundDownloadModule.java
Normal file
309
android/src/main/java/com/eko/RNBackgroundDownloadModule.java
Normal file
@@ -0,0 +1,309 @@
|
|||||||
|
|
||||||
|
package com.eko;
|
||||||
|
|
||||||
|
import android.annotation.SuppressLint;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.SharedPreferences;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import com.ekoapp.BuildConfig;
|
||||||
|
import com.facebook.react.bridge.Arguments;
|
||||||
|
import com.facebook.react.bridge.Promise;
|
||||||
|
import com.facebook.react.bridge.ReactApplicationContext;
|
||||||
|
import com.facebook.react.bridge.ReactContextBaseJavaModule;
|
||||||
|
import com.facebook.react.bridge.ReactMethod;
|
||||||
|
import com.facebook.react.bridge.ReadableMap;
|
||||||
|
import com.facebook.react.bridge.WritableArray;
|
||||||
|
import com.facebook.react.bridge.WritableMap;
|
||||||
|
import com.facebook.react.modules.core.DeviceEventManagerModule;
|
||||||
|
import com.tonyodev.fetch2.Download;
|
||||||
|
import com.tonyodev.fetch2.Error;
|
||||||
|
import com.tonyodev.fetch2.Fetch;
|
||||||
|
import com.tonyodev.fetch2.FetchListener;
|
||||||
|
import com.tonyodev.fetch2.Func;
|
||||||
|
import com.tonyodev.fetch2.NetworkType;
|
||||||
|
import com.tonyodev.fetch2.Priority;
|
||||||
|
import com.tonyodev.fetch2.Request;
|
||||||
|
import com.tonyodev.fetch2.Status;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.FileOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.ObjectInputStream;
|
||||||
|
import java.io.ObjectOutputStream;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class RNBackgroundDownloadModule extends ReactContextBaseJavaModule {
|
||||||
|
|
||||||
|
private static final int TASK_RUNNING = 0;
|
||||||
|
private static final int TASK_SUSPENDED = 1;
|
||||||
|
private static final int TASK_CANCELING = 2;
|
||||||
|
private static final int TASK_COMPLETED = 3;
|
||||||
|
|
||||||
|
private static Map<Status, Integer> stateMap = new HashMap<Status, Integer>() {{
|
||||||
|
put(Status.DOWNLOADING, TASK_RUNNING);
|
||||||
|
put(Status.COMPLETED, TASK_COMPLETED);
|
||||||
|
put(Status.PAUSED, TASK_SUSPENDED);
|
||||||
|
put(Status.QUEUED, TASK_RUNNING);
|
||||||
|
put(Status.CANCELLED, TASK_CANCELING);
|
||||||
|
put(Status.FAILED, TASK_CANCELING);
|
||||||
|
put(Status.REMOVED, TASK_CANCELING);
|
||||||
|
put(Status.DELETED, TASK_CANCELING);
|
||||||
|
put(Status.NONE, TASK_CANCELING);
|
||||||
|
}};
|
||||||
|
|
||||||
|
private Fetch fetch;
|
||||||
|
private Map<String, Integer> idToRequestId = new HashMap<>();
|
||||||
|
@SuppressLint("UseSparseArrays")
|
||||||
|
private Map<Integer, TaskConfig> requestIdToConfig = new HashMap<>();
|
||||||
|
private DeviceEventManagerModule.RCTDeviceEventEmitter ee;
|
||||||
|
private Date lastProgressReport = new Date();
|
||||||
|
private WritableArray progressReports = Arguments.createArray();
|
||||||
|
|
||||||
|
public RNFBGDModule(ReactApplicationContext reactContext) {
|
||||||
|
super(reactContext);
|
||||||
|
|
||||||
|
loadConfigMap();
|
||||||
|
fetch = new Fetch.Builder(this.getReactApplicationContext(), "RNBackgroundDownload")
|
||||||
|
.setDownloadConcurrentLimit(4)
|
||||||
|
.build();
|
||||||
|
fetch.addListener(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCatalystInstanceDestroy() {
|
||||||
|
fetch.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() {
|
||||||
|
return "RNBackgroundDownload";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void initialize() {
|
||||||
|
ee = getReactApplicationContext().getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasConstants() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
@Override
|
||||||
|
public Map<String, Object> getConstants() {
|
||||||
|
Map<String, Object> constants = new HashMap<>();
|
||||||
|
constants.put("documents", this.getReactApplicationContext().getFilesDir().getAbsolutePath());
|
||||||
|
constants.put("TaskRunning", TASK_RUNNING);
|
||||||
|
constants.put("TaskSuspended", TASK_SUSPENDED);
|
||||||
|
constants.put("TaskCanceling", TASK_CANCELING);
|
||||||
|
constants.put("TaskCompleted", TASK_COMPLETED);
|
||||||
|
constants.put("PriorityHigh", Priority.HIGH.getValue());
|
||||||
|
constants.put("PriorityNormal", Priority.NORMAL.getValue());
|
||||||
|
constants.put("PriorityLow", Priority.LOW.getValue());
|
||||||
|
constants.put("OnlyWifi", NetworkType.WIFI_ONLY.getValue());
|
||||||
|
constants.put("AllNetworks", NetworkType.ALL.getValue());
|
||||||
|
return constants;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void removeFromMaps(int requestId) {
|
||||||
|
TaskConfig config = requestIdToConfig.get(requestId);
|
||||||
|
if (config != null) {
|
||||||
|
idToRequestId.remove(config.id);
|
||||||
|
requestIdToConfig.remove(requestId);
|
||||||
|
|
||||||
|
saveConfigMap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void saveConfigMap() {
|
||||||
|
File file = new File(this.getReactApplicationContext().getFilesDir(), "RNFileBackgroundDownload_configMap");
|
||||||
|
try {
|
||||||
|
ObjectOutputStream outputStream = new ObjectOutputStream(new FileOutputStream(file));
|
||||||
|
outputStream.writeObject(requestIdToConfig);
|
||||||
|
outputStream.flush();
|
||||||
|
outputStream.close();
|
||||||
|
} catch (IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loadConfigMap() {
|
||||||
|
File file = new File(this.getReactApplicationContext().getFilesDir(), "RNFileBackgroundDownload_configMap");
|
||||||
|
try {
|
||||||
|
ObjectInputStream inputStream = new ObjectInputStream(new FileInputStream(file));
|
||||||
|
requestIdToConfig = (Map<Integer, TaskConfig>) inputStream.readObject();
|
||||||
|
} catch (IOException | ClassNotFoundException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// JS Methods
|
||||||
|
@ReactMethod
|
||||||
|
public void download(ReadableMap options) {
|
||||||
|
String id = options.getString("id");
|
||||||
|
String url = options.getString("url");
|
||||||
|
String destination = options.getString("destination");
|
||||||
|
|
||||||
|
if (id == null || url == null || destination == null) {
|
||||||
|
Log.e(getName(), "id, url and destination must be set");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
TaskConfig config = new TaskConfig(id);
|
||||||
|
Request request = new Request(url, destination);
|
||||||
|
request.setPriority(options.hasKey("priority") ? Priority.valueOf(options.getInt("priority")) : Priority.NORMAL);
|
||||||
|
request.setNetworkType(options.hasKey("network") ? NetworkType.valueOf(options.getInt("network")) : NetworkType.ALL);
|
||||||
|
fetch.enqueue(request, null, null);
|
||||||
|
|
||||||
|
idToRequestId.put(id, request.getId());
|
||||||
|
requestIdToConfig.put(request.getId(), config);
|
||||||
|
saveConfigMap();
|
||||||
|
}
|
||||||
|
|
||||||
|
@ReactMethod
|
||||||
|
public void puaseTask(String identifier) {
|
||||||
|
Integer requestId = idToRequestId.get(identifier);
|
||||||
|
if (requestId != null) {
|
||||||
|
fetch.pause(requestId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@ReactMethod
|
||||||
|
public void resumeTask(String identifier) {
|
||||||
|
Integer requestId = idToRequestId.get(identifier);
|
||||||
|
if (requestId != null) {
|
||||||
|
fetch.resume(requestId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@ReactMethod
|
||||||
|
public void stopTask(String identifier) {
|
||||||
|
Integer requestId = idToRequestId.get(identifier);
|
||||||
|
if (requestId != null) {
|
||||||
|
fetch.cancel(requestId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@ReactMethod
|
||||||
|
public void checkForExistingDownloads(final Promise promise) {
|
||||||
|
fetch.getDownloads(new Func<List<? extends Download>>() {
|
||||||
|
@Override
|
||||||
|
public void call(List<? extends Download> downloads) {
|
||||||
|
WritableArray foundIds = Arguments.createArray();
|
||||||
|
|
||||||
|
for (Download download : downloads) {
|
||||||
|
if (requestIdToConfig.containsKey(download.getId())) {
|
||||||
|
TaskConfig config = requestIdToConfig.get(download.getId());
|
||||||
|
WritableMap params = Arguments.createMap();
|
||||||
|
params.putString("id", config.id);
|
||||||
|
params.putInt("state", stateMap.get(download.getStatus()));
|
||||||
|
params.putInt("bytesWritten", (int)download.getDownloaded());
|
||||||
|
params.putInt("totalBytes", (int)download.getTotal());
|
||||||
|
params.putDouble("percent", ((double)download.getProgress()) / 100);
|
||||||
|
|
||||||
|
foundIds.pushMap(params);
|
||||||
|
|
||||||
|
idToRequestId.put(config.id, download.getId());
|
||||||
|
config.reportedBegin = true;
|
||||||
|
} else {
|
||||||
|
fetch.delete(download.getId());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
promise.resolve(foundIds);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch API
|
||||||
|
@Override
|
||||||
|
public void onQueued(Download download) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCompleted(Download download) {
|
||||||
|
WritableMap params = Arguments.createMap();
|
||||||
|
params.putString("id", requestIdToConfig.get(download.getId()).id);
|
||||||
|
ee.emit("downloadComplete", params);
|
||||||
|
|
||||||
|
removeFromMaps(download.getId());
|
||||||
|
fetch.remove(download.getId());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onError(Download download) {
|
||||||
|
Error error = download.getError();
|
||||||
|
Throwable throwable = error.getThrowable();
|
||||||
|
|
||||||
|
Log.e(getName(), error.toString());
|
||||||
|
WritableMap params = Arguments.createMap();
|
||||||
|
params.putString("id", requestIdToConfig.get(download.getId()).id);
|
||||||
|
if (error == Error.UNKNOWN && throwable != null) {
|
||||||
|
params.putString("error", throwable.getLocalizedMessage());
|
||||||
|
} else {
|
||||||
|
params.putString("error", error.toString());
|
||||||
|
}
|
||||||
|
ee.emit("downloadFailed", params);
|
||||||
|
|
||||||
|
removeFromMaps(download.getId());
|
||||||
|
fetch.remove(download.getId());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onProgress(Download download, long l, long l1) {
|
||||||
|
TaskConfig config = requestIdToConfig.get(download.getId());
|
||||||
|
WritableMap params = Arguments.createMap();
|
||||||
|
params.putString("id", config.id);
|
||||||
|
|
||||||
|
if (!config.reportedBegin) {
|
||||||
|
params.putInt("expctedBytes", (int)download.getTotal());
|
||||||
|
ee.emit("downloadBegin", params);
|
||||||
|
config.reportedBegin = true;
|
||||||
|
} else {
|
||||||
|
params.putInt("written", (int)download.getDownloaded());
|
||||||
|
params.putInt("total", (int)download.getTotal());
|
||||||
|
params.putDouble("percent", ((double)download.getProgress()) / 100);
|
||||||
|
progressReports.pushMap(params);
|
||||||
|
Date now = new Date();
|
||||||
|
if (now.getTime() - lastProgressReport.getTime() > 1500) {
|
||||||
|
ee.emit("downloadProgress", progressReports);
|
||||||
|
lastProgressReport = now;
|
||||||
|
progressReports = Arguments.createArray();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onPaused(Download download) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onResumed(Download download) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCancelled(Download download) {
|
||||||
|
removeFromMaps(download.getId());
|
||||||
|
fetch.delete(download.getId());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRemoved(Download download) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onDeleted(Download download) {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
|
||||||
|
package com.eko;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import com.facebook.react.ReactPackage;
|
||||||
|
import com.facebook.react.bridge.NativeModule;
|
||||||
|
import com.facebook.react.bridge.ReactApplicationContext;
|
||||||
|
import com.facebook.react.uimanager.ViewManager;
|
||||||
|
import com.facebook.react.bridge.JavaScriptModule;
|
||||||
|
public class RNBackgroundDownloadPackage implements ReactPackage {
|
||||||
|
@Override
|
||||||
|
public List<NativeModule> createNativeModules(ReactApplicationContext reactContext) {
|
||||||
|
return Arrays.<NativeModule>asList(new RNBackgroundDownloadModule(reactContext));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<Class<? extends JavaScriptModule>> createJSModules() {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<ViewManager> createViewManagers(ReactApplicationContext reactContext) {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
}
|
||||||
90
index.js
Normal file
90
index.js
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
import { NativeModules, NativeEventEmitter } from 'react-native';
|
||||||
|
const { RNBackgroundDownload } = NativeModules;
|
||||||
|
const RNBackgroundDownloadEmitter = new NativeEventEmitter(RNBackgroundDownload);
|
||||||
|
import DownloadTask from './lib/downloadTask';
|
||||||
|
|
||||||
|
const tasksMap = new Map();
|
||||||
|
|
||||||
|
RNBackgroundDownloadEmitter.addListener('downloadProgress', events => {
|
||||||
|
for (let event of events) {
|
||||||
|
let task = tasksMap.get(event.id);
|
||||||
|
task && task._onProgress(event.percent, event.written, event.total);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
RNBackgroundDownloadEmitter.addListener('downloadComplete', event => {
|
||||||
|
let task = tasksMap.get(event.id);
|
||||||
|
task && task._onDone(event.location);
|
||||||
|
tasksMap.delete(event.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
RNBackgroundDownloadEmitter.addListener('downloadFailed', event => {
|
||||||
|
let task = tasksMap.get(event.id);
|
||||||
|
task && task._onError(event.error);
|
||||||
|
tasksMap.delete(event.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
RNBackgroundDownloadEmitter.addListener('downloadBegin', event => {
|
||||||
|
console.log('GOT downloadBegin', event);
|
||||||
|
let task = tasksMap.get(event.id);
|
||||||
|
task && task._onBegin(event.expctedBytes);
|
||||||
|
});
|
||||||
|
|
||||||
|
export function checkForExistingDownloads() {
|
||||||
|
return RNBackgroundDownload.checkForExistingDownloads()
|
||||||
|
.then(foundTasks => {
|
||||||
|
console.log('Fond lost downloads!!: ', foundTasks);
|
||||||
|
return foundTasks.map(taskInfo => {
|
||||||
|
let task = new DownloadTask(taskInfo);
|
||||||
|
if (taskInfo.state === RNBackgroundDownload.TaskRunning) {
|
||||||
|
task.state = 'DOWNLOADING';
|
||||||
|
} else if (taskInfo.state === RNBackgroundDownload.TaskSuspended) {
|
||||||
|
task.state = 'PAUSED';
|
||||||
|
} else if (taskInfo.state === RNBackgroundDownload.TaskCanceling) {
|
||||||
|
task.stop();
|
||||||
|
return null;
|
||||||
|
} else if (taskInfo.state === RNBackgroundDownload.TaskCompleted) {
|
||||||
|
if (taskInfo.bytesWritten === taskInfo.totalBytes) {
|
||||||
|
task.state = 'DONE';
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tasksMap.set(taskInfo.id, task);
|
||||||
|
return task;
|
||||||
|
}).filter(task => task !== null);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function download(options) {
|
||||||
|
if (!options.id || !options.url || !options.destination) {
|
||||||
|
throw new Error('[RNBackgroundDownload] id, url and destination are required');
|
||||||
|
}
|
||||||
|
RNBackgroundDownload.download(options);
|
||||||
|
let task = new DownloadTask(options.id);
|
||||||
|
tasksMap.set(options.id, task);
|
||||||
|
return task;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const directories = {
|
||||||
|
documents: RNBackgroundDownload.documents
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Network = {
|
||||||
|
WIFI_ONLY: RNBackgroundDownload.OnlyWifi,
|
||||||
|
ALL: RNBackgroundDownload.AllNetworks
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Priority = {
|
||||||
|
HIGH: RNBackgroundDownload.PriorityHigh,
|
||||||
|
MEDIUM: RNBackgroundDownload.PriorityNormal,
|
||||||
|
LOW: RNBackgroundDownload.PriorityLow
|
||||||
|
};
|
||||||
|
|
||||||
|
export default {
|
||||||
|
download,
|
||||||
|
checkForExistingDownloads,
|
||||||
|
directories,
|
||||||
|
Network,
|
||||||
|
Priority
|
||||||
|
};
|
||||||
24
ios/RNBackgroundDownload.h
Normal file
24
ios/RNBackgroundDownload.h
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
//
|
||||||
|
// RNFileBackgroundDownload.h
|
||||||
|
// EkoApp
|
||||||
|
//
|
||||||
|
// Created by Elad Gil on 20/11/2017.
|
||||||
|
// Copyright © 2017 Eko. All rights reserved.
|
||||||
|
//
|
||||||
|
//
|
||||||
|
#import <Foundation/Foundation.h>
|
||||||
|
#if __has_include(<React/RCTBridgeModule.h>)
|
||||||
|
#import <React/RCTBridgeModule.h>
|
||||||
|
#import <React/RCTEventEmitter.h>
|
||||||
|
#elif __has_include("RCTBridgeModule.h")
|
||||||
|
#import "RCTBridgeModule.h"
|
||||||
|
#import "RCTEventEmitter.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef void (^CompletionHandler)();
|
||||||
|
|
||||||
|
@interface RNFileBackgroundDownload : RCTEventEmitter <RCTBridgeModule, NSURLSessionDelegate, NSURLSessionDownloadDelegate>
|
||||||
|
|
||||||
|
+ (void)setCompletionHandlerWithIdentifier: (NSString *)identifier completionHandler: (CompletionHandler)completionHandler;
|
||||||
|
|
||||||
|
@end
|
||||||
271
ios/RNBackgroundDownload.m
Normal file
271
ios/RNBackgroundDownload.m
Normal file
@@ -0,0 +1,271 @@
|
|||||||
|
//
|
||||||
|
// RNFileBackgroundDownload.m
|
||||||
|
// EkoApp
|
||||||
|
//
|
||||||
|
// Created by Elad Gil on 20/11/2017.
|
||||||
|
// Copyright © 2017 Eko. All rights reserved.
|
||||||
|
//
|
||||||
|
//
|
||||||
|
#import "RNBackgroundDownload.h"
|
||||||
|
#import "TaskConfig.h"
|
||||||
|
|
||||||
|
#define URL_TO_CONFIG_MAP_KEY @"com.eko.bgdownloadmap"
|
||||||
|
|
||||||
|
static CompletionHandler storedCompletionHandler;
|
||||||
|
|
||||||
|
@implementation RNBackgroundDownload {
|
||||||
|
NSURLSession *urlSession;
|
||||||
|
NSURLSessionConfiguration *sessionConfig;
|
||||||
|
NSMutableDictionary<NSString *, TaskConfig *> *urlToConfigMap;
|
||||||
|
NSMutableDictionary<NSURLSessionTask *, TaskConfig *> *taskToConfigMap;
|
||||||
|
NSMutableDictionary<NSString *, NSURLSessionDownloadTask *> *idToTaskMap;
|
||||||
|
NSMutableDictionary<NSString *, NSData *> *idToResumeDataMap;
|
||||||
|
NSMutableDictionary<NSString *, NSNumber *> *idToPercentMap;
|
||||||
|
NSOperationQueue *downloadOperationsQueue;
|
||||||
|
NSDate *lastProgressReport;
|
||||||
|
NSMutableArray<NSDictionary *> *progressReports;
|
||||||
|
}
|
||||||
|
|
||||||
|
RCT_EXPORT_MODULE();
|
||||||
|
|
||||||
|
- (dispatch_queue_t)methodQueue
|
||||||
|
{
|
||||||
|
return dispatch_queue_create("com.eko.backgrounddownload", DISPATCH_QUEUE_SERIAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
- (BOOL)requiresMainQueueSetup {
|
||||||
|
return YES;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (NSArray<NSString *> *)supportedEvents {
|
||||||
|
return @[@"downloadComplete", @"downloadProgress", @"downloadFailed", @"downloadBegin"];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (NSDictionary *)constantsToExport {
|
||||||
|
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
|
||||||
|
|
||||||
|
return @{
|
||||||
|
@"documents": [paths firstObject],
|
||||||
|
@"TaskRunning": @(NSURLSessionTaskStateRunning),
|
||||||
|
@"TaskSuspended": @(NSURLSessionTaskStateSuspended),
|
||||||
|
@"TaskCanceling": @(NSURLSessionTaskStateCanceling),
|
||||||
|
@"TaskCompleted": @(NSURLSessionTaskStateCompleted)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
- (id) init {
|
||||||
|
self = [super init];
|
||||||
|
if (self) {
|
||||||
|
urlToConfigMap = [self deserialize:[[NSUserDefaults standardUserDefaults] objectForKey:URL_TO_CONFIG_MAP_KEY]];
|
||||||
|
if (urlToConfigMap == nil) {
|
||||||
|
urlToConfigMap = [[NSMutableDictionary alloc] init];
|
||||||
|
}
|
||||||
|
taskToConfigMap = [[NSMutableDictionary alloc] init];
|
||||||
|
idToTaskMap = [[NSMutableDictionary alloc] init];
|
||||||
|
idToResumeDataMap= [[NSMutableDictionary alloc] init];
|
||||||
|
idToPercentMap = [[NSMutableDictionary alloc] init];
|
||||||
|
NSString *bundleIdentifier = [[NSBundle mainBundle] bundleIdentifier];
|
||||||
|
NSString *sessonIdentifier = [bundleIdentifier stringByAppendingString:@".backgrounddownloadtask"];
|
||||||
|
sessionConfig = [NSURLSessionConfiguration backgroundSessionConfigurationWithIdentifier:sessonIdentifier];
|
||||||
|
downloadOperationsQueue = [[NSOperationQueue alloc] init];
|
||||||
|
progressReports = [[NSMutableArray alloc] init];
|
||||||
|
lastProgressReport = [[NSDate alloc] init];
|
||||||
|
}
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)lazyInitSession {
|
||||||
|
if (urlSession == nil) {
|
||||||
|
urlSession = [NSURLSession sessionWithConfiguration:sessionConfig delegate:self delegateQueue:downloadOperationsQueue];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)removeTaskFromMap: (NSURLSessionTask *)task {
|
||||||
|
TaskConfig *taskConfig = taskToConfigMap[task];
|
||||||
|
[taskToConfigMap removeObjectForKey:task];
|
||||||
|
[urlToConfigMap removeObjectForKey:task.currentRequest.URL.absoluteString];
|
||||||
|
[[NSUserDefaults standardUserDefaults] setObject:[self serialize: urlToConfigMap] forKey:URL_TO_CONFIG_MAP_KEY];
|
||||||
|
if (taskConfig) {
|
||||||
|
[idToTaskMap removeObjectForKey:taskConfig.id];
|
||||||
|
[idToPercentMap removeObjectForKey:taskConfig.id];
|
||||||
|
}
|
||||||
|
if (taskToConfigMap.count == 0) {
|
||||||
|
[urlSession invalidateAndCancel];
|
||||||
|
urlSession = nil;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
+ (void)setCompletionHandlerWithIdentifier: (NSString *)identifier completionHandler: (CompletionHandler)completionHandler {
|
||||||
|
NSString *bundleIdentifier = [[NSBundle mainBundle] bundleIdentifier];
|
||||||
|
NSString *sessonIdentifier = [bundleIdentifier stringByAppendingString:@".backgrounddownloadtask"];
|
||||||
|
if ([sessonIdentifier isEqualToString:identifier]) {
|
||||||
|
storedCompletionHandler = completionHandler;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#pragma mark - JS exported methods
|
||||||
|
RCT_EXPORT_METHOD(download: (NSDictionary *) options) {
|
||||||
|
NSString *identifier = options[@"id"];
|
||||||
|
NSString *url = options[@"url"];
|
||||||
|
NSString *destination = options[@"destination"];
|
||||||
|
if (identifier == nil || url == nil || destination == nil) {
|
||||||
|
NSLog(@"[RNFileBackgroundDownload] [Error] id, url and destination must be set");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
[self lazyInitSession];
|
||||||
|
NSURLSessionDownloadTask *task = [urlSession downloadTaskWithURL:[NSURL URLWithString:url]];
|
||||||
|
TaskConfig *taskConfig = [[TaskConfig alloc] initWithDictionary: @{@"id": identifier, @"destination": destination}];
|
||||||
|
taskToConfigMap[task] = taskConfig;
|
||||||
|
idToTaskMap[identifier] = task;
|
||||||
|
idToPercentMap[identifier] = @0.0;
|
||||||
|
|
||||||
|
[task resume];
|
||||||
|
}
|
||||||
|
|
||||||
|
RCT_EXPORT_METHOD(puaseTask: (NSString *)identifier) {
|
||||||
|
NSURLSessionDownloadTask *task = idToTaskMap[identifier];
|
||||||
|
if (task != nil && task.state == NSURLSessionTaskStateRunning) {
|
||||||
|
[task suspend];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RCT_EXPORT_METHOD(resumeTask: (NSString *)identifier) {
|
||||||
|
NSURLSessionDownloadTask *task = idToTaskMap[identifier];
|
||||||
|
if (task != nil && task.state == NSURLSessionTaskStateSuspended) {
|
||||||
|
[task resume];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RCT_EXPORT_METHOD(stopTask: (NSString *)identifier) {
|
||||||
|
NSURLSessionDownloadTask *task = idToTaskMap[identifier];
|
||||||
|
if (task != nil) {
|
||||||
|
[task cancel];
|
||||||
|
[self removeTaskFromMap:task];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RCT_EXPORT_METHOD(checkForExistingDownloads: (RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject) {
|
||||||
|
[self lazyInitSession];
|
||||||
|
[urlSession getTasksWithCompletionHandler:^(NSArray<NSURLSessionDataTask *> * _Nonnull dataTasks, NSArray<NSURLSessionUploadTask *> * _Nonnull uploadTasks, NSArray<NSURLSessionDownloadTask *> * _Nonnull downloadTasks) {
|
||||||
|
NSMutableArray *idsFound = [[NSMutableArray alloc] init];
|
||||||
|
for (NSURLSessionDownloadTask *foundTask in downloadTasks) {
|
||||||
|
NSURLSessionDownloadTask __strong *task = foundTask;
|
||||||
|
NSLog(@"Found task with url: %@", task.currentRequest.URL.absoluteString);
|
||||||
|
TaskConfig *taskConfig = urlToConfigMap[task.currentRequest.URL.absoluteString];
|
||||||
|
if (taskConfig) {
|
||||||
|
if (task.state == NSURLSessionTaskStateCompleted && task.countOfBytesReceived < task.countOfBytesExpectedToReceive) {
|
||||||
|
if (task.error && task.error.code == -999 && task.error.userInfo[NSURLSessionDownloadTaskResumeData] != nil) {
|
||||||
|
task = [urlSession downloadTaskWithResumeData:task.error.userInfo[NSURLSessionDownloadTaskResumeData]];
|
||||||
|
} else {
|
||||||
|
task = [urlSession downloadTaskWithURL:foundTask.currentRequest.URL];
|
||||||
|
}
|
||||||
|
[task resume];
|
||||||
|
}
|
||||||
|
NSNumber *percent = foundTask.countOfBytesExpectedToReceive > 0 ? [NSNumber numberWithFloat:(float)task.countOfBytesReceived/(float)foundTask.countOfBytesExpectedToReceive] : @0.0;
|
||||||
|
[idsFound addObject:@{
|
||||||
|
@"id": taskConfig.id,
|
||||||
|
@"state": [NSNumber numberWithInt: task.state],
|
||||||
|
@"bytesWritten": [NSNumber numberWithLongLong:task.countOfBytesReceived],
|
||||||
|
@"totalBytes": [NSNumber numberWithLongLong:foundTask.countOfBytesExpectedToReceive],
|
||||||
|
@"percent": percent
|
||||||
|
}];
|
||||||
|
taskConfig.reportedBegin = YES;
|
||||||
|
taskToConfigMap[task] = taskConfig;
|
||||||
|
idToTaskMap[taskConfig.id] = task;
|
||||||
|
idToPercentMap[taskConfig.id] = percent;
|
||||||
|
} else {
|
||||||
|
[task cancel];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
resolve(idsFound);
|
||||||
|
}];
|
||||||
|
}
|
||||||
|
|
||||||
|
#pragma mark - NSURLSessionDownloadDelegate methods
|
||||||
|
- (void)URLSession:(nonnull NSURLSession *)session downloadTask:(nonnull NSURLSessionDownloadTask *)downloadTask didFinishDownloadingToURL:(nonnull NSURL *)location {
|
||||||
|
TaskConfig *taskCofig = taskToConfigMap[downloadTask];
|
||||||
|
if (taskCofig != nil) {
|
||||||
|
NSFileManager *fileManager = [NSFileManager defaultManager];
|
||||||
|
NSURL *destURL = [NSURL fileURLWithPath:taskCofig.destination];
|
||||||
|
[fileManager createDirectoryAtURL:[destURL URLByDeletingLastPathComponent] withIntermediateDirectories:YES attributes:nil error:nil];
|
||||||
|
[fileManager removeItemAtURL:destURL error:nil];
|
||||||
|
NSError *moveError;
|
||||||
|
BOOL moved = [fileManager moveItemAtURL:location toURL:destURL error:&moveError];
|
||||||
|
if (self.bridge) {
|
||||||
|
if (moved) {
|
||||||
|
[self sendEventWithName:@"downloadComplete" body:@{@"id": taskCofig.id}];
|
||||||
|
} else {
|
||||||
|
[self sendEventWithName:@"downloadFailed" body:@{@"id": taskCofig.id, @"error": [moveError localizedDescription]}];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
[self removeTaskFromMap:downloadTask];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didResumeAtOffset:(int64_t)fileOffset expectedTotalBytes:(int64_t)expectedTotalBytes {
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite {
|
||||||
|
TaskConfig *taskCofig = taskToConfigMap[downloadTask];
|
||||||
|
if (taskCofig != nil) {
|
||||||
|
if (!taskCofig.reportedBegin) {
|
||||||
|
[self sendEventWithName:@"downloadBegin" body:@{@"id": taskCofig.id, @"expctedBytes": [NSNumber numberWithLongLong: totalBytesExpectedToWrite]}];
|
||||||
|
urlToConfigMap[downloadTask.currentRequest.URL.absoluteString] = taskCofig;
|
||||||
|
[[NSUserDefaults standardUserDefaults] setObject:[self serialize: urlToConfigMap] forKey:URL_TO_CONFIG_MAP_KEY];
|
||||||
|
taskCofig.reportedBegin = YES;
|
||||||
|
}
|
||||||
|
|
||||||
|
NSNumber *prevPercent = idToPercentMap[taskCofig.id];
|
||||||
|
NSNumber *percent = [NSNumber numberWithFloat:(float)totalBytesWritten/(float)totalBytesExpectedToWrite];
|
||||||
|
if ([percent floatValue] - [prevPercent floatValue] > 0.01f) {
|
||||||
|
[progressReports addObject:@{@"id": taskCofig.id, @"written": [NSNumber numberWithLongLong: totalBytesWritten], @"total": [NSNumber numberWithLongLong: totalBytesExpectedToWrite], @"percent": percent}];
|
||||||
|
idToPercentMap[taskCofig.id] = percent;
|
||||||
|
}
|
||||||
|
|
||||||
|
NSDate *now = [[NSDate alloc] init];
|
||||||
|
if ([now timeIntervalSinceDate:lastProgressReport] > 1.5 && progressReports.count > 0) {
|
||||||
|
if (self.bridge) {
|
||||||
|
[self sendEventWithName:@"downloadProgress" body:progressReports];
|
||||||
|
}
|
||||||
|
lastProgressReport = now;
|
||||||
|
[progressReports removeAllObjects];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
|
||||||
|
TaskConfig *taskCofig = taskToConfigMap[task];
|
||||||
|
if (error != nil && error.code != -999 && taskCofig != nil) {
|
||||||
|
if (self.bridge) {
|
||||||
|
[self sendEventWithName:@"downloadFailed" body:@{@"id": taskCofig.id, @"error": [error localizedDescription]}];
|
||||||
|
}
|
||||||
|
[self removeTaskFromMap:task];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)URLSessionDidFinishEventsForBackgroundURLSession:(NSURLSession *)session {
|
||||||
|
NSLog(@"in URLSessionDidFinishEventsForBackgroundURLSession");
|
||||||
|
if (storedCompletionHandler) {
|
||||||
|
[[NSOperationQueue mainQueue] addOperationWithBlock:^{
|
||||||
|
storedCompletionHandler();
|
||||||
|
storedCompletionHandler = nil;
|
||||||
|
}];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#pragma mark - serialization
|
||||||
|
- (NSData *)serialize: (id)obj {
|
||||||
|
return [NSKeyedArchiver archivedDataWithRootObject:obj];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (id)deserialize: (NSData *)data {
|
||||||
|
if (data == nil) {
|
||||||
|
return nil;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [NSKeyedUnarchiver unarchiveObjectWithData:data];
|
||||||
|
}
|
||||||
|
|
||||||
|
@end
|
||||||
24
ios/RNBackgroundDownload.podspec
Normal file
24
ios/RNBackgroundDownload.podspec
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
|
||||||
|
Pod::Spec.new do |s|
|
||||||
|
s.name = "RNBackgroundDownload"
|
||||||
|
s.version = "1.0.0"
|
||||||
|
s.summary = "RNBackgroundDownload"
|
||||||
|
s.description = <<-DESC
|
||||||
|
RNBackgroundDownload
|
||||||
|
DESC
|
||||||
|
s.homepage = ""
|
||||||
|
s.license = "MIT"
|
||||||
|
# s.license = { :type => "MIT", :file => "FILE_LICENSE" }
|
||||||
|
s.author = { "author" => "author@domain.cn" }
|
||||||
|
s.platform = :ios, "7.0"
|
||||||
|
s.source = { :git => "https://github.com/author/RNBackgroundDownload.git", :tag => "master" }
|
||||||
|
s.source_files = "RNBackgroundDownload/**/*.{h,m}"
|
||||||
|
s.requires_arc = true
|
||||||
|
|
||||||
|
|
||||||
|
s.dependency "React"
|
||||||
|
#s.dependency "others"
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
252
ios/RNBackgroundDownload.xcodeproj/project.pbxproj
Normal file
252
ios/RNBackgroundDownload.xcodeproj/project.pbxproj
Normal file
@@ -0,0 +1,252 @@
|
|||||||
|
// !$*UTF8*$!
|
||||||
|
{
|
||||||
|
archiveVersion = 1;
|
||||||
|
classes = {
|
||||||
|
};
|
||||||
|
objectVersion = 46;
|
||||||
|
objects = {
|
||||||
|
|
||||||
|
/* Begin PBXBuildFile section */
|
||||||
|
B3E7B58A1CC2AC0600A0062D /* RNBackgroundDownload.m in Sources */ = {isa = PBXBuildFile; fileRef = B3E7B5891CC2AC0600A0062D /* RNBackgroundDownload.m */; };
|
||||||
|
/* End PBXBuildFile section */
|
||||||
|
|
||||||
|
/* Begin PBXCopyFilesBuildPhase section */
|
||||||
|
58B511D91A9E6C8500147676 /* CopyFiles */ = {
|
||||||
|
isa = PBXCopyFilesBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
dstPath = "include/$(PRODUCT_NAME)";
|
||||||
|
dstSubfolderSpec = 16;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
/* End PBXCopyFilesBuildPhase section */
|
||||||
|
|
||||||
|
/* Begin PBXFileReference section */
|
||||||
|
134814201AA4EA6300B7C361 /* libRNBackgroundDownload.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = libRNBackgroundDownload.a; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||||
|
B3E7B5881CC2AC0600A0062D /* RNBackgroundDownload.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = RNBackgroundDownload.h; sourceTree = "<group>"; };
|
||||||
|
B3E7B5891CC2AC0600A0062D /* RNBackgroundDownload.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = RNBackgroundDownload.m; sourceTree = "<group>"; };
|
||||||
|
/* End PBXFileReference section */
|
||||||
|
|
||||||
|
/* Begin PBXFrameworksBuildPhase section */
|
||||||
|
58B511D81A9E6C8500147676 /* Frameworks */ = {
|
||||||
|
isa = PBXFrameworksBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
/* End PBXFrameworksBuildPhase section */
|
||||||
|
|
||||||
|
/* Begin PBXGroup section */
|
||||||
|
134814211AA4EA7D00B7C361 /* Products */ = {
|
||||||
|
isa = PBXGroup;
|
||||||
|
children = (
|
||||||
|
134814201AA4EA6300B7C361 /* libRNBackgroundDownload.a */,
|
||||||
|
);
|
||||||
|
name = Products;
|
||||||
|
sourceTree = "<group>";
|
||||||
|
};
|
||||||
|
58B511D21A9E6C8500147676 = {
|
||||||
|
isa = PBXGroup;
|
||||||
|
children = (
|
||||||
|
B3E7B5881CC2AC0600A0062D /* RNBackgroundDownload.h */,
|
||||||
|
B3E7B5891CC2AC0600A0062D /* RNBackgroundDownload.m */,
|
||||||
|
134814211AA4EA7D00B7C361 /* Products */,
|
||||||
|
);
|
||||||
|
sourceTree = "<group>";
|
||||||
|
};
|
||||||
|
/* End PBXGroup section */
|
||||||
|
|
||||||
|
/* Begin PBXNativeTarget section */
|
||||||
|
58B511DA1A9E6C8500147676 /* RNBackgroundDownload */ = {
|
||||||
|
isa = PBXNativeTarget;
|
||||||
|
buildConfigurationList = 58B511EF1A9E6C8500147676 /* Build configuration list for PBXNativeTarget "RNBackgroundDownload" */;
|
||||||
|
buildPhases = (
|
||||||
|
58B511D71A9E6C8500147676 /* Sources */,
|
||||||
|
58B511D81A9E6C8500147676 /* Frameworks */,
|
||||||
|
58B511D91A9E6C8500147676 /* CopyFiles */,
|
||||||
|
);
|
||||||
|
buildRules = (
|
||||||
|
);
|
||||||
|
dependencies = (
|
||||||
|
);
|
||||||
|
name = RNBackgroundDownload;
|
||||||
|
productName = RCTDataManager;
|
||||||
|
productReference = 134814201AA4EA6300B7C361 /* libRNBackgroundDownload.a */;
|
||||||
|
productType = "com.apple.product-type.library.static";
|
||||||
|
};
|
||||||
|
/* End PBXNativeTarget section */
|
||||||
|
|
||||||
|
/* Begin PBXProject section */
|
||||||
|
58B511D31A9E6C8500147676 /* Project object */ = {
|
||||||
|
isa = PBXProject;
|
||||||
|
attributes = {
|
||||||
|
LastUpgradeCheck = 0610;
|
||||||
|
ORGANIZATIONNAME = Facebook;
|
||||||
|
TargetAttributes = {
|
||||||
|
58B511DA1A9E6C8500147676 = {
|
||||||
|
CreatedOnToolsVersion = 6.1.1;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
buildConfigurationList = 58B511D61A9E6C8500147676 /* Build configuration list for PBXProject "RNBackgroundDownload" */;
|
||||||
|
compatibilityVersion = "Xcode 3.2";
|
||||||
|
developmentRegion = English;
|
||||||
|
hasScannedForEncodings = 0;
|
||||||
|
knownRegions = (
|
||||||
|
en,
|
||||||
|
);
|
||||||
|
mainGroup = 58B511D21A9E6C8500147676;
|
||||||
|
productRefGroup = 58B511D21A9E6C8500147676;
|
||||||
|
projectDirPath = "";
|
||||||
|
projectRoot = "";
|
||||||
|
targets = (
|
||||||
|
58B511DA1A9E6C8500147676 /* RNBackgroundDownload */,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
/* End PBXProject section */
|
||||||
|
|
||||||
|
/* Begin PBXSourcesBuildPhase section */
|
||||||
|
58B511D71A9E6C8500147676 /* Sources */ = {
|
||||||
|
isa = PBXSourcesBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
B3E7B58A1CC2AC0600A0062D /* RNBackgroundDownload.m in Sources */,
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
/* End PBXSourcesBuildPhase section */
|
||||||
|
|
||||||
|
/* Begin XCBuildConfiguration section */
|
||||||
|
58B511ED1A9E6C8500147676 /* Debug */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||||
|
CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
|
||||||
|
CLANG_CXX_LIBRARY = "libc++";
|
||||||
|
CLANG_ENABLE_MODULES = YES;
|
||||||
|
CLANG_ENABLE_OBJC_ARC = YES;
|
||||||
|
CLANG_WARN_BOOL_CONVERSION = YES;
|
||||||
|
CLANG_WARN_CONSTANT_CONVERSION = YES;
|
||||||
|
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
|
||||||
|
CLANG_WARN_EMPTY_BODY = YES;
|
||||||
|
CLANG_WARN_ENUM_CONVERSION = YES;
|
||||||
|
CLANG_WARN_INT_CONVERSION = YES;
|
||||||
|
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
|
||||||
|
CLANG_WARN_UNREACHABLE_CODE = YES;
|
||||||
|
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
|
||||||
|
COPY_PHASE_STRIP = NO;
|
||||||
|
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||||
|
GCC_C_LANGUAGE_STANDARD = gnu99;
|
||||||
|
GCC_DYNAMIC_NO_PIC = NO;
|
||||||
|
GCC_OPTIMIZATION_LEVEL = 0;
|
||||||
|
GCC_PREPROCESSOR_DEFINITIONS = (
|
||||||
|
"DEBUG=1",
|
||||||
|
"$(inherited)",
|
||||||
|
);
|
||||||
|
GCC_SYMBOLS_PRIVATE_EXTERN = NO;
|
||||||
|
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||||
|
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
|
||||||
|
GCC_WARN_UNDECLARED_SELECTOR = YES;
|
||||||
|
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||||
|
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||||
|
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||||
|
IPHONEOS_DEPLOYMENT_TARGET = 7.0;
|
||||||
|
MTL_ENABLE_DEBUG_INFO = YES;
|
||||||
|
ONLY_ACTIVE_ARCH = YES;
|
||||||
|
SDKROOT = iphoneos;
|
||||||
|
};
|
||||||
|
name = Debug;
|
||||||
|
};
|
||||||
|
58B511EE1A9E6C8500147676 /* Release */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||||
|
CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
|
||||||
|
CLANG_CXX_LIBRARY = "libc++";
|
||||||
|
CLANG_ENABLE_MODULES = YES;
|
||||||
|
CLANG_ENABLE_OBJC_ARC = YES;
|
||||||
|
CLANG_WARN_BOOL_CONVERSION = YES;
|
||||||
|
CLANG_WARN_CONSTANT_CONVERSION = YES;
|
||||||
|
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
|
||||||
|
CLANG_WARN_EMPTY_BODY = YES;
|
||||||
|
CLANG_WARN_ENUM_CONVERSION = YES;
|
||||||
|
CLANG_WARN_INT_CONVERSION = YES;
|
||||||
|
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
|
||||||
|
CLANG_WARN_UNREACHABLE_CODE = YES;
|
||||||
|
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
|
||||||
|
COPY_PHASE_STRIP = YES;
|
||||||
|
ENABLE_NS_ASSERTIONS = NO;
|
||||||
|
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||||
|
GCC_C_LANGUAGE_STANDARD = gnu99;
|
||||||
|
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||||
|
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
|
||||||
|
GCC_WARN_UNDECLARED_SELECTOR = YES;
|
||||||
|
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||||
|
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||||
|
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||||
|
IPHONEOS_DEPLOYMENT_TARGET = 7.0;
|
||||||
|
MTL_ENABLE_DEBUG_INFO = NO;
|
||||||
|
SDKROOT = iphoneos;
|
||||||
|
VALIDATE_PRODUCT = YES;
|
||||||
|
};
|
||||||
|
name = Release;
|
||||||
|
};
|
||||||
|
58B511F01A9E6C8500147676 /* Debug */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
HEADER_SEARCH_PATHS = (
|
||||||
|
"$(inherited)",
|
||||||
|
/Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/include,
|
||||||
|
"$(SRCROOT)/../../../React/**",
|
||||||
|
"$(SRCROOT)/../../react-native/React/**",
|
||||||
|
);
|
||||||
|
LIBRARY_SEARCH_PATHS = "$(inherited)";
|
||||||
|
OTHER_LDFLAGS = "-ObjC";
|
||||||
|
PRODUCT_NAME = RNBackgroundDownload;
|
||||||
|
SKIP_INSTALL = YES;
|
||||||
|
};
|
||||||
|
name = Debug;
|
||||||
|
};
|
||||||
|
58B511F11A9E6C8500147676 /* Release */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
HEADER_SEARCH_PATHS = (
|
||||||
|
"$(inherited)",
|
||||||
|
/Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/include,
|
||||||
|
"$(SRCROOT)/../../../React/**",
|
||||||
|
"$(SRCROOT)/../../react-native/React/**",
|
||||||
|
);
|
||||||
|
LIBRARY_SEARCH_PATHS = "$(inherited)";
|
||||||
|
OTHER_LDFLAGS = "-ObjC";
|
||||||
|
PRODUCT_NAME = RNBackgroundDownload;
|
||||||
|
SKIP_INSTALL = YES;
|
||||||
|
};
|
||||||
|
name = Release;
|
||||||
|
};
|
||||||
|
/* End XCBuildConfiguration section */
|
||||||
|
|
||||||
|
/* Begin XCConfigurationList section */
|
||||||
|
58B511D61A9E6C8500147676 /* Build configuration list for PBXProject "RNBackgroundDownload" */ = {
|
||||||
|
isa = XCConfigurationList;
|
||||||
|
buildConfigurations = (
|
||||||
|
58B511ED1A9E6C8500147676 /* Debug */,
|
||||||
|
58B511EE1A9E6C8500147676 /* Release */,
|
||||||
|
);
|
||||||
|
defaultConfigurationIsVisible = 0;
|
||||||
|
defaultConfigurationName = Release;
|
||||||
|
};
|
||||||
|
58B511EF1A9E6C8500147676 /* Build configuration list for PBXNativeTarget "RNBackgroundDownload" */ = {
|
||||||
|
isa = XCConfigurationList;
|
||||||
|
buildConfigurations = (
|
||||||
|
58B511F01A9E6C8500147676 /* Debug */,
|
||||||
|
58B511F11A9E6C8500147676 /* Release */,
|
||||||
|
);
|
||||||
|
defaultConfigurationIsVisible = 0;
|
||||||
|
defaultConfigurationName = Release;
|
||||||
|
};
|
||||||
|
/* End XCConfigurationList section */
|
||||||
|
};
|
||||||
|
rootObject = 58B511D31A9E6C8500147676 /* Project object */;
|
||||||
|
}
|
||||||
77
lib/downloadTask.js
Normal file
77
lib/downloadTask.js
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
import { NativeModules } from 'react-native';
|
||||||
|
const { RNBackgroundDownload } = NativeModules;
|
||||||
|
|
||||||
|
export default class DownloadTask {
|
||||||
|
state = 'PENDING';
|
||||||
|
percent = 0;
|
||||||
|
bytesWritten = 0;
|
||||||
|
totalBytes = 0;
|
||||||
|
|
||||||
|
constructor(taskInfo) {
|
||||||
|
if (typeof taskInfo === 'string') {
|
||||||
|
this.id = taskInfo;
|
||||||
|
} else {
|
||||||
|
this.id = taskInfo.id;
|
||||||
|
this.percent = taskInfo.percent;
|
||||||
|
this.bytesWritten = taskInfo.bytesWritten;
|
||||||
|
this.totalBytes = taskInfo.totalBytes;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
begin(handler) {
|
||||||
|
this._beginHandler = handler;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
progress(handler) {
|
||||||
|
this._progressHandler = handler;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
done(handler) {
|
||||||
|
this._doneHandler = handler;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
error(handler) {
|
||||||
|
this._errorHandler = handler;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
_onBegin(expectedBytes) {
|
||||||
|
this.state = 'DOWNLOADING';
|
||||||
|
this._beginHandler && this._beginHandler(expectedBytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
_onProgress(percent, bytesWritten, totalBytes) {
|
||||||
|
this.percent = percent;
|
||||||
|
this.bytesWritten = bytesWritten;
|
||||||
|
this.totalBytes = totalBytes;
|
||||||
|
this._progressHandler && this._progressHandler(percent, bytesWritten, totalBytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
_onDone() {
|
||||||
|
this.state = 'DONE';
|
||||||
|
this._doneHandler && this._doneHandler();
|
||||||
|
}
|
||||||
|
|
||||||
|
_onError(error) {
|
||||||
|
this.state = 'FAILED';
|
||||||
|
this._errorHandler && this._errorHandler(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
pause() {
|
||||||
|
this.state = 'PAUSED';
|
||||||
|
RNBackgroundDownload.puaseTask(this.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
resume() {
|
||||||
|
this.state = 'DOWNLOADING';
|
||||||
|
RNBackgroundDownload.resumeTask(this.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
stop() {
|
||||||
|
this.state = 'STOPPED';
|
||||||
|
RNBackgroundDownload.stopTask(this.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
18
package.json
Normal file
18
package.json
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
|
||||||
|
{
|
||||||
|
"name": "react-native-background-download",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "",
|
||||||
|
"main": "index.js",
|
||||||
|
"scripts": {
|
||||||
|
"test": "echo \"Error: no test specified\" && exit 1"
|
||||||
|
},
|
||||||
|
"keywords": [
|
||||||
|
"react-native"
|
||||||
|
],
|
||||||
|
"author": "",
|
||||||
|
"license": "",
|
||||||
|
"peerDependencies": {
|
||||||
|
"react-native": "^0.41.2"
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user