├── .gitignore ├── .idea └── vcs.xml ├── .npmignore ├── LICENSE ├── README.md ├── examples └── offline-image │ ├── .babelrc │ ├── .flowconfig │ ├── .gitignore │ ├── .watchmanconfig │ ├── App.test.js │ ├── README.md │ ├── android │ ├── app │ │ ├── BUCK │ │ ├── build.gradle │ │ ├── proguard-rules.pro │ │ └── src │ │ │ └── main │ │ │ ├── AndroidManifest.xml │ │ │ ├── java │ │ │ └── com │ │ │ │ └── offlineimage │ │ │ │ ├── MainActivity.java │ │ │ │ └── MainApplication.java │ │ │ └── res │ │ │ ├── mipmap-hdpi │ │ │ └── ic_launcher.png │ │ │ ├── mipmap-mdpi │ │ │ └── ic_launcher.png │ │ │ ├── mipmap-xhdpi │ │ │ └── ic_launcher.png │ │ │ ├── mipmap-xxhdpi │ │ │ └── ic_launcher.png │ │ │ └── values │ │ │ ├── strings.xml │ │ │ └── styles.xml │ ├── build.gradle │ ├── gradle.properties │ ├── gradle │ │ └── wrapper │ │ │ ├── gradle-wrapper.jar │ │ │ └── gradle-wrapper.properties │ ├── gradlew │ ├── gradlew.bat │ ├── keystores │ │ ├── BUCK │ │ └── debug.keystore.properties │ └── settings.gradle │ ├── index.android.js │ ├── index.ios.js │ ├── ios │ ├── offlineimage-tvOS │ │ └── Info.plist │ ├── offlineimage-tvOSTests │ │ └── Info.plist │ ├── offlineimage.xcodeproj │ │ ├── project.pbxproj │ │ └── xcshareddata │ │ │ └── xcschemes │ │ │ ├── offlineimage-tvOS.xcscheme │ │ │ └── offlineimage.xcscheme │ ├── offlineimage │ │ ├── AppDelegate.h │ │ ├── AppDelegate.m │ │ ├── Base.lproj │ │ │ └── LaunchScreen.xib │ │ ├── Images.xcassets │ │ │ └── AppIcon.appiconset │ │ │ │ └── Contents.json │ │ ├── Info.plist │ │ └── main.m │ └── offlineimageTests │ │ ├── Info.plist │ │ └── offlineimageTests.m │ ├── package.json │ ├── src │ ├── assets │ │ ├── images.js │ │ ├── images │ │ │ └── fallbackSource.png │ │ └── package.json │ └── index.js │ └── yarn.lock ├── package.json ├── screenshot.png ├── src ├── OfflineImage.js ├── OfflineImageStore.js ├── actions.js ├── constants.js ├── epics.js ├── index.js ├── reducer.js └── utils.js └── yarn.lock /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | 8 | # Runtime data 9 | pids 10 | *.pid 11 | *.seed 12 | *.pid.lock 13 | 14 | # Directory for instrumented libs generated by jscoverage/JSCover 15 | lib-cov 16 | 17 | # Coverage directory used by tools like istanbul 18 | coverage 19 | 20 | # nyc test coverage 21 | .nyc_output 22 | 23 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 24 | .grunt 25 | 26 | # Bower dependency directory (https://bower.io/) 27 | bower_components 28 | 29 | # node-waf configuration 30 | .lock-wscript 31 | 32 | # Compiled binary addons (http://nodejs.org/api/addons.html) 33 | build/Release 34 | 35 | # Dependency directories 36 | node_modules/ 37 | jspm_packages/ 38 | 39 | # Typescript v1 declaration files 40 | typings/ 41 | 42 | # Optional npm cache directory 43 | .npm 44 | 45 | # Optional eslint cache 46 | .eslintcache 47 | 48 | # Optional REPL history 49 | .node_repl_history 50 | 51 | # Output of 'npm pack' 52 | *.tgz 53 | 54 | # Yarn Integrity file 55 | .yarn-integrity 56 | 57 | # dotenv environment variables file 58 | .env 59 | 60 | # IntelliJ 61 | .idea 62 | 63 | -------------------------------------------------------------------------------- /.idea/vcs.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /.npmignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | .idea 3 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2017 code & co. 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # react-native-image-offline 2 | 3 | React Native library for iOS and Android offline image storage. This library provides most of the capabilities for an application to display pre-loaded images when offline. 4 | (This library has a dependency on **rn-fetch-blob**. Refer [here](https://github.com/joltup/rn-fetch-blob) for more details about the library.) 5 | 6 | ## Features 7 | * Define your own offline storage name! 8 | * Pre load the images 9 | * Automatically remove expired images from offline storage. 10 | * Supports a fallback source static image that will be shown if a source object has a 'uri' but is unable to download the image or is unable to find it in the offline storage. 11 | * Use custom Image components. 12 | * You can always re-fetch the image by specifying `reloadImage={true}` irrespective of the image already being stored offline, this way you can refresh/load the most recently updated images. 13 | * Option to clear offline storage 14 | 15 | 16 | ## Installation 17 | This library has a dependency on `rn-fetch-blob`, please refer to their [installation instructions](https://github.com/joltup/rn-fetch-blob#user-content-installation) 18 | 19 | **Using yarn** 20 | 21 | `$ yarn add react-native-image-offline` 22 | 23 | Note: Do not forget to run `react-native link` after adding `rn-fetch-blob` dependency. 24 | 25 | **Using npm** 26 | 27 | `$ npm install react-native-image-offline --save` 28 | 29 | 30 | ## Usage 31 | 32 | 33 | ##### `restore` 34 | First and foremeost, to use this library it is important to call the `restore` function so that you can get the completion status back. See the basic example usage. 35 | 36 | `OfflineImageStore.restore({}, () => {})` 37 | 38 | First argument is configuration object, example 39 | 40 | ``` 41 | { 42 | name: 'My_Image_gallery', 43 | imageRemoveTimeout: 120, // expire image after 120 seconds, default is 3 days if you don't provide this property. 44 | debugMode: true, 45 | } 46 | ``` 47 | Second argument is the callback function 48 | ``` 49 | const restoreCompletion = () => { // Callback function 50 | console.log('Restore completed !'); 51 | // Restore completed!! 52 | this.setState({ reStoreCompleted: true }); 53 | } 54 | ``` 55 | 56 | ### Configuration properties 57 | ##### name 58 | `name` configuration property used to define the application offline store directory name 59 | ##### imageRemoveTimeout 60 | `imageRemoveTimeout` is to provide image expiry time in seconds 61 | ##### debugMode 62 | `debugMode` set to `true` to view the debug logs 63 | 64 | ### Complete example usage 65 | ``` 66 | export default class App extends React.Component { 67 | 68 | constructor(props) { 69 | super(props); 70 | this.state = { 71 | reStoreCompleted: false, 72 | }; 73 | } 74 | componentWillMount() { 75 | OfflineImageStore.restore({ 76 | name: 'My_Image_gallery', 77 | imageRemoveTimeout: 120, // expire image after 120 seconds, default is 3 days if you don't provide this property. 78 | debugMode: true, 79 | }, () => { // Callback function 80 | console.log('Restore completed and callback called !'); 81 | // Restore completed!! 82 | this.setState({ reStoreCompleted: true }); 83 | 84 | // Preload images 85 | // Note: We recommend call this method on `restore` completion! 86 | OfflineImageStore.preLoad([ 87 | 'https://wallpaperbrowse.com/media/images/mobileswall-047.jpg', 88 | 'https://wallpaperbrowse.com/media/images/wallpaper-for-mobile-13.jpg', 89 | 'https://wallpaperbrowse.com/media/images/tvrcnkbcgeirbxcmsbfz.jpg', 90 | 'https://wallpaperbrowse.com/media/images/hd-wallpapers-1080p-for-mobile-2015.jpg', 91 | 'https://wallpaperbrowse.com/media/images/mobileswall-043.jpg', 92 | 'https://wallpaperbrowse.com/media/images/hd-wallpapers-for-mobile-2015.png', 93 | 'https://wallpaperbrowse.com/media/images/download_ZNNDLIt.jpg' 94 | ]); 95 | }); 96 | } 97 | 98 | render() { 99 | if (!this.state.reStoreCompleted) { 100 | return ( 101 | 111 | ); 112 | } 113 | return ( 114 | 115 | React native offline image 116 | 122 | { 125 | console.log('Loading finished for image with path: ', sourceUri) 126 | }} 127 | reloadImage = { true } 128 | resizeMode={'cover'} 129 | fallbackSource={ Images.fallbackSource } 130 | style={ { width: '99%', height: 110, margin: 5 } } 131 | source={ { uri: 'https://wallpaperbrowse.com/media/images/wallpaper-for-mobile-13.jpg' } }/> 132 | { 135 | console.log('Loading finished for image with path: ', sourceUri) 136 | }} 137 | resizeMode={'cover'} 138 | style={ { width: '99%', height: 110, margin: 5 } } 139 | source={ { uri: 'https://wallpaperbrowse.com/media/images/tvrcnkbcgeirbxcmsbfz.jpg' } }/> 140 | { 143 | console.log('Loading finished for image with path: ', sourceUri) 144 | }} 145 | resizeMode={'cover'} 146 | style={ { width: '99%', height: 110, margin: 5 } } 147 | source={ { uri: 'https://wallpaperbrowse.com/media/images/mobileswall-043.jpg' } }/> 148 | { 151 | console.log('Loading finished for image with path: ', sourceUri) 152 | }} 153 | resizeMode={'cover'} 154 | fallbackSource={ Images.fallbackSource } 155 | style={ { width: '99%', height: 110, margin: 5 } } 156 | source={ { uri: 'https://wallpaperbrowse.com/media/images/butterfly-wallpaper_SzlKJB8.jpeg' } }/> 157 | 158 | ); 159 | } 160 | } 161 | ``` 162 | 163 | ### `preload` 164 | The recommended approach to preload images is to call after `restore`. You could call this method anywhere from the code. For instance, this library can be used with `redux-observable`, here is the code snippet! 165 | 166 | ``` 167 | const loadShoppingCartEpic = (action$, store, { getJSON }) => 168 | action$.ofType(LOAD_SHOPPING_CART) 169 | .flatMap(action => { 170 | return getJSON(`${API_BASE_URL}/api/cart`) 171 | .map(res => { 172 | if (res.metadata.code === 200) { 173 | // Preload image after successful response 174 | // These images download and persist offline. 175 | OfflineImageStore.preLoad([ 176 | 'res.content.image1.link', 177 | 'res.content.image2.link', 178 | ]); 179 | return loadShoppingCartSuccess(res.content); 180 | } else { 181 | return loadShoppingCartFailure(); 182 | } 183 | }) 184 | ._catch(error => Observable.of(loadShoppingCartFailure)); 185 | }) 186 | ``` 187 | 188 | ### OfflineImage with static source 189 | ``` 190 | 194 | ... 195 | 196 | 197 | ``` 198 | 199 | ### OfflineImage with fallback/placeholder image 200 | You can use a fallback image as a default image to show when unable to download the image or if the image not available in the offline storage. 201 | ``` 202 | 207 | ... 208 | 209 | 210 | ``` 211 | 212 | ### Clear offline store 213 | You can clear complete offline store at any point of time using 214 | ``` 215 | // Clean all the images 216 | OfflineImageStore.clearStore((err) => { 217 | if (!err) { 218 | console.log('Hurray!! clearStore completed callback called'); 219 | } 220 | }); 221 | ``` 222 | 223 | ## Development/Contributions 224 | 225 | #### Credits 226 | Thanks to [https://wallpaperbrowse.com/](https://wallpaperbrowse.com/). These image uris are only used in a sample example application. 227 | -------------------------------------------------------------------------------- /examples/offline-image/.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "presets": [ 3 | "babel-preset-react-native-stage-0/decorator-support" 4 | ], 5 | "env": { 6 | "development": { 7 | "plugins": [ 8 | "transform-react-jsx-source" 9 | ] 10 | } 11 | } 12 | } -------------------------------------------------------------------------------- /examples/offline-image/.flowconfig: -------------------------------------------------------------------------------- 1 | [ignore] 2 | ; We fork some components by platform 3 | .*/*[.]android.js 4 | 5 | ; Ignore "BUCK" generated dirs 6 | /\.buckd/ 7 | 8 | ; Ignore unexpected extra "@providesModule" 9 | .*/node_modules/.*/node_modules/fbjs/.* 10 | 11 | ; Ignore duplicate module providers 12 | ; For RN Apps installed via npm, "Libraries" folder is inside 13 | ; "node_modules/react-native" but in the source repo it is in the root 14 | .*/Libraries/react-native/React.js 15 | .*/Libraries/react-native/ReactNative.js 16 | 17 | ; Additional create-react-native-app ignores 18 | 19 | ; Ignore duplicate module providers 20 | .*/node_modules/fbemitter/lib/* 21 | 22 | ; Ignore misbehaving dev-dependencies 23 | .*/node_modules/xdl/build/* 24 | .*/node_modules/reqwest/tests/* 25 | 26 | ; Ignore missing expo-sdk dependencies (temporarily) 27 | ; https://github.com/expo/expo/issues/162 28 | .*/node_modules/expo/src/* 29 | 30 | ; Ignore react-native-fbads dependency of the expo sdk 31 | .*/node_modules/react-native-fbads/* 32 | 33 | [include] 34 | 35 | [libs] 36 | node_modules/react-native/Libraries/react-native/react-native-interface.js 37 | node_modules/react-native/flow 38 | flow/ 39 | 40 | [options] 41 | module.system=haste 42 | 43 | emoji=true 44 | 45 | experimental.strict_type_args=true 46 | 47 | munge_underscores=true 48 | 49 | module.name_mapper='^[./a-zA-Z0-9$_-]+\.\(bmp\|gif\|jpg\|jpeg\|png\|psd\|svg\|webp\|m4v\|mov\|mp4\|mpeg\|mpg\|webm\|aac\|aiff\|caf\|m4a\|mp3\|wav\|html\|pdf\)$' -> 'RelativeImageStub' 50 | 51 | suppress_type=$FlowIssue 52 | suppress_type=$FlowFixMe 53 | suppress_type=$FixMe 54 | 55 | suppress_comment=\\(.\\|\n\\)*\\$FlowFixMe\\($\\|[^(]\\|(\\(>=0\\.\\(4[0-9]\\|[1-3][0-9]\\|[0-9]\\).[0-9]\\)? *\\(site=[a-z,_]*react_native[a-z,_]*\\)?)\\) 56 | suppress_comment=\\(.\\|\n\\)*\\$FlowIssue\\((\\(>=0\\.\\(4[0-9]\\|[1-3][0-9]\\|[0-9]\\).[0-9]\\)? *\\(site=[a-z,_]*react_native[a-z,_]*\\)?)\\)?:? #[0-9]+ 57 | suppress_comment=\\(.\\|\n\\)*\\$FlowFixedInNextDeploy 58 | suppress_comment=\\(.\\|\n\\)*\\$FlowExpectedError 59 | 60 | unsafe.enable_getters_and_setters=true 61 | 62 | [version] 63 | ^0.49.1 64 | -------------------------------------------------------------------------------- /examples/offline-image/.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | 8 | # Runtime data 9 | pids 10 | *.pid 11 | *.seed 12 | *.pid.lock 13 | 14 | # Directory for instrumented libs generated by jscoverage/JSCover 15 | lib-cov 16 | 17 | # Coverage directory used by tools like istanbul 18 | coverage 19 | 20 | # nyc test coverage 21 | .nyc_output 22 | 23 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 24 | .grunt 25 | 26 | # Bower dependency directory (https://bower.io/) 27 | bower_components 28 | 29 | # node-waf configuration 30 | .lock-wscript 31 | 32 | # Compiled binary addons (http://nodejs.org/api/addons.html) 33 | build/Release 34 | 35 | # Dependency directories 36 | node_modules/ 37 | jspm_packages/ 38 | 39 | # Typescript v1 declaration files 40 | typings/ 41 | 42 | # Optional npm cache directory 43 | .npm 44 | 45 | # Optional eslint cache 46 | .eslintcache 47 | 48 | # Optional REPL history 49 | .node_repl_history 50 | 51 | # Output of 'npm pack' 52 | *.tgz 53 | 54 | # Yarn Integrity file 55 | .yarn-integrity 56 | 57 | # dotenv environment variables file 58 | .env 59 | .env.dev 60 | 61 | ======= 62 | # OSX 63 | # 64 | .DS_Store 65 | 66 | # Xcode 67 | # 68 | build/ 69 | *.pbxuser 70 | !default.pbxuser 71 | *.mode1v3 72 | !default.mode1v3 73 | *.mode2v3 74 | !default.mode2v3 75 | *.perspectivev3 76 | !default.perspectivev3 77 | xcuserdata 78 | *.xccheckout 79 | *.moved-aside 80 | DerivedData 81 | *.hmap 82 | *.ipa 83 | *.xcuserstate 84 | project.xcworkspace 85 | 86 | # Android/IntelliJ 87 | # 88 | build/ 89 | .idea 90 | .gradle 91 | local.properties 92 | *.iml 93 | 94 | # node.js 95 | # 96 | node_modules/ 97 | npm-debug.log 98 | yarn-error.log 99 | 100 | # BUCK 101 | buck-out/ 102 | \.buckd/ 103 | *.keystore 104 | 105 | # fastlane 106 | # 107 | # It is recommended to not store the screenshots in the git repo. Instead, use fastlane to re-generate the 108 | # screenshots whenever they are needed. 109 | # For more information about the recommended setup visit: 110 | # https://github.com/fastlane/fastlane/blob/master/fastlane/docs/Gitignore.md 111 | 112 | fastlane/report.xml 113 | fastlane/Preview.html 114 | fastlane/screenshots 115 | -------------------------------------------------------------------------------- /examples/offline-image/.watchmanconfig: -------------------------------------------------------------------------------- 1 | {} 2 | -------------------------------------------------------------------------------- /examples/offline-image/App.test.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import App from './App'; 3 | 4 | import renderer from 'react-test-renderer'; 5 | 6 | it('renders without crashing', () => { 7 | const rendered = renderer.create().toJSON(); 8 | expect(rendered).toBeTruthy(); 9 | }); 10 | -------------------------------------------------------------------------------- /examples/offline-image/README.md: -------------------------------------------------------------------------------- 1 | ## Sample examples 2 | 3 | -------------------------------------------------------------------------------- /examples/offline-image/android/app/BUCK: -------------------------------------------------------------------------------- 1 | # To learn about Buck see [Docs](https://buckbuild.com/). 2 | # To run your application with Buck: 3 | # - install Buck 4 | # - `npm start` - to start the packager 5 | # - `cd android` 6 | # - `keytool -genkey -v -keystore keystores/debug.keystore -storepass android -alias androiddebugkey -keypass android -dname "CN=Android Debug,O=Android,C=US"` 7 | # - `./gradlew :app:copyDownloadableDepsToLibs` - make all Gradle compile dependencies available to Buck 8 | # - `buck install -r android/app` - compile, install and run application 9 | # 10 | 11 | lib_deps = [] 12 | 13 | for jarfile in glob(['libs/*.jar']): 14 | name = 'jars__' + jarfile[jarfile.rindex('/') + 1: jarfile.rindex('.jar')] 15 | lib_deps.append(':' + name) 16 | prebuilt_jar( 17 | name = name, 18 | binary_jar = jarfile, 19 | ) 20 | 21 | for aarfile in glob(['libs/*.aar']): 22 | name = 'aars__' + aarfile[aarfile.rindex('/') + 1: aarfile.rindex('.aar')] 23 | lib_deps.append(':' + name) 24 | android_prebuilt_aar( 25 | name = name, 26 | aar = aarfile, 27 | ) 28 | 29 | android_library( 30 | name = "all-libs", 31 | exported_deps = lib_deps, 32 | ) 33 | 34 | android_library( 35 | name = "app-code", 36 | srcs = glob([ 37 | "src/main/java/**/*.java", 38 | ]), 39 | deps = [ 40 | ":all-libs", 41 | ":build_config", 42 | ":res", 43 | ], 44 | ) 45 | 46 | android_build_config( 47 | name = "build_config", 48 | package = "com.offlineimage", 49 | ) 50 | 51 | android_resource( 52 | name = "res", 53 | package = "com.offlineimage", 54 | res = "src/main/res", 55 | ) 56 | 57 | android_binary( 58 | name = "app", 59 | keystore = "//android/keystores:debug", 60 | manifest = "src/main/AndroidManifest.xml", 61 | package_type = "debug", 62 | deps = [ 63 | ":app-code", 64 | ], 65 | ) 66 | -------------------------------------------------------------------------------- /examples/offline-image/android/app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: "com.android.application" 2 | 3 | import com.android.build.OutputFile 4 | 5 | /** 6 | * The react.gradle file registers a task for each build variant (e.g. bundleDebugJsAndAssets 7 | * and bundleReleaseJsAndAssets). 8 | * These basically call `react-native bundle` with the correct arguments during the Android build 9 | * cycle. By default, bundleDebugJsAndAssets is skipped, as in debug/dev mode we prefer to load the 10 | * bundle directly from the development server. Below you can see all the possible configurations 11 | * and their defaults. If you decide to add a configuration block, make sure to add it before the 12 | * `apply from: "../../node_modules/react-native/react.gradle"` line. 13 | * 14 | * project.ext.react = [ 15 | * // the name of the generated asset file containing your JS bundle 16 | * bundleAssetName: "index.android.bundle", 17 | * 18 | * // the entry file for bundle generation 19 | * entryFile: "index.android.js", 20 | * 21 | * // whether to bundle JS and assets in debug mode 22 | * bundleInDebug: false, 23 | * 24 | * // whether to bundle JS and assets in release mode 25 | * bundleInRelease: true, 26 | * 27 | * // whether to bundle JS and assets in another build variant (if configured). 28 | * // See http://tools.android.com/tech-docs/new-build-system/user-guide#TOC-Build-Variants 29 | * // The configuration property can be in the following formats 30 | * // 'bundleIn${productFlavor}${buildType}' 31 | * // 'bundleIn${buildType}' 32 | * // bundleInFreeDebug: true, 33 | * // bundleInPaidRelease: true, 34 | * // bundleInBeta: true, 35 | * 36 | * // whether to disable dev mode in custom build variants (by default only disabled in release) 37 | * // for example: to disable dev mode in the staging build type (if configured) 38 | * devDisabledInStaging: true, 39 | * // The configuration property can be in the following formats 40 | * // 'devDisabledIn${productFlavor}${buildType}' 41 | * // 'devDisabledIn${buildType}' 42 | * 43 | * // the root of your project, i.e. where "package.json" lives 44 | * root: "../../", 45 | * 46 | * // where to put the JS bundle asset in debug mode 47 | * jsBundleDirDebug: "$buildDir/intermediates/assets/debug", 48 | * 49 | * // where to put the JS bundle asset in release mode 50 | * jsBundleDirRelease: "$buildDir/intermediates/assets/release", 51 | * 52 | * // where to put drawable resources / React Native assets, e.g. the ones you use via 53 | * // require('./image.png')), in debug mode 54 | * resourcesDirDebug: "$buildDir/intermediates/res/merged/debug", 55 | * 56 | * // where to put drawable resources / React Native assets, e.g. the ones you use via 57 | * // require('./image.png')), in release mode 58 | * resourcesDirRelease: "$buildDir/intermediates/res/merged/release", 59 | * 60 | * // by default the gradle tasks are skipped if none of the JS files or assets change; this means 61 | * // that we don't look at files in android/ or ios/ to determine whether the tasks are up to 62 | * // date; if you have any other folders that you want to ignore for performance reasons (gradle 63 | * // indexes the entire tree), add them here. Alternatively, if you have JS files in android/ 64 | * // for example, you might want to remove it from here. 65 | * inputExcludes: ["android/**", "ios/**"], 66 | * 67 | * // override which node gets called and with what additional arguments 68 | * nodeExecutableAndArgs: ["node"], 69 | * 70 | * // supply additional arguments to the packager 71 | * extraPackagerArgs: [] 72 | * ] 73 | */ 74 | 75 | apply from: "../../node_modules/react-native/react.gradle" 76 | 77 | /** 78 | * Set this to true to create two separate APKs instead of one: 79 | * - An APK that only works on ARM devices 80 | * - An APK that only works on x86 devices 81 | * The advantage is the size of the APK is reduced by about 4MB. 82 | * Upload all the APKs to the Play Store and people will download 83 | * the correct one based on the CPU architecture of their device. 84 | */ 85 | def enableSeparateBuildPerCPUArchitecture = false 86 | 87 | /** 88 | * Run Proguard to shrink the Java bytecode in release builds. 89 | */ 90 | def enableProguardInReleaseBuilds = false 91 | 92 | android { 93 | compileSdkVersion 23 94 | buildToolsVersion "23.0.1" 95 | 96 | defaultConfig { 97 | applicationId "com.offlineimage" 98 | minSdkVersion 16 99 | targetSdkVersion 22 100 | versionCode 1 101 | versionName "1.0" 102 | ndk { 103 | abiFilters "armeabi-v7a", "x86" 104 | } 105 | } 106 | splits { 107 | abi { 108 | reset() 109 | enable enableSeparateBuildPerCPUArchitecture 110 | universalApk false // If true, also generate a universal APK 111 | include "armeabi-v7a", "x86" 112 | } 113 | } 114 | buildTypes { 115 | release { 116 | minifyEnabled enableProguardInReleaseBuilds 117 | proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro" 118 | } 119 | } 120 | // applicationVariants are e.g. debug, release 121 | applicationVariants.all { variant -> 122 | variant.outputs.each { output -> 123 | // For each separate APK per architecture, set a unique version code as described here: 124 | // http://tools.android.com/tech-docs/new-build-system/user-guide/apk-splits 125 | def versionCodes = ["armeabi-v7a":1, "x86":2] 126 | def abi = output.getFilter(OutputFile.ABI) 127 | if (abi != null) { // null for the universal-debug, universal-release variants 128 | output.versionCodeOverride = 129 | versionCodes.get(abi) * 1048576 + defaultConfig.versionCode 130 | } 131 | } 132 | } 133 | } 134 | 135 | dependencies { 136 | compile project(':rn-fetch-blob') 137 | compile fileTree(dir: "libs", include: ["*.jar"]) 138 | compile "com.android.support:appcompat-v7:23.0.1" 139 | compile "com.facebook.react:react-native:+" // From node_modules 140 | } 141 | 142 | // Run this once to be able to run the application with BUCK 143 | // puts all compile dependencies into folder libs for BUCK to use 144 | task copyDownloadableDepsToLibs(type: Copy) { 145 | from configurations.compile 146 | into 'libs' 147 | } 148 | -------------------------------------------------------------------------------- /examples/offline-image/android/app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # By default, the flags in this file are appended to flags specified 3 | # in /usr/local/Cellar/android-sdk/24.3.3/tools/proguard/proguard-android.txt 4 | # You can edit the include path and order by changing the proguardFiles 5 | # directive in build.gradle. 6 | # 7 | # For more details, see 8 | # http://developer.android.com/guide/developing/tools/proguard.html 9 | 10 | # Add any project specific keep options here: 11 | 12 | # If your project uses WebView with JS, uncomment the following 13 | # and specify the fully qualified class name to the JavaScript interface 14 | # class: 15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 16 | # public *; 17 | #} 18 | 19 | # Disabling obfuscation is useful if you collect stack traces from production crashes 20 | # (unless you are using a system that supports de-obfuscate the stack traces). 21 | -dontobfuscate 22 | 23 | # React Native 24 | 25 | # Keep our interfaces so they can be used by other ProGuard rules. 26 | # See http://sourceforge.net/p/proguard/bugs/466/ 27 | -keep,allowobfuscation @interface com.facebook.proguard.annotations.DoNotStrip 28 | -keep,allowobfuscation @interface com.facebook.proguard.annotations.KeepGettersAndSetters 29 | -keep,allowobfuscation @interface com.facebook.common.internal.DoNotStrip 30 | 31 | # Do not strip any method/class that is annotated with @DoNotStrip 32 | -keep @com.facebook.proguard.annotations.DoNotStrip class * 33 | -keep @com.facebook.common.internal.DoNotStrip class * 34 | -keepclassmembers class * { 35 | @com.facebook.proguard.annotations.DoNotStrip *; 36 | @com.facebook.common.internal.DoNotStrip *; 37 | } 38 | 39 | -keepclassmembers @com.facebook.proguard.annotations.KeepGettersAndSetters class * { 40 | void set*(***); 41 | *** get*(); 42 | } 43 | 44 | -keep class * extends com.facebook.react.bridge.JavaScriptModule { *; } 45 | -keep class * extends com.facebook.react.bridge.NativeModule { *; } 46 | -keepclassmembers,includedescriptorclasses class * { native ; } 47 | -keepclassmembers class * { @com.facebook.react.uimanager.UIProp ; } 48 | -keepclassmembers class * { @com.facebook.react.uimanager.annotations.ReactProp ; } 49 | -keepclassmembers class * { @com.facebook.react.uimanager.annotations.ReactPropGroup ; } 50 | 51 | -dontwarn com.facebook.react.** 52 | 53 | # TextLayoutBuilder uses a non-public Android constructor within StaticLayout. 54 | # See libs/proxy/src/main/java/com/facebook/fbui/textlayoutbuilder/proxy for details. 55 | -dontwarn android.text.StaticLayout 56 | 57 | # okhttp 58 | 59 | -keepattributes Signature 60 | -keepattributes *Annotation* 61 | -keep class okhttp3.** { *; } 62 | -keep interface okhttp3.** { *; } 63 | -dontwarn okhttp3.** 64 | 65 | # okio 66 | 67 | -keep class sun.misc.Unsafe { *; } 68 | -dontwarn java.nio.file.* 69 | -dontwarn org.codehaus.mojo.animal_sniffer.IgnoreJRERequirement 70 | -dontwarn okio.** 71 | -------------------------------------------------------------------------------- /examples/offline-image/android/app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 5 | 6 | 7 | 8 | 9 | 10 | 13 | 14 | 20 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | -------------------------------------------------------------------------------- /examples/offline-image/android/app/src/main/java/com/offlineimage/MainActivity.java: -------------------------------------------------------------------------------- 1 | package com.offlineimage; 2 | 3 | import com.facebook.react.ReactActivity; 4 | 5 | public class MainActivity extends ReactActivity { 6 | 7 | /** 8 | * Returns the name of the main component registered from JavaScript. 9 | * This is used to schedule rendering of the component. 10 | */ 11 | @Override 12 | protected String getMainComponentName() { 13 | return "offlineimage"; 14 | } 15 | } 16 | -------------------------------------------------------------------------------- /examples/offline-image/android/app/src/main/java/com/offlineimage/MainApplication.java: -------------------------------------------------------------------------------- 1 | package com.offlineimage; 2 | 3 | import android.app.Application; 4 | 5 | import com.facebook.react.ReactApplication; 6 | import com.RNFetchBlob.RNFetchBlobPackage; 7 | import com.facebook.react.ReactNativeHost; 8 | import com.facebook.react.ReactPackage; 9 | import com.facebook.react.shell.MainReactPackage; 10 | import com.facebook.soloader.SoLoader; 11 | 12 | import java.util.Arrays; 13 | import java.util.List; 14 | 15 | public class MainApplication extends Application implements ReactApplication { 16 | 17 | private final ReactNativeHost mReactNativeHost = new ReactNativeHost(this) { 18 | @Override 19 | public boolean getUseDeveloperSupport() { 20 | return BuildConfig.DEBUG; 21 | } 22 | 23 | @Override 24 | protected List getPackages() { 25 | return Arrays.asList( 26 | new MainReactPackage(), 27 | new RNFetchBlobPackage() 28 | ); 29 | } 30 | }; 31 | 32 | @Override 33 | public ReactNativeHost getReactNativeHost() { 34 | return mReactNativeHost; 35 | } 36 | 37 | @Override 38 | public void onCreate() { 39 | super.onCreate(); 40 | SoLoader.init(this, /* native exopackage */ false); 41 | } 42 | } 43 | -------------------------------------------------------------------------------- /examples/offline-image/android/app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/code-and-co/react-native-image-offline/82fb5aab47b8c9743ec3c7edd2994fdc509a92cd/examples/offline-image/android/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /examples/offline-image/android/app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/code-and-co/react-native-image-offline/82fb5aab47b8c9743ec3c7edd2994fdc509a92cd/examples/offline-image/android/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /examples/offline-image/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/code-and-co/react-native-image-offline/82fb5aab47b8c9743ec3c7edd2994fdc509a92cd/examples/offline-image/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /examples/offline-image/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/code-and-co/react-native-image-offline/82fb5aab47b8c9743ec3c7edd2994fdc509a92cd/examples/offline-image/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /examples/offline-image/android/app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | React Native Offline Images 3 | 4 | -------------------------------------------------------------------------------- /examples/offline-image/android/app/src/main/res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /examples/offline-image/android/build.gradle: -------------------------------------------------------------------------------- 1 | // Top-level build file where you can add configuration options common to all sub-projects/modules. 2 | 3 | buildscript { 4 | repositories { 5 | jcenter() 6 | } 7 | dependencies { 8 | classpath 'com.android.tools.build:gradle:2.2.3' 9 | 10 | // NOTE: Do not place your application dependencies here; they belong 11 | // in the individual module build.gradle files 12 | } 13 | } 14 | 15 | allprojects { 16 | repositories { 17 | mavenLocal() 18 | jcenter() 19 | maven { 20 | // All of React Native (JS, Obj-C sources, Android binaries) is installed from npm 21 | url "$rootDir/../node_modules/react-native/android" 22 | } 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /examples/offline-image/android/gradle.properties: -------------------------------------------------------------------------------- 1 | # Project-wide Gradle settings. 2 | 3 | # IDE (e.g. Android Studio) users: 4 | # Gradle settings configured through the IDE *will override* 5 | # any settings specified in this file. 6 | 7 | # For more details on how to configure your build environment visit 8 | # http://www.gradle.org/docs/current/userguide/build_environment.html 9 | 10 | # Specifies the JVM arguments used for the daemon process. 11 | # The setting is particularly useful for tweaking memory settings. 12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m 13 | # org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8 14 | 15 | # When configured, Gradle will run in incubating parallel mode. 16 | # This option should only be used with decoupled projects. More details, visit 17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects 18 | # org.gradle.parallel=true 19 | 20 | android.useDeprecatedNdk=true 21 | -------------------------------------------------------------------------------- /examples/offline-image/android/gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/code-and-co/react-native-image-offline/82fb5aab47b8c9743ec3c7edd2994fdc509a92cd/examples/offline-image/android/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /examples/offline-image/android/gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | distributionBase=GRADLE_USER_HOME 2 | distributionPath=wrapper/dists 3 | zipStoreBase=GRADLE_USER_HOME 4 | zipStorePath=wrapper/dists 5 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.14.1-all.zip 6 | -------------------------------------------------------------------------------- /examples/offline-image/android/gradlew: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | ############################################################################## 4 | ## 5 | ## Gradle start up script for UN*X 6 | ## 7 | ############################################################################## 8 | 9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 10 | DEFAULT_JVM_OPTS="" 11 | 12 | APP_NAME="Gradle" 13 | APP_BASE_NAME=`basename "$0"` 14 | 15 | # Use the maximum available, or set MAX_FD != -1 to use that value. 16 | MAX_FD="maximum" 17 | 18 | warn ( ) { 19 | echo "$*" 20 | } 21 | 22 | die ( ) { 23 | echo 24 | echo "$*" 25 | echo 26 | exit 1 27 | } 28 | 29 | # OS specific support (must be 'true' or 'false'). 30 | cygwin=false 31 | msys=false 32 | darwin=false 33 | case "`uname`" in 34 | CYGWIN* ) 35 | cygwin=true 36 | ;; 37 | Darwin* ) 38 | darwin=true 39 | ;; 40 | MINGW* ) 41 | msys=true 42 | ;; 43 | esac 44 | 45 | # For Cygwin, ensure paths are in UNIX format before anything is touched. 46 | if $cygwin ; then 47 | [ -n "$JAVA_HOME" ] && JAVA_HOME=`cygpath --unix "$JAVA_HOME"` 48 | fi 49 | 50 | # Attempt to set APP_HOME 51 | # Resolve links: $0 may be a link 52 | PRG="$0" 53 | # Need this for relative symlinks. 54 | while [ -h "$PRG" ] ; do 55 | ls=`ls -ld "$PRG"` 56 | link=`expr "$ls" : '.*-> \(.*\)$'` 57 | if expr "$link" : '/.*' > /dev/null; then 58 | PRG="$link" 59 | else 60 | PRG=`dirname "$PRG"`"/$link" 61 | fi 62 | done 63 | SAVED="`pwd`" 64 | cd "`dirname \"$PRG\"`/" >&- 65 | APP_HOME="`pwd -P`" 66 | cd "$SAVED" >&- 67 | 68 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 69 | 70 | # Determine the Java command to use to start the JVM. 71 | if [ -n "$JAVA_HOME" ] ; then 72 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 73 | # IBM's JDK on AIX uses strange locations for the executables 74 | JAVACMD="$JAVA_HOME/jre/sh/java" 75 | else 76 | JAVACMD="$JAVA_HOME/bin/java" 77 | fi 78 | if [ ! -x "$JAVACMD" ] ; then 79 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 80 | 81 | Please set the JAVA_HOME variable in your environment to match the 82 | location of your Java installation." 83 | fi 84 | else 85 | JAVACMD="java" 86 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 87 | 88 | Please set the JAVA_HOME variable in your environment to match the 89 | location of your Java installation." 90 | fi 91 | 92 | # Increase the maximum file descriptors if we can. 93 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then 94 | MAX_FD_LIMIT=`ulimit -H -n` 95 | if [ $? -eq 0 ] ; then 96 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 97 | MAX_FD="$MAX_FD_LIMIT" 98 | fi 99 | ulimit -n $MAX_FD 100 | if [ $? -ne 0 ] ; then 101 | warn "Could not set maximum file descriptor limit: $MAX_FD" 102 | fi 103 | else 104 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 105 | fi 106 | fi 107 | 108 | # For Darwin, add options to specify how the application appears in the dock 109 | if $darwin; then 110 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 111 | fi 112 | 113 | # For Cygwin, switch paths to Windows format before running java 114 | if $cygwin ; then 115 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 116 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 117 | 118 | # We build the pattern for arguments to be converted via cygpath 119 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 120 | SEP="" 121 | for dir in $ROOTDIRSRAW ; do 122 | ROOTDIRS="$ROOTDIRS$SEP$dir" 123 | SEP="|" 124 | done 125 | OURCYGPATTERN="(^($ROOTDIRS))" 126 | # Add a user-defined pattern to the cygpath arguments 127 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then 128 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 129 | fi 130 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 131 | i=0 132 | for arg in "$@" ; do 133 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 134 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 135 | 136 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 137 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 138 | else 139 | eval `echo args$i`="\"$arg\"" 140 | fi 141 | i=$((i+1)) 142 | done 143 | case $i in 144 | (0) set -- ;; 145 | (1) set -- "$args0" ;; 146 | (2) set -- "$args0" "$args1" ;; 147 | (3) set -- "$args0" "$args1" "$args2" ;; 148 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;; 149 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 150 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 151 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 152 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 153 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 154 | esac 155 | fi 156 | 157 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules 158 | function splitJvmOpts() { 159 | JVM_OPTS=("$@") 160 | } 161 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS 162 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME" 163 | 164 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" 165 | -------------------------------------------------------------------------------- /examples/offline-image/android/gradlew.bat: -------------------------------------------------------------------------------- 1 | @if "%DEBUG%" == "" @echo off 2 | @rem ########################################################################## 3 | @rem 4 | @rem Gradle startup script for Windows 5 | @rem 6 | @rem ########################################################################## 7 | 8 | @rem Set local scope for the variables with windows NT shell 9 | if "%OS%"=="Windows_NT" setlocal 10 | 11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 12 | set DEFAULT_JVM_OPTS= 13 | 14 | set DIRNAME=%~dp0 15 | if "%DIRNAME%" == "" set DIRNAME=. 16 | set APP_BASE_NAME=%~n0 17 | set APP_HOME=%DIRNAME% 18 | 19 | @rem Find java.exe 20 | if defined JAVA_HOME goto findJavaFromJavaHome 21 | 22 | set JAVA_EXE=java.exe 23 | %JAVA_EXE% -version >NUL 2>&1 24 | if "%ERRORLEVEL%" == "0" goto init 25 | 26 | echo. 27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 28 | echo. 29 | echo Please set the JAVA_HOME variable in your environment to match the 30 | echo location of your Java installation. 31 | 32 | goto fail 33 | 34 | :findJavaFromJavaHome 35 | set JAVA_HOME=%JAVA_HOME:"=% 36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 37 | 38 | if exist "%JAVA_EXE%" goto init 39 | 40 | echo. 41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 42 | echo. 43 | echo Please set the JAVA_HOME variable in your environment to match the 44 | echo location of your Java installation. 45 | 46 | goto fail 47 | 48 | :init 49 | @rem Get command-line arguments, handling Windowz variants 50 | 51 | if not "%OS%" == "Windows_NT" goto win9xME_args 52 | if "%@eval[2+2]" == "4" goto 4NT_args 53 | 54 | :win9xME_args 55 | @rem Slurp the command line arguments. 56 | set CMD_LINE_ARGS= 57 | set _SKIP=2 58 | 59 | :win9xME_args_slurp 60 | if "x%~1" == "x" goto execute 61 | 62 | set CMD_LINE_ARGS=%* 63 | goto execute 64 | 65 | :4NT_args 66 | @rem Get arguments from the 4NT Shell from JP Software 67 | set CMD_LINE_ARGS=%$ 68 | 69 | :execute 70 | @rem Setup the command line 71 | 72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 73 | 74 | @rem Execute Gradle 75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% 76 | 77 | :end 78 | @rem End local scope for the variables with windows NT shell 79 | if "%ERRORLEVEL%"=="0" goto mainEnd 80 | 81 | :fail 82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 83 | rem the _cmd.exe /c_ return code! 84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 85 | exit /b 1 86 | 87 | :mainEnd 88 | if "%OS%"=="Windows_NT" endlocal 89 | 90 | :omega 91 | -------------------------------------------------------------------------------- /examples/offline-image/android/keystores/BUCK: -------------------------------------------------------------------------------- 1 | keystore( 2 | name = "debug", 3 | properties = "debug.keystore.properties", 4 | store = "debug.keystore", 5 | visibility = [ 6 | "PUBLIC", 7 | ], 8 | ) 9 | -------------------------------------------------------------------------------- /examples/offline-image/android/keystores/debug.keystore.properties: -------------------------------------------------------------------------------- 1 | key.store=debug.keystore 2 | key.alias=androiddebugkey 3 | key.store.password=android 4 | key.alias.password=android 5 | -------------------------------------------------------------------------------- /examples/offline-image/android/settings.gradle: -------------------------------------------------------------------------------- 1 | rootProject.name = 'offlineimage' 2 | include ':rn-fetch-blob' 3 | project(':rn-fetch-blob').projectDir = new File(rootProject.projectDir, '../node_modules/rn-fetch-blob/android') 4 | 5 | include ':app' 6 | -------------------------------------------------------------------------------- /examples/offline-image/index.android.js: -------------------------------------------------------------------------------- 1 | import { AppRegistry } from 'react-native'; 2 | import App from './src'; 3 | 4 | AppRegistry.registerComponent('offlineimage', () => App); 5 | -------------------------------------------------------------------------------- /examples/offline-image/index.ios.js: -------------------------------------------------------------------------------- 1 | import { AppRegistry } from 'react-native'; 2 | import App from './src'; 3 | 4 | AppRegistry.registerComponent('offlineimage', () => App); 5 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage-tvOS/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleExecutable 8 | $(EXECUTABLE_NAME) 9 | CFBundleIdentifier 10 | org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier) 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | $(PRODUCT_NAME) 15 | CFBundlePackageType 16 | APPL 17 | CFBundleShortVersionString 18 | 1.0 19 | CFBundleSignature 20 | ???? 21 | CFBundleVersion 22 | 1 23 | LSRequiresIPhoneOS 24 | 25 | UILaunchStoryboardName 26 | LaunchScreen 27 | UIRequiredDeviceCapabilities 28 | 29 | armv7 30 | 31 | UISupportedInterfaceOrientations 32 | 33 | UIInterfaceOrientationPortrait 34 | UIInterfaceOrientationLandscapeLeft 35 | UIInterfaceOrientationLandscapeRight 36 | 37 | UIViewControllerBasedStatusBarAppearance 38 | 39 | NSLocationWhenInUseUsageDescription 40 | 41 | NSAppTransportSecurity 42 | 43 | 44 | NSExceptionDomains 45 | 46 | localhost 47 | 48 | NSExceptionAllowsInsecureHTTPLoads 49 | 50 | 51 | 52 | 53 | 54 | 55 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage-tvOSTests/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleExecutable 8 | $(EXECUTABLE_NAME) 9 | CFBundleIdentifier 10 | org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier) 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | $(PRODUCT_NAME) 15 | CFBundlePackageType 16 | BNDL 17 | CFBundleShortVersionString 18 | 1.0 19 | CFBundleSignature 20 | ???? 21 | CFBundleVersion 22 | 1 23 | 24 | 25 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage.xcodeproj/project.pbxproj: -------------------------------------------------------------------------------- 1 | // !$*UTF8*$! 2 | { 3 | archiveVersion = 1; 4 | classes = { 5 | }; 6 | objectVersion = 46; 7 | objects = { 8 | /* Begin PBXBuildFile section */ 9 | 00C302E51ABCBA2D00DB3ED1 /* libRCTActionSheet.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302AC1ABCB8CE00DB3ED1 /* libRCTActionSheet.a */; }; 10 | 00C302E71ABCBA2D00DB3ED1 /* libRCTGeolocation.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302BA1ABCB90400DB3ED1 /* libRCTGeolocation.a */; }; 11 | 00C302E81ABCBA2D00DB3ED1 /* libRCTImage.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */; }; 12 | 00C302E91ABCBA2D00DB3ED1 /* libRCTNetwork.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */; }; 13 | 00C302EA1ABCBA2D00DB3ED1 /* libRCTVibration.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */; }; 14 | 00E356F31AD99517003FC87E /* offlineimageTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 00E356F21AD99517003FC87E /* offlineimageTests.m */; }; 15 | 133E29F31AD74F7200F7D852 /* libRCTLinking.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 78C398B91ACF4ADC00677621 /* libRCTLinking.a */; }; 16 | 139105C61AF99C1200B5F7CC /* libRCTSettings.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */; }; 17 | 139FDEF61B0652A700C62182 /* libRCTWebSocket.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */; }; 18 | 13B07FBC1A68108700A75B9A /* AppDelegate.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB01A68108700A75B9A /* AppDelegate.m */; }; 19 | 13B07FBD1A68108700A75B9A /* LaunchScreen.xib in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB11A68108700A75B9A /* LaunchScreen.xib */; }; 20 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB51A68108700A75B9A /* Images.xcassets */; }; 21 | 13B07FC11A68108700A75B9A /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB71A68108700A75B9A /* main.m */; }; 22 | 140ED2AC1D01E1AD002B40FF /* libReact.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 146834041AC3E56700842450 /* libReact.a */; }; 23 | 146834051AC3E58100842450 /* libReact.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 146834041AC3E56700842450 /* libReact.a */; }; 24 | 2D02E4BC1E0B4A80006451C7 /* AppDelegate.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB01A68108700A75B9A /* AppDelegate.m */; }; 25 | 2D02E4BD1E0B4A84006451C7 /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB51A68108700A75B9A /* Images.xcassets */; }; 26 | 2D02E4BF1E0B4AB3006451C7 /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB71A68108700A75B9A /* main.m */; }; 27 | 2D02E4C21E0B4AEC006451C7 /* libRCTAnimation-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 5E9157351DD0AC6500FF2AA8 /* libRCTAnimation-tvOS.a */; }; 28 | 2D02E4C31E0B4AEC006451C7 /* libRCTImage-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E841DF850E9000B6D8A /* libRCTImage-tvOS.a */; }; 29 | 2D02E4C41E0B4AEC006451C7 /* libRCTLinking-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E881DF850E9000B6D8A /* libRCTLinking-tvOS.a */; }; 30 | 2D02E4C51E0B4AEC006451C7 /* libRCTNetwork-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E8C1DF850E9000B6D8A /* libRCTNetwork-tvOS.a */; }; 31 | 2D02E4C61E0B4AEC006451C7 /* libRCTSettings-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E901DF850E9000B6D8A /* libRCTSettings-tvOS.a */; }; 32 | 2D02E4C71E0B4AEC006451C7 /* libRCTText-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E941DF850E9000B6D8A /* libRCTText-tvOS.a */; }; 33 | 2D02E4C81E0B4AEC006451C7 /* libRCTWebSocket-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E991DF850E9000B6D8A /* libRCTWebSocket-tvOS.a */; }; 34 | 2D02E4C91E0B4AEC006451C7 /* libReact.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3EA31DF850E9000B6D8A /* libReact.a */; }; 35 | 2DCD954D1E0B4F2C00145EB5 /* offlineimageTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 00E356F21AD99517003FC87E /* offlineimageTests.m */; }; 36 | 5E9157361DD0AC6A00FF2AA8 /* libRCTAnimation.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 5E9157331DD0AC6500FF2AA8 /* libRCTAnimation.a */; }; 37 | 832341BD1AAA6AB300B99B32 /* libRCTText.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 832341B51AAA6A8300B99B32 /* libRCTText.a */; }; 38 | C3B5687602144040A606066B /* libRNFetchBlob.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 9407ED48B25E4C6D85F3498A /* libRNFetchBlob.a */; }; 39 | /* End PBXBuildFile section */ 40 | 41 | /* Begin PBXContainerItemProxy section */ 42 | 00C302AB1ABCB8CE00DB3ED1 /* PBXContainerItemProxy */ = { 43 | isa = PBXContainerItemProxy; 44 | containerPortal = 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */; 45 | proxyType = 2; 46 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 47 | remoteInfo = RCTActionSheet; 48 | }; 49 | 00C302B91ABCB90400DB3ED1 /* PBXContainerItemProxy */ = { 50 | isa = PBXContainerItemProxy; 51 | containerPortal = 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */; 52 | proxyType = 2; 53 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 54 | remoteInfo = RCTGeolocation; 55 | }; 56 | 00C302BF1ABCB91800DB3ED1 /* PBXContainerItemProxy */ = { 57 | isa = PBXContainerItemProxy; 58 | containerPortal = 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */; 59 | proxyType = 2; 60 | remoteGlobalIDString = 58B5115D1A9E6B3D00147676; 61 | remoteInfo = RCTImage; 62 | }; 63 | 00C302DB1ABCB9D200DB3ED1 /* PBXContainerItemProxy */ = { 64 | isa = PBXContainerItemProxy; 65 | containerPortal = 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */; 66 | proxyType = 2; 67 | remoteGlobalIDString = 58B511DB1A9E6C8500147676; 68 | remoteInfo = RCTNetwork; 69 | }; 70 | 00C302E31ABCB9EE00DB3ED1 /* PBXContainerItemProxy */ = { 71 | isa = PBXContainerItemProxy; 72 | containerPortal = 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */; 73 | proxyType = 2; 74 | remoteGlobalIDString = 832C81801AAF6DEF007FA2F7; 75 | remoteInfo = RCTVibration; 76 | }; 77 | 00E356F41AD99517003FC87E /* PBXContainerItemProxy */ = { 78 | isa = PBXContainerItemProxy; 79 | containerPortal = 83CBB9F71A601CBA00E9B192 /* Project object */; 80 | proxyType = 1; 81 | remoteGlobalIDString = 13B07F861A680F5B00A75B9A; 82 | remoteInfo = offlineimage; 83 | }; 84 | 139105C01AF99BAD00B5F7CC /* PBXContainerItemProxy */ = { 85 | isa = PBXContainerItemProxy; 86 | containerPortal = 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */; 87 | proxyType = 2; 88 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 89 | remoteInfo = RCTSettings; 90 | }; 91 | 139FDEF31B06529B00C62182 /* PBXContainerItemProxy */ = { 92 | isa = PBXContainerItemProxy; 93 | containerPortal = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */; 94 | proxyType = 2; 95 | remoteGlobalIDString = 3C86DF461ADF2C930047B81A; 96 | remoteInfo = RCTWebSocket; 97 | }; 98 | 146834031AC3E56700842450 /* PBXContainerItemProxy */ = { 99 | isa = PBXContainerItemProxy; 100 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 101 | proxyType = 2; 102 | remoteGlobalIDString = 83CBBA2E1A601D0E00E9B192; 103 | remoteInfo = React; 104 | }; 105 | 2D02E4911E0B4A5D006451C7 /* PBXContainerItemProxy */ = { 106 | isa = PBXContainerItemProxy; 107 | containerPortal = 83CBB9F71A601CBA00E9B192 /* Project object */; 108 | proxyType = 1; 109 | remoteGlobalIDString = 2D02E47A1E0B4A5D006451C7; 110 | remoteInfo = "offlineimage-tvOS"; 111 | }; 112 | 3DAD3E831DF850E9000B6D8A /* PBXContainerItemProxy */ = { 113 | isa = PBXContainerItemProxy; 114 | containerPortal = 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */; 115 | proxyType = 2; 116 | remoteGlobalIDString = 2D2A283A1D9B042B00D4039D; 117 | remoteInfo = "RCTImage-tvOS"; 118 | }; 119 | 3DAD3E871DF850E9000B6D8A /* PBXContainerItemProxy */ = { 120 | isa = PBXContainerItemProxy; 121 | containerPortal = 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */; 122 | proxyType = 2; 123 | remoteGlobalIDString = 2D2A28471D9B043800D4039D; 124 | remoteInfo = "RCTLinking-tvOS"; 125 | }; 126 | 3DAD3E8B1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 127 | isa = PBXContainerItemProxy; 128 | containerPortal = 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */; 129 | proxyType = 2; 130 | remoteGlobalIDString = 2D2A28541D9B044C00D4039D; 131 | remoteInfo = "RCTNetwork-tvOS"; 132 | }; 133 | 3DAD3E8F1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 134 | isa = PBXContainerItemProxy; 135 | containerPortal = 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */; 136 | proxyType = 2; 137 | remoteGlobalIDString = 2D2A28611D9B046600D4039D; 138 | remoteInfo = "RCTSettings-tvOS"; 139 | }; 140 | 3DAD3E931DF850E9000B6D8A /* PBXContainerItemProxy */ = { 141 | isa = PBXContainerItemProxy; 142 | containerPortal = 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */; 143 | proxyType = 2; 144 | remoteGlobalIDString = 2D2A287B1D9B048500D4039D; 145 | remoteInfo = "RCTText-tvOS"; 146 | }; 147 | 3DAD3E981DF850E9000B6D8A /* PBXContainerItemProxy */ = { 148 | isa = PBXContainerItemProxy; 149 | containerPortal = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */; 150 | proxyType = 2; 151 | remoteGlobalIDString = 2D2A28881D9B049200D4039D; 152 | remoteInfo = "RCTWebSocket-tvOS"; 153 | }; 154 | 3DAD3EA21DF850E9000B6D8A /* PBXContainerItemProxy */ = { 155 | isa = PBXContainerItemProxy; 156 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 157 | proxyType = 2; 158 | remoteGlobalIDString = 2D2A28131D9B038B00D4039D; 159 | remoteInfo = "React-tvOS"; 160 | }; 161 | 3DAD3EA41DF850E9000B6D8A /* PBXContainerItemProxy */ = { 162 | isa = PBXContainerItemProxy; 163 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 164 | proxyType = 2; 165 | remoteGlobalIDString = 3D3C059A1DE3340900C268FA; 166 | remoteInfo = yoga; 167 | }; 168 | 3DAD3EA61DF850E9000B6D8A /* PBXContainerItemProxy */ = { 169 | isa = PBXContainerItemProxy; 170 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 171 | proxyType = 2; 172 | remoteGlobalIDString = 3D3C06751DE3340C00C268FA; 173 | remoteInfo = "yoga-tvOS"; 174 | }; 175 | 3DAD3EA81DF850E9000B6D8A /* PBXContainerItemProxy */ = { 176 | isa = PBXContainerItemProxy; 177 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 178 | proxyType = 2; 179 | remoteGlobalIDString = 3D3CD9251DE5FBEC00167DC4; 180 | remoteInfo = cxxreact; 181 | }; 182 | 3DAD3EAA1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 183 | isa = PBXContainerItemProxy; 184 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 185 | proxyType = 2; 186 | remoteGlobalIDString = 3D3CD9321DE5FBEE00167DC4; 187 | remoteInfo = "cxxreact-tvOS"; 188 | }; 189 | 3DAD3EAC1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 190 | isa = PBXContainerItemProxy; 191 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 192 | proxyType = 2; 193 | remoteGlobalIDString = 3D3CD90B1DE5FBD600167DC4; 194 | remoteInfo = jschelpers; 195 | }; 196 | 3DAD3EAE1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 197 | isa = PBXContainerItemProxy; 198 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 199 | proxyType = 2; 200 | remoteGlobalIDString = 3D3CD9181DE5FBD800167DC4; 201 | remoteInfo = "jschelpers-tvOS"; 202 | }; 203 | 5E9157321DD0AC6500FF2AA8 /* PBXContainerItemProxy */ = { 204 | isa = PBXContainerItemProxy; 205 | containerPortal = 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */; 206 | proxyType = 2; 207 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 208 | remoteInfo = RCTAnimation; 209 | }; 210 | 5E9157341DD0AC6500FF2AA8 /* PBXContainerItemProxy */ = { 211 | isa = PBXContainerItemProxy; 212 | containerPortal = 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */; 213 | proxyType = 2; 214 | remoteGlobalIDString = 2D2A28201D9B03D100D4039D; 215 | remoteInfo = "RCTAnimation-tvOS"; 216 | }; 217 | 78C398B81ACF4ADC00677621 /* PBXContainerItemProxy */ = { 218 | isa = PBXContainerItemProxy; 219 | containerPortal = 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */; 220 | proxyType = 2; 221 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 222 | remoteInfo = RCTLinking; 223 | }; 224 | 832341B41AAA6A8300B99B32 /* PBXContainerItemProxy */ = { 225 | isa = PBXContainerItemProxy; 226 | containerPortal = 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */; 227 | proxyType = 2; 228 | remoteGlobalIDString = 58B5119B1A9E6C1200147676; 229 | remoteInfo = RCTText; 230 | }; 231 | /* End PBXContainerItemProxy section */ 232 | 233 | /* Begin PBXFileReference section */ 234 | 008F07F21AC5B25A0029DE68 /* main.jsbundle */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; path = main.jsbundle; sourceTree = ""; }; 235 | 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTActionSheet.xcodeproj; path = "../node_modules/react-native/Libraries/ActionSheetIOS/RCTActionSheet.xcodeproj"; sourceTree = ""; }; 236 | 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTGeolocation.xcodeproj; path = "../node_modules/react-native/Libraries/Geolocation/RCTGeolocation.xcodeproj"; sourceTree = ""; }; 237 | 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTImage.xcodeproj; path = "../node_modules/react-native/Libraries/Image/RCTImage.xcodeproj"; sourceTree = ""; }; 238 | 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTNetwork.xcodeproj; path = "../node_modules/react-native/Libraries/Network/RCTNetwork.xcodeproj"; sourceTree = ""; }; 239 | 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTVibration.xcodeproj; path = "../node_modules/react-native/Libraries/Vibration/RCTVibration.xcodeproj"; sourceTree = ""; }; 240 | 00E356EE1AD99517003FC87E /* offlineimageTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = offlineimageTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; 241 | 00E356F11AD99517003FC87E /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; 242 | 00E356F21AD99517003FC87E /* offlineimageTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = offlineimageTests.m; sourceTree = ""; }; 243 | 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTSettings.xcodeproj; path = "../node_modules/react-native/Libraries/Settings/RCTSettings.xcodeproj"; sourceTree = ""; }; 244 | 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTWebSocket.xcodeproj; path = "../node_modules/react-native/Libraries/WebSocket/RCTWebSocket.xcodeproj"; sourceTree = ""; }; 245 | 13B07F961A680F5B00A75B9A /* offlineimage.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = offlineimage.app; sourceTree = BUILT_PRODUCTS_DIR; }; 246 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = AppDelegate.h; path = offlineimage/AppDelegate.h; sourceTree = ""; }; 247 | 13B07FB01A68108700A75B9A /* AppDelegate.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = AppDelegate.m; path = offlineimage/AppDelegate.m; sourceTree = ""; }; 248 | 13B07FB21A68108700A75B9A /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = Base.lproj/LaunchScreen.xib; sourceTree = ""; }; 249 | 13B07FB51A68108700A75B9A /* Images.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Images.xcassets; path = offlineimage/Images.xcassets; sourceTree = ""; }; 250 | 13B07FB61A68108700A75B9A /* Info.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = Info.plist; path = offlineimage/Info.plist; sourceTree = ""; }; 251 | 13B07FB71A68108700A75B9A /* main.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = main.m; path = offlineimage/main.m; sourceTree = ""; }; 252 | 146833FF1AC3E56700842450 /* React.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = React.xcodeproj; path = "../node_modules/react-native/React/React.xcodeproj"; sourceTree = ""; }; 253 | 2D02E47B1E0B4A5D006451C7 /* offlineimage-tvOS.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = "offlineimage-tvOS.app"; sourceTree = BUILT_PRODUCTS_DIR; }; 254 | 2D02E4901E0B4A5D006451C7 /* offlineimage-tvOSTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = "offlineimage-tvOSTests.xctest"; sourceTree = BUILT_PRODUCTS_DIR; }; 255 | 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTAnimation.xcodeproj; path = "../node_modules/react-native/Libraries/NativeAnimation/RCTAnimation.xcodeproj"; sourceTree = ""; }; 256 | 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTLinking.xcodeproj; path = "../node_modules/react-native/Libraries/LinkingIOS/RCTLinking.xcodeproj"; sourceTree = ""; }; 257 | 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTText.xcodeproj; path = "../node_modules/react-native/Libraries/Text/RCTText.xcodeproj"; sourceTree = ""; }; 258 | 7502282FC1F24C038877D71A /* RNFetchBlob.xcodeproj */ = {isa = PBXFileReference; name = "RNFetchBlob.xcodeproj"; path = "../node_modules/rn-fetch-blob/ios/RNFetchBlob.xcodeproj"; sourceTree = ""; fileEncoding = undefined; lastKnownFileType = wrapper.pb-project; explicitFileType = undefined; includeInIndex = 0; }; 259 | 9407ED48B25E4C6D85F3498A /* libRNFetchBlob.a */ = {isa = PBXFileReference; name = "libRNFetchBlob.a"; path = "libRNFetchBlob.a"; sourceTree = ""; fileEncoding = undefined; lastKnownFileType = archive.ar; explicitFileType = undefined; includeInIndex = 0; }; 260 | /* End PBXFileReference section */ 261 | 262 | /* Begin PBXFrameworksBuildPhase section */ 263 | 00E356EB1AD99517003FC87E /* Frameworks */ = { 264 | isa = PBXFrameworksBuildPhase; 265 | buildActionMask = 2147483647; 266 | files = ( 267 | 140ED2AC1D01E1AD002B40FF /* libReact.a in Frameworks */, 268 | ); 269 | runOnlyForDeploymentPostprocessing = 0; 270 | }; 271 | 13B07F8C1A680F5B00A75B9A /* Frameworks */ = { 272 | isa = PBXFrameworksBuildPhase; 273 | buildActionMask = 2147483647; 274 | files = ( 275 | 146834051AC3E58100842450 /* libReact.a in Frameworks */, 276 | 5E9157361DD0AC6A00FF2AA8 /* libRCTAnimation.a in Frameworks */, 277 | 00C302E51ABCBA2D00DB3ED1 /* libRCTActionSheet.a in Frameworks */, 278 | 00C302E71ABCBA2D00DB3ED1 /* libRCTGeolocation.a in Frameworks */, 279 | 00C302E81ABCBA2D00DB3ED1 /* libRCTImage.a in Frameworks */, 280 | 133E29F31AD74F7200F7D852 /* libRCTLinking.a in Frameworks */, 281 | 00C302E91ABCBA2D00DB3ED1 /* libRCTNetwork.a in Frameworks */, 282 | 139105C61AF99C1200B5F7CC /* libRCTSettings.a in Frameworks */, 283 | 832341BD1AAA6AB300B99B32 /* libRCTText.a in Frameworks */, 284 | 00C302EA1ABCBA2D00DB3ED1 /* libRCTVibration.a in Frameworks */, 285 | 139FDEF61B0652A700C62182 /* libRCTWebSocket.a in Frameworks */, 286 | C3B5687602144040A606066B /* libRNFetchBlob.a in Frameworks */, 287 | ); 288 | runOnlyForDeploymentPostprocessing = 0; 289 | }; 290 | 2D02E4781E0B4A5D006451C7 /* Frameworks */ = { 291 | isa = PBXFrameworksBuildPhase; 292 | buildActionMask = 2147483647; 293 | files = ( 294 | 2D02E4C91E0B4AEC006451C7 /* libReact.a in Frameworks */, 295 | 2D02E4C21E0B4AEC006451C7 /* libRCTAnimation-tvOS.a in Frameworks */, 296 | 2D02E4C31E0B4AEC006451C7 /* libRCTImage-tvOS.a in Frameworks */, 297 | 2D02E4C41E0B4AEC006451C7 /* libRCTLinking-tvOS.a in Frameworks */, 298 | 2D02E4C51E0B4AEC006451C7 /* libRCTNetwork-tvOS.a in Frameworks */, 299 | 2D02E4C61E0B4AEC006451C7 /* libRCTSettings-tvOS.a in Frameworks */, 300 | 2D02E4C71E0B4AEC006451C7 /* libRCTText-tvOS.a in Frameworks */, 301 | 2D02E4C81E0B4AEC006451C7 /* libRCTWebSocket-tvOS.a in Frameworks */, 302 | ); 303 | runOnlyForDeploymentPostprocessing = 0; 304 | }; 305 | 2D02E48D1E0B4A5D006451C7 /* Frameworks */ = { 306 | isa = PBXFrameworksBuildPhase; 307 | buildActionMask = 2147483647; 308 | files = ( 309 | ); 310 | runOnlyForDeploymentPostprocessing = 0; 311 | }; 312 | /* End PBXFrameworksBuildPhase section */ 313 | 314 | /* Begin PBXGroup section */ 315 | 00C302A81ABCB8CE00DB3ED1 /* Products */ = { 316 | isa = PBXGroup; 317 | children = ( 318 | 00C302AC1ABCB8CE00DB3ED1 /* libRCTActionSheet.a */, 319 | ); 320 | name = Products; 321 | sourceTree = ""; 322 | }; 323 | 00C302B61ABCB90400DB3ED1 /* Products */ = { 324 | isa = PBXGroup; 325 | children = ( 326 | 00C302BA1ABCB90400DB3ED1 /* libRCTGeolocation.a */, 327 | ); 328 | name = Products; 329 | sourceTree = ""; 330 | }; 331 | 00C302BC1ABCB91800DB3ED1 /* Products */ = { 332 | isa = PBXGroup; 333 | children = ( 334 | 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */, 335 | 3DAD3E841DF850E9000B6D8A /* libRCTImage-tvOS.a */, 336 | ); 337 | name = Products; 338 | sourceTree = ""; 339 | }; 340 | 00C302D41ABCB9D200DB3ED1 /* Products */ = { 341 | isa = PBXGroup; 342 | children = ( 343 | 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */, 344 | 3DAD3E8C1DF850E9000B6D8A /* libRCTNetwork-tvOS.a */, 345 | ); 346 | name = Products; 347 | sourceTree = ""; 348 | }; 349 | 00C302E01ABCB9EE00DB3ED1 /* Products */ = { 350 | isa = PBXGroup; 351 | children = ( 352 | 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */, 353 | ); 354 | name = Products; 355 | sourceTree = ""; 356 | }; 357 | 00E356EF1AD99517003FC87E /* offlineimageTests */ = { 358 | isa = PBXGroup; 359 | children = ( 360 | 00E356F21AD99517003FC87E /* offlineimageTests.m */, 361 | 00E356F01AD99517003FC87E /* Supporting Files */, 362 | ); 363 | path = offlineimageTests; 364 | sourceTree = ""; 365 | }; 366 | 00E356F01AD99517003FC87E /* Supporting Files */ = { 367 | isa = PBXGroup; 368 | children = ( 369 | 00E356F11AD99517003FC87E /* Info.plist */, 370 | ); 371 | name = "Supporting Files"; 372 | sourceTree = ""; 373 | }; 374 | 139105B71AF99BAD00B5F7CC /* Products */ = { 375 | isa = PBXGroup; 376 | children = ( 377 | 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */, 378 | 3DAD3E901DF850E9000B6D8A /* libRCTSettings-tvOS.a */, 379 | ); 380 | name = Products; 381 | sourceTree = ""; 382 | }; 383 | 139FDEE71B06529A00C62182 /* Products */ = { 384 | isa = PBXGroup; 385 | children = ( 386 | 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */, 387 | 3DAD3E991DF850E9000B6D8A /* libRCTWebSocket-tvOS.a */, 388 | ); 389 | name = Products; 390 | sourceTree = ""; 391 | }; 392 | 13B07FAE1A68108700A75B9A /* offlineimage */ = { 393 | isa = PBXGroup; 394 | children = ( 395 | 008F07F21AC5B25A0029DE68 /* main.jsbundle */, 396 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */, 397 | 13B07FB01A68108700A75B9A /* AppDelegate.m */, 398 | 13B07FB51A68108700A75B9A /* Images.xcassets */, 399 | 13B07FB61A68108700A75B9A /* Info.plist */, 400 | 13B07FB11A68108700A75B9A /* LaunchScreen.xib */, 401 | 13B07FB71A68108700A75B9A /* main.m */, 402 | ); 403 | name = offlineimage; 404 | sourceTree = ""; 405 | }; 406 | 146834001AC3E56700842450 /* Products */ = { 407 | isa = PBXGroup; 408 | children = ( 409 | 146834041AC3E56700842450 /* libReact.a */, 410 | 3DAD3EA31DF850E9000B6D8A /* libReact.a */, 411 | 3DAD3EA51DF850E9000B6D8A /* libyoga.a */, 412 | 3DAD3EA71DF850E9000B6D8A /* libyoga.a */, 413 | 3DAD3EA91DF850E9000B6D8A /* libcxxreact.a */, 414 | 3DAD3EAB1DF850E9000B6D8A /* libcxxreact.a */, 415 | 3DAD3EAD1DF850E9000B6D8A /* libjschelpers.a */, 416 | 3DAD3EAF1DF850E9000B6D8A /* libjschelpers.a */, 417 | ); 418 | name = Products; 419 | sourceTree = ""; 420 | }; 421 | 5E91572E1DD0AC6500FF2AA8 /* Products */ = { 422 | isa = PBXGroup; 423 | children = ( 424 | 5E9157331DD0AC6500FF2AA8 /* libRCTAnimation.a */, 425 | 5E9157351DD0AC6500FF2AA8 /* libRCTAnimation-tvOS.a */, 426 | ); 427 | name = Products; 428 | sourceTree = ""; 429 | }; 430 | 78C398B11ACF4ADC00677621 /* Products */ = { 431 | isa = PBXGroup; 432 | children = ( 433 | 78C398B91ACF4ADC00677621 /* libRCTLinking.a */, 434 | 3DAD3E881DF850E9000B6D8A /* libRCTLinking-tvOS.a */, 435 | ); 436 | name = Products; 437 | sourceTree = ""; 438 | }; 439 | 832341AE1AAA6A7D00B99B32 /* Libraries */ = { 440 | isa = PBXGroup; 441 | children = ( 442 | 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */, 443 | 146833FF1AC3E56700842450 /* React.xcodeproj */, 444 | 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */, 445 | 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */, 446 | 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */, 447 | 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */, 448 | 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */, 449 | 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */, 450 | 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */, 451 | 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */, 452 | 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */, 453 | 7502282FC1F24C038877D71A /* RNFetchBlob.xcodeproj */, 454 | ); 455 | name = Libraries; 456 | sourceTree = ""; 457 | }; 458 | 832341B11AAA6A8300B99B32 /* Products */ = { 459 | isa = PBXGroup; 460 | children = ( 461 | 832341B51AAA6A8300B99B32 /* libRCTText.a */, 462 | 3DAD3E941DF850E9000B6D8A /* libRCTText-tvOS.a */, 463 | ); 464 | name = Products; 465 | sourceTree = ""; 466 | }; 467 | 83CBB9F61A601CBA00E9B192 = { 468 | isa = PBXGroup; 469 | children = ( 470 | 13B07FAE1A68108700A75B9A /* offlineimage */, 471 | 832341AE1AAA6A7D00B99B32 /* Libraries */, 472 | 00E356EF1AD99517003FC87E /* offlineimageTests */, 473 | 83CBBA001A601CBA00E9B192 /* Products */, 474 | ); 475 | indentWidth = 2; 476 | sourceTree = ""; 477 | tabWidth = 2; 478 | }; 479 | 83CBBA001A601CBA00E9B192 /* Products */ = { 480 | isa = PBXGroup; 481 | children = ( 482 | 13B07F961A680F5B00A75B9A /* offlineimage.app */, 483 | 00E356EE1AD99517003FC87E /* offlineimageTests.xctest */, 484 | 2D02E47B1E0B4A5D006451C7 /* offlineimage-tvOS.app */, 485 | 2D02E4901E0B4A5D006451C7 /* offlineimage-tvOSTests.xctest */, 486 | ); 487 | name = Products; 488 | sourceTree = ""; 489 | }; 490 | /* End PBXGroup section */ 491 | 492 | /* Begin PBXNativeTarget section */ 493 | 00E356ED1AD99517003FC87E /* offlineimageTests */ = { 494 | isa = PBXNativeTarget; 495 | buildConfigurationList = 00E357021AD99517003FC87E /* Build configuration list for PBXNativeTarget "offlineimageTests" */; 496 | buildPhases = ( 497 | 00E356EA1AD99517003FC87E /* Sources */, 498 | 00E356EB1AD99517003FC87E /* Frameworks */, 499 | 00E356EC1AD99517003FC87E /* Resources */, 500 | ); 501 | buildRules = ( 502 | ); 503 | dependencies = ( 504 | 00E356F51AD99517003FC87E /* PBXTargetDependency */, 505 | ); 506 | name = offlineimageTests; 507 | productName = offlineimageTests; 508 | productReference = 00E356EE1AD99517003FC87E /* offlineimageTests.xctest */; 509 | productType = "com.apple.product-type.bundle.unit-test"; 510 | }; 511 | 13B07F861A680F5B00A75B9A /* offlineimage */ = { 512 | isa = PBXNativeTarget; 513 | buildConfigurationList = 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "offlineimage" */; 514 | buildPhases = ( 515 | 13B07F871A680F5B00A75B9A /* Sources */, 516 | 13B07F8C1A680F5B00A75B9A /* Frameworks */, 517 | 13B07F8E1A680F5B00A75B9A /* Resources */, 518 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */, 519 | ); 520 | buildRules = ( 521 | ); 522 | dependencies = ( 523 | ); 524 | name = offlineimage; 525 | productName = "Hello World"; 526 | productReference = 13B07F961A680F5B00A75B9A /* offlineimage.app */; 527 | productType = "com.apple.product-type.application"; 528 | }; 529 | 2D02E47A1E0B4A5D006451C7 /* offlineimage-tvOS */ = { 530 | isa = PBXNativeTarget; 531 | buildConfigurationList = 2D02E4BA1E0B4A5E006451C7 /* Build configuration list for PBXNativeTarget "offlineimage-tvOS" */; 532 | buildPhases = ( 533 | 2D02E4771E0B4A5D006451C7 /* Sources */, 534 | 2D02E4781E0B4A5D006451C7 /* Frameworks */, 535 | 2D02E4791E0B4A5D006451C7 /* Resources */, 536 | 2D02E4CB1E0B4B27006451C7 /* Bundle React Native Code And Images */, 537 | ); 538 | buildRules = ( 539 | ); 540 | dependencies = ( 541 | ); 542 | name = "offlineimage-tvOS"; 543 | productName = "offlineimage-tvOS"; 544 | productReference = 2D02E47B1E0B4A5D006451C7 /* offlineimage-tvOS.app */; 545 | productType = "com.apple.product-type.application"; 546 | }; 547 | 2D02E48F1E0B4A5D006451C7 /* offlineimage-tvOSTests */ = { 548 | isa = PBXNativeTarget; 549 | buildConfigurationList = 2D02E4BB1E0B4A5E006451C7 /* Build configuration list for PBXNativeTarget "offlineimage-tvOSTests" */; 550 | buildPhases = ( 551 | 2D02E48C1E0B4A5D006451C7 /* Sources */, 552 | 2D02E48D1E0B4A5D006451C7 /* Frameworks */, 553 | 2D02E48E1E0B4A5D006451C7 /* Resources */, 554 | ); 555 | buildRules = ( 556 | ); 557 | dependencies = ( 558 | 2D02E4921E0B4A5D006451C7 /* PBXTargetDependency */, 559 | ); 560 | name = "offlineimage-tvOSTests"; 561 | productName = "offlineimage-tvOSTests"; 562 | productReference = 2D02E4901E0B4A5D006451C7 /* offlineimage-tvOSTests.xctest */; 563 | productType = "com.apple.product-type.bundle.unit-test"; 564 | }; 565 | /* End PBXNativeTarget section */ 566 | 567 | /* Begin PBXProject section */ 568 | 83CBB9F71A601CBA00E9B192 /* Project object */ = { 569 | isa = PBXProject; 570 | attributes = { 571 | LastUpgradeCheck = 610; 572 | ORGANIZATIONNAME = Facebook; 573 | TargetAttributes = { 574 | 00E356ED1AD99517003FC87E = { 575 | CreatedOnToolsVersion = 6.2; 576 | TestTargetID = 13B07F861A680F5B00A75B9A; 577 | }; 578 | 2D02E47A1E0B4A5D006451C7 = { 579 | CreatedOnToolsVersion = 8.2.1; 580 | ProvisioningStyle = Automatic; 581 | }; 582 | 2D02E48F1E0B4A5D006451C7 = { 583 | CreatedOnToolsVersion = 8.2.1; 584 | ProvisioningStyle = Automatic; 585 | TestTargetID = 2D02E47A1E0B4A5D006451C7; 586 | }; 587 | }; 588 | }; 589 | buildConfigurationList = 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "offlineimage" */; 590 | compatibilityVersion = "Xcode 3.2"; 591 | developmentRegion = English; 592 | hasScannedForEncodings = 0; 593 | knownRegions = ( 594 | en, 595 | Base, 596 | ); 597 | mainGroup = 83CBB9F61A601CBA00E9B192; 598 | productRefGroup = 83CBBA001A601CBA00E9B192 /* Products */; 599 | projectDirPath = ""; 600 | projectReferences = ( 601 | { 602 | ProductGroup = 00C302A81ABCB8CE00DB3ED1 /* Products */; 603 | ProjectRef = 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */; 604 | }, 605 | { 606 | ProductGroup = 5E91572E1DD0AC6500FF2AA8 /* Products */; 607 | ProjectRef = 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */; 608 | }, 609 | { 610 | ProductGroup = 00C302B61ABCB90400DB3ED1 /* Products */; 611 | ProjectRef = 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */; 612 | }, 613 | { 614 | ProductGroup = 00C302BC1ABCB91800DB3ED1 /* Products */; 615 | ProjectRef = 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */; 616 | }, 617 | { 618 | ProductGroup = 78C398B11ACF4ADC00677621 /* Products */; 619 | ProjectRef = 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */; 620 | }, 621 | { 622 | ProductGroup = 00C302D41ABCB9D200DB3ED1 /* Products */; 623 | ProjectRef = 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */; 624 | }, 625 | { 626 | ProductGroup = 139105B71AF99BAD00B5F7CC /* Products */; 627 | ProjectRef = 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */; 628 | }, 629 | { 630 | ProductGroup = 832341B11AAA6A8300B99B32 /* Products */; 631 | ProjectRef = 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */; 632 | }, 633 | { 634 | ProductGroup = 00C302E01ABCB9EE00DB3ED1 /* Products */; 635 | ProjectRef = 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */; 636 | }, 637 | { 638 | ProductGroup = 139FDEE71B06529A00C62182 /* Products */; 639 | ProjectRef = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */; 640 | }, 641 | { 642 | ProductGroup = 146834001AC3E56700842450 /* Products */; 643 | ProjectRef = 146833FF1AC3E56700842450 /* React.xcodeproj */; 644 | }, 645 | ); 646 | projectRoot = ""; 647 | targets = ( 648 | 13B07F861A680F5B00A75B9A /* offlineimage */, 649 | 00E356ED1AD99517003FC87E /* offlineimageTests */, 650 | 2D02E47A1E0B4A5D006451C7 /* offlineimage-tvOS */, 651 | 2D02E48F1E0B4A5D006451C7 /* offlineimage-tvOSTests */, 652 | ); 653 | }; 654 | /* End PBXProject section */ 655 | 656 | /* Begin PBXReferenceProxy section */ 657 | 00C302AC1ABCB8CE00DB3ED1 /* libRCTActionSheet.a */ = { 658 | isa = PBXReferenceProxy; 659 | fileType = archive.ar; 660 | path = libRCTActionSheet.a; 661 | remoteRef = 00C302AB1ABCB8CE00DB3ED1 /* PBXContainerItemProxy */; 662 | sourceTree = BUILT_PRODUCTS_DIR; 663 | }; 664 | 00C302BA1ABCB90400DB3ED1 /* libRCTGeolocation.a */ = { 665 | isa = PBXReferenceProxy; 666 | fileType = archive.ar; 667 | path = libRCTGeolocation.a; 668 | remoteRef = 00C302B91ABCB90400DB3ED1 /* PBXContainerItemProxy */; 669 | sourceTree = BUILT_PRODUCTS_DIR; 670 | }; 671 | 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */ = { 672 | isa = PBXReferenceProxy; 673 | fileType = archive.ar; 674 | path = libRCTImage.a; 675 | remoteRef = 00C302BF1ABCB91800DB3ED1 /* PBXContainerItemProxy */; 676 | sourceTree = BUILT_PRODUCTS_DIR; 677 | }; 678 | 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */ = { 679 | isa = PBXReferenceProxy; 680 | fileType = archive.ar; 681 | path = libRCTNetwork.a; 682 | remoteRef = 00C302DB1ABCB9D200DB3ED1 /* PBXContainerItemProxy */; 683 | sourceTree = BUILT_PRODUCTS_DIR; 684 | }; 685 | 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */ = { 686 | isa = PBXReferenceProxy; 687 | fileType = archive.ar; 688 | path = libRCTVibration.a; 689 | remoteRef = 00C302E31ABCB9EE00DB3ED1 /* PBXContainerItemProxy */; 690 | sourceTree = BUILT_PRODUCTS_DIR; 691 | }; 692 | 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */ = { 693 | isa = PBXReferenceProxy; 694 | fileType = archive.ar; 695 | path = libRCTSettings.a; 696 | remoteRef = 139105C01AF99BAD00B5F7CC /* PBXContainerItemProxy */; 697 | sourceTree = BUILT_PRODUCTS_DIR; 698 | }; 699 | 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */ = { 700 | isa = PBXReferenceProxy; 701 | fileType = archive.ar; 702 | path = libRCTWebSocket.a; 703 | remoteRef = 139FDEF31B06529B00C62182 /* PBXContainerItemProxy */; 704 | sourceTree = BUILT_PRODUCTS_DIR; 705 | }; 706 | 146834041AC3E56700842450 /* libReact.a */ = { 707 | isa = PBXReferenceProxy; 708 | fileType = archive.ar; 709 | path = libReact.a; 710 | remoteRef = 146834031AC3E56700842450 /* PBXContainerItemProxy */; 711 | sourceTree = BUILT_PRODUCTS_DIR; 712 | }; 713 | 3DAD3E841DF850E9000B6D8A /* libRCTImage-tvOS.a */ = { 714 | isa = PBXReferenceProxy; 715 | fileType = archive.ar; 716 | path = "libRCTImage-tvOS.a"; 717 | remoteRef = 3DAD3E831DF850E9000B6D8A /* PBXContainerItemProxy */; 718 | sourceTree = BUILT_PRODUCTS_DIR; 719 | }; 720 | 3DAD3E881DF850E9000B6D8A /* libRCTLinking-tvOS.a */ = { 721 | isa = PBXReferenceProxy; 722 | fileType = archive.ar; 723 | path = "libRCTLinking-tvOS.a"; 724 | remoteRef = 3DAD3E871DF850E9000B6D8A /* PBXContainerItemProxy */; 725 | sourceTree = BUILT_PRODUCTS_DIR; 726 | }; 727 | 3DAD3E8C1DF850E9000B6D8A /* libRCTNetwork-tvOS.a */ = { 728 | isa = PBXReferenceProxy; 729 | fileType = archive.ar; 730 | path = "libRCTNetwork-tvOS.a"; 731 | remoteRef = 3DAD3E8B1DF850E9000B6D8A /* PBXContainerItemProxy */; 732 | sourceTree = BUILT_PRODUCTS_DIR; 733 | }; 734 | 3DAD3E901DF850E9000B6D8A /* libRCTSettings-tvOS.a */ = { 735 | isa = PBXReferenceProxy; 736 | fileType = archive.ar; 737 | path = "libRCTSettings-tvOS.a"; 738 | remoteRef = 3DAD3E8F1DF850E9000B6D8A /* PBXContainerItemProxy */; 739 | sourceTree = BUILT_PRODUCTS_DIR; 740 | }; 741 | 3DAD3E941DF850E9000B6D8A /* libRCTText-tvOS.a */ = { 742 | isa = PBXReferenceProxy; 743 | fileType = archive.ar; 744 | path = "libRCTText-tvOS.a"; 745 | remoteRef = 3DAD3E931DF850E9000B6D8A /* PBXContainerItemProxy */; 746 | sourceTree = BUILT_PRODUCTS_DIR; 747 | }; 748 | 3DAD3E991DF850E9000B6D8A /* libRCTWebSocket-tvOS.a */ = { 749 | isa = PBXReferenceProxy; 750 | fileType = archive.ar; 751 | path = "libRCTWebSocket-tvOS.a"; 752 | remoteRef = 3DAD3E981DF850E9000B6D8A /* PBXContainerItemProxy */; 753 | sourceTree = BUILT_PRODUCTS_DIR; 754 | }; 755 | 3DAD3EA31DF850E9000B6D8A /* libReact.a */ = { 756 | isa = PBXReferenceProxy; 757 | fileType = archive.ar; 758 | path = libReact.a; 759 | remoteRef = 3DAD3EA21DF850E9000B6D8A /* PBXContainerItemProxy */; 760 | sourceTree = BUILT_PRODUCTS_DIR; 761 | }; 762 | 3DAD3EA51DF850E9000B6D8A /* libyoga.a */ = { 763 | isa = PBXReferenceProxy; 764 | fileType = archive.ar; 765 | path = libyoga.a; 766 | remoteRef = 3DAD3EA41DF850E9000B6D8A /* PBXContainerItemProxy */; 767 | sourceTree = BUILT_PRODUCTS_DIR; 768 | }; 769 | 3DAD3EA71DF850E9000B6D8A /* libyoga.a */ = { 770 | isa = PBXReferenceProxy; 771 | fileType = archive.ar; 772 | path = libyoga.a; 773 | remoteRef = 3DAD3EA61DF850E9000B6D8A /* PBXContainerItemProxy */; 774 | sourceTree = BUILT_PRODUCTS_DIR; 775 | }; 776 | 3DAD3EA91DF850E9000B6D8A /* libcxxreact.a */ = { 777 | isa = PBXReferenceProxy; 778 | fileType = archive.ar; 779 | path = libcxxreact.a; 780 | remoteRef = 3DAD3EA81DF850E9000B6D8A /* PBXContainerItemProxy */; 781 | sourceTree = BUILT_PRODUCTS_DIR; 782 | }; 783 | 3DAD3EAB1DF850E9000B6D8A /* libcxxreact.a */ = { 784 | isa = PBXReferenceProxy; 785 | fileType = archive.ar; 786 | path = libcxxreact.a; 787 | remoteRef = 3DAD3EAA1DF850E9000B6D8A /* PBXContainerItemProxy */; 788 | sourceTree = BUILT_PRODUCTS_DIR; 789 | }; 790 | 3DAD3EAD1DF850E9000B6D8A /* libjschelpers.a */ = { 791 | isa = PBXReferenceProxy; 792 | fileType = archive.ar; 793 | path = libjschelpers.a; 794 | remoteRef = 3DAD3EAC1DF850E9000B6D8A /* PBXContainerItemProxy */; 795 | sourceTree = BUILT_PRODUCTS_DIR; 796 | }; 797 | 3DAD3EAF1DF850E9000B6D8A /* libjschelpers.a */ = { 798 | isa = PBXReferenceProxy; 799 | fileType = archive.ar; 800 | path = libjschelpers.a; 801 | remoteRef = 3DAD3EAE1DF850E9000B6D8A /* PBXContainerItemProxy */; 802 | sourceTree = BUILT_PRODUCTS_DIR; 803 | }; 804 | 5E9157331DD0AC6500FF2AA8 /* libRCTAnimation.a */ = { 805 | isa = PBXReferenceProxy; 806 | fileType = archive.ar; 807 | path = libRCTAnimation.a; 808 | remoteRef = 5E9157321DD0AC6500FF2AA8 /* PBXContainerItemProxy */; 809 | sourceTree = BUILT_PRODUCTS_DIR; 810 | }; 811 | 5E9157351DD0AC6500FF2AA8 /* libRCTAnimation-tvOS.a */ = { 812 | isa = PBXReferenceProxy; 813 | fileType = archive.ar; 814 | path = "libRCTAnimation-tvOS.a"; 815 | remoteRef = 5E9157341DD0AC6500FF2AA8 /* PBXContainerItemProxy */; 816 | sourceTree = BUILT_PRODUCTS_DIR; 817 | }; 818 | 78C398B91ACF4ADC00677621 /* libRCTLinking.a */ = { 819 | isa = PBXReferenceProxy; 820 | fileType = archive.ar; 821 | path = libRCTLinking.a; 822 | remoteRef = 78C398B81ACF4ADC00677621 /* PBXContainerItemProxy */; 823 | sourceTree = BUILT_PRODUCTS_DIR; 824 | }; 825 | 832341B51AAA6A8300B99B32 /* libRCTText.a */ = { 826 | isa = PBXReferenceProxy; 827 | fileType = archive.ar; 828 | path = libRCTText.a; 829 | remoteRef = 832341B41AAA6A8300B99B32 /* PBXContainerItemProxy */; 830 | sourceTree = BUILT_PRODUCTS_DIR; 831 | }; 832 | /* End PBXReferenceProxy section */ 833 | 834 | /* Begin PBXResourcesBuildPhase section */ 835 | 00E356EC1AD99517003FC87E /* Resources */ = { 836 | isa = PBXResourcesBuildPhase; 837 | buildActionMask = 2147483647; 838 | files = ( 839 | ); 840 | runOnlyForDeploymentPostprocessing = 0; 841 | }; 842 | 13B07F8E1A680F5B00A75B9A /* Resources */ = { 843 | isa = PBXResourcesBuildPhase; 844 | buildActionMask = 2147483647; 845 | files = ( 846 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */, 847 | 13B07FBD1A68108700A75B9A /* LaunchScreen.xib in Resources */, 848 | ); 849 | runOnlyForDeploymentPostprocessing = 0; 850 | }; 851 | 2D02E4791E0B4A5D006451C7 /* Resources */ = { 852 | isa = PBXResourcesBuildPhase; 853 | buildActionMask = 2147483647; 854 | files = ( 855 | 2D02E4BD1E0B4A84006451C7 /* Images.xcassets in Resources */, 856 | ); 857 | runOnlyForDeploymentPostprocessing = 0; 858 | }; 859 | 2D02E48E1E0B4A5D006451C7 /* Resources */ = { 860 | isa = PBXResourcesBuildPhase; 861 | buildActionMask = 2147483647; 862 | files = ( 863 | ); 864 | runOnlyForDeploymentPostprocessing = 0; 865 | }; 866 | /* End PBXResourcesBuildPhase section */ 867 | 868 | /* Begin PBXShellScriptBuildPhase section */ 869 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */ = { 870 | isa = PBXShellScriptBuildPhase; 871 | buildActionMask = 2147483647; 872 | files = ( 873 | ); 874 | inputPaths = ( 875 | ); 876 | name = "Bundle React Native code and images"; 877 | outputPaths = ( 878 | ); 879 | runOnlyForDeploymentPostprocessing = 0; 880 | shellPath = /bin/sh; 881 | shellScript = "export NODE_BINARY=node\n../node_modules/react-native/scripts/react-native-xcode.sh"; 882 | }; 883 | 2D02E4CB1E0B4B27006451C7 /* Bundle React Native Code And Images */ = { 884 | isa = PBXShellScriptBuildPhase; 885 | buildActionMask = 2147483647; 886 | files = ( 887 | ); 888 | inputPaths = ( 889 | ); 890 | name = "Bundle React Native Code And Images"; 891 | outputPaths = ( 892 | ); 893 | runOnlyForDeploymentPostprocessing = 0; 894 | shellPath = /bin/sh; 895 | shellScript = "export NODE_BINARY=node\n../node_modules/react-native/scripts/react-native-xcode.sh"; 896 | }; 897 | /* End PBXShellScriptBuildPhase section */ 898 | 899 | /* Begin PBXSourcesBuildPhase section */ 900 | 00E356EA1AD99517003FC87E /* Sources */ = { 901 | isa = PBXSourcesBuildPhase; 902 | buildActionMask = 2147483647; 903 | files = ( 904 | 00E356F31AD99517003FC87E /* offlineimageTests.m in Sources */, 905 | ); 906 | runOnlyForDeploymentPostprocessing = 0; 907 | }; 908 | 13B07F871A680F5B00A75B9A /* Sources */ = { 909 | isa = PBXSourcesBuildPhase; 910 | buildActionMask = 2147483647; 911 | files = ( 912 | 13B07FBC1A68108700A75B9A /* AppDelegate.m in Sources */, 913 | 13B07FC11A68108700A75B9A /* main.m in Sources */, 914 | ); 915 | runOnlyForDeploymentPostprocessing = 0; 916 | }; 917 | 2D02E4771E0B4A5D006451C7 /* Sources */ = { 918 | isa = PBXSourcesBuildPhase; 919 | buildActionMask = 2147483647; 920 | files = ( 921 | 2D02E4BF1E0B4AB3006451C7 /* main.m in Sources */, 922 | 2D02E4BC1E0B4A80006451C7 /* AppDelegate.m in Sources */, 923 | ); 924 | runOnlyForDeploymentPostprocessing = 0; 925 | }; 926 | 2D02E48C1E0B4A5D006451C7 /* Sources */ = { 927 | isa = PBXSourcesBuildPhase; 928 | buildActionMask = 2147483647; 929 | files = ( 930 | 2DCD954D1E0B4F2C00145EB5 /* offlineimageTests.m in Sources */, 931 | ); 932 | runOnlyForDeploymentPostprocessing = 0; 933 | }; 934 | /* End PBXSourcesBuildPhase section */ 935 | 936 | /* Begin PBXTargetDependency section */ 937 | 00E356F51AD99517003FC87E /* PBXTargetDependency */ = { 938 | isa = PBXTargetDependency; 939 | target = 13B07F861A680F5B00A75B9A /* offlineimage */; 940 | targetProxy = 00E356F41AD99517003FC87E /* PBXContainerItemProxy */; 941 | }; 942 | 2D02E4921E0B4A5D006451C7 /* PBXTargetDependency */ = { 943 | isa = PBXTargetDependency; 944 | target = 2D02E47A1E0B4A5D006451C7 /* offlineimage-tvOS */; 945 | targetProxy = 2D02E4911E0B4A5D006451C7 /* PBXContainerItemProxy */; 946 | }; 947 | /* End PBXTargetDependency section */ 948 | 949 | /* Begin PBXVariantGroup section */ 950 | 13B07FB11A68108700A75B9A /* LaunchScreen.xib */ = { 951 | isa = PBXVariantGroup; 952 | children = ( 953 | 13B07FB21A68108700A75B9A /* Base */, 954 | ); 955 | name = LaunchScreen.xib; 956 | path = offlineimage; 957 | sourceTree = ""; 958 | }; 959 | /* End PBXVariantGroup section */ 960 | 961 | /* Begin XCBuildConfiguration section */ 962 | 00E356F61AD99517003FC87E /* Debug */ = { 963 | isa = XCBuildConfiguration; 964 | buildSettings = { 965 | BUNDLE_LOADER = "$(TEST_HOST)"; 966 | GCC_PREPROCESSOR_DEFINITIONS = ( 967 | "DEBUG=1", 968 | "$(inherited)", 969 | ); 970 | INFOPLIST_FILE = offlineimageTests/Info.plist; 971 | IPHONEOS_DEPLOYMENT_TARGET = 8.0; 972 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks"; 973 | OTHER_LDFLAGS = ( 974 | "-ObjC", 975 | "-lc++", 976 | ); 977 | PRODUCT_NAME = "$(TARGET_NAME)"; 978 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/offlineimage.app/offlineimage"; 979 | LIBRARY_SEARCH_PATHS = ( 980 | "$(inherited)", 981 | "\"$(SRCROOT)/$(TARGET_NAME)\"", 982 | ); 983 | HEADER_SEARCH_PATHS = ( 984 | "$(inherited)", 985 | "$(SRCROOT)/../node_modules/rn-fetch-blob/ios/**", 986 | ); 987 | }; 988 | name = Debug; 989 | }; 990 | 00E356F71AD99517003FC87E /* Release */ = { 991 | isa = XCBuildConfiguration; 992 | buildSettings = { 993 | BUNDLE_LOADER = "$(TEST_HOST)"; 994 | COPY_PHASE_STRIP = NO; 995 | INFOPLIST_FILE = offlineimageTests/Info.plist; 996 | IPHONEOS_DEPLOYMENT_TARGET = 8.0; 997 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks"; 998 | OTHER_LDFLAGS = ( 999 | "-ObjC", 1000 | "-lc++", 1001 | ); 1002 | PRODUCT_NAME = "$(TARGET_NAME)"; 1003 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/offlineimage.app/offlineimage"; 1004 | LIBRARY_SEARCH_PATHS = ( 1005 | "$(inherited)", 1006 | "\"$(SRCROOT)/$(TARGET_NAME)\"", 1007 | ); 1008 | HEADER_SEARCH_PATHS = ( 1009 | "$(inherited)", 1010 | "$(SRCROOT)/../node_modules/rn-fetch-blob/ios/**", 1011 | ); 1012 | }; 1013 | name = Release; 1014 | }; 1015 | 13B07F941A680F5B00A75B9A /* Debug */ = { 1016 | isa = XCBuildConfiguration; 1017 | buildSettings = { 1018 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 1019 | CURRENT_PROJECT_VERSION = 1; 1020 | DEAD_CODE_STRIPPING = NO; 1021 | INFOPLIST_FILE = offlineimage/Info.plist; 1022 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 1023 | OTHER_LDFLAGS = ( 1024 | "$(inherited)", 1025 | "-ObjC", 1026 | "-lc++", 1027 | ); 1028 | PRODUCT_NAME = offlineimage; 1029 | VERSIONING_SYSTEM = "apple-generic"; 1030 | HEADER_SEARCH_PATHS = ( 1031 | "$(inherited)", 1032 | "$(SRCROOT)/../node_modules/rn-fetch-blob/ios/**", 1033 | ); 1034 | }; 1035 | name = Debug; 1036 | }; 1037 | 13B07F951A680F5B00A75B9A /* Release */ = { 1038 | isa = XCBuildConfiguration; 1039 | buildSettings = { 1040 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 1041 | CURRENT_PROJECT_VERSION = 1; 1042 | INFOPLIST_FILE = offlineimage/Info.plist; 1043 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 1044 | OTHER_LDFLAGS = ( 1045 | "$(inherited)", 1046 | "-ObjC", 1047 | "-lc++", 1048 | ); 1049 | PRODUCT_NAME = offlineimage; 1050 | VERSIONING_SYSTEM = "apple-generic"; 1051 | HEADER_SEARCH_PATHS = ( 1052 | "$(inherited)", 1053 | "$(SRCROOT)/../node_modules/rn-fetch-blob/ios/**", 1054 | ); 1055 | }; 1056 | name = Release; 1057 | }; 1058 | 2D02E4971E0B4A5E006451C7 /* Debug */ = { 1059 | isa = XCBuildConfiguration; 1060 | buildSettings = { 1061 | ASSETCATALOG_COMPILER_APPICON_NAME = "App Icon & Top Shelf Image"; 1062 | ASSETCATALOG_COMPILER_LAUNCHIMAGE_NAME = LaunchImage; 1063 | CLANG_ANALYZER_NONNULL = YES; 1064 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 1065 | CLANG_WARN_INFINITE_RECURSION = YES; 1066 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1067 | DEBUG_INFORMATION_FORMAT = dwarf; 1068 | ENABLE_TESTABILITY = YES; 1069 | GCC_NO_COMMON_BLOCKS = YES; 1070 | INFOPLIST_FILE = "offlineimage-tvOS/Info.plist"; 1071 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 1072 | OTHER_LDFLAGS = ( 1073 | "-ObjC", 1074 | "-lc++", 1075 | ); 1076 | PRODUCT_BUNDLE_IDENTIFIER = "com.facebook.REACT.offlineimage-tvOS"; 1077 | PRODUCT_NAME = "$(TARGET_NAME)"; 1078 | SDKROOT = appletvos; 1079 | TARGETED_DEVICE_FAMILY = 3; 1080 | TVOS_DEPLOYMENT_TARGET = 9.2; 1081 | LIBRARY_SEARCH_PATHS = ( 1082 | "$(inherited)", 1083 | "\"$(SRCROOT)/$(TARGET_NAME)\"", 1084 | ); 1085 | HEADER_SEARCH_PATHS = ( 1086 | "$(inherited)", 1087 | "$(SRCROOT)/../node_modules/rn-fetch-blob/ios/**", 1088 | ); 1089 | }; 1090 | name = Debug; 1091 | }; 1092 | 2D02E4981E0B4A5E006451C7 /* Release */ = { 1093 | isa = XCBuildConfiguration; 1094 | buildSettings = { 1095 | ASSETCATALOG_COMPILER_APPICON_NAME = "App Icon & Top Shelf Image"; 1096 | ASSETCATALOG_COMPILER_LAUNCHIMAGE_NAME = LaunchImage; 1097 | CLANG_ANALYZER_NONNULL = YES; 1098 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 1099 | CLANG_WARN_INFINITE_RECURSION = YES; 1100 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1101 | COPY_PHASE_STRIP = NO; 1102 | DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; 1103 | GCC_NO_COMMON_BLOCKS = YES; 1104 | INFOPLIST_FILE = "offlineimage-tvOS/Info.plist"; 1105 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 1106 | OTHER_LDFLAGS = ( 1107 | "-ObjC", 1108 | "-lc++", 1109 | ); 1110 | PRODUCT_BUNDLE_IDENTIFIER = "com.facebook.REACT.offlineimage-tvOS"; 1111 | PRODUCT_NAME = "$(TARGET_NAME)"; 1112 | SDKROOT = appletvos; 1113 | TARGETED_DEVICE_FAMILY = 3; 1114 | TVOS_DEPLOYMENT_TARGET = 9.2; 1115 | LIBRARY_SEARCH_PATHS = ( 1116 | "$(inherited)", 1117 | "\"$(SRCROOT)/$(TARGET_NAME)\"", 1118 | ); 1119 | HEADER_SEARCH_PATHS = ( 1120 | "$(inherited)", 1121 | "$(SRCROOT)/../node_modules/rn-fetch-blob/ios/**", 1122 | ); 1123 | }; 1124 | name = Release; 1125 | }; 1126 | 2D02E4991E0B4A5E006451C7 /* Debug */ = { 1127 | isa = XCBuildConfiguration; 1128 | buildSettings = { 1129 | BUNDLE_LOADER = "$(TEST_HOST)"; 1130 | CLANG_ANALYZER_NONNULL = YES; 1131 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 1132 | CLANG_WARN_INFINITE_RECURSION = YES; 1133 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1134 | DEBUG_INFORMATION_FORMAT = dwarf; 1135 | ENABLE_TESTABILITY = YES; 1136 | GCC_NO_COMMON_BLOCKS = YES; 1137 | INFOPLIST_FILE = "offlineimage-tvOSTests/Info.plist"; 1138 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks"; 1139 | PRODUCT_BUNDLE_IDENTIFIER = "com.facebook.REACT.offlineimage-tvOSTests"; 1140 | PRODUCT_NAME = "$(TARGET_NAME)"; 1141 | SDKROOT = appletvos; 1142 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/offlineimage-tvOS.app/offlineimage-tvOS"; 1143 | TVOS_DEPLOYMENT_TARGET = 10.1; 1144 | LIBRARY_SEARCH_PATHS = ( 1145 | "$(inherited)", 1146 | "\"$(SRCROOT)/$(TARGET_NAME)\"", 1147 | ); 1148 | }; 1149 | name = Debug; 1150 | }; 1151 | 2D02E49A1E0B4A5E006451C7 /* Release */ = { 1152 | isa = XCBuildConfiguration; 1153 | buildSettings = { 1154 | BUNDLE_LOADER = "$(TEST_HOST)"; 1155 | CLANG_ANALYZER_NONNULL = YES; 1156 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 1157 | CLANG_WARN_INFINITE_RECURSION = YES; 1158 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1159 | COPY_PHASE_STRIP = NO; 1160 | DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; 1161 | GCC_NO_COMMON_BLOCKS = YES; 1162 | INFOPLIST_FILE = "offlineimage-tvOSTests/Info.plist"; 1163 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks"; 1164 | PRODUCT_BUNDLE_IDENTIFIER = "com.facebook.REACT.offlineimage-tvOSTests"; 1165 | PRODUCT_NAME = "$(TARGET_NAME)"; 1166 | SDKROOT = appletvos; 1167 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/offlineimage-tvOS.app/offlineimage-tvOS"; 1168 | TVOS_DEPLOYMENT_TARGET = 10.1; 1169 | LIBRARY_SEARCH_PATHS = ( 1170 | "$(inherited)", 1171 | "\"$(SRCROOT)/$(TARGET_NAME)\"", 1172 | ); 1173 | }; 1174 | name = Release; 1175 | }; 1176 | 83CBBA201A601CBA00E9B192 /* Debug */ = { 1177 | isa = XCBuildConfiguration; 1178 | buildSettings = { 1179 | ALWAYS_SEARCH_USER_PATHS = NO; 1180 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; 1181 | CLANG_CXX_LIBRARY = "libc++"; 1182 | CLANG_ENABLE_MODULES = YES; 1183 | CLANG_ENABLE_OBJC_ARC = YES; 1184 | CLANG_WARN_BOOL_CONVERSION = YES; 1185 | CLANG_WARN_CONSTANT_CONVERSION = YES; 1186 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 1187 | CLANG_WARN_EMPTY_BODY = YES; 1188 | CLANG_WARN_ENUM_CONVERSION = YES; 1189 | CLANG_WARN_INT_CONVERSION = YES; 1190 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 1191 | CLANG_WARN_UNREACHABLE_CODE = YES; 1192 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 1193 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; 1194 | COPY_PHASE_STRIP = NO; 1195 | ENABLE_STRICT_OBJC_MSGSEND = YES; 1196 | GCC_C_LANGUAGE_STANDARD = gnu99; 1197 | GCC_DYNAMIC_NO_PIC = NO; 1198 | GCC_OPTIMIZATION_LEVEL = 0; 1199 | GCC_PREPROCESSOR_DEFINITIONS = ( 1200 | "DEBUG=1", 1201 | "$(inherited)", 1202 | ); 1203 | GCC_SYMBOLS_PRIVATE_EXTERN = NO; 1204 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 1205 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 1206 | GCC_WARN_UNDECLARED_SELECTOR = YES; 1207 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 1208 | GCC_WARN_UNUSED_FUNCTION = YES; 1209 | GCC_WARN_UNUSED_VARIABLE = YES; 1210 | IPHONEOS_DEPLOYMENT_TARGET = 8.0; 1211 | MTL_ENABLE_DEBUG_INFO = YES; 1212 | ONLY_ACTIVE_ARCH = YES; 1213 | SDKROOT = iphoneos; 1214 | }; 1215 | name = Debug; 1216 | }; 1217 | 83CBBA211A601CBA00E9B192 /* Release */ = { 1218 | isa = XCBuildConfiguration; 1219 | buildSettings = { 1220 | ALWAYS_SEARCH_USER_PATHS = NO; 1221 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; 1222 | CLANG_CXX_LIBRARY = "libc++"; 1223 | CLANG_ENABLE_MODULES = YES; 1224 | CLANG_ENABLE_OBJC_ARC = YES; 1225 | CLANG_WARN_BOOL_CONVERSION = YES; 1226 | CLANG_WARN_CONSTANT_CONVERSION = YES; 1227 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 1228 | CLANG_WARN_EMPTY_BODY = YES; 1229 | CLANG_WARN_ENUM_CONVERSION = YES; 1230 | CLANG_WARN_INT_CONVERSION = YES; 1231 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 1232 | CLANG_WARN_UNREACHABLE_CODE = YES; 1233 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 1234 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; 1235 | COPY_PHASE_STRIP = YES; 1236 | ENABLE_NS_ASSERTIONS = NO; 1237 | ENABLE_STRICT_OBJC_MSGSEND = YES; 1238 | GCC_C_LANGUAGE_STANDARD = gnu99; 1239 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 1240 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 1241 | GCC_WARN_UNDECLARED_SELECTOR = YES; 1242 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 1243 | GCC_WARN_UNUSED_FUNCTION = YES; 1244 | GCC_WARN_UNUSED_VARIABLE = YES; 1245 | IPHONEOS_DEPLOYMENT_TARGET = 8.0; 1246 | MTL_ENABLE_DEBUG_INFO = NO; 1247 | SDKROOT = iphoneos; 1248 | VALIDATE_PRODUCT = YES; 1249 | }; 1250 | name = Release; 1251 | }; 1252 | /* End XCBuildConfiguration section */ 1253 | 1254 | /* Begin XCConfigurationList section */ 1255 | 00E357021AD99517003FC87E /* Build configuration list for PBXNativeTarget "offlineimageTests" */ = { 1256 | isa = XCConfigurationList; 1257 | buildConfigurations = ( 1258 | 00E356F61AD99517003FC87E /* Debug */, 1259 | 00E356F71AD99517003FC87E /* Release */, 1260 | ); 1261 | defaultConfigurationIsVisible = 0; 1262 | defaultConfigurationName = Release; 1263 | }; 1264 | 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "offlineimage" */ = { 1265 | isa = XCConfigurationList; 1266 | buildConfigurations = ( 1267 | 13B07F941A680F5B00A75B9A /* Debug */, 1268 | 13B07F951A680F5B00A75B9A /* Release */, 1269 | ); 1270 | defaultConfigurationIsVisible = 0; 1271 | defaultConfigurationName = Release; 1272 | }; 1273 | 2D02E4BA1E0B4A5E006451C7 /* Build configuration list for PBXNativeTarget "offlineimage-tvOS" */ = { 1274 | isa = XCConfigurationList; 1275 | buildConfigurations = ( 1276 | 2D02E4971E0B4A5E006451C7 /* Debug */, 1277 | 2D02E4981E0B4A5E006451C7 /* Release */, 1278 | ); 1279 | defaultConfigurationIsVisible = 0; 1280 | defaultConfigurationName = Release; 1281 | }; 1282 | 2D02E4BB1E0B4A5E006451C7 /* Build configuration list for PBXNativeTarget "offlineimage-tvOSTests" */ = { 1283 | isa = XCConfigurationList; 1284 | buildConfigurations = ( 1285 | 2D02E4991E0B4A5E006451C7 /* Debug */, 1286 | 2D02E49A1E0B4A5E006451C7 /* Release */, 1287 | ); 1288 | defaultConfigurationIsVisible = 0; 1289 | defaultConfigurationName = Release; 1290 | }; 1291 | 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "offlineimage" */ = { 1292 | isa = XCConfigurationList; 1293 | buildConfigurations = ( 1294 | 83CBBA201A601CBA00E9B192 /* Debug */, 1295 | 83CBBA211A601CBA00E9B192 /* Release */, 1296 | ); 1297 | defaultConfigurationIsVisible = 0; 1298 | defaultConfigurationName = Release; 1299 | }; 1300 | /* End XCConfigurationList section */ 1301 | }; 1302 | rootObject = 83CBB9F71A601CBA00E9B192 /* Project object */; 1303 | } 1304 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage.xcodeproj/xcshareddata/xcschemes/offlineimage-tvOS.xcscheme: -------------------------------------------------------------------------------- 1 | 2 | 5 | 8 | 9 | 15 | 21 | 22 | 23 | 29 | 35 | 36 | 37 | 43 | 49 | 50 | 51 | 52 | 53 | 58 | 59 | 61 | 67 | 68 | 69 | 70 | 71 | 77 | 78 | 79 | 80 | 81 | 82 | 92 | 94 | 100 | 101 | 102 | 103 | 104 | 105 | 111 | 113 | 119 | 120 | 121 | 122 | 124 | 125 | 128 | 129 | 130 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage.xcodeproj/xcshareddata/xcschemes/offlineimage.xcscheme: -------------------------------------------------------------------------------- 1 | 2 | 5 | 8 | 9 | 15 | 21 | 22 | 23 | 29 | 35 | 36 | 37 | 43 | 49 | 50 | 51 | 52 | 53 | 58 | 59 | 61 | 67 | 68 | 69 | 70 | 71 | 77 | 78 | 79 | 80 | 81 | 82 | 92 | 94 | 100 | 101 | 102 | 103 | 104 | 105 | 111 | 113 | 119 | 120 | 121 | 122 | 124 | 125 | 128 | 129 | 130 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage/AppDelegate.h: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) 2015-present, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | */ 9 | 10 | #import 11 | 12 | @interface AppDelegate : UIResponder 13 | 14 | @property (nonatomic, strong) UIWindow *window; 15 | 16 | @end 17 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage/AppDelegate.m: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) 2015-present, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | */ 9 | 10 | #import "AppDelegate.h" 11 | 12 | #import 13 | #import 14 | 15 | @implementation AppDelegate 16 | 17 | - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 18 | { 19 | NSURL *jsCodeLocation; 20 | 21 | jsCodeLocation = [[RCTBundleURLProvider sharedSettings] jsBundleURLForBundleRoot:@"index.ios" fallbackResource:nil]; 22 | 23 | RCTRootView *rootView = [[RCTRootView alloc] initWithBundleURL:jsCodeLocation 24 | moduleName:@"offlineimage" 25 | initialProperties:nil 26 | launchOptions:launchOptions]; 27 | rootView.backgroundColor = [[UIColor alloc] initWithRed:1.0f green:1.0f blue:1.0f alpha:1]; 28 | 29 | self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds]; 30 | UIViewController *rootViewController = [UIViewController new]; 31 | rootViewController.view = rootView; 32 | self.window.rootViewController = rootViewController; 33 | [self.window makeKeyAndVisible]; 34 | return YES; 35 | } 36 | 37 | @end 38 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage/Base.lproj/LaunchScreen.xib: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 21 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage/Images.xcassets/AppIcon.appiconset/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "images" : [ 3 | { 4 | "idiom" : "iphone", 5 | "size" : "29x29", 6 | "scale" : "2x" 7 | }, 8 | { 9 | "idiom" : "iphone", 10 | "size" : "29x29", 11 | "scale" : "3x" 12 | }, 13 | { 14 | "idiom" : "iphone", 15 | "size" : "40x40", 16 | "scale" : "2x" 17 | }, 18 | { 19 | "idiom" : "iphone", 20 | "size" : "40x40", 21 | "scale" : "3x" 22 | }, 23 | { 24 | "idiom" : "iphone", 25 | "size" : "60x60", 26 | "scale" : "2x" 27 | }, 28 | { 29 | "idiom" : "iphone", 30 | "size" : "60x60", 31 | "scale" : "3x" 32 | } 33 | ], 34 | "info" : { 35 | "version" : 1, 36 | "author" : "xcode" 37 | } 38 | } -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleDisplayName 8 | React Native Offline Images 9 | CFBundleExecutable 10 | $(EXECUTABLE_NAME) 11 | CFBundleIdentifier 12 | org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier) 13 | CFBundleInfoDictionaryVersion 14 | 6.0 15 | CFBundleName 16 | $(PRODUCT_NAME) 17 | CFBundlePackageType 18 | APPL 19 | CFBundleShortVersionString 20 | 1.0 21 | CFBundleSignature 22 | ???? 23 | CFBundleVersion 24 | 1 25 | LSRequiresIPhoneOS 26 | 27 | UILaunchStoryboardName 28 | LaunchScreen 29 | UIRequiredDeviceCapabilities 30 | 31 | armv7 32 | 33 | UISupportedInterfaceOrientations 34 | 35 | UIInterfaceOrientationPortrait 36 | UIInterfaceOrientationLandscapeLeft 37 | UIInterfaceOrientationLandscapeRight 38 | 39 | UIViewControllerBasedStatusBarAppearance 40 | 41 | NSLocationWhenInUseUsageDescription 42 | 43 | NSAppTransportSecurity 44 | 45 | 46 | NSExceptionDomains 47 | 48 | localhost 49 | 50 | NSExceptionAllowsInsecureHTTPLoads 51 | 52 | 53 | 54 | 55 | 56 | 57 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimage/main.m: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) 2015-present, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | */ 9 | 10 | #import 11 | 12 | #import "AppDelegate.h" 13 | 14 | int main(int argc, char * argv[]) { 15 | @autoreleasepool { 16 | return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class])); 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimageTests/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleExecutable 8 | $(EXECUTABLE_NAME) 9 | CFBundleIdentifier 10 | org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier) 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | $(PRODUCT_NAME) 15 | CFBundlePackageType 16 | BNDL 17 | CFBundleShortVersionString 18 | 1.0 19 | CFBundleSignature 20 | ???? 21 | CFBundleVersion 22 | 1 23 | 24 | 25 | -------------------------------------------------------------------------------- /examples/offline-image/ios/offlineimageTests/offlineimageTests.m: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) 2015-present, Facebook, Inc. 3 | * All rights reserved. 4 | * 5 | * This source code is licensed under the BSD-style license found in the 6 | * LICENSE file in the root directory of this source tree. An additional grant 7 | * of patent rights can be found in the PATENTS file in the same directory. 8 | */ 9 | 10 | #import 11 | #import 12 | 13 | #import 14 | #import 15 | 16 | #define TIMEOUT_SECONDS 600 17 | #define TEXT_TO_LOOK_FOR @"Welcome to React Native!" 18 | 19 | @interface offlineimageTests : XCTestCase 20 | 21 | @end 22 | 23 | @implementation offlineimageTests 24 | 25 | - (BOOL)findSubviewInView:(UIView *)view matching:(BOOL(^)(UIView *view))test 26 | { 27 | if (test(view)) { 28 | return YES; 29 | } 30 | for (UIView *subview in [view subviews]) { 31 | if ([self findSubviewInView:subview matching:test]) { 32 | return YES; 33 | } 34 | } 35 | return NO; 36 | } 37 | 38 | - (void)testRendersWelcomeScreen 39 | { 40 | UIViewController *vc = [[[RCTSharedApplication() delegate] window] rootViewController]; 41 | NSDate *date = [NSDate dateWithTimeIntervalSinceNow:TIMEOUT_SECONDS]; 42 | BOOL foundElement = NO; 43 | 44 | __block NSString *redboxError = nil; 45 | RCTSetLogFunction(^(RCTLogLevel level, RCTLogSource source, NSString *fileName, NSNumber *lineNumber, NSString *message) { 46 | if (level >= RCTLogLevelError) { 47 | redboxError = message; 48 | } 49 | }); 50 | 51 | while ([date timeIntervalSinceNow] > 0 && !foundElement && !redboxError) { 52 | [[NSRunLoop mainRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]]; 53 | [[NSRunLoop mainRunLoop] runMode:NSRunLoopCommonModes beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]]; 54 | 55 | foundElement = [self findSubviewInView:vc.view matching:^BOOL(UIView *view) { 56 | if ([view.accessibilityLabel isEqualToString:TEXT_TO_LOOK_FOR]) { 57 | return YES; 58 | } 59 | return NO; 60 | }]; 61 | } 62 | 63 | RCTSetLogFunction(RCTDefaultLogFunction); 64 | 65 | XCTAssertNil(redboxError, @"RedBox error: %@", redboxError); 66 | XCTAssertTrue(foundElement, @"Couldn't find element with text '%@' in %d seconds", TEXT_TO_LOOK_FOR, TIMEOUT_SECONDS); 67 | } 68 | 69 | 70 | @end 71 | -------------------------------------------------------------------------------- /examples/offline-image/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "offline-image", 3 | "version": "0.1.0", 4 | "private": true, 5 | "scripts": { 6 | "start": "react-native start", 7 | "clean-ios": "rm -rf $TMPDIR/react-* && watchman watch-del-all && rm -rf ios/build/* && rm -rf node_modules/ && yarn cache clean && yarn install", 8 | "droid": "react-native run-android", 9 | "ios": "react-native run-ios", 10 | "clean-droid": "rm -rf $TMPDIR/react-* && watchman watch-del-all && rm -rf android/build/* && rm -rf node_modules/ && yarn cache clean && yarn install", 11 | "test": "node node_modules/jest/bin/jest.js --watch" 12 | }, 13 | "jest": { 14 | "preset": "jest-expo" 15 | }, 16 | "dependencies": { 17 | "react": "16.0.0-alpha.12", 18 | "react-native": "^0.47.0", 19 | "react-native-image-offline": "file:../..", 20 | "rn-fetch-blob": "^0.10.12" 21 | }, 22 | "devDependencies": { 23 | "babel-preset-react-native-stage-0": "^1.0.1", 24 | "jest-expo": "~20.0.0", 25 | "react-test-renderer": "16.0.0-alpha.12" 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /examples/offline-image/src/assets/images.js: -------------------------------------------------------------------------------- 1 | const images = { 2 | fallbackSource: require('./images/fallbackSource.png'), 3 | }; 4 | 5 | export default images; -------------------------------------------------------------------------------- /examples/offline-image/src/assets/images/fallbackSource.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/code-and-co/react-native-image-offline/82fb5aab47b8c9743ec3c7edd2994fdc509a92cd/examples/offline-image/src/assets/images/fallbackSource.png -------------------------------------------------------------------------------- /examples/offline-image/src/assets/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@assets", 3 | "version": "0.0.1", 4 | "dependencies": { 5 | 6 | } 7 | } 8 | -------------------------------------------------------------------------------- /examples/offline-image/src/index.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import { ActivityIndicator, Dimensions, ScrollView, TouchableOpacity, StyleSheet, Text, View } from 'react-native'; 3 | import { OfflineImage, OfflineImageStore } from 'react-native-image-offline'; 4 | import Images from '@assets/images'; 5 | 6 | const width = Dimensions.get('window').width; 7 | export default class App extends React.Component { 8 | 9 | constructor(props) { 10 | super(props); 11 | this.state = { 12 | reStoreCompleted: false, 13 | loadImages: false, 14 | }; 15 | } 16 | 17 | componentWillMount() { 18 | OfflineImageStore.restore({ 19 | name: 'My_Image_gallery', 20 | imageRemoveTimeout: 120, // expire image after 120 seconds, default is 3 days if you don't provide this property. 21 | debugMode: true, 22 | }, () => { 23 | console.log('Restore completed and callback called !'); 24 | // Restore completed!! 25 | this.setState({ reStoreCompleted: true }); 26 | 27 | // Preload images 28 | // Note: We recommend call this method on `restore` completion! 29 | OfflineImageStore.preLoad([ 30 | 'https://wallpaperbrowse.com/media/images/mobileswall-047.jpg', 31 | 'https://wallpaperbrowse.com/media/images/wallpaper-for-mobile-13.jpg', 32 | 'https://wallpaperbrowse.com/media/images/tvrcnkbcgeirbxcmsbfz.jpg', 33 | 'https://wallpaperbrowse.com/media/images/hd-wallpapers-1080p-for-mobile-2015.jpg', 34 | 'https://wallpaperbrowse.com/media/images/mobileswall-043.jpg', 35 | 'https://wallpaperbrowse.com/media/images/hd-wallpapers-for-mobile-2015.png', 36 | 'https://wallpaperbrowse.com/media/images/download_ZNNDLIt.jpg' 37 | ]); 38 | }); 39 | } 40 | 41 | /** 42 | * This is just testing logic to show `removeExpiredImages` and `clearStore` usage!! 43 | */ 44 | componentDidMount() { 45 | // Clean all the images 46 | 47 | // OfflineImageStore.clearStore(() => { 48 | // console.log('Hurray!! clearStore completed callback called'); 49 | // }); 50 | } 51 | 52 | render() { 53 | if (!this.state.reStoreCompleted) { 54 | return ( 55 | 65 | ); 66 | } 67 | if(this.state.reStoreCompleted && !this.state.loadImages) { 68 | return ( 69 | 70 | { 73 | this.setState({ 74 | loadImages: true 75 | })} 76 | }> 77 | Click to Load Images! 78 | 79 | 80 | ); 81 | } 82 | 83 | return ( 84 | 85 | React native offline image 86 | 92 | { 95 | console.log('Loading finished for image with path: ', sourceUri) 96 | }} 97 | reloadImage = { true } 98 | resizeMode={'cover'} 99 | fallbackSource={ Images.fallbackSource } 100 | style={ { width: '99%', height: 110, margin: 5 } } 101 | source={ { uri: 'https://wallpaperbrowse.com/media/images/wallpaper-for-mobile-13.jpg' } }/> 102 | { 105 | console.log('Loading finished for image with path: ', sourceUri) 106 | }} 107 | resizeMode={'cover'} 108 | style={ { width: '99%', height: 110, margin: 5 } } 109 | source={ { uri: 'https://wallpaperbrowse.com/media/images/tvrcnkbcgeirbxcmsbfz.jpg' } }/> 110 | { 113 | console.log('Loading finished for image with path: ', sourceUri) 114 | }} 115 | resizeMode={'cover'} 116 | style={ { width: '99%', height: 110, margin: 5 } } 117 | source={ { uri: 'https://wallpaperbrowse.com/media/images/mobileswall-043.jpg' } }/> 118 | { 121 | console.log('Loading finished for image with path: ', sourceUri) 122 | }} 123 | resizeMode={'cover'} 124 | fallbackSource={ Images.fallbackSource } 125 | style={ { width: '99%', height: 110, margin: 5 } } 126 | source={ { uri: 'https://wallpaperbrowse.com/media/images/butterfly-wallpaper_SzlKJB8.jpeg' } }/> 127 | 128 | ); 129 | } 130 | } 131 | 132 | const styles = StyleSheet.create({ 133 | scrollView: { 134 | height: '100%', 135 | }, 136 | container: { 137 | flex: 1, 138 | backgroundColor: '#fff', 139 | alignItems: 'center', 140 | justifyContent: 'center', 141 | }, 142 | 143 | }); 144 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "react-native-image-offline", 3 | "version": "0.1.9", 4 | "main": "src/index.js", 5 | "repository": "https://github.com/code-and-co/react-native-image-offline.git", 6 | "author": "code&co.", 7 | "license": "MIT", 8 | "keywords": [ 9 | "react-native", 10 | "image offline", 11 | "image persist", 12 | "image cache" 13 | ], 14 | "peerDependencies": { 15 | "react": "*", 16 | "react-native": "*", 17 | "rn-fetch-blob": "0.10.*" 18 | }, 19 | "dependencies": { 20 | "crypto-js": "^3.1.9-1" 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /screenshot.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/code-and-co/react-native-image-offline/82fb5aab47b8c9743ec3c7edd2994fdc509a92cd/screenshot.png -------------------------------------------------------------------------------- /src/OfflineImage.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import PropTypes from 'prop-types'; 3 | import { ImageBackground, Platform } from 'react-native'; 4 | 5 | import offlineImageStore from './OfflineImageStore'; 6 | 7 | const FILE_PREFIX = Platform.OS === 'ios' ? '' : 'file://'; 8 | 9 | /** 10 | * Wrapper class for React Image {@link https://facebook.github.io/react-native/docs/image.html}. 11 | * This component can get the cached image's device file path as source path. 12 | */ 13 | class OfflineImage extends React.Component { 14 | 15 | constructor(props) { 16 | super(props); 17 | this.state = { 18 | path: undefined, 19 | }; 20 | 21 | this.handler = this.handler.bind(this); 22 | } 23 | 24 | /** 25 | * Callback function triggered after image downloaded or if already exist in offline store 26 | */ 27 | handler = (sourceUri, path) => { 28 | const { onLoadEnd, source } = this.props; 29 | if (source && source.uri && source.uri === sourceUri) { 30 | this.setState({ path: path }); 31 | if (path && onLoadEnd) { 32 | onLoadEnd(sourceUri); 33 | } 34 | } 35 | }; 36 | 37 | // FIXME : Get rid of return true and improve render component logic 38 | shouldComponentUpdate(nextProps, nextState) { 39 | return true 40 | // if (!nextState.path) return true 41 | // return this.state.path !== nextState.path || this.props.opacity !== nextProps.opacity; 42 | } 43 | 44 | componentWillReceiveProps(nextProps) { 45 | const nextSource = nextProps.source; 46 | const reloadImage = nextProps.reloadImage; 47 | 48 | const source = this.props.source; 49 | if (nextSource.uri !== source.uri){ 50 | const offlinePath = offlineImageStore.getImageOfflinePath(nextSource.uri); 51 | this.setState({ path: offlinePath }); 52 | offlineImageStore.subscribe(source, this.handler, reloadImage); 53 | } 54 | } 55 | 56 | componentWillUnmount(){ 57 | const { source } = this.props; 58 | if (source.uri) { 59 | // Subscribe so that we can re-render once image downloaded! 60 | offlineImageStore.unsubscribe(source, this.handler); 61 | } 62 | } 63 | 64 | componentWillMount() { 65 | /** 66 | * Always download and update image in offline store if 'reloadImage' === 'always', however 67 | * Case 1: Show offline image if already exist 68 | * Case 2: Show Fallback image if given until image gets downloaded 69 | * Case 3: Never cache image if property 'reloadImage' === never 70 | */ 71 | const { source, reloadImage } = this.props; 72 | 73 | // TODO: check source type as 'ImageURISource' 74 | // Download only if property 'uri' exists 75 | if (source.uri) { 76 | // Get image offline path if already exist else it returns undefined 77 | const offlinePath = offlineImageStore.getImageOfflinePath(source.uri); 78 | this.setState({ path: offlinePath }); 79 | 80 | // Subscribe so that we can re-render once image downloaded! 81 | offlineImageStore.subscribe(source, this.handler, reloadImage); 82 | } 83 | } 84 | 85 | // this.props.fallBackSource // Show default image as fallbackImage(If exist) until actual image has been loaded. 86 | render() { 87 | const { fallbackSource, source, component } = this.props; 88 | let sourceImage = source; 89 | 90 | // Replace source.uri with offline image path instead waiting for image to download from server 91 | if (source.uri) { 92 | if (this.state.path) { 93 | sourceImage = { 94 | uri: FILE_PREFIX + this.state.path, 95 | }; 96 | } else if (fallbackSource) { // Show fallback image until we download actual image if not able to download show fallback image only! 97 | sourceImage = fallbackSource; 98 | } 99 | } else { 100 | sourceImage = fallbackSource; 101 | } 102 | 103 | const componentProps = { 104 | ...this.props, 105 | source: this.state.path ? sourceImage : source, 106 | }; 107 | 108 | if (component) { 109 | const Component = component; 110 | return ( 111 | { this.props.children } 112 | ); 113 | } 114 | 115 | // Default component would be 'ImageBackground' to render 116 | return ( 117 | { this.props.children } 118 | ); 119 | } 120 | 121 | } 122 | 123 | OfflineImage.propTypes = { 124 | //fallbackSource: PropTypes.int, 125 | component: PropTypes.func, 126 | reloadImage: PropTypes.bool, 127 | onLoadEnd: PropTypes.func, 128 | }; 129 | 130 | export default OfflineImage; 131 | -------------------------------------------------------------------------------- /src/OfflineImageStore.js: -------------------------------------------------------------------------------- 1 | import { AsyncStorage } from 'react-native'; 2 | import RNFetchBlob from 'rn-fetch-blob'; 3 | 4 | const SHA1 = require('crypto-js/sha1'); 5 | 6 | /** 7 | * Primary class responsible with all operations required to communicate with Offline Store! 8 | * 9 | */ 10 | class OfflineImageStore { 11 | 12 | // TODOs 13 | // A component should only subscribe only once 14 | // Check necessities of using async functions instead of normal ones 15 | 16 | constructor(name, storeImageTimeout) { 17 | if (!OfflineImageStore.instance) { 18 | OfflineImageStore.instance = this; 19 | this.entries = {}; 20 | 21 | this.store = { 22 | name,// Application should set their own application store name. 23 | // Offline Image removed after given time in seconds. 24 | // Default: 3 days 25 | storeImageTimeout, 26 | debugMode: false, 27 | }; 28 | // If it is `true` then we will remove expired images after given `storeImageTimeout` 29 | this.handlers = {}; 30 | 31 | this.restore = this.restore.bind(this); 32 | } 33 | 34 | return OfflineImageStore.instance; 35 | } 36 | 37 | /** 38 | * Gives the Offline store cache base directory 39 | */ 40 | getBaseDir = () => { 41 | return RNFetchBlob.fs.dirs.CacheDir + '/' + this.store.name; 42 | }; 43 | 44 | /** 45 | * This would be the method to be called on app start so that you could prepare application offline 46 | * image dictionary with existing image uris and its local store path. 47 | * 48 | * Pass onCompletion callback function to get the restore completion state. 49 | */ 50 | async restore (config, onRestoreCompletion) { 51 | if (config.name === undefined || config.name.length === 0) { 52 | throw 'Offline image store name is missing'; 53 | } 54 | 55 | this.store.name = config.name; 56 | 57 | if (config.imageRemoveTimeout) { 58 | this.store.storeImageTimeout = config.imageRemoveTimeout; 59 | } 60 | 61 | if (config.debugMode === true) { 62 | this.store.debugMode = true; 63 | } 64 | 65 | // Restore existing entries: 66 | AsyncStorage.getItem(`@${this.store.name}:uris`, (err, uris) => { // On `getItems` completion 67 | 68 | if (this.store.debugMode) { 69 | console.log('Restored offline images entry dictionary'); 70 | } 71 | // Assign uris to entry list cache(`this.entries`) 72 | Object.assign(this.entries, JSON.parse(uris)); 73 | 74 | // Remove Expired images from offline store and then call user given callback completion method ! 75 | this._removeExpiredImages(onRestoreCompletion); 76 | }); 77 | 78 | }; 79 | 80 | /** 81 | * Removes all the images in the offline store. 82 | */ 83 | clearStore = (onRestoreCompletion) => { 84 | 85 | // Check if the folder exists 86 | return RNFetchBlob.fs.exists(this.getBaseDir()) 87 | .then((exists) => { 88 | // If folder does not exists, no need to unlink it 89 | if (!exists) { 90 | return; 91 | } 92 | // Remove from offline store 93 | return RNFetchBlob.fs.unlink(this.getBaseDir()) 94 | }) 95 | .then(() => { // On completion 96 | if (this.store.debugMode) { 97 | console.log('Removed offline image store completely!'); 98 | } 99 | // Empty all entries so that we should update offline Async storage 100 | Object.keys(this.entries).forEach(key => delete this.entries[key]); 101 | 102 | // Update offline Async storage 103 | this._updateAsyncStorage(onRestoreCompletion); 104 | return null; 105 | }) 106 | .catch((err) => { 107 | if (this.store.debugMode) { 108 | console.log('unable to remove offline store', err); 109 | } 110 | 111 | // Call callback with the error 112 | onRestoreCompletion(err); 113 | }); 114 | }; 115 | 116 | /** 117 | * This method expects one or more list of image uris as a array to preload them. 118 | * This is very useful specially if your application want to support offline. 119 | * Note: We recommend call this method after `restore` 120 | */ 121 | preLoad = async (uris) => { 122 | if (uris === undefined && !Array.isArray(uris)) { 123 | throw 'uris should not be undefined and should be array type'; 124 | } 125 | uris.forEach((uri) => { 126 | // If image not exist already, then download 127 | if (!this.entries[uri]) { 128 | this._downloadImage({ 'uri': uri }); 129 | return; 130 | } 131 | 132 | // Exists but Base Dir changed , may be due to update new app version or whatever 133 | if (this.entries[uri]) { 134 | const entry = this.entries[uri]; 135 | // Only exist if base directory matches 136 | if (entry.basePath !== this.getBaseDir()) { 137 | this._downloadImage({ 'uri': uri }); 138 | } 139 | } 140 | }); 141 | }; 142 | 143 | subscribe = async (source, handler, reloadImage) => { 144 | const { uri } = source; 145 | 146 | if (!this.handlers[uri]) { 147 | this.handlers[uri] = [handler]; 148 | } else { 149 | this.handlers[uri].push(handler); 150 | } 151 | 152 | // Get the image if already exist else download and notify! 153 | this._getImage(source, reloadImage); 154 | }; 155 | 156 | // Un subscribe all the handlers for the given source uri 157 | unsubscribe = async (source) => { 158 | const { uri } = source; 159 | 160 | if (this.handlers[uri]) { 161 | delete this.handlers[uri]; 162 | } 163 | }; 164 | 165 | /** 166 | * Check whether given uri already exist in our offline cache! 167 | * @param uri uri to check in offline cache list 168 | */ 169 | isImageExistOffline = (uri) => { 170 | return this.entries[uri] !== undefined; 171 | }; 172 | 173 | _getExpiredImages = () => { 174 | const toBeRemovedImages = []; 175 | const uriList = Object.keys(this.entries); 176 | uriList.forEach((uri) => { 177 | const createdPlusDaysDate = this._addTime(this.entries[uri].createdOn, this.store.storeImageTimeout); 178 | // Image created date + EXPIRED_AFTER_DAYS is < current Date, then remove the image 179 | if (createdPlusDaysDate < new Date()) { 180 | toBeRemovedImages.push(uri); 181 | } 182 | }); 183 | 184 | return toBeRemovedImages; 185 | }; 186 | 187 | /** 188 | * Removes the downloaded offline images which are greater then given 'storeImageTimeout' in the config. 189 | */ 190 | _removeExpiredImages = (onRestoreCompletion) => { 191 | const toBeRemovedImagePromises = []; 192 | const uriListToRemove = this._getExpiredImages(); 193 | if (this.store.debugMode) { 194 | console.log('uris to remove from offline store', uriListToRemove); 195 | } 196 | uriListToRemove.forEach((uri) => { 197 | // Remove image from cache 198 | const unlinkPromise = RNFetchBlob.fs.unlink(this.entries[uri].basePath + '/' + this.entries[uri].localUriPath) 199 | .then(() => { 200 | // Delete entry from cache so that we should remove from offline Async storage 201 | delete this.entries[uri]; 202 | }) 203 | .catch((err) => { 204 | if (this.store.debugMode) { 205 | console.log('unable to remove image', uri, err); 206 | } 207 | }); 208 | toBeRemovedImagePromises.push(unlinkPromise); 209 | }); 210 | 211 | if (toBeRemovedImagePromises.length > 0) { 212 | if (this.store.debugMode) { 213 | console.log('Found images to remove:'); 214 | } 215 | Promise.all(toBeRemovedImagePromises) 216 | .then((results) => { 217 | if (this.store.debugMode) { 218 | console.log('removeExpiredImages completed callback'); 219 | } 220 | 221 | // Update AsyncStorage with removed entries 222 | this._updateAsyncStorage(onRestoreCompletion); 223 | return null; 224 | }) 225 | .catch((err) => { 226 | if (this.store.debugMode) { 227 | console.log('removeExpiredImages error'); 228 | } 229 | onRestoreCompletion(err); 230 | }); 231 | } else { // Nothing to remove so just trigger callback! 232 | if (this.store.debugMode) { 233 | console.log('No images to remove:'); 234 | } 235 | onRestoreCompletion(); 236 | } 237 | }; 238 | 239 | /** 240 | * Update AsyncStorage with entries cache and trigger callback. 241 | */ 242 | _updateAsyncStorage = (onRestoreCompletionCallback) => { 243 | AsyncStorage.setItem(`@${this.store.name}:uris`, JSON.stringify(this.entries), (err) => { 244 | if (onRestoreCompletionCallback) { 245 | err ? onRestoreCompletionCallback(err) : onRestoreCompletionCallback(); 246 | } 247 | }); 248 | }; 249 | 250 | getImageOfflinePath = (uri) => { 251 | if (this.entries[uri]) { 252 | const entry = this.entries[uri]; 253 | // Only exist if base directory matches 254 | if (entry.basePath === this.getBaseDir()) { 255 | if (this.store.debugMode) { 256 | console.log('Image exist offline', this.entries[uri].localUriPath); 257 | } 258 | return this.entries[uri].basePath + '/' + this.entries[uri].localUriPath; 259 | } 260 | } 261 | if (this.store.debugMode) { 262 | console.log('Image not exist offline', uri); 263 | } 264 | return undefined; 265 | }; 266 | 267 | _getImage = (source, reloadImage) => { 268 | // Image already exist 269 | if (this.entries[source.uri]) { 270 | const entry = this.entries[source.uri]; 271 | // Only exist if base directory matches 272 | if (entry.basePath === this.getBaseDir()) { 273 | if (this.store.debugMode) { 274 | console.log('Image exist offline', source.uri); 275 | } 276 | // Notify subscribed handler 277 | this._notify(source.uri); 278 | 279 | // Reload image: 280 | // Update existing image in offline store as server side image could have updated! 281 | if (reloadImage) { 282 | if (this.store.debugMode) { 283 | console.log('reloadImage is set to true for uri:', source.uri); 284 | } 285 | this._downloadImage(source); 286 | } 287 | } else { 288 | this._downloadImage(source); 289 | } 290 | return; 291 | } 292 | 293 | if (this.store.debugMode) { 294 | console.log('Image not exist offline', source.uri); 295 | } 296 | this._downloadImage(source); 297 | }; 298 | 299 | _downloadImage = (source) => { 300 | if (!source.uri) return 301 | const method = source.method ? source.method : 'GET'; 302 | const imageFilePath = this._getImageFileName(source.uri); 303 | if (!imageFilePath) return 304 | RNFetchBlob 305 | .config({ 306 | path: this.getBaseDir() + '/' + imageFilePath 307 | }) 308 | .fetch(method, source.uri, source.headers) 309 | .then(() => { 310 | // Add entry to entry list!! 311 | this._addEntry(source.uri, imageFilePath); 312 | // Notify subscribed handler AND Persist entries to AsyncStorage for offline 313 | this._updateOfflineStore(source.uri).done(); 314 | return null; 315 | }).catch(() => { 316 | if (this.store.debugMode) { 317 | console.log('Failed to download image', source.uri); 318 | } 319 | }); 320 | }; 321 | 322 | _notify = (uri) => { 323 | const handlers = this.handlers[uri]; 324 | if (handlers && handlers.length > 0) { 325 | handlers.forEach(handler => { 326 | if (this.store.debugMode) { 327 | console.log('Notify handler called', uri); 328 | } 329 | handler(uri, this.entries[uri].basePath + '/' + this.entries[uri].localUriPath); 330 | }); 331 | } 332 | }; 333 | 334 | _getImageFileName = (uri) => { 335 | if (!uri || !uri.substring) return 336 | let path = uri.substring(uri.lastIndexOf('/')); 337 | path = path.indexOf('?') === -1 ? path : path.substring(path.lastIndexOf('.'), path.indexOf('?')); 338 | const imageExtension = path.indexOf('.') === -1 ? '.jpg' : path.substring(path.indexOf('.')); 339 | 340 | const localFilePath = SHA1(uri) + imageExtension; 341 | //console.log('getImageFilePath: ', localFilePath); 342 | 343 | return localFilePath; 344 | }; 345 | 346 | _addEntry = (uri, imageFilePath) => { 347 | // Save Downloaded date when image downloads for first time 348 | if (this.entries[uri] === undefined) { 349 | this.entries[uri] = { 350 | createdOn: new Date().toString(), 351 | basePath: this.getBaseDir(), 352 | localUriPath: imageFilePath, 353 | }; 354 | } else { 355 | const imageUri = this.entries[uri]; 356 | this.entries[uri] = { 357 | ...imageUri, 358 | basePath: this.getBaseDir(), 359 | localUriPath: imageFilePath, 360 | }; 361 | } 362 | }; 363 | 364 | _updateOfflineStore = async (uri) => { 365 | try { 366 | await AsyncStorage.setItem(`@${this.store.name}:uris`, JSON.stringify(this.entries)); 367 | // Notify subscribed handler 368 | this._notify(uri); 369 | } catch (error) { 370 | if (this.store.debugMode) { 371 | // Error saving data 372 | console.log(' Offline image entry update failed', error); 373 | } 374 | } 375 | }; 376 | 377 | _addTime = (date, seconds) => { 378 | var result = new Date(date); 379 | result.setSeconds(result.getSeconds() + seconds); 380 | return result; 381 | }; 382 | } 383 | 384 | const instance = new OfflineImageStore('RN_Default_ImageStore', 259200); 385 | 386 | Object.freeze(instance); 387 | 388 | export default instance; 389 | -------------------------------------------------------------------------------- /src/actions.js: -------------------------------------------------------------------------------- 1 | import OFFLINE_IMAGES from './constants'; 2 | 3 | exports.downloadImageOffline = (source) => ({ 4 | type: OFFLINE_IMAGES.DOWNLOAD_IMAGE_OFFLINE, 5 | payload: source, 6 | }); 7 | 8 | exports.downloadImageOfflineSuccess = (uri, localImageFilePath) => ({ 9 | type: OFFLINE_IMAGES.DOWNLOAD_IMAGE_OFFLINE_SUCCESS, 10 | payload: { 11 | uri, 12 | localImageFilePath, 13 | } 14 | }); 15 | 16 | exports.downloadImageOfflineFailure = (error) => ({ 17 | type: OFFLINE_IMAGES.DOWNLOAD_IMAGE_OFFLINE_FAILED, 18 | payload: error, 19 | }); 20 | 21 | exports.downloadImageOfflineNetworkError = () => ({ 22 | type: OFFLINE_IMAGES.DOWNLOAD_IMAGE_OFFLINE_NETWORK_ERROR, 23 | }); 24 | 25 | exports.downloadImageOfflineTimeout = () => ({ 26 | type: OFFLINE_IMAGES.DOWNLOAD_IMAGE_OFFLINE_TIMEOUT, 27 | }); -------------------------------------------------------------------------------- /src/constants.js: -------------------------------------------------------------------------------- 1 | exports.DOWNLOAD_IMAGE_OFFLINE = 'DOWNLOAD_IMAGE_OFFLINE'; 2 | exports.DOWNLOAD_IMAGE_OFFLINE_SUCCESS = 'DOWNLOAD_IMAGE_OFFLINE_SUCCESS'; 3 | exports.DOWNLOAD_IMAGE_OFFLINE_FAILED = 'DOWNLOAD_IMAGE_OFFLINE_FAILED'; 4 | exports.DOWNLOAD_IMAGE_OFFLINE_NETWORK_ERROR = 'DOWNLOAD_IMAGE_OFFLINE_NETWORK_ERROR'; 5 | exports.DOWNLOAD_IMAGE_OFFLINE_TIMEOUT = 'DOWNLOAD_IMAGE_OFFLINE_TIMEOUT'; -------------------------------------------------------------------------------- /src/epics.js: -------------------------------------------------------------------------------- 1 | import { Observable } from 'rxjs/Observable'; 2 | import { of } from 'rxjs/add/observable/of'; 3 | import { delay } from 'rxjs/add/operator/delay'; 4 | import { takeUntil } from 'rxjs/add/operator/takeUntil'; 5 | import { merge } from 'rxjs/add/observable/merge'; 6 | import { concat } from 'rxjs/add/observable/concat'; 7 | import { flatMap, mergeMap } from 'rxjs/add/operator/mergeMap'; 8 | import { map } from 'rxjs/add/operator/map'; 9 | import { _catch } from 'rxjs/add/operator/catch'; 10 | import RNFetchBlob from 'rn-fetch-blob'; 11 | import OFFLINE_IMAGES from './constants'; 12 | 13 | import { downloadImageOfflineFailure, downloadImageOfflineNetworkError, downloadImageOfflineSuccess } from './actions'; 14 | 15 | import { getImageFilePath } from './utils'; 16 | 17 | const defaultTimeout = 5000; 18 | 19 | const downloadImageEpic = (action$, store) => 20 | action$.ofType(OFFLINE_IMAGES.DOWNLOAD_IMAGE_OFFLINE) 21 | .flatMap(action => { 22 | console.log('downloadImage', 'Entry'); 23 | console.log('downloadImage', action.payload); 24 | // TODO: Replace NetInfo 25 | // if (!store.getState().networkReducer.isConnected) { 26 | // console.log('downloadImage', 'Network not connected'); 27 | // return Observable.of(downloadImageOfflineNetworkError()); 28 | // } 29 | 30 | const source = action.payload; 31 | 32 | const method = source.method ? source.method : 'GET'; 33 | const imageFilePath = getImageFilePath(source.uri); 34 | 35 | RNFetchBlob 36 | .config({ 37 | path: imageFilePath 38 | }) 39 | .fetch(method, source.uri, source.headers) 40 | .then(() => { 41 | console.log('RNFetchBlob', 'success', imageFilePath, source.uri); 42 | store.dispatch(downloadImageOfflineSuccess(source.uri, imageFilePath)); 43 | }).catch(() => { 44 | console.log('RNFetchBlob', 'failure', imageFilePath, source.uri); 45 | RNFetchBlob.fs.unlink(imageFilePath); 46 | store.dispatch(downloadImageOfflineFailure()); 47 | }); 48 | 49 | return Observable.of({ 50 | type: 'DOWNLOAD_IMAGE_OFFLINE_REQUEST_SENT' 51 | }); 52 | 53 | }); 54 | 55 | export default downloadImageEpic; 56 | 57 | -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | import OfflineImage from './OfflineImage'; 2 | //import {getImageFilePath} from './utils'; 3 | //import offlineImageReducer from './reducer'; 4 | //import offlineImageEpic from './epics'; 5 | import OfflineImageStore from './OfflineImageStore'; 6 | 7 | export { 8 | OfflineImage, 9 | OfflineImageStore, 10 | }; -------------------------------------------------------------------------------- /src/reducer.js: -------------------------------------------------------------------------------- 1 | import OFFLINE_IMAGES from './constants'; 2 | import { REHYDRATE } from 'redux-persist/constants'; 3 | 4 | const initialState = { 5 | uris: {} 6 | }; 7 | 8 | const offlineImageReducer = (state = initialState, action) => { 9 | switch (action.type) { 10 | case OFFLINE_IMAGES.DOWNLOAD_IMAGE_OFFLINE_SUCCESS: 11 | return { 12 | ...state, 13 | uris: { 14 | ...state.uris, 15 | [action.payload.uri]: action.payload.localImageFilePath, 16 | } 17 | }; 18 | case REHYDRATE: 19 | const rehydratedState = action.payload.offlineImageReducer; 20 | if (rehydratedState) { 21 | return { 22 | ...state, 23 | ...rehydratedState, 24 | }; 25 | } 26 | return { ...state }; 27 | default: 28 | return { ...state } 29 | } 30 | }; 31 | 32 | export default offlineImageReducer; -------------------------------------------------------------------------------- /src/utils.js: -------------------------------------------------------------------------------- 1 | import RNFetchBlob from 'rn-fetch-blob'; 2 | 3 | const SHA1 = require('crypto-js/sha1'); 4 | 5 | // TODO : How user can configure this?? 6 | const IMAGE_OFFLINE_STORE_BASE_DIR = RNFetchBlob.fs.dirs.CacheDir + '/TODO_App'; 7 | 8 | exports.getImageFilePath = (uri) => { 9 | let path = uri.substring(uri.lastIndexOf('/')); 10 | path = path.indexOf('?') === -1 ? path : path.substring(path.lastIndexOf('.'), path.indexOf('?')); 11 | const imageExtension = path.indexOf('.') === -1 ? '.jpg' : path.substring(path.indexOf('.')); 12 | 13 | console.log('IMAGE_OFFLINE_STORE_BASE_DIR', IMAGE_OFFLINE_STORE_BASE_DIR); 14 | 15 | return IMAGE_OFFLINE_STORE_BASE_DIR + '/' + SHA1(uri) + imageExtension; 16 | }; -------------------------------------------------------------------------------- /yarn.lock: -------------------------------------------------------------------------------- 1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. 2 | # yarn lockfile v1 3 | 4 | 5 | crypto-js@^3.1.9-1: 6 | version "3.1.9-1" 7 | resolved "https://registry.yarnpkg.com/crypto-js/-/crypto-js-3.1.9-1.tgz#fda19e761fc077e01ffbfdc6e9fdfc59e8806cd8" 8 | --------------------------------------------------------------------------------