├── .buckconfig ├── .flowconfig ├── .gitattributes ├── .gitignore ├── .watchmanconfig ├── App.js ├── README.md ├── __tests__ └── App-test.js ├── android ├── app │ ├── BUCK │ ├── build.gradle │ ├── build_defs.bzl │ ├── proguard-rules.pro │ └── src │ │ ├── debug │ │ └── AndroidManifest.xml │ │ └── main │ │ ├── AndroidManifest.xml │ │ ├── java │ │ └── com │ │ │ └── modaldemo │ │ │ ├── MainActivity.java │ │ │ ├── MainApplication.java │ │ │ ├── MainPackage.java │ │ │ └── fullscreen │ │ │ ├── FullScreenDialog.java │ │ │ ├── FullScreenModalHostHelper.java │ │ │ ├── FullScreenModalHostShadowNode.java │ │ │ ├── FullScreenModalManager.java │ │ │ ├── FullScreenModalView.java │ │ │ └── util │ │ │ ├── AndroidBug5497Workaround.java │ │ │ └── StatusBarUtil.java │ │ └── res │ │ ├── mipmap-hdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-mdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-xhdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-xxhdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.png │ │ ├── mipmap-xxxhdpi │ │ ├── ic_launcher.png │ │ └── ic_launcher_round.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 ├── app.json ├── app ├── FullScreenModal │ ├── FullScreenModal.android.js │ └── FullScreenModal.ios.js └── ModalView.js ├── babel.config.js ├── index.js ├── ios ├── ModalDemo-tvOS │ └── Info.plist ├── ModalDemo-tvOSTests │ └── Info.plist ├── ModalDemo.xcodeproj │ ├── project.pbxproj │ └── xcshareddata │ │ └── xcschemes │ │ ├── ModalDemo-tvOS.xcscheme │ │ └── ModalDemo.xcscheme ├── ModalDemo │ ├── AppDelegate.h │ ├── AppDelegate.m │ ├── Base.lproj │ │ └── LaunchScreen.xib │ ├── Images.xcassets │ │ ├── AppIcon.appiconset │ │ │ └── Contents.json │ │ └── Contents.json │ ├── Info.plist │ └── main.m └── ModalDemoTests │ ├── Info.plist │ └── ModalDemoTests.m ├── metro.config.js ├── package-lock.json ├── package.json └── yarn.lock /.buckconfig: -------------------------------------------------------------------------------- 1 | 2 | [android] 3 | target = Google Inc.:Google APIs:23 4 | 5 | [maven_repositories] 6 | central = https://repo1.maven.org/maven2 7 | -------------------------------------------------------------------------------- /.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 | 16 | ; Ignore polyfills 17 | .*/Libraries/polyfills/.* 18 | 19 | ; Ignore metro 20 | .*/node_modules/metro/.* 21 | 22 | [include] 23 | 24 | [libs] 25 | node_modules/react-native/Libraries/react-native/react-native-interface.js 26 | node_modules/react-native/flow/ 27 | 28 | [options] 29 | emoji=true 30 | 31 | esproposal.optional_chaining=enable 32 | esproposal.nullish_coalescing=enable 33 | 34 | module.system=haste 35 | module.system.haste.use_name_reducers=true 36 | # get basename 37 | module.system.haste.name_reducers='^.*/\([a-zA-Z0-9$_.-]+\.js\(\.flow\)?\)$' -> '\1' 38 | # strip .js or .js.flow suffix 39 | module.system.haste.name_reducers='^\(.*\)\.js\(\.flow\)?$' -> '\1' 40 | # strip .ios suffix 41 | module.system.haste.name_reducers='^\(.*\)\.ios$' -> '\1' 42 | module.system.haste.name_reducers='^\(.*\)\.android$' -> '\1' 43 | module.system.haste.name_reducers='^\(.*\)\.native$' -> '\1' 44 | module.system.haste.paths.blacklist=.*/__tests__/.* 45 | module.system.haste.paths.blacklist=.*/__mocks__/.* 46 | module.system.haste.paths.blacklist=/node_modules/react-native/Libraries/Animated/src/polyfills/.* 47 | module.system.haste.paths.whitelist=/node_modules/react-native/Libraries/.* 48 | 49 | munge_underscores=true 50 | 51 | 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' 52 | 53 | module.file_ext=.js 54 | module.file_ext=.jsx 55 | module.file_ext=.json 56 | module.file_ext=.native.js 57 | 58 | suppress_type=$FlowIssue 59 | suppress_type=$FlowFixMe 60 | suppress_type=$FlowFixMeProps 61 | suppress_type=$FlowFixMeState 62 | 63 | suppress_comment=\\(.\\|\n\\)*\\$FlowFixMe\\($\\|[^(]\\|(\\(\\)? *\\(site=[a-z,_]*react_native[a-z,_]*\\)?)\\) 64 | suppress_comment=\\(.\\|\n\\)*\\$FlowIssue\\((\\(\\)? *\\(site=[a-z,_]*react_native[a-z,_]*\\)?)\\)?:? #[0-9]+ 65 | suppress_comment=\\(.\\|\n\\)*\\$FlowFixedInNextDeploy 66 | suppress_comment=\\(.\\|\n\\)*\\$FlowExpectedError 67 | 68 | [version] 69 | ^0.92.0 70 | -------------------------------------------------------------------------------- /.gitattributes: -------------------------------------------------------------------------------- 1 | *.pbxproj -text 2 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # OSX 2 | # 3 | .DS_Store 4 | 5 | # Xcode 6 | # 7 | build/ 8 | *.pbxuser 9 | !default.pbxuser 10 | *.mode1v3 11 | !default.mode1v3 12 | *.mode2v3 13 | !default.mode2v3 14 | *.perspectivev3 15 | !default.perspectivev3 16 | xcuserdata 17 | *.xccheckout 18 | *.moved-aside 19 | DerivedData 20 | *.hmap 21 | *.ipa 22 | *.xcuserstate 23 | project.xcworkspace 24 | 25 | # Android/IntelliJ 26 | # 27 | build/ 28 | .idea 29 | .gradle 30 | local.properties 31 | *.iml 32 | 33 | # node.js 34 | # 35 | node_modules/ 36 | npm-debug.log 37 | yarn-error.log 38 | 39 | # BUCK 40 | buck-out/ 41 | \.buckd/ 42 | *.keystore 43 | 44 | # fastlane 45 | # 46 | # It is recommended to not store the screenshots in the git repo. Instead, use fastlane to re-generate the 47 | # screenshots whenever they are needed. 48 | # For more information about the recommended setup visit: 49 | # https://docs.fastlane.tools/best-practices/source-control/ 50 | 51 | */fastlane/report.xml 52 | */fastlane/Preview.html 53 | */fastlane/screenshots 54 | 55 | # Bundle artifact 56 | *.jsbundle 57 | -------------------------------------------------------------------------------- /.watchmanconfig: -------------------------------------------------------------------------------- 1 | {} -------------------------------------------------------------------------------- /App.js: -------------------------------------------------------------------------------- 1 | import React, {Component} from 'react'; 2 | import {StyleSheet, Button, View, Text, Dimensions} from 'react-native'; 3 | import ModalView from './app/ModalView' 4 | 5 | const {width} = Dimensions.get('window') 6 | 7 | export default class App extends Component { 8 | 9 | constructor(props) { 10 | super(props); 11 | this.state = { 12 | modalType: 1, 13 | useReactModal: false, 14 | } 15 | } 16 | 17 | render() { 18 | return ( 19 | 20 | this._onPressModal(1, true)}>RN 自带 Modal 实现 21 | this._onPressModal(1, false)}>全屏 Modal 实现 22 | this._onPressModal(2, false)}>RootSiblings Modal 实现 23 | this._onPressModal(3, false)}>View Modal 实现 24 | 25 | this.modalView = c} 27 | animationType={'slide'} 28 | onRequestClose={() => this._disMiss()} 29 | useReactModal={this.state.useReactModal} 30 | modalType={this.state.modalType}> 31 | 32 | this._disMiss()} style={{position: 'absolute', top: 0, left: 0, right: 0, bottom: 0}}/> 33 | 34 | 欢迎打开 Modal 35 | 36 | 37 | 38 | 39 | ); 40 | } 41 | 42 | /** 43 | * @param modalType 弹窗类型 44 | * @param useReactModal android 是否使用 RN Modal 的实现 45 | * **/ 46 | _onPressModal = (modalType, useReactModal = false) => { 47 | this.setState({modalType: modalType, useReactModal: useReactModal}, () => this._show()) 48 | } 49 | 50 | _show = () => { 51 | this.modalView && this.modalView.show() 52 | } 53 | 54 | _disMiss = () => { 55 | this.modalView && this.modalView.disMiss() 56 | } 57 | } 58 | 59 | const styles = StyleSheet.create({ 60 | container: { 61 | flex: 1, 62 | justifyContent: 'center', 63 | alignItems: 'center', 64 | backgroundColor: '#F5FCFF', 65 | }, 66 | }); 67 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # ModalViewDemo 2 | React Native 添加可以在 Android 上进行全屏展示的 Modal 控件,并使用 RootSiblings 和 View 对整体的 Modal 控件进行一个封装使用 3 | -------------------------------------------------------------------------------- /__tests__/App-test.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @format 3 | */ 4 | 5 | import 'react-native'; 6 | import React from 'react'; 7 | import App from '../App'; 8 | 9 | // Note: test renderer must be required after react-native. 10 | import renderer from 'react-test-renderer'; 11 | 12 | it('renders correctly', () => { 13 | renderer.create(); 14 | }); 15 | -------------------------------------------------------------------------------- /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 | load(":build_defs.bzl", "create_aar_targets", "create_jar_targets") 12 | 13 | lib_deps = [] 14 | 15 | create_aar_targets(glob(["libs/*.aar"])) 16 | 17 | create_jar_targets(glob(["libs/*.jar"])) 18 | 19 | android_library( 20 | name = "all-libs", 21 | exported_deps = lib_deps, 22 | ) 23 | 24 | android_library( 25 | name = "app-code", 26 | srcs = glob([ 27 | "src/main/java/**/*.java", 28 | ]), 29 | deps = [ 30 | ":all-libs", 31 | ":build_config", 32 | ":res", 33 | ], 34 | ) 35 | 36 | android_build_config( 37 | name = "build_config", 38 | package = "com.modaldemo", 39 | ) 40 | 41 | android_resource( 42 | name = "res", 43 | package = "com.modaldemo", 44 | res = "src/main/res", 45 | ) 46 | 47 | android_binary( 48 | name = "app", 49 | keystore = "//android/keystores:debug", 50 | manifest = "src/main/AndroidManifest.xml", 51 | package_type = "debug", 52 | deps = [ 53 | ":app-code", 54 | ], 55 | ) 56 | -------------------------------------------------------------------------------- /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 | project.ext.react = [ 76 | entryFile: "index.js" 77 | ] 78 | 79 | apply from: "../../node_modules/react-native/react.gradle" 80 | 81 | /** 82 | * Set this to true to create two separate APKs instead of one: 83 | * - An APK that only works on ARM devices 84 | * - An APK that only works on x86 devices 85 | * The advantage is the size of the APK is reduced by about 4MB. 86 | * Upload all the APKs to the Play Store and people will download 87 | * the correct one based on the CPU architecture of their device. 88 | */ 89 | def enableSeparateBuildPerCPUArchitecture = false 90 | 91 | /** 92 | * Run Proguard to shrink the Java bytecode in release builds. 93 | */ 94 | def enableProguardInReleaseBuilds = false 95 | 96 | android { 97 | compileSdkVersion rootProject.ext.compileSdkVersion 98 | 99 | compileOptions { 100 | sourceCompatibility JavaVersion.VERSION_1_8 101 | targetCompatibility JavaVersion.VERSION_1_8 102 | } 103 | 104 | defaultConfig { 105 | applicationId "com.modaldemo" 106 | minSdkVersion rootProject.ext.minSdkVersion 107 | targetSdkVersion rootProject.ext.targetSdkVersion 108 | versionCode 1 109 | versionName "1.0" 110 | } 111 | splits { 112 | abi { 113 | reset() 114 | enable enableSeparateBuildPerCPUArchitecture 115 | universalApk false // If true, also generate a universal APK 116 | include "armeabi-v7a", "x86", "arm64-v8a", "x86_64" 117 | } 118 | } 119 | buildTypes { 120 | release { 121 | minifyEnabled enableProguardInReleaseBuilds 122 | proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro" 123 | } 124 | } 125 | // applicationVariants are e.g. debug, release 126 | applicationVariants.all { variant -> 127 | variant.outputs.each { output -> 128 | // For each separate APK per architecture, set a unique version code as described here: 129 | // http://tools.android.com/tech-docs/new-build-system/user-guide/apk-splits 130 | def versionCodes = ["armeabi-v7a":1, "x86":2, "arm64-v8a": 3, "x86_64": 4] 131 | def abi = output.getFilter(OutputFile.ABI) 132 | if (abi != null) { // null for the universal-debug, universal-release variants 133 | output.versionCodeOverride = 134 | versionCodes.get(abi) * 1048576 + defaultConfig.versionCode 135 | } 136 | } 137 | } 138 | } 139 | 140 | dependencies { 141 | implementation fileTree(dir: "libs", include: ["*.jar"]) 142 | implementation "com.android.support:appcompat-v7:${rootProject.ext.supportLibVersion}" 143 | implementation "com.facebook.react:react-native:+" // From node_modules 144 | } 145 | 146 | // Run this once to be able to run the application with BUCK 147 | // puts all compile dependencies into folder libs for BUCK to use 148 | task copyDownloadableDepsToLibs(type: Copy) { 149 | from configurations.compile 150 | into 'libs' 151 | } 152 | -------------------------------------------------------------------------------- /android/app/build_defs.bzl: -------------------------------------------------------------------------------- 1 | """Helper definitions to glob .aar and .jar targets""" 2 | 3 | def create_aar_targets(aarfiles): 4 | for aarfile in aarfiles: 5 | name = "aars__" + aarfile[aarfile.rindex("/") + 1:aarfile.rindex(".aar")] 6 | lib_deps.append(":" + name) 7 | android_prebuilt_aar( 8 | name = name, 9 | aar = aarfile, 10 | ) 11 | 12 | def create_jar_targets(jarfiles): 13 | for jarfile in jarfiles: 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 | -------------------------------------------------------------------------------- /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 | -------------------------------------------------------------------------------- /android/app/src/debug/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /android/app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 3 | 4 | 5 | 6 | 13 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/MainActivity.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo; 2 | 3 | import android.os.Bundle; 4 | 5 | import com.facebook.react.ReactActivity; 6 | import com.modaldemo.fullscreen.util.AndroidBug5497Workaround; 7 | import com.modaldemo.fullscreen.util.StatusBarUtil; 8 | 9 | public class MainActivity extends ReactActivity { 10 | /** 11 | * Returns the name of the main component registered from JavaScript. 12 | * This is used to schedule rendering of the component. 13 | */ 14 | @Override 15 | protected String getMainComponentName() { 16 | return "ModalDemo"; 17 | } 18 | 19 | @Override 20 | protected void onCreate(final Bundle savedInstanceState) { 21 | super.onCreate(savedInstanceState); 22 | StatusBarUtil.setTransparent(getWindow()); 23 | StatusBarUtil.setLightMode(getWindow()); 24 | AndroidBug5497Workaround.assistActivity(this); 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/MainApplication.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo; 2 | 3 | import android.app.Application; 4 | 5 | import com.facebook.react.ReactApplication; 6 | import com.facebook.react.ReactNativeHost; 7 | import com.facebook.react.ReactPackage; 8 | import com.facebook.react.shell.MainReactPackage; 9 | import com.facebook.soloader.SoLoader; 10 | 11 | import java.util.Arrays; 12 | import java.util.List; 13 | 14 | public class MainApplication extends Application implements ReactApplication { 15 | private final ReactNativeHost mReactNativeHost = new ReactNativeHost(this) { 16 | @Override 17 | public boolean getUseDeveloperSupport() { 18 | return BuildConfig.DEBUG; 19 | } 20 | 21 | @Override 22 | protected List getPackages() { 23 | return Arrays.asList( 24 | new MainReactPackage(), 25 | new MainPackage() 26 | ); 27 | } 28 | 29 | @Override 30 | protected String getJSMainModuleName() { 31 | return "index"; 32 | } 33 | }; 34 | 35 | @Override 36 | public ReactNativeHost getReactNativeHost() { 37 | return mReactNativeHost; 38 | } 39 | 40 | @Override 41 | public void onCreate() { 42 | super.onCreate(); 43 | SoLoader.init(this, /* native exopackage */ false); 44 | } 45 | } 46 | -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/MainPackage.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo; 2 | 3 | import com.facebook.react.ReactPackage; 4 | import com.facebook.react.bridge.NativeModule; 5 | import com.facebook.react.bridge.ReactApplicationContext; 6 | import com.facebook.react.uimanager.ViewManager; 7 | import com.modaldemo.fullscreen.FullScreenModalManager; 8 | 9 | import java.util.ArrayList; 10 | import java.util.Collections; 11 | import java.util.List; 12 | 13 | import javax.annotation.Nonnull; 14 | 15 | /** 16 | * @author linhe 17 | */ 18 | public class MainPackage implements ReactPackage { 19 | @Nonnull 20 | @Override 21 | public List createNativeModules(@Nonnull final ReactApplicationContext reactContext) { 22 | return Collections.emptyList(); 23 | } 24 | 25 | @Override 26 | public List createViewManagers(final ReactApplicationContext reactContext) { 27 | List viewManagers = new ArrayList<>(); 28 | viewManagers.add(new FullScreenModalManager()); 29 | return viewManagers; 30 | } 31 | } 32 | -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/fullscreen/FullScreenDialog.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo.fullscreen; 2 | 3 | import android.app.Dialog; 4 | import android.content.Context; 5 | import android.support.annotation.NonNull; 6 | import android.support.annotation.StyleRes; 7 | import android.view.View; 8 | 9 | import com.modaldemo.fullscreen.util.AndroidBug5497Workaround; 10 | import com.modaldemo.fullscreen.util.StatusBarUtil; 11 | 12 | /** 13 | * @author linhe 14 | * 一个可以全屏显示的弹窗控件,从状态栏处开始布局 15 | */ 16 | public class FullScreenDialog extends Dialog { 17 | private boolean isDarkMode; 18 | private View rootView; 19 | 20 | public void setDarkMode(boolean isDarkMode) { 21 | this.isDarkMode = isDarkMode; 22 | } 23 | 24 | public FullScreenDialog(@NonNull Context context, @StyleRes int themeResId) { 25 | super(context, themeResId); 26 | } 27 | 28 | @Override 29 | public void setContentView(@NonNull View view) { 30 | super.setContentView(view); 31 | this.rootView = view; 32 | } 33 | 34 | @Override 35 | public void show() { 36 | super.show(); 37 | StatusBarUtil.setTransparent(getWindow()); 38 | if (isDarkMode) { 39 | StatusBarUtil.setDarkMode(getWindow()); 40 | } else { 41 | StatusBarUtil.setLightMode(getWindow()); 42 | } 43 | AndroidBug5497Workaround.assistView(rootView, getWindow()); 44 | } 45 | } 46 | -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/fullscreen/FullScreenModalHostHelper.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo.fullscreen; 2 | 3 | import android.annotation.TargetApi; 4 | import android.content.Context; 5 | import android.content.res.Resources; 6 | import android.graphics.Point; 7 | import android.view.Display; 8 | import android.view.WindowManager; 9 | 10 | import com.facebook.infer.annotation.Assertions; 11 | 12 | /** 13 | * @author linhe 14 | */ 15 | public class FullScreenModalHostHelper { 16 | private static final Point MIN_POINT = new Point(); 17 | private static final Point MAX_POINT = new Point(); 18 | private static final Point SIZE_POINT = new Point(); 19 | 20 | /** 21 | * To get the size of the screen, we use information from the WindowManager and 22 | * default Display. We don't use DisplayMetricsHolder, or Display#getSize() because 23 | * they return values that include the status bar. We only want the values of what 24 | * will actually be shown on screen. 25 | * We use Display#getSize() to determine if the screen is in portrait or landscape. 26 | * We don't use getRotation because the 'natural' rotation will be portrait on phones 27 | * and landscape on tablets. 28 | * This should only be called on the native modules/shadow nodes thread. 29 | */ 30 | @TargetApi(16) 31 | public static Point getModalHostSize(Context context) { 32 | WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE); 33 | Display display = Assertions.assertNotNull(wm).getDefaultDisplay(); 34 | // getCurrentSizeRange will return the min and max width and height that the window can be 35 | display.getCurrentSizeRange(MIN_POINT, MAX_POINT); 36 | // getSize will return the dimensions of the screen in its current orientation 37 | display.getSize(SIZE_POINT); 38 | 39 | // We need to add the status bar height to the height if we have a fullscreen window, 40 | // because Display.getCurrentSizeRange doesn't include it. 41 | Resources resources = context.getResources(); 42 | int statusBarId = resources.getIdentifier("status_bar_height", "dimen", "android"); 43 | int statusBarHeight = 0; 44 | if (statusBarId > 0) { //获取状态栏的高度 45 | statusBarHeight = (int) resources.getDimension(statusBarId); 46 | } 47 | 48 | if (SIZE_POINT.x < SIZE_POINT.y) { 49 | // If we are vertical the width value comes from min width and height comes from max height 50 | return new Point(MIN_POINT.x, MAX_POINT.y + statusBarHeight); 51 | } else { 52 | // If we are horizontal the width value comes from max width and height comes from min height 53 | return new Point(MAX_POINT.x, MIN_POINT.y + statusBarHeight); 54 | } 55 | } 56 | } 57 | 58 | -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/fullscreen/FullScreenModalHostShadowNode.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo.fullscreen; 2 | 3 | import android.graphics.Point; 4 | 5 | import com.facebook.react.uimanager.LayoutShadowNode; 6 | import com.facebook.react.uimanager.ReactShadowNodeImpl; 7 | 8 | /** 9 | * @author linhe 10 | */ 11 | public class FullScreenModalHostShadowNode extends LayoutShadowNode { 12 | @Override 13 | public void addChildAt(ReactShadowNodeImpl child, int i) { 14 | super.addChildAt(child, i); 15 | Point modalSize = FullScreenModalHostHelper.getModalHostSize(getThemedContext()); 16 | child.setStyleWidth(modalSize.x); 17 | child.setStyleHeight(modalSize.y); 18 | } 19 | } -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/fullscreen/FullScreenModalManager.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo.fullscreen; 2 | 3 | import android.content.DialogInterface; 4 | import android.support.annotation.Nullable; 5 | 6 | import com.facebook.react.common.MapBuilder; 7 | import com.facebook.react.uimanager.LayoutShadowNode; 8 | import com.facebook.react.uimanager.ThemedReactContext; 9 | import com.facebook.react.uimanager.ViewGroupManager; 10 | import com.facebook.react.uimanager.annotations.ReactProp; 11 | import com.facebook.react.uimanager.events.RCTEventEmitter; 12 | 13 | import java.util.Map; 14 | 15 | /** 16 | * @author linhe 17 | * 可以进行全屏显示的dialog,用来代替RN的modal 18 | */ 19 | public class FullScreenModalManager extends ViewGroupManager { 20 | @Override 21 | public String getName() { 22 | return "RCTFullScreenModalHostView"; 23 | } 24 | 25 | public enum Events { 26 | ON_SHOW("onFullScreenShow"), 27 | ON_REQUEST_CLOSE("onFullScreenRequstClose"); 28 | private final String mName; 29 | 30 | Events(final String name) { 31 | mName = name; 32 | } 33 | 34 | @Override 35 | public String toString() { 36 | return mName; 37 | } 38 | } 39 | 40 | @Override 41 | @Nullable 42 | public Map getExportedCustomDirectEventTypeConstants() { 43 | MapBuilder.Builder builder = MapBuilder.builder(); 44 | for (Events event : Events.values()) { 45 | builder.put(event.toString(), MapBuilder.of("registrationName", event.toString())); 46 | } 47 | return builder.build(); 48 | } 49 | 50 | @Override 51 | protected FullScreenModalView createViewInstance(ThemedReactContext reactContext) { 52 | final FullScreenModalView view = new FullScreenModalView(reactContext); 53 | final RCTEventEmitter mEventEmitter = reactContext.getJSModule(RCTEventEmitter.class); 54 | view.setOnRequestCloseListener(new FullScreenModalView.OnRequestCloseListener() { 55 | @Override 56 | public void onRequestClose(DialogInterface dialog) { 57 | mEventEmitter.receiveEvent(view.getId(), Events.ON_REQUEST_CLOSE.toString(), null); 58 | } 59 | }); 60 | view.setOnShowListener(new DialogInterface.OnShowListener() { 61 | @Override 62 | public void onShow(DialogInterface dialog) { 63 | mEventEmitter.receiveEvent(view.getId(), Events.ON_SHOW.toString(), null); 64 | } 65 | }); 66 | return view; 67 | } 68 | 69 | @Override 70 | public LayoutShadowNode createShadowNodeInstance() { 71 | return new FullScreenModalHostShadowNode(); 72 | } 73 | 74 | @Override 75 | public Class getShadowNodeClass() { 76 | return FullScreenModalHostShadowNode.class; 77 | } 78 | 79 | @Override 80 | public void onDropViewInstance(FullScreenModalView view) { 81 | super.onDropViewInstance(view); 82 | view.onDropInstance(); 83 | } 84 | 85 | @ReactProp(name = "autoKeyboard") 86 | public void setAutoKeyboard(FullScreenModalView view, boolean autoKeyboard) { 87 | view.setAutoKeyboard(autoKeyboard); 88 | } 89 | 90 | @ReactProp(name = "isDarkMode") 91 | public void setDarkMode(FullScreenModalView view, boolean isDarkMode) { 92 | view.setDarkMode(isDarkMode); 93 | } 94 | 95 | @ReactProp(name = "animationType") 96 | public void setAnimationType(FullScreenModalView view, String animationType) { 97 | view.setAnimationType(animationType); 98 | } 99 | 100 | @ReactProp(name = "transparent") 101 | public void setTransparent(FullScreenModalView view, boolean transparent) { 102 | view.setTransparent(transparent); 103 | } 104 | 105 | @ReactProp(name = "hardwareAccelerated") 106 | public void setHardwareAccelerated(FullScreenModalView view, boolean hardwareAccelerated) { 107 | view.setHardwareAccelerated(hardwareAccelerated); 108 | } 109 | 110 | @Override 111 | protected void onAfterUpdateTransaction(FullScreenModalView view) { 112 | super.onAfterUpdateTransaction(view); 113 | view.showOrUpdate(); 114 | } 115 | } 116 | 117 | -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/fullscreen/FullScreenModalView.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo.fullscreen; 2 | 3 | import android.app.Activity; 4 | import android.app.Dialog; 5 | import android.content.Context; 6 | import android.content.DialogInterface; 7 | import android.support.annotation.Nullable; 8 | import android.support.annotation.VisibleForTesting; 9 | import android.text.TextUtils; 10 | import android.view.KeyEvent; 11 | import android.view.MotionEvent; 12 | import android.view.View; 13 | import android.view.ViewGroup; 14 | import android.view.WindowManager; 15 | import android.view.accessibility.AccessibilityEvent; 16 | import android.widget.FrameLayout; 17 | 18 | import com.facebook.infer.annotation.Assertions; 19 | import com.facebook.react.bridge.LifecycleEventListener; 20 | import com.facebook.react.bridge.ReactContext; 21 | import com.facebook.react.uimanager.JSTouchDispatcher; 22 | import com.facebook.react.uimanager.RootView; 23 | import com.facebook.react.uimanager.UIManagerModule; 24 | import com.facebook.react.uimanager.events.EventDispatcher; 25 | import com.facebook.react.views.view.ReactViewGroup; 26 | import com.modaldemo.R; 27 | 28 | import java.util.ArrayList; 29 | 30 | /** 31 | * @author linhe 32 | */ 33 | public class FullScreenModalView extends ViewGroup implements LifecycleEventListener { 34 | // This listener is called when the user presses KeyEvent.KEYCODE_BACK 35 | // An event is then passed to JS which can either close or not close the Modal by setting the 36 | // visible property 37 | public interface OnRequestCloseListener { 38 | void onRequestClose(DialogInterface dialog); 39 | } 40 | 41 | private DialogRootViewGroup mHostView; 42 | private @Nullable 43 | FullScreenDialog mDialog; 44 | private boolean mTransparent; 45 | private String mAnimationType; 46 | private boolean mHardwareAccelerated; 47 | // Set this flag to true if changing a particular property on the view requires a new Dialog to 48 | // be created. For instance, animation does since it affects Dialog creation through the theme 49 | // but transparency does not since we can access the window to update the property. 50 | private boolean mPropertyRequiresNewDialog; 51 | private boolean isDarkMode = false; //默认状态栏为白底黑字 52 | private boolean autoKeyboard = false; 53 | private @Nullable 54 | DialogInterface.OnShowListener mOnShowListener; 55 | private @Nullable 56 | OnRequestCloseListener mOnRequestCloseListener; 57 | 58 | public FullScreenModalView(Context context) { 59 | super(context); 60 | ((ReactContext) context).addLifecycleEventListener(this); 61 | 62 | mHostView = new DialogRootViewGroup(context); 63 | } 64 | 65 | @Override 66 | protected void onLayout(boolean changed, int l, int t, int r, int b) { 67 | // Do nothing as we are laid out by UIManager 68 | } 69 | 70 | @Override 71 | public void addView(View child, int index) { 72 | mHostView.addView(child, index); 73 | } 74 | 75 | @Override 76 | public int getChildCount() { 77 | return mHostView.getChildCount(); 78 | } 79 | 80 | @Override 81 | public View getChildAt(int index) { 82 | return mHostView.getChildAt(index); 83 | } 84 | 85 | @Override 86 | public void removeView(View child) { 87 | mHostView.removeView(child); 88 | } 89 | 90 | @Override 91 | public void removeViewAt(int index) { 92 | View child = getChildAt(index); 93 | mHostView.removeView(child); 94 | } 95 | 96 | @Override 97 | public void addChildrenForAccessibility(ArrayList outChildren) { 98 | // Explicitly override this to prevent accessibility events being passed down to children 99 | // Those will be handled by the mHostView which lives in the dialog 100 | } 101 | 102 | @Override 103 | public boolean dispatchPopulateAccessibilityEvent(AccessibilityEvent event) { 104 | // Explicitly override this to prevent accessibility events being passed down to children 105 | // Those will be handled by the mHostView which lives in the dialog 106 | return false; 107 | } 108 | 109 | public void onDropInstance() { 110 | ((ReactContext) getContext()).removeLifecycleEventListener(this); 111 | dismiss(); 112 | } 113 | 114 | private void dismiss() { 115 | if (mDialog != null) { 116 | mDialog.dismiss(); 117 | mDialog = null; 118 | 119 | // We need to remove the mHostView from the parent 120 | // It is possible we are dismissing this dialog and reattaching the hostView to another 121 | ViewGroup parent = (ViewGroup) mHostView.getParent(); 122 | parent.removeViewAt(0); 123 | } 124 | } 125 | 126 | protected void setAutoKeyboard(boolean autoKeyboard) { 127 | this.autoKeyboard = autoKeyboard; 128 | } 129 | 130 | protected void setDarkMode(boolean isDarkMode) { 131 | this.isDarkMode = isDarkMode; 132 | } 133 | 134 | protected void setOnRequestCloseListener(OnRequestCloseListener listener) { 135 | mOnRequestCloseListener = listener; 136 | } 137 | 138 | protected void setOnShowListener(DialogInterface.OnShowListener listener) { 139 | mOnShowListener = listener; 140 | } 141 | 142 | protected void setTransparent(boolean transparent) { 143 | mTransparent = transparent; 144 | } 145 | 146 | protected void setAnimationType(String animationType) { 147 | mAnimationType = animationType; 148 | mPropertyRequiresNewDialog = true; 149 | } 150 | 151 | protected void setHardwareAccelerated(boolean hardwareAccelerated) { 152 | mHardwareAccelerated = hardwareAccelerated; 153 | mPropertyRequiresNewDialog = true; 154 | } 155 | 156 | @Override 157 | public void onHostResume() { 158 | // We show the dialog again when the host resumes 159 | showOrUpdate(); 160 | } 161 | 162 | @Override 163 | public void onHostPause() { 164 | // We dismiss the dialog and reconstitute it onHostResume 165 | dismiss(); 166 | } 167 | 168 | @Override 169 | public void onHostDestroy() { 170 | // Drop the instance if the host is destroyed which will dismiss the dialog 171 | onDropInstance(); 172 | } 173 | 174 | @VisibleForTesting 175 | public @Nullable 176 | Dialog getDialog() { 177 | return mDialog; 178 | } 179 | 180 | /** 181 | * showOrUpdate will display the Dialog. It is called by the manager once all properties are set 182 | * because we need to know all of them before creating the Dialog. It is also smart during 183 | * updates if the changed properties can be applied directly to the Dialog or require the 184 | * recreation of a new Dialog. 185 | */ 186 | protected void showOrUpdate() { 187 | // If the existing Dialog is currently up, we may need to redraw it or we may be able to update 188 | // the property without having to recreate the dialog 189 | if (mDialog != null) { 190 | if (mPropertyRequiresNewDialog) { 191 | dismiss(); 192 | } else { 193 | updateProperties(); 194 | return; 195 | } 196 | } 197 | 198 | // Reset the flag since we are going to create a new dialog 199 | mPropertyRequiresNewDialog = false; 200 | int theme = R.style.Theme_FullScreenDialog; 201 | if (TextUtils.isEmpty(mAnimationType) || mAnimationType.equals("none")) { 202 | theme = R.style.Theme_FullScreenDialog; 203 | } else if (mAnimationType.equals("fade")) { 204 | theme = R.style.Theme_FullScreenDialogAnimatedFade; 205 | } else if (mAnimationType.equals("slide")) { 206 | theme = R.style.Theme_FullScreenDialogAnimatedSlide; 207 | } 208 | mDialog = new FullScreenDialog(getContext(), theme); 209 | mDialog.setDarkMode(isDarkMode); 210 | mDialog.setContentView(getContentView()); 211 | updateProperties(); 212 | 213 | mDialog.setOnShowListener(mOnShowListener); 214 | mDialog.setOnKeyListener(new DialogInterface.OnKeyListener() { 215 | @Override 216 | public boolean onKey(DialogInterface dialog, int keyCode, KeyEvent event) { 217 | if (event.getAction() == KeyEvent.ACTION_UP) { 218 | // We need to stop the BACK button from closing the dialog by default so we capture that 219 | // event and instead inform JS so that it can make the decision as to whether or not to 220 | // allow the back button to close the dialog. If it chooses to, it can just set visible 221 | // to false on the Modal and the Modal will go away 222 | if (keyCode == KeyEvent.KEYCODE_BACK) { 223 | Assertions.assertNotNull(mOnRequestCloseListener, "setOnRequestCloseListener must be called " + 224 | "by the manager"); 225 | mOnRequestCloseListener.onRequestClose(dialog); 226 | return true; 227 | } else { 228 | // We redirect the rest of the key events to the current activity, since the activity 229 | // expects to receive those events and react to them, ie. in the case of the dev menu 230 | Activity currentActivity = ((ReactContext) getContext()).getCurrentActivity(); 231 | if (currentActivity != null) { 232 | return currentActivity.onKeyUp(keyCode, event); 233 | } 234 | } 235 | } 236 | return false; 237 | } 238 | }); 239 | 240 | int keyboardType = WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE; 241 | if (autoKeyboard) { 242 | keyboardType |= WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_VISIBLE; 243 | } 244 | mDialog.getWindow().setSoftInputMode(keyboardType); 245 | 246 | if (mHardwareAccelerated) { 247 | mDialog.getWindow().addFlags(WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED); 248 | } 249 | mDialog.show(); 250 | } 251 | 252 | /** 253 | * Returns the view that will be the root view of the dialog. We are wrapping this in a 254 | * FrameLayout because this is the system's way of notifying us that the dialog size has changed. 255 | * This has the pleasant side-effect of us not having to preface all Modals with 256 | * "top: statusBarHeight", since that margin will be included in the FrameLayout. 257 | */ 258 | private View getContentView() { 259 | FrameLayout frameLayout = new FrameLayout(getContext()); 260 | frameLayout.setLayoutParams(new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT)); 261 | frameLayout.addView(mHostView); 262 | return frameLayout; 263 | } 264 | 265 | /** 266 | * updateProperties will update the properties that do not require us to recreate the dialog 267 | * Properties that do require us to recreate the dialog should set mPropertyRequiresNewDialog to 268 | * true when the property changes 269 | */ 270 | private void updateProperties() { 271 | Assertions.assertNotNull(mDialog, "mDialog must exist when we call updateProperties"); 272 | 273 | if (mTransparent) { 274 | mDialog.getWindow().clearFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND); 275 | } else { 276 | mDialog.getWindow().setDimAmount(0.5f); 277 | mDialog.getWindow().setFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND, 278 | WindowManager.LayoutParams.FLAG_DIM_BEHIND); 279 | } 280 | } 281 | 282 | /** 283 | * DialogRootViewGroup is the ViewGroup which contains all the children of a Modal. It gets all 284 | * child information forwarded from FullScreenModalView and uses that to create children. It is 285 | * also responsible for acting as a RootView and handling touch events. It does this the same 286 | * way as ReactRootView. 287 | *

288 | * To get layout to work properly, we need to layout all the elements within the Modal as if they 289 | * can fill the entire window. To do that, we need to explicitly set the styleWidth and 290 | * styleHeight on the LayoutShadowNode to be the window size. This is done through the 291 | * UIManagerModule, and will then cause the children to layout as if they can fill the window. 292 | */ 293 | static class DialogRootViewGroup extends ReactViewGroup implements RootView { 294 | private final JSTouchDispatcher mJSTouchDispatcher = new JSTouchDispatcher(this); 295 | 296 | public DialogRootViewGroup(Context context) { 297 | super(context); 298 | } 299 | 300 | @Override 301 | protected void onSizeChanged(final int w, final int h, int oldw, int oldh) { 302 | super.onSizeChanged(w, h, oldw, oldh); 303 | if (getChildCount() > 0) { 304 | final int viewTag = getChildAt(0).getId(); 305 | ReactContext reactContext = (ReactContext) getContext(); 306 | reactContext.runOnNativeModulesQueueThread(new Runnable() { 307 | @Override 308 | public void run() { 309 | ((ReactContext) getContext()).getNativeModule(UIManagerModule.class) 310 | .updateNodeSize(viewTag, w, h); 311 | } 312 | }); 313 | } 314 | } 315 | 316 | @Override 317 | public boolean onInterceptTouchEvent(MotionEvent event) { 318 | mJSTouchDispatcher.handleTouchEvent(event, getEventDispatcher()); 319 | return super.onInterceptTouchEvent(event); 320 | } 321 | 322 | @Override 323 | public boolean onTouchEvent(MotionEvent event) { 324 | mJSTouchDispatcher.handleTouchEvent(event, getEventDispatcher()); 325 | super.onTouchEvent(event); 326 | // In case when there is no children interested in handling touch event, we return true from 327 | // the root view in order to receive subsequent events related to that gesture 328 | return true; 329 | } 330 | 331 | @Override 332 | public void onChildStartedNativeGesture(MotionEvent androidEvent) { 333 | mJSTouchDispatcher.onChildStartedNativeGesture(androidEvent, getEventDispatcher()); 334 | } 335 | 336 | @Override 337 | public void handleException(Throwable t) { 338 | ((ReactContext) getContext()).handleException(new RuntimeException(t)); 339 | } 340 | 341 | @Override 342 | public void requestDisallowInterceptTouchEvent(boolean disallowIntercept) { 343 | // No-op - override in order to still receive events to onInterceptTouchEvent 344 | // even when some other view disallow that 345 | } 346 | 347 | private EventDispatcher getEventDispatcher() { 348 | ReactContext reactContext = (ReactContext) getContext(); 349 | return reactContext.getNativeModule(UIManagerModule.class).getEventDispatcher(); 350 | } 351 | } 352 | } 353 | -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/fullscreen/util/AndroidBug5497Workaround.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo.fullscreen.util; 2 | 3 | import android.app.Activity; 4 | import android.graphics.Rect; 5 | import android.os.Build; 6 | import android.view.View; 7 | import android.view.ViewGroup; 8 | import android.view.ViewTreeObserver; 9 | import android.view.Window; 10 | import android.widget.FrameLayout; 11 | 12 | /** 13 | * @author linhe 14 | */ 15 | public class AndroidBug5497Workaround { 16 | public static void assistActivity(Activity activity) { 17 | new AndroidBug5497Workaround(activity); 18 | } 19 | 20 | public static void assistView(View rootView, Window window) { 21 | new AndroidBug5497Workaround(rootView, window); 22 | } 23 | 24 | private Window window; 25 | private View mChildOfContent; 26 | private int usableHeightPrevious; 27 | private ViewGroup.LayoutParams frameLayoutParams; 28 | 29 | private AndroidBug5497Workaround(View rootView, Window window) { 30 | this.window = window; 31 | mChildOfContent = rootView; 32 | mChildOfContent.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() { 33 | @Override 34 | public void onGlobalLayout() { 35 | possiblyResizeChildOfContent(); 36 | } 37 | }); 38 | frameLayoutParams = mChildOfContent.getLayoutParams(); 39 | } 40 | 41 | private AndroidBug5497Workaround(Activity activity) { 42 | window = activity.getWindow(); 43 | FrameLayout content = activity.findViewById(android.R.id.content); 44 | mChildOfContent = content.getChildAt(0); 45 | mChildOfContent.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() { 46 | @Override 47 | public void onGlobalLayout() { 48 | possiblyResizeChildOfContent(); 49 | } 50 | }); 51 | frameLayoutParams = mChildOfContent.getLayoutParams(); 52 | } 53 | 54 | private void possiblyResizeChildOfContent() { 55 | int usableHeightNow = computeUsableHeight(); 56 | if (usableHeightNow != usableHeightPrevious) { 57 | //如果两次高度不一致 58 | //将计算的可视高度设置成视图的高度 59 | frameLayoutParams.height = usableHeightNow; 60 | mChildOfContent.requestLayout();//请求重新布局 61 | usableHeightPrevious = usableHeightNow; 62 | } 63 | } 64 | 65 | private int computeUsableHeight() { 66 | Rect frame = new Rect(); 67 | window.getDecorView().getWindowVisibleDisplayFrame(frame); 68 | int statusBarHeight = frame.top; 69 | Rect r = new Rect(); 70 | mChildOfContent.getWindowVisibleDisplayFrame(r); 71 | //这个判断是为了解决19之后的版本在弹出软键盘时,键盘和推上去的布局(adjustResize)之间有黑色区域的问题 72 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { 73 | return (r.bottom - r.top) + statusBarHeight; 74 | } 75 | return (r.bottom - r.top); 76 | } 77 | } 78 | -------------------------------------------------------------------------------- /android/app/src/main/java/com/modaldemo/fullscreen/util/StatusBarUtil.java: -------------------------------------------------------------------------------- 1 | package com.modaldemo.fullscreen.util; 2 | 3 | import android.annotation.TargetApi; 4 | import android.graphics.Color; 5 | import android.os.Build; 6 | import android.support.annotation.NonNull; 7 | import android.view.View; 8 | import android.view.Window; 9 | import android.view.WindowManager; 10 | 11 | 12 | import java.lang.reflect.Field; 13 | import java.lang.reflect.Method; 14 | 15 | /** 16 | * @author linhe 17 | */ 18 | public class StatusBarUtil { 19 | /** 20 | * 设置状态栏全透明 21 | * 22 | * @param window 需要设置的window 23 | */ 24 | public static void setTransparent(Window window) { 25 | if (Build.VERSION.SDK_INT < Build.VERSION_CODES.KITKAT) { 26 | return; 27 | } 28 | transparentStatusBar(window); 29 | } 30 | 31 | @TargetApi(Build.VERSION_CODES.M) 32 | public static void setLightMode(Window window) { 33 | setMIUIStatusBarDarkIcon(window, true); 34 | setMeizuStatusBarDarkIcon(window, true); 35 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { 36 | window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN); 37 | } 38 | } 39 | 40 | @TargetApi(Build.VERSION_CODES.M) 41 | public static void setDarkMode(Window window) { 42 | setMIUIStatusBarDarkIcon(window, false); 43 | setMeizuStatusBarDarkIcon(window, false); 44 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { 45 | window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN); 46 | } 47 | } 48 | 49 | /** 50 | * 修改 MIUI V6 以上状态栏颜色 51 | */ 52 | private static void setMIUIStatusBarDarkIcon(@NonNull Window window, boolean darkIcon) { 53 | Class clazz = window.getClass(); 54 | try { 55 | Class layoutParams = Class.forName("android.view.MiuiWindowManager$LayoutParams"); 56 | Field field = layoutParams.getField("EXTRA_FLAG_STATUS_BAR_DARK_MODE"); 57 | int darkModeFlag = field.getInt(layoutParams); 58 | Method extraFlagField = clazz.getMethod("setExtraFlags", int.class, int.class); 59 | extraFlagField.invoke(window, darkIcon ? darkModeFlag : 0, darkModeFlag); 60 | } catch (Exception e) { 61 | e.printStackTrace(); 62 | } 63 | } 64 | 65 | /** 66 | * 修改魅族状态栏字体颜色 Flyme 4.0 67 | */ 68 | private static void setMeizuStatusBarDarkIcon(@NonNull Window window, boolean darkIcon) { 69 | try { 70 | WindowManager.LayoutParams lp = window.getAttributes(); 71 | Field darkFlag = WindowManager.LayoutParams.class.getDeclaredField("MEIZU_FLAG_DARK_STATUS_BAR_ICON"); 72 | Field meizuFlags = WindowManager.LayoutParams.class.getDeclaredField("meizuFlags"); 73 | darkFlag.setAccessible(true); 74 | meizuFlags.setAccessible(true); 75 | int bit = darkFlag.getInt(null); 76 | int value = meizuFlags.getInt(lp); 77 | if (darkIcon) { 78 | value |= bit; 79 | } else { 80 | value &= ~bit; 81 | } 82 | meizuFlags.setInt(lp, value); 83 | window.setAttributes(lp); 84 | } catch (Exception e) { 85 | e.printStackTrace(); 86 | } 87 | } 88 | 89 | /** 90 | * 使状态栏透明 91 | */ 92 | @TargetApi(Build.VERSION_CODES.KITKAT) 93 | private static void transparentStatusBar(Window window) { 94 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { 95 | View decorView = window.getDecorView(); 96 | int option = View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_LAYOUT_STABLE; 97 | decorView.setSystemUiVisibility(option); 98 | window.setStatusBarColor(Color.TRANSPARENT); 99 | } else { 100 | window.addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS); 101 | } 102 | } 103 | } -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /android/app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | ModalDemo 3 | 4 | -------------------------------------------------------------------------------- /android/app/src/main/res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /android/build.gradle: -------------------------------------------------------------------------------- 1 | // Top-level build file where you can add configuration options common to all sub-projects/modules. 2 | 3 | buildscript { 4 | ext { 5 | buildToolsVersion = "28.0.3" 6 | minSdkVersion = 16 7 | compileSdkVersion = 28 8 | targetSdkVersion = 28 9 | supportLibVersion = "28.0.0" 10 | } 11 | repositories { 12 | google() 13 | jcenter() 14 | } 15 | dependencies { 16 | classpath 'com.android.tools.build:gradle:3.3.1' 17 | 18 | // NOTE: Do not place your application dependencies here; they belong 19 | // in the individual module build.gradle files 20 | } 21 | } 22 | 23 | allprojects { 24 | repositories { 25 | mavenLocal() 26 | google() 27 | jcenter() 28 | maven { 29 | // All of React Native (JS, Obj-C sources, Android binaries) is installed from npm 30 | url "$rootDir/../node_modules/react-native/android" 31 | } 32 | } 33 | } 34 | -------------------------------------------------------------------------------- /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 | -------------------------------------------------------------------------------- /android/gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/hzl123456/ModalViewDemo/b9c44fa58ab0dae140ee6cc3e4102d966fe9d5a9/android/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /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-4.10.2-all.zip 6 | -------------------------------------------------------------------------------- /android/gradlew: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env sh 2 | 3 | ############################################################################## 4 | ## 5 | ## Gradle start up script for UN*X 6 | ## 7 | ############################################################################## 8 | 9 | # Attempt to set APP_HOME 10 | # Resolve links: $0 may be a link 11 | PRG="$0" 12 | # Need this for relative symlinks. 13 | while [ -h "$PRG" ] ; do 14 | ls=`ls -ld "$PRG"` 15 | link=`expr "$ls" : '.*-> \(.*\)$'` 16 | if expr "$link" : '/.*' > /dev/null; then 17 | PRG="$link" 18 | else 19 | PRG=`dirname "$PRG"`"/$link" 20 | fi 21 | done 22 | SAVED="`pwd`" 23 | cd "`dirname \"$PRG\"`/" >/dev/null 24 | APP_HOME="`pwd -P`" 25 | cd "$SAVED" >/dev/null 26 | 27 | APP_NAME="Gradle" 28 | APP_BASE_NAME=`basename "$0"` 29 | 30 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 31 | DEFAULT_JVM_OPTS="" 32 | 33 | # Use the maximum available, or set MAX_FD != -1 to use that value. 34 | MAX_FD="maximum" 35 | 36 | warn () { 37 | echo "$*" 38 | } 39 | 40 | die () { 41 | echo 42 | echo "$*" 43 | echo 44 | exit 1 45 | } 46 | 47 | # OS specific support (must be 'true' or 'false'). 48 | cygwin=false 49 | msys=false 50 | darwin=false 51 | nonstop=false 52 | case "`uname`" in 53 | CYGWIN* ) 54 | cygwin=true 55 | ;; 56 | Darwin* ) 57 | darwin=true 58 | ;; 59 | MINGW* ) 60 | msys=true 61 | ;; 62 | NONSTOP* ) 63 | nonstop=true 64 | ;; 65 | esac 66 | 67 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 68 | 69 | # Determine the Java command to use to start the JVM. 70 | if [ -n "$JAVA_HOME" ] ; then 71 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 72 | # IBM's JDK on AIX uses strange locations for the executables 73 | JAVACMD="$JAVA_HOME/jre/sh/java" 74 | else 75 | JAVACMD="$JAVA_HOME/bin/java" 76 | fi 77 | if [ ! -x "$JAVACMD" ] ; then 78 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 79 | 80 | Please set the JAVA_HOME variable in your environment to match the 81 | location of your Java installation." 82 | fi 83 | else 84 | JAVACMD="java" 85 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 86 | 87 | Please set the JAVA_HOME variable in your environment to match the 88 | location of your Java installation." 89 | fi 90 | 91 | # Increase the maximum file descriptors if we can. 92 | if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then 93 | MAX_FD_LIMIT=`ulimit -H -n` 94 | if [ $? -eq 0 ] ; then 95 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 96 | MAX_FD="$MAX_FD_LIMIT" 97 | fi 98 | ulimit -n $MAX_FD 99 | if [ $? -ne 0 ] ; then 100 | warn "Could not set maximum file descriptor limit: $MAX_FD" 101 | fi 102 | else 103 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 104 | fi 105 | fi 106 | 107 | # For Darwin, add options to specify how the application appears in the dock 108 | if $darwin; then 109 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 110 | fi 111 | 112 | # For Cygwin, switch paths to Windows format before running java 113 | if $cygwin ; then 114 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 115 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 116 | JAVACMD=`cygpath --unix "$JAVACMD"` 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 | # Escape application args 158 | save () { 159 | for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done 160 | echo " " 161 | } 162 | APP_ARGS=$(save "$@") 163 | 164 | # Collect all arguments for the java command, following the shell quoting and substitution rules 165 | eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" 166 | 167 | # by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong 168 | if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then 169 | cd "$(dirname "$0")" 170 | fi 171 | 172 | exec "$JAVACMD" "$@" 173 | -------------------------------------------------------------------------------- /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 | set DIRNAME=%~dp0 12 | if "%DIRNAME%" == "" set DIRNAME=. 13 | set APP_BASE_NAME=%~n0 14 | set APP_HOME=%DIRNAME% 15 | 16 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 17 | set DEFAULT_JVM_OPTS= 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 Windows variants 50 | 51 | if not "%OS%" == "Windows_NT" goto win9xME_args 52 | 53 | :win9xME_args 54 | @rem Slurp the command line arguments. 55 | set CMD_LINE_ARGS= 56 | set _SKIP=2 57 | 58 | :win9xME_args_slurp 59 | if "x%~1" == "x" goto execute 60 | 61 | set CMD_LINE_ARGS=%* 62 | 63 | :execute 64 | @rem Setup the command line 65 | 66 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 67 | 68 | @rem Execute Gradle 69 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% 70 | 71 | :end 72 | @rem End local scope for the variables with windows NT shell 73 | if "%ERRORLEVEL%"=="0" goto mainEnd 74 | 75 | :fail 76 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 77 | rem the _cmd.exe /c_ return code! 78 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 79 | exit /b 1 80 | 81 | :mainEnd 82 | if "%OS%"=="Windows_NT" endlocal 83 | 84 | :omega 85 | -------------------------------------------------------------------------------- /android/keystores/BUCK: -------------------------------------------------------------------------------- 1 | keystore( 2 | name = "debug", 3 | properties = "debug.keystore.properties", 4 | store = "debug.keystore", 5 | visibility = [ 6 | "PUBLIC", 7 | ], 8 | ) 9 | -------------------------------------------------------------------------------- /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 | -------------------------------------------------------------------------------- /android/settings.gradle: -------------------------------------------------------------------------------- 1 | rootProject.name = 'ModalDemo' 2 | 3 | include ':app' 4 | -------------------------------------------------------------------------------- /app.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ModalDemo", 3 | "displayName": "ModalDemo" 4 | } -------------------------------------------------------------------------------- /app/FullScreenModal/FullScreenModal.android.js: -------------------------------------------------------------------------------- 1 | /** 2 | * android 全屏展示的 Modal 3 | * @Author: linhe 4 | * @Date: 2019-04-08 15:17 5 | */ 6 | import React, {Component} from 'react'; 7 | import {requireNativeComponent, View} from 'react-native'; 8 | import PropTypes from 'prop-types'; 9 | 10 | const FullScreenModal = requireNativeComponent('RCTFullScreenModalHostView', null); 11 | 12 | export default class FullScreenModalViewAndroid extends Component { 13 | _shouldSetResponder = () => { 14 | return true; 15 | } 16 | 17 | static propTypes = { 18 | isDarkMode: PropTypes.bool, // false 表示白底黑字,true 表示黑底白字 19 | autoKeyboard: PropTypes.bool, // 未知原因的坑,modal中的edittext自动弹起键盘要设置这个参数为true 20 | }; 21 | 22 | render() { 23 | if (this.props.visible === false) { 24 | return null; 25 | } 26 | const containerStyles = { 27 | backgroundColor: this.props.transparent ? 'transparent' : 'white', 28 | }; 29 | return ( 30 | this.props.onShow && this.props.onShow()} 34 | onFullScreenRequstClose={() => this.props.onRequestClose && this.props.onRequestClose()}> 35 | 36 | {this.props.children} 37 | 38 | 39 | ); 40 | } 41 | } 42 | -------------------------------------------------------------------------------- /app/FullScreenModal/FullScreenModal.ios.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @Author: linhe 3 | * @Date: 2019-04-09 14:15 4 | */ 5 | import {Modal} from 'react-native'; 6 | 7 | export default Modal 8 | 9 | -------------------------------------------------------------------------------- /app/ModalView.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @Author: linhe 3 | * @Date: 2019-05-12 10:11 4 | * 5 | * 因为ios端同时只能存在一个Modal,并且Modal多次显示隐藏会有很奇怪的bug 6 | * 7 | * 为了兼容ios的使用,这里需要封装一个ModalView 8 | * 9 | * Android 依旧使用 React Native Modal 来进行实现 10 | * ios 的话采用 RootSiblings 配合进行使用 11 | * 12 | * 这个是因为有的modal里面还需要跳转到其他界面 13 | * 这个时候主要要将该View放到最外边的层级才可以 14 | * 15 | * modalType:1 //表示使用Modal进行实现 16 | * 2 //表示使用RootSiblings进行实现 17 | * 3 //表示使用View进行实现 18 | * 注意:默认情况下 Android 使用的是1,ios使用的是2 19 | * 20 | * 同时采用与 React Native Modal 相同的API 21 | */ 22 | 'use strict'; 23 | import React, {Component} from "react"; 24 | import {Animated, BackHandler, Platform, Easing, StyleSheet, Dimensions, Modal} from "react-native"; 25 | import PropTypes from 'prop-types' 26 | import RootSiblings from 'react-native-root-siblings'; 27 | import FullScreenModal from './FullScreenModal/FullScreenModal' 28 | 29 | const {height} = Dimensions.get('window') 30 | const animationShortTime = 250 //动画时长为250ms 31 | const DEVICE_BACK_EVENT = 'hardwareBackPress'; 32 | 33 | export default class ModalView extends Component { 34 | 35 | static propTypes = { 36 | isDarkMode: PropTypes.bool, // false 表示白底黑字,true 表示黑底白字 37 | autoKeyboard: PropTypes.bool, // 未知原因的坑,modal中的edittext自动弹起键盘要设置这个参数为true 38 | useReactModal: PropTypes.bool, // 是否使用 RN Modal 进行实现 39 | modalType: PropTypes.number // modalType 类型,默认 android 为 1,ios 为 2 40 | }; 41 | 42 | static defaultProps = { 43 | isDarkMode: false, 44 | autoKeyboard: false, 45 | useReactModal: false, 46 | modalType: (Platform.OS === 'android' ? 1 : 2) // 默认 android 为1,ios 为2 47 | }; 48 | 49 | constructor(props) { 50 | super(props); 51 | this.state = { 52 | visible: false, 53 | animationSlide: new Animated.Value(0), 54 | animationFade: new Animated.Value(0) 55 | }; 56 | } 57 | 58 | render() { 59 | const {modalType} = this.props 60 | if (modalType === 1) { //modal实现 61 | return this._renderModal() 62 | } else if (modalType === 2) { //RootSiblings实现 63 | this.RootSiblings && this.RootSiblings.update(this._renderRootSiblings()) 64 | return null 65 | } else { //View的实现 66 | return this._renderView() 67 | } 68 | } 69 | 70 | _renderModal = () => { 71 | const ModalView = this.props.useReactModal ? Modal : FullScreenModal 72 | return ( 73 | { 78 | if (this.props.onRequestClose) { 79 | this.props.onRequestClose() 80 | } else { 81 | this.disMiss() 82 | } 83 | }}> 84 | {this.props.children} 85 | 86 | ) 87 | } 88 | 89 | _renderRootSiblings = () => { 90 | return ( 91 | 101 | {this.props.children} 102 | 103 | ); 104 | } 105 | 106 | _renderView = () => { 107 | if (this.state.visible) { 108 | return ( 109 | 119 | {this.props.children} 120 | 121 | ); 122 | } else { 123 | return null 124 | } 125 | } 126 | 127 | show = (callback) => { 128 | if (this.isShow()) { 129 | return 130 | } 131 | const {modalType, animationType} = this.props 132 | if (modalType === 1) { //modal 133 | this.setState({visible: true}, () => callback && callback()) 134 | } else if (modalType === 2) { //RootSiblings 135 | this.RootSiblings = new RootSiblings(this._renderRootSiblings(), () => { 136 | if (animationType === 'fade') { 137 | this._animationFadeIn(callback) 138 | } else if (animationType === 'slide') { 139 | this._animationSlideIn(callback) 140 | } else { 141 | this._animationNoneIn(callback) 142 | } 143 | }); 144 | // 这里需要监听 back 键 145 | this._addHandleBack() 146 | } else { //view 147 | if (animationType === 'fade') { 148 | this.setState({visible: true}, () => this._animationFadeIn(callback)) 149 | } else if (animationType === 'slide') { 150 | this.setState({visible: true}, () => this._animationSlideIn(callback)) 151 | } else { 152 | this.setState({visible: true}, () => this._animationNoneIn(callback)) 153 | } 154 | // 这里需要监听 back 键 155 | this._addHandleBack() 156 | } 157 | } 158 | 159 | disMiss = (callback) => { 160 | if (!this.isShow()) { 161 | return 162 | } 163 | const {modalType, animationType} = this.props 164 | if (modalType === 1) { //modal 165 | this.setState({visible: false}, () => callback && callback()) 166 | } else { //RootSiblings和View 167 | if (animationType === 'fade') { 168 | this._animationFadeOut(callback) 169 | } else if (animationType === 'slide') { 170 | this._animationSlideOut(callback) 171 | } else { 172 | this._animationNoneOut(callback) 173 | } 174 | // 移除 back 键的监听 175 | this._removeHandleBack() 176 | } 177 | } 178 | 179 | isShow = () => { 180 | const {modalType} = this.props 181 | if (modalType === 1 || modalType === 3) { //modal和view 182 | return this.state.visible 183 | } else { //RootSiblings 184 | return !!this.RootSiblings 185 | } 186 | } 187 | 188 | _addHandleBack = () => { 189 | if (Platform.OS === 'ios') { 190 | return 191 | } 192 | // 监听back键 193 | this.handleBack = BackHandler.addEventListener(DEVICE_BACK_EVENT, () => { 194 | const {onRequestClose} = this.props 195 | if (onRequestClose) { 196 | onRequestClose() 197 | } else { 198 | this.disMiss() 199 | } 200 | return true 201 | }); 202 | } 203 | 204 | _removeHandleBack = () => { 205 | if (Platform.OS === 'ios') { 206 | return 207 | } 208 | this.handleBack && this.handleBack.remove() 209 | } 210 | 211 | _animationNoneIn = (callback) => { 212 | this.state.animationSlide.setValue(1) 213 | this.state.animationFade.setValue(1) 214 | callback && callback() 215 | } 216 | 217 | _animationNoneOut = (callback) => { 218 | this._animationCallback(callback); 219 | } 220 | 221 | _animationSlideIn = (callback) => { 222 | this.state.animationSlide.setValue(0) 223 | this.state.animationFade.setValue(1) 224 | Animated.timing(this.state.animationSlide, { 225 | easing: Easing.in(), 226 | duration: animationShortTime, 227 | toValue: 1, 228 | }).start(() => callback && callback()); 229 | } 230 | 231 | _animationSlideOut = (callback) => { 232 | this.state.animationSlide.setValue(1) 233 | this.state.animationFade.setValue(1) 234 | Animated.timing(this.state.animationSlide, { 235 | easing: Easing.in(), 236 | duration: animationShortTime, 237 | toValue: 0, 238 | }).start(() => this._animationCallback(callback)); 239 | } 240 | 241 | _animationFadeIn = (callback) => { 242 | this.state.animationSlide.setValue(1) 243 | this.state.animationFade.setValue(0) 244 | Animated.timing(this.state.animationFade, { 245 | easing: Easing.in(), 246 | duration: animationShortTime, 247 | toValue: 1, 248 | }).start(() => callback && callback()); 249 | } 250 | 251 | _animationFadeOut = (callback) => { 252 | this.state.animationSlide.setValue(1) 253 | this.state.animationFade.setValue(1) 254 | Animated.timing(this.state.animationFade, { 255 | easing: Easing.in(), 256 | duration: animationShortTime, 257 | toValue: 0, 258 | }).start(() => this._animationCallback(callback)); 259 | } 260 | 261 | _animationCallback = (callback) => { 262 | if (this.props.modalType === 2) {//RootSiblings 263 | this.RootSiblings && this.RootSiblings.destroy(() => { 264 | callback && callback() 265 | this.RootSiblings = undefined 266 | }) 267 | } else { //view 268 | this.setState({visible: false}, () => callback && callback()) 269 | } 270 | } 271 | } 272 | 273 | const styles = StyleSheet.create({ 274 | root: { 275 | position: 'absolute', 276 | left: 0, 277 | top: 0, 278 | right: 0, 279 | bottom: 0, 280 | } 281 | }); -------------------------------------------------------------------------------- /babel.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | presets: ['module:metro-react-native-babel-preset'], 3 | }; 4 | -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @format 3 | */ 4 | import {AppRegistry} from 'react-native'; 5 | import App from './App'; 6 | import {name as appName} from './app.json'; 7 | 8 | AppRegistry.registerComponent(appName, () => App); 9 | -------------------------------------------------------------------------------- /ios/ModalDemo-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 | -------------------------------------------------------------------------------- /ios/ModalDemo-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 | -------------------------------------------------------------------------------- /ios/ModalDemo.xcodeproj/project.pbxproj: -------------------------------------------------------------------------------- 1 | // !$*UTF8*$! 2 | { 3 | archiveVersion = 1; 4 | classes = { 5 | }; 6 | objectVersion = 46; 7 | objects = { 8 | 9 | /* Begin PBXBuildFile section */ 10 | 00C302E51ABCBA2D00DB3ED1 /* libRCTActionSheet.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302AC1ABCB8CE00DB3ED1 /* libRCTActionSheet.a */; }; 11 | 00C302E71ABCBA2D00DB3ED1 /* libRCTGeolocation.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302BA1ABCB90400DB3ED1 /* libRCTGeolocation.a */; }; 12 | 00C302E81ABCBA2D00DB3ED1 /* libRCTImage.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */; }; 13 | 00C302E91ABCBA2D00DB3ED1 /* libRCTNetwork.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */; }; 14 | 00C302EA1ABCBA2D00DB3ED1 /* libRCTVibration.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */; }; 15 | 00E356F31AD99517003FC87E /* ModalDemoTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 00E356F21AD99517003FC87E /* ModalDemoTests.m */; }; 16 | 11D1A2F320CAFA9E000508D9 /* libRCTAnimation.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 5E9157331DD0AC6500FF2AA8 /* libRCTAnimation.a */; }; 17 | 133E29F31AD74F7200F7D852 /* libRCTLinking.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 78C398B91ACF4ADC00677621 /* libRCTLinking.a */; }; 18 | 139105C61AF99C1200B5F7CC /* libRCTSettings.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */; }; 19 | 139FDEF61B0652A700C62182 /* libRCTWebSocket.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */; }; 20 | 13B07FBC1A68108700A75B9A /* AppDelegate.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB01A68108700A75B9A /* AppDelegate.m */; }; 21 | 13B07FBD1A68108700A75B9A /* LaunchScreen.xib in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB11A68108700A75B9A /* LaunchScreen.xib */; }; 22 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB51A68108700A75B9A /* Images.xcassets */; }; 23 | 13B07FC11A68108700A75B9A /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB71A68108700A75B9A /* main.m */; }; 24 | 140ED2AC1D01E1AD002B40FF /* libReact.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 146834041AC3E56700842450 /* libReact.a */; }; 25 | 146834051AC3E58100842450 /* libReact.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 146834041AC3E56700842450 /* libReact.a */; }; 26 | 2D02E4BC1E0B4A80006451C7 /* AppDelegate.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB01A68108700A75B9A /* AppDelegate.m */; }; 27 | 2D02E4BD1E0B4A84006451C7 /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB51A68108700A75B9A /* Images.xcassets */; }; 28 | 2D02E4BF1E0B4AB3006451C7 /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB71A68108700A75B9A /* main.m */; }; 29 | 2D02E4C21E0B4AEC006451C7 /* libRCTAnimation.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 5E9157351DD0AC6500FF2AA8 /* libRCTAnimation.a */; }; 30 | 2D02E4C31E0B4AEC006451C7 /* libRCTImage-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E841DF850E9000B6D8A /* libRCTImage-tvOS.a */; }; 31 | 2D02E4C41E0B4AEC006451C7 /* libRCTLinking-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E881DF850E9000B6D8A /* libRCTLinking-tvOS.a */; }; 32 | 2D02E4C51E0B4AEC006451C7 /* libRCTNetwork-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E8C1DF850E9000B6D8A /* libRCTNetwork-tvOS.a */; }; 33 | 2D02E4C61E0B4AEC006451C7 /* libRCTSettings-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E901DF850E9000B6D8A /* libRCTSettings-tvOS.a */; }; 34 | 2D02E4C71E0B4AEC006451C7 /* libRCTText-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E941DF850E9000B6D8A /* libRCTText-tvOS.a */; }; 35 | 2D02E4C81E0B4AEC006451C7 /* libRCTWebSocket-tvOS.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3E991DF850E9000B6D8A /* libRCTWebSocket-tvOS.a */; }; 36 | 2D16E6881FA4F8E400B85C8A /* libReact.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 2D16E6891FA4F8E400B85C8A /* libReact.a */; }; 37 | 2DCD954D1E0B4F2C00145EB5 /* ModalDemoTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 00E356F21AD99517003FC87E /* ModalDemoTests.m */; }; 38 | 2DF0FFEE2056DD460020B375 /* libReact.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3DAD3EA31DF850E9000B6D8A /* libReact.a */; }; 39 | 832341BD1AAA6AB300B99B32 /* libRCTText.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 832341B51AAA6A8300B99B32 /* libRCTText.a */; }; 40 | ADBDB9381DFEBF1600ED6528 /* libRCTBlob.a in Frameworks */ = {isa = PBXBuildFile; fileRef = ADBDB9271DFEBF0700ED6528 /* libRCTBlob.a */; }; 41 | ED297163215061F000B7C4FE /* JavaScriptCore.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = ED297162215061F000B7C4FE /* JavaScriptCore.framework */; }; 42 | ED2971652150620600B7C4FE /* JavaScriptCore.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = ED2971642150620600B7C4FE /* JavaScriptCore.framework */; }; 43 | /* End PBXBuildFile section */ 44 | 45 | /* Begin PBXContainerItemProxy section */ 46 | 00C302AB1ABCB8CE00DB3ED1 /* PBXContainerItemProxy */ = { 47 | isa = PBXContainerItemProxy; 48 | containerPortal = 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */; 49 | proxyType = 2; 50 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 51 | remoteInfo = RCTActionSheet; 52 | }; 53 | 00C302B91ABCB90400DB3ED1 /* PBXContainerItemProxy */ = { 54 | isa = PBXContainerItemProxy; 55 | containerPortal = 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */; 56 | proxyType = 2; 57 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 58 | remoteInfo = RCTGeolocation; 59 | }; 60 | 00C302BF1ABCB91800DB3ED1 /* PBXContainerItemProxy */ = { 61 | isa = PBXContainerItemProxy; 62 | containerPortal = 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */; 63 | proxyType = 2; 64 | remoteGlobalIDString = 58B5115D1A9E6B3D00147676; 65 | remoteInfo = RCTImage; 66 | }; 67 | 00C302DB1ABCB9D200DB3ED1 /* PBXContainerItemProxy */ = { 68 | isa = PBXContainerItemProxy; 69 | containerPortal = 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */; 70 | proxyType = 2; 71 | remoteGlobalIDString = 58B511DB1A9E6C8500147676; 72 | remoteInfo = RCTNetwork; 73 | }; 74 | 00C302E31ABCB9EE00DB3ED1 /* PBXContainerItemProxy */ = { 75 | isa = PBXContainerItemProxy; 76 | containerPortal = 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */; 77 | proxyType = 2; 78 | remoteGlobalIDString = 832C81801AAF6DEF007FA2F7; 79 | remoteInfo = RCTVibration; 80 | }; 81 | 00E356F41AD99517003FC87E /* PBXContainerItemProxy */ = { 82 | isa = PBXContainerItemProxy; 83 | containerPortal = 83CBB9F71A601CBA00E9B192 /* Project object */; 84 | proxyType = 1; 85 | remoteGlobalIDString = 13B07F861A680F5B00A75B9A; 86 | remoteInfo = ModalDemo; 87 | }; 88 | 139105C01AF99BAD00B5F7CC /* PBXContainerItemProxy */ = { 89 | isa = PBXContainerItemProxy; 90 | containerPortal = 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */; 91 | proxyType = 2; 92 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 93 | remoteInfo = RCTSettings; 94 | }; 95 | 139FDEF31B06529B00C62182 /* PBXContainerItemProxy */ = { 96 | isa = PBXContainerItemProxy; 97 | containerPortal = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */; 98 | proxyType = 2; 99 | remoteGlobalIDString = 3C86DF461ADF2C930047B81A; 100 | remoteInfo = RCTWebSocket; 101 | }; 102 | 146834031AC3E56700842450 /* PBXContainerItemProxy */ = { 103 | isa = PBXContainerItemProxy; 104 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 105 | proxyType = 2; 106 | remoteGlobalIDString = 83CBBA2E1A601D0E00E9B192; 107 | remoteInfo = React; 108 | }; 109 | 2D02E4911E0B4A5D006451C7 /* PBXContainerItemProxy */ = { 110 | isa = PBXContainerItemProxy; 111 | containerPortal = 83CBB9F71A601CBA00E9B192 /* Project object */; 112 | proxyType = 1; 113 | remoteGlobalIDString = 2D02E47A1E0B4A5D006451C7; 114 | remoteInfo = "ModalDemo-tvOS"; 115 | }; 116 | 2D16E6711FA4F8DC00B85C8A /* PBXContainerItemProxy */ = { 117 | isa = PBXContainerItemProxy; 118 | containerPortal = ADBDB91F1DFEBF0600ED6528 /* RCTBlob.xcodeproj */; 119 | proxyType = 2; 120 | remoteGlobalIDString = ADD01A681E09402E00F6D226; 121 | remoteInfo = "RCTBlob-tvOS"; 122 | }; 123 | 2D16E6831FA4F8DC00B85C8A /* PBXContainerItemProxy */ = { 124 | isa = PBXContainerItemProxy; 125 | containerPortal = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */; 126 | proxyType = 2; 127 | remoteGlobalIDString = 3DBE0D001F3B181A0099AA32; 128 | remoteInfo = fishhook; 129 | }; 130 | 2D16E6851FA4F8DC00B85C8A /* PBXContainerItemProxy */ = { 131 | isa = PBXContainerItemProxy; 132 | containerPortal = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */; 133 | proxyType = 2; 134 | remoteGlobalIDString = 3DBE0D0D1F3B181C0099AA32; 135 | remoteInfo = "fishhook-tvOS"; 136 | }; 137 | 2DF0FFDE2056DD460020B375 /* PBXContainerItemProxy */ = { 138 | isa = PBXContainerItemProxy; 139 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 140 | proxyType = 2; 141 | remoteGlobalIDString = EBF21BDC1FC498900052F4D5; 142 | remoteInfo = jsinspector; 143 | }; 144 | 2DF0FFE02056DD460020B375 /* PBXContainerItemProxy */ = { 145 | isa = PBXContainerItemProxy; 146 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 147 | proxyType = 2; 148 | remoteGlobalIDString = EBF21BFA1FC4989A0052F4D5; 149 | remoteInfo = "jsinspector-tvOS"; 150 | }; 151 | 2DF0FFE22056DD460020B375 /* PBXContainerItemProxy */ = { 152 | isa = PBXContainerItemProxy; 153 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 154 | proxyType = 2; 155 | remoteGlobalIDString = 139D7ECE1E25DB7D00323FB7; 156 | remoteInfo = "third-party"; 157 | }; 158 | 2DF0FFE42056DD460020B375 /* PBXContainerItemProxy */ = { 159 | isa = PBXContainerItemProxy; 160 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 161 | proxyType = 2; 162 | remoteGlobalIDString = 3D383D3C1EBD27B6005632C8; 163 | remoteInfo = "third-party-tvOS"; 164 | }; 165 | 2DF0FFE62056DD460020B375 /* PBXContainerItemProxy */ = { 166 | isa = PBXContainerItemProxy; 167 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 168 | proxyType = 2; 169 | remoteGlobalIDString = 139D7E881E25C6D100323FB7; 170 | remoteInfo = "double-conversion"; 171 | }; 172 | 2DF0FFE82056DD460020B375 /* PBXContainerItemProxy */ = { 173 | isa = PBXContainerItemProxy; 174 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 175 | proxyType = 2; 176 | remoteGlobalIDString = 3D383D621EBD27B9005632C8; 177 | remoteInfo = "double-conversion-tvOS"; 178 | }; 179 | 2DF0FFEA2056DD460020B375 /* PBXContainerItemProxy */ = { 180 | isa = PBXContainerItemProxy; 181 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 182 | proxyType = 2; 183 | remoteGlobalIDString = 9936F3131F5F2E4B0010BF04; 184 | remoteInfo = privatedata; 185 | }; 186 | 2DF0FFEC2056DD460020B375 /* PBXContainerItemProxy */ = { 187 | isa = PBXContainerItemProxy; 188 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 189 | proxyType = 2; 190 | remoteGlobalIDString = 9936F32F1F5F2E5B0010BF04; 191 | remoteInfo = "privatedata-tvOS"; 192 | }; 193 | 3DAD3E831DF850E9000B6D8A /* PBXContainerItemProxy */ = { 194 | isa = PBXContainerItemProxy; 195 | containerPortal = 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */; 196 | proxyType = 2; 197 | remoteGlobalIDString = 2D2A283A1D9B042B00D4039D; 198 | remoteInfo = "RCTImage-tvOS"; 199 | }; 200 | 3DAD3E871DF850E9000B6D8A /* PBXContainerItemProxy */ = { 201 | isa = PBXContainerItemProxy; 202 | containerPortal = 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */; 203 | proxyType = 2; 204 | remoteGlobalIDString = 2D2A28471D9B043800D4039D; 205 | remoteInfo = "RCTLinking-tvOS"; 206 | }; 207 | 3DAD3E8B1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 208 | isa = PBXContainerItemProxy; 209 | containerPortal = 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */; 210 | proxyType = 2; 211 | remoteGlobalIDString = 2D2A28541D9B044C00D4039D; 212 | remoteInfo = "RCTNetwork-tvOS"; 213 | }; 214 | 3DAD3E8F1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 215 | isa = PBXContainerItemProxy; 216 | containerPortal = 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */; 217 | proxyType = 2; 218 | remoteGlobalIDString = 2D2A28611D9B046600D4039D; 219 | remoteInfo = "RCTSettings-tvOS"; 220 | }; 221 | 3DAD3E931DF850E9000B6D8A /* PBXContainerItemProxy */ = { 222 | isa = PBXContainerItemProxy; 223 | containerPortal = 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */; 224 | proxyType = 2; 225 | remoteGlobalIDString = 2D2A287B1D9B048500D4039D; 226 | remoteInfo = "RCTText-tvOS"; 227 | }; 228 | 3DAD3E981DF850E9000B6D8A /* PBXContainerItemProxy */ = { 229 | isa = PBXContainerItemProxy; 230 | containerPortal = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */; 231 | proxyType = 2; 232 | remoteGlobalIDString = 2D2A28881D9B049200D4039D; 233 | remoteInfo = "RCTWebSocket-tvOS"; 234 | }; 235 | 3DAD3EA21DF850E9000B6D8A /* PBXContainerItemProxy */ = { 236 | isa = PBXContainerItemProxy; 237 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 238 | proxyType = 2; 239 | remoteGlobalIDString = 2D2A28131D9B038B00D4039D; 240 | remoteInfo = "React-tvOS"; 241 | }; 242 | 3DAD3EA41DF850E9000B6D8A /* PBXContainerItemProxy */ = { 243 | isa = PBXContainerItemProxy; 244 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 245 | proxyType = 2; 246 | remoteGlobalIDString = 3D3C059A1DE3340900C268FA; 247 | remoteInfo = yoga; 248 | }; 249 | 3DAD3EA61DF850E9000B6D8A /* PBXContainerItemProxy */ = { 250 | isa = PBXContainerItemProxy; 251 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 252 | proxyType = 2; 253 | remoteGlobalIDString = 3D3C06751DE3340C00C268FA; 254 | remoteInfo = "yoga-tvOS"; 255 | }; 256 | 3DAD3EA81DF850E9000B6D8A /* PBXContainerItemProxy */ = { 257 | isa = PBXContainerItemProxy; 258 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 259 | proxyType = 2; 260 | remoteGlobalIDString = 3D3CD9251DE5FBEC00167DC4; 261 | remoteInfo = cxxreact; 262 | }; 263 | 3DAD3EAA1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 264 | isa = PBXContainerItemProxy; 265 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 266 | proxyType = 2; 267 | remoteGlobalIDString = 3D3CD9321DE5FBEE00167DC4; 268 | remoteInfo = "cxxreact-tvOS"; 269 | }; 270 | 3DAD3EAC1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 271 | isa = PBXContainerItemProxy; 272 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 273 | proxyType = 2; 274 | remoteGlobalIDString = 3D3CD90B1DE5FBD600167DC4; 275 | remoteInfo = jschelpers; 276 | }; 277 | 3DAD3EAE1DF850E9000B6D8A /* PBXContainerItemProxy */ = { 278 | isa = PBXContainerItemProxy; 279 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */; 280 | proxyType = 2; 281 | remoteGlobalIDString = 3D3CD9181DE5FBD800167DC4; 282 | remoteInfo = "jschelpers-tvOS"; 283 | }; 284 | 5E9157321DD0AC6500FF2AA8 /* PBXContainerItemProxy */ = { 285 | isa = PBXContainerItemProxy; 286 | containerPortal = 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */; 287 | proxyType = 2; 288 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 289 | remoteInfo = RCTAnimation; 290 | }; 291 | 5E9157341DD0AC6500FF2AA8 /* PBXContainerItemProxy */ = { 292 | isa = PBXContainerItemProxy; 293 | containerPortal = 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */; 294 | proxyType = 2; 295 | remoteGlobalIDString = 2D2A28201D9B03D100D4039D; 296 | remoteInfo = "RCTAnimation-tvOS"; 297 | }; 298 | 78C398B81ACF4ADC00677621 /* PBXContainerItemProxy */ = { 299 | isa = PBXContainerItemProxy; 300 | containerPortal = 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */; 301 | proxyType = 2; 302 | remoteGlobalIDString = 134814201AA4EA6300B7C361; 303 | remoteInfo = RCTLinking; 304 | }; 305 | 832341B41AAA6A8300B99B32 /* PBXContainerItemProxy */ = { 306 | isa = PBXContainerItemProxy; 307 | containerPortal = 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */; 308 | proxyType = 2; 309 | remoteGlobalIDString = 58B5119B1A9E6C1200147676; 310 | remoteInfo = RCTText; 311 | }; 312 | ADBDB9261DFEBF0700ED6528 /* PBXContainerItemProxy */ = { 313 | isa = PBXContainerItemProxy; 314 | containerPortal = ADBDB91F1DFEBF0600ED6528 /* RCTBlob.xcodeproj */; 315 | proxyType = 2; 316 | remoteGlobalIDString = 358F4ED71D1E81A9004DF814; 317 | remoteInfo = RCTBlob; 318 | }; 319 | /* End PBXContainerItemProxy section */ 320 | 321 | /* Begin PBXFileReference section */ 322 | 008F07F21AC5B25A0029DE68 /* main.jsbundle */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; path = main.jsbundle; sourceTree = ""; }; 323 | 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTActionSheet.xcodeproj; path = "../node_modules/react-native/Libraries/ActionSheetIOS/RCTActionSheet.xcodeproj"; sourceTree = ""; }; 324 | 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTGeolocation.xcodeproj; path = "../node_modules/react-native/Libraries/Geolocation/RCTGeolocation.xcodeproj"; sourceTree = ""; }; 325 | 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTImage.xcodeproj; path = "../node_modules/react-native/Libraries/Image/RCTImage.xcodeproj"; sourceTree = ""; }; 326 | 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTNetwork.xcodeproj; path = "../node_modules/react-native/Libraries/Network/RCTNetwork.xcodeproj"; sourceTree = ""; }; 327 | 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTVibration.xcodeproj; path = "../node_modules/react-native/Libraries/Vibration/RCTVibration.xcodeproj"; sourceTree = ""; }; 328 | 00E356EE1AD99517003FC87E /* ModalDemoTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = ModalDemoTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; 329 | 00E356F11AD99517003FC87E /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; 330 | 00E356F21AD99517003FC87E /* ModalDemoTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = ModalDemoTests.m; sourceTree = ""; }; 331 | 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTSettings.xcodeproj; path = "../node_modules/react-native/Libraries/Settings/RCTSettings.xcodeproj"; sourceTree = ""; }; 332 | 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTWebSocket.xcodeproj; path = "../node_modules/react-native/Libraries/WebSocket/RCTWebSocket.xcodeproj"; sourceTree = ""; }; 333 | 13B07F961A680F5B00A75B9A /* ModalDemo.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = ModalDemo.app; sourceTree = BUILT_PRODUCTS_DIR; }; 334 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = AppDelegate.h; path = ModalDemo/AppDelegate.h; sourceTree = ""; }; 335 | 13B07FB01A68108700A75B9A /* AppDelegate.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = AppDelegate.m; path = ModalDemo/AppDelegate.m; sourceTree = ""; }; 336 | 13B07FB21A68108700A75B9A /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = Base.lproj/LaunchScreen.xib; sourceTree = ""; }; 337 | 13B07FB51A68108700A75B9A /* Images.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Images.xcassets; path = ModalDemo/Images.xcassets; sourceTree = ""; }; 338 | 13B07FB61A68108700A75B9A /* Info.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = Info.plist; path = ModalDemo/Info.plist; sourceTree = ""; }; 339 | 13B07FB71A68108700A75B9A /* main.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = main.m; path = ModalDemo/main.m; sourceTree = ""; }; 340 | 146833FF1AC3E56700842450 /* React.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = React.xcodeproj; path = "../node_modules/react-native/React/React.xcodeproj"; sourceTree = ""; }; 341 | 2D02E47B1E0B4A5D006451C7 /* ModalDemo-tvOS.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = "ModalDemo-tvOS.app"; sourceTree = BUILT_PRODUCTS_DIR; }; 342 | 2D02E4901E0B4A5D006451C7 /* ModalDemo-tvOSTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = "ModalDemo-tvOSTests.xctest"; sourceTree = BUILT_PRODUCTS_DIR; }; 343 | 2D16E6891FA4F8E400B85C8A /* libReact.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; path = libReact.a; sourceTree = BUILT_PRODUCTS_DIR; }; 344 | 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTAnimation.xcodeproj; path = "../node_modules/react-native/Libraries/NativeAnimation/RCTAnimation.xcodeproj"; sourceTree = ""; }; 345 | 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTLinking.xcodeproj; path = "../node_modules/react-native/Libraries/LinkingIOS/RCTLinking.xcodeproj"; sourceTree = ""; }; 346 | 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTText.xcodeproj; path = "../node_modules/react-native/Libraries/Text/RCTText.xcodeproj"; sourceTree = ""; }; 347 | ADBDB91F1DFEBF0600ED6528 /* RCTBlob.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTBlob.xcodeproj; path = "../node_modules/react-native/Libraries/Blob/RCTBlob.xcodeproj"; sourceTree = ""; }; 348 | ED297162215061F000B7C4FE /* JavaScriptCore.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = JavaScriptCore.framework; path = System/Library/Frameworks/JavaScriptCore.framework; sourceTree = SDKROOT; }; 349 | ED2971642150620600B7C4FE /* JavaScriptCore.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = JavaScriptCore.framework; path = Platforms/AppleTVOS.platform/Developer/SDKs/AppleTVOS12.0.sdk/System/Library/Frameworks/JavaScriptCore.framework; sourceTree = DEVELOPER_DIR; }; 350 | /* End PBXFileReference section */ 351 | 352 | /* Begin PBXFrameworksBuildPhase section */ 353 | 00E356EB1AD99517003FC87E /* Frameworks */ = { 354 | isa = PBXFrameworksBuildPhase; 355 | buildActionMask = 2147483647; 356 | files = ( 357 | 140ED2AC1D01E1AD002B40FF /* libReact.a in Frameworks */, 358 | ); 359 | runOnlyForDeploymentPostprocessing = 0; 360 | }; 361 | 13B07F8C1A680F5B00A75B9A /* Frameworks */ = { 362 | isa = PBXFrameworksBuildPhase; 363 | buildActionMask = 2147483647; 364 | files = ( 365 | ED297163215061F000B7C4FE /* JavaScriptCore.framework in Frameworks */, 366 | ADBDB9381DFEBF1600ED6528 /* libRCTBlob.a in Frameworks */, 367 | 11D1A2F320CAFA9E000508D9 /* libRCTAnimation.a in Frameworks */, 368 | 146834051AC3E58100842450 /* libReact.a in Frameworks */, 369 | 00C302E51ABCBA2D00DB3ED1 /* libRCTActionSheet.a in Frameworks */, 370 | 00C302E71ABCBA2D00DB3ED1 /* libRCTGeolocation.a in Frameworks */, 371 | 00C302E81ABCBA2D00DB3ED1 /* libRCTImage.a in Frameworks */, 372 | 133E29F31AD74F7200F7D852 /* libRCTLinking.a in Frameworks */, 373 | 00C302E91ABCBA2D00DB3ED1 /* libRCTNetwork.a in Frameworks */, 374 | 139105C61AF99C1200B5F7CC /* libRCTSettings.a in Frameworks */, 375 | 832341BD1AAA6AB300B99B32 /* libRCTText.a in Frameworks */, 376 | 00C302EA1ABCBA2D00DB3ED1 /* libRCTVibration.a in Frameworks */, 377 | 139FDEF61B0652A700C62182 /* libRCTWebSocket.a in Frameworks */, 378 | ); 379 | runOnlyForDeploymentPostprocessing = 0; 380 | }; 381 | 2D02E4781E0B4A5D006451C7 /* Frameworks */ = { 382 | isa = PBXFrameworksBuildPhase; 383 | buildActionMask = 2147483647; 384 | files = ( 385 | ED2971652150620600B7C4FE /* JavaScriptCore.framework in Frameworks */, 386 | 2D16E6881FA4F8E400B85C8A /* libReact.a in Frameworks */, 387 | 2D02E4C21E0B4AEC006451C7 /* libRCTAnimation.a in Frameworks */, 388 | 2D02E4C31E0B4AEC006451C7 /* libRCTImage-tvOS.a in Frameworks */, 389 | 2D02E4C41E0B4AEC006451C7 /* libRCTLinking-tvOS.a in Frameworks */, 390 | 2D02E4C51E0B4AEC006451C7 /* libRCTNetwork-tvOS.a in Frameworks */, 391 | 2D02E4C61E0B4AEC006451C7 /* libRCTSettings-tvOS.a in Frameworks */, 392 | 2D02E4C71E0B4AEC006451C7 /* libRCTText-tvOS.a in Frameworks */, 393 | 2D02E4C81E0B4AEC006451C7 /* libRCTWebSocket-tvOS.a in Frameworks */, 394 | ); 395 | runOnlyForDeploymentPostprocessing = 0; 396 | }; 397 | 2D02E48D1E0B4A5D006451C7 /* Frameworks */ = { 398 | isa = PBXFrameworksBuildPhase; 399 | buildActionMask = 2147483647; 400 | files = ( 401 | 2DF0FFEE2056DD460020B375 /* libReact.a in Frameworks */, 402 | ); 403 | runOnlyForDeploymentPostprocessing = 0; 404 | }; 405 | /* End PBXFrameworksBuildPhase section */ 406 | 407 | /* Begin PBXGroup section */ 408 | 00C302A81ABCB8CE00DB3ED1 /* Products */ = { 409 | isa = PBXGroup; 410 | children = ( 411 | 00C302AC1ABCB8CE00DB3ED1 /* libRCTActionSheet.a */, 412 | ); 413 | name = Products; 414 | sourceTree = ""; 415 | }; 416 | 00C302B61ABCB90400DB3ED1 /* Products */ = { 417 | isa = PBXGroup; 418 | children = ( 419 | 00C302BA1ABCB90400DB3ED1 /* libRCTGeolocation.a */, 420 | ); 421 | name = Products; 422 | sourceTree = ""; 423 | }; 424 | 00C302BC1ABCB91800DB3ED1 /* Products */ = { 425 | isa = PBXGroup; 426 | children = ( 427 | 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */, 428 | 3DAD3E841DF850E9000B6D8A /* libRCTImage-tvOS.a */, 429 | ); 430 | name = Products; 431 | sourceTree = ""; 432 | }; 433 | 00C302D41ABCB9D200DB3ED1 /* Products */ = { 434 | isa = PBXGroup; 435 | children = ( 436 | 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */, 437 | 3DAD3E8C1DF850E9000B6D8A /* libRCTNetwork-tvOS.a */, 438 | ); 439 | name = Products; 440 | sourceTree = ""; 441 | }; 442 | 00C302E01ABCB9EE00DB3ED1 /* Products */ = { 443 | isa = PBXGroup; 444 | children = ( 445 | 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */, 446 | ); 447 | name = Products; 448 | sourceTree = ""; 449 | }; 450 | 00E356EF1AD99517003FC87E /* ModalDemoTests */ = { 451 | isa = PBXGroup; 452 | children = ( 453 | 00E356F21AD99517003FC87E /* ModalDemoTests.m */, 454 | 00E356F01AD99517003FC87E /* Supporting Files */, 455 | ); 456 | path = ModalDemoTests; 457 | sourceTree = ""; 458 | }; 459 | 00E356F01AD99517003FC87E /* Supporting Files */ = { 460 | isa = PBXGroup; 461 | children = ( 462 | 00E356F11AD99517003FC87E /* Info.plist */, 463 | ); 464 | name = "Supporting Files"; 465 | sourceTree = ""; 466 | }; 467 | 139105B71AF99BAD00B5F7CC /* Products */ = { 468 | isa = PBXGroup; 469 | children = ( 470 | 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */, 471 | 3DAD3E901DF850E9000B6D8A /* libRCTSettings-tvOS.a */, 472 | ); 473 | name = Products; 474 | sourceTree = ""; 475 | }; 476 | 139FDEE71B06529A00C62182 /* Products */ = { 477 | isa = PBXGroup; 478 | children = ( 479 | 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */, 480 | 3DAD3E991DF850E9000B6D8A /* libRCTWebSocket-tvOS.a */, 481 | 2D16E6841FA4F8DC00B85C8A /* libfishhook.a */, 482 | 2D16E6861FA4F8DC00B85C8A /* libfishhook-tvOS.a */, 483 | ); 484 | name = Products; 485 | sourceTree = ""; 486 | }; 487 | 13B07FAE1A68108700A75B9A /* ModalDemo */ = { 488 | isa = PBXGroup; 489 | children = ( 490 | 008F07F21AC5B25A0029DE68 /* main.jsbundle */, 491 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */, 492 | 13B07FB01A68108700A75B9A /* AppDelegate.m */, 493 | 13B07FB51A68108700A75B9A /* Images.xcassets */, 494 | 13B07FB61A68108700A75B9A /* Info.plist */, 495 | 13B07FB11A68108700A75B9A /* LaunchScreen.xib */, 496 | 13B07FB71A68108700A75B9A /* main.m */, 497 | ); 498 | name = ModalDemo; 499 | sourceTree = ""; 500 | }; 501 | 146834001AC3E56700842450 /* Products */ = { 502 | isa = PBXGroup; 503 | children = ( 504 | 146834041AC3E56700842450 /* libReact.a */, 505 | 3DAD3EA31DF850E9000B6D8A /* libReact.a */, 506 | 3DAD3EA51DF850E9000B6D8A /* libyoga.a */, 507 | 3DAD3EA71DF850E9000B6D8A /* libyoga.a */, 508 | 3DAD3EA91DF850E9000B6D8A /* libcxxreact.a */, 509 | 3DAD3EAB1DF850E9000B6D8A /* libcxxreact.a */, 510 | 3DAD3EAD1DF850E9000B6D8A /* libjschelpers.a */, 511 | 3DAD3EAF1DF850E9000B6D8A /* libjschelpers.a */, 512 | 2DF0FFDF2056DD460020B375 /* libjsinspector.a */, 513 | 2DF0FFE12056DD460020B375 /* libjsinspector-tvOS.a */, 514 | 2DF0FFE32056DD460020B375 /* libthird-party.a */, 515 | 2DF0FFE52056DD460020B375 /* libthird-party.a */, 516 | 2DF0FFE72056DD460020B375 /* libdouble-conversion.a */, 517 | 2DF0FFE92056DD460020B375 /* libdouble-conversion.a */, 518 | 2DF0FFEB2056DD460020B375 /* libprivatedata.a */, 519 | 2DF0FFED2056DD460020B375 /* libprivatedata-tvOS.a */, 520 | ); 521 | name = Products; 522 | sourceTree = ""; 523 | }; 524 | 2D16E6871FA4F8E400B85C8A /* Frameworks */ = { 525 | isa = PBXGroup; 526 | children = ( 527 | ED297162215061F000B7C4FE /* JavaScriptCore.framework */, 528 | ED2971642150620600B7C4FE /* JavaScriptCore.framework */, 529 | 2D16E6891FA4F8E400B85C8A /* libReact.a */, 530 | ); 531 | name = Frameworks; 532 | sourceTree = ""; 533 | }; 534 | 5E91572E1DD0AC6500FF2AA8 /* Products */ = { 535 | isa = PBXGroup; 536 | children = ( 537 | 5E9157331DD0AC6500FF2AA8 /* libRCTAnimation.a */, 538 | 5E9157351DD0AC6500FF2AA8 /* libRCTAnimation.a */, 539 | ); 540 | name = Products; 541 | sourceTree = ""; 542 | }; 543 | 78C398B11ACF4ADC00677621 /* Products */ = { 544 | isa = PBXGroup; 545 | children = ( 546 | 78C398B91ACF4ADC00677621 /* libRCTLinking.a */, 547 | 3DAD3E881DF850E9000B6D8A /* libRCTLinking-tvOS.a */, 548 | ); 549 | name = Products; 550 | sourceTree = ""; 551 | }; 552 | 832341AE1AAA6A7D00B99B32 /* Libraries */ = { 553 | isa = PBXGroup; 554 | children = ( 555 | 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */, 556 | 146833FF1AC3E56700842450 /* React.xcodeproj */, 557 | 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */, 558 | ADBDB91F1DFEBF0600ED6528 /* RCTBlob.xcodeproj */, 559 | 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */, 560 | 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */, 561 | 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */, 562 | 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */, 563 | 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */, 564 | 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */, 565 | 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */, 566 | 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */, 567 | ); 568 | name = Libraries; 569 | sourceTree = ""; 570 | }; 571 | 832341B11AAA6A8300B99B32 /* Products */ = { 572 | isa = PBXGroup; 573 | children = ( 574 | 832341B51AAA6A8300B99B32 /* libRCTText.a */, 575 | 3DAD3E941DF850E9000B6D8A /* libRCTText-tvOS.a */, 576 | ); 577 | name = Products; 578 | sourceTree = ""; 579 | }; 580 | 83CBB9F61A601CBA00E9B192 = { 581 | isa = PBXGroup; 582 | children = ( 583 | 13B07FAE1A68108700A75B9A /* ModalDemo */, 584 | 832341AE1AAA6A7D00B99B32 /* Libraries */, 585 | 00E356EF1AD99517003FC87E /* ModalDemoTests */, 586 | 83CBBA001A601CBA00E9B192 /* Products */, 587 | 2D16E6871FA4F8E400B85C8A /* Frameworks */, 588 | ); 589 | indentWidth = 2; 590 | sourceTree = ""; 591 | tabWidth = 2; 592 | usesTabs = 0; 593 | }; 594 | 83CBBA001A601CBA00E9B192 /* Products */ = { 595 | isa = PBXGroup; 596 | children = ( 597 | 13B07F961A680F5B00A75B9A /* ModalDemo.app */, 598 | 00E356EE1AD99517003FC87E /* ModalDemoTests.xctest */, 599 | 2D02E47B1E0B4A5D006451C7 /* ModalDemo-tvOS.app */, 600 | 2D02E4901E0B4A5D006451C7 /* ModalDemo-tvOSTests.xctest */, 601 | ); 602 | name = Products; 603 | sourceTree = ""; 604 | }; 605 | ADBDB9201DFEBF0600ED6528 /* Products */ = { 606 | isa = PBXGroup; 607 | children = ( 608 | ADBDB9271DFEBF0700ED6528 /* libRCTBlob.a */, 609 | 2D16E6721FA4F8DC00B85C8A /* libRCTBlob-tvOS.a */, 610 | ); 611 | name = Products; 612 | sourceTree = ""; 613 | }; 614 | /* End PBXGroup section */ 615 | 616 | /* Begin PBXNativeTarget section */ 617 | 00E356ED1AD99517003FC87E /* ModalDemoTests */ = { 618 | isa = PBXNativeTarget; 619 | buildConfigurationList = 00E357021AD99517003FC87E /* Build configuration list for PBXNativeTarget "ModalDemoTests" */; 620 | buildPhases = ( 621 | 00E356EA1AD99517003FC87E /* Sources */, 622 | 00E356EB1AD99517003FC87E /* Frameworks */, 623 | 00E356EC1AD99517003FC87E /* Resources */, 624 | ); 625 | buildRules = ( 626 | ); 627 | dependencies = ( 628 | 00E356F51AD99517003FC87E /* PBXTargetDependency */, 629 | ); 630 | name = ModalDemoTests; 631 | productName = ModalDemoTests; 632 | productReference = 00E356EE1AD99517003FC87E /* ModalDemoTests.xctest */; 633 | productType = "com.apple.product-type.bundle.unit-test"; 634 | }; 635 | 13B07F861A680F5B00A75B9A /* ModalDemo */ = { 636 | isa = PBXNativeTarget; 637 | buildConfigurationList = 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "ModalDemo" */; 638 | buildPhases = ( 639 | 13B07F871A680F5B00A75B9A /* Sources */, 640 | 13B07F8C1A680F5B00A75B9A /* Frameworks */, 641 | 13B07F8E1A680F5B00A75B9A /* Resources */, 642 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */, 643 | ); 644 | buildRules = ( 645 | ); 646 | dependencies = ( 647 | ); 648 | name = ModalDemo; 649 | productName = "Hello World"; 650 | productReference = 13B07F961A680F5B00A75B9A /* ModalDemo.app */; 651 | productType = "com.apple.product-type.application"; 652 | }; 653 | 2D02E47A1E0B4A5D006451C7 /* ModalDemo-tvOS */ = { 654 | isa = PBXNativeTarget; 655 | buildConfigurationList = 2D02E4BA1E0B4A5E006451C7 /* Build configuration list for PBXNativeTarget "ModalDemo-tvOS" */; 656 | buildPhases = ( 657 | 2D02E4771E0B4A5D006451C7 /* Sources */, 658 | 2D02E4781E0B4A5D006451C7 /* Frameworks */, 659 | 2D02E4791E0B4A5D006451C7 /* Resources */, 660 | 2D02E4CB1E0B4B27006451C7 /* Bundle React Native Code And Images */, 661 | ); 662 | buildRules = ( 663 | ); 664 | dependencies = ( 665 | ); 666 | name = "ModalDemo-tvOS"; 667 | productName = "ModalDemo-tvOS"; 668 | productReference = 2D02E47B1E0B4A5D006451C7 /* ModalDemo-tvOS.app */; 669 | productType = "com.apple.product-type.application"; 670 | }; 671 | 2D02E48F1E0B4A5D006451C7 /* ModalDemo-tvOSTests */ = { 672 | isa = PBXNativeTarget; 673 | buildConfigurationList = 2D02E4BB1E0B4A5E006451C7 /* Build configuration list for PBXNativeTarget "ModalDemo-tvOSTests" */; 674 | buildPhases = ( 675 | 2D02E48C1E0B4A5D006451C7 /* Sources */, 676 | 2D02E48D1E0B4A5D006451C7 /* Frameworks */, 677 | 2D02E48E1E0B4A5D006451C7 /* Resources */, 678 | ); 679 | buildRules = ( 680 | ); 681 | dependencies = ( 682 | 2D02E4921E0B4A5D006451C7 /* PBXTargetDependency */, 683 | ); 684 | name = "ModalDemo-tvOSTests"; 685 | productName = "ModalDemo-tvOSTests"; 686 | productReference = 2D02E4901E0B4A5D006451C7 /* ModalDemo-tvOSTests.xctest */; 687 | productType = "com.apple.product-type.bundle.unit-test"; 688 | }; 689 | /* End PBXNativeTarget section */ 690 | 691 | /* Begin PBXProject section */ 692 | 83CBB9F71A601CBA00E9B192 /* Project object */ = { 693 | isa = PBXProject; 694 | attributes = { 695 | LastUpgradeCheck = 0940; 696 | ORGANIZATIONNAME = Facebook; 697 | TargetAttributes = { 698 | 00E356ED1AD99517003FC87E = { 699 | CreatedOnToolsVersion = 6.2; 700 | TestTargetID = 13B07F861A680F5B00A75B9A; 701 | }; 702 | 2D02E47A1E0B4A5D006451C7 = { 703 | CreatedOnToolsVersion = 8.2.1; 704 | ProvisioningStyle = Automatic; 705 | }; 706 | 2D02E48F1E0B4A5D006451C7 = { 707 | CreatedOnToolsVersion = 8.2.1; 708 | ProvisioningStyle = Automatic; 709 | TestTargetID = 2D02E47A1E0B4A5D006451C7; 710 | }; 711 | }; 712 | }; 713 | buildConfigurationList = 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "ModalDemo" */; 714 | compatibilityVersion = "Xcode 3.2"; 715 | developmentRegion = English; 716 | hasScannedForEncodings = 0; 717 | knownRegions = ( 718 | en, 719 | Base, 720 | ); 721 | mainGroup = 83CBB9F61A601CBA00E9B192; 722 | productRefGroup = 83CBBA001A601CBA00E9B192 /* Products */; 723 | projectDirPath = ""; 724 | projectReferences = ( 725 | { 726 | ProductGroup = 00C302A81ABCB8CE00DB3ED1 /* Products */; 727 | ProjectRef = 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */; 728 | }, 729 | { 730 | ProductGroup = 5E91572E1DD0AC6500FF2AA8 /* Products */; 731 | ProjectRef = 5E91572D1DD0AC6500FF2AA8 /* RCTAnimation.xcodeproj */; 732 | }, 733 | { 734 | ProductGroup = ADBDB9201DFEBF0600ED6528 /* Products */; 735 | ProjectRef = ADBDB91F1DFEBF0600ED6528 /* RCTBlob.xcodeproj */; 736 | }, 737 | { 738 | ProductGroup = 00C302B61ABCB90400DB3ED1 /* Products */; 739 | ProjectRef = 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */; 740 | }, 741 | { 742 | ProductGroup = 00C302BC1ABCB91800DB3ED1 /* Products */; 743 | ProjectRef = 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */; 744 | }, 745 | { 746 | ProductGroup = 78C398B11ACF4ADC00677621 /* Products */; 747 | ProjectRef = 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */; 748 | }, 749 | { 750 | ProductGroup = 00C302D41ABCB9D200DB3ED1 /* Products */; 751 | ProjectRef = 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */; 752 | }, 753 | { 754 | ProductGroup = 139105B71AF99BAD00B5F7CC /* Products */; 755 | ProjectRef = 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */; 756 | }, 757 | { 758 | ProductGroup = 832341B11AAA6A8300B99B32 /* Products */; 759 | ProjectRef = 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */; 760 | }, 761 | { 762 | ProductGroup = 00C302E01ABCB9EE00DB3ED1 /* Products */; 763 | ProjectRef = 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */; 764 | }, 765 | { 766 | ProductGroup = 139FDEE71B06529A00C62182 /* Products */; 767 | ProjectRef = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */; 768 | }, 769 | { 770 | ProductGroup = 146834001AC3E56700842450 /* Products */; 771 | ProjectRef = 146833FF1AC3E56700842450 /* React.xcodeproj */; 772 | }, 773 | ); 774 | projectRoot = ""; 775 | targets = ( 776 | 13B07F861A680F5B00A75B9A /* ModalDemo */, 777 | 00E356ED1AD99517003FC87E /* ModalDemoTests */, 778 | 2D02E47A1E0B4A5D006451C7 /* ModalDemo-tvOS */, 779 | 2D02E48F1E0B4A5D006451C7 /* ModalDemo-tvOSTests */, 780 | ); 781 | }; 782 | /* End PBXProject section */ 783 | 784 | /* Begin PBXReferenceProxy section */ 785 | 00C302AC1ABCB8CE00DB3ED1 /* libRCTActionSheet.a */ = { 786 | isa = PBXReferenceProxy; 787 | fileType = archive.ar; 788 | path = libRCTActionSheet.a; 789 | remoteRef = 00C302AB1ABCB8CE00DB3ED1 /* PBXContainerItemProxy */; 790 | sourceTree = BUILT_PRODUCTS_DIR; 791 | }; 792 | 00C302BA1ABCB90400DB3ED1 /* libRCTGeolocation.a */ = { 793 | isa = PBXReferenceProxy; 794 | fileType = archive.ar; 795 | path = libRCTGeolocation.a; 796 | remoteRef = 00C302B91ABCB90400DB3ED1 /* PBXContainerItemProxy */; 797 | sourceTree = BUILT_PRODUCTS_DIR; 798 | }; 799 | 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */ = { 800 | isa = PBXReferenceProxy; 801 | fileType = archive.ar; 802 | path = libRCTImage.a; 803 | remoteRef = 00C302BF1ABCB91800DB3ED1 /* PBXContainerItemProxy */; 804 | sourceTree = BUILT_PRODUCTS_DIR; 805 | }; 806 | 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */ = { 807 | isa = PBXReferenceProxy; 808 | fileType = archive.ar; 809 | path = libRCTNetwork.a; 810 | remoteRef = 00C302DB1ABCB9D200DB3ED1 /* PBXContainerItemProxy */; 811 | sourceTree = BUILT_PRODUCTS_DIR; 812 | }; 813 | 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */ = { 814 | isa = PBXReferenceProxy; 815 | fileType = archive.ar; 816 | path = libRCTVibration.a; 817 | remoteRef = 00C302E31ABCB9EE00DB3ED1 /* PBXContainerItemProxy */; 818 | sourceTree = BUILT_PRODUCTS_DIR; 819 | }; 820 | 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */ = { 821 | isa = PBXReferenceProxy; 822 | fileType = archive.ar; 823 | path = libRCTSettings.a; 824 | remoteRef = 139105C01AF99BAD00B5F7CC /* PBXContainerItemProxy */; 825 | sourceTree = BUILT_PRODUCTS_DIR; 826 | }; 827 | 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */ = { 828 | isa = PBXReferenceProxy; 829 | fileType = archive.ar; 830 | path = libRCTWebSocket.a; 831 | remoteRef = 139FDEF31B06529B00C62182 /* PBXContainerItemProxy */; 832 | sourceTree = BUILT_PRODUCTS_DIR; 833 | }; 834 | 146834041AC3E56700842450 /* libReact.a */ = { 835 | isa = PBXReferenceProxy; 836 | fileType = archive.ar; 837 | path = libReact.a; 838 | remoteRef = 146834031AC3E56700842450 /* PBXContainerItemProxy */; 839 | sourceTree = BUILT_PRODUCTS_DIR; 840 | }; 841 | 2D16E6721FA4F8DC00B85C8A /* libRCTBlob-tvOS.a */ = { 842 | isa = PBXReferenceProxy; 843 | fileType = archive.ar; 844 | path = "libRCTBlob-tvOS.a"; 845 | remoteRef = 2D16E6711FA4F8DC00B85C8A /* PBXContainerItemProxy */; 846 | sourceTree = BUILT_PRODUCTS_DIR; 847 | }; 848 | 2D16E6841FA4F8DC00B85C8A /* libfishhook.a */ = { 849 | isa = PBXReferenceProxy; 850 | fileType = archive.ar; 851 | path = libfishhook.a; 852 | remoteRef = 2D16E6831FA4F8DC00B85C8A /* PBXContainerItemProxy */; 853 | sourceTree = BUILT_PRODUCTS_DIR; 854 | }; 855 | 2D16E6861FA4F8DC00B85C8A /* libfishhook-tvOS.a */ = { 856 | isa = PBXReferenceProxy; 857 | fileType = archive.ar; 858 | path = "libfishhook-tvOS.a"; 859 | remoteRef = 2D16E6851FA4F8DC00B85C8A /* PBXContainerItemProxy */; 860 | sourceTree = BUILT_PRODUCTS_DIR; 861 | }; 862 | 2DF0FFDF2056DD460020B375 /* libjsinspector.a */ = { 863 | isa = PBXReferenceProxy; 864 | fileType = archive.ar; 865 | path = libjsinspector.a; 866 | remoteRef = 2DF0FFDE2056DD460020B375 /* PBXContainerItemProxy */; 867 | sourceTree = BUILT_PRODUCTS_DIR; 868 | }; 869 | 2DF0FFE12056DD460020B375 /* libjsinspector-tvOS.a */ = { 870 | isa = PBXReferenceProxy; 871 | fileType = archive.ar; 872 | path = "libjsinspector-tvOS.a"; 873 | remoteRef = 2DF0FFE02056DD460020B375 /* PBXContainerItemProxy */; 874 | sourceTree = BUILT_PRODUCTS_DIR; 875 | }; 876 | 2DF0FFE32056DD460020B375 /* libthird-party.a */ = { 877 | isa = PBXReferenceProxy; 878 | fileType = archive.ar; 879 | path = "libthird-party.a"; 880 | remoteRef = 2DF0FFE22056DD460020B375 /* PBXContainerItemProxy */; 881 | sourceTree = BUILT_PRODUCTS_DIR; 882 | }; 883 | 2DF0FFE52056DD460020B375 /* libthird-party.a */ = { 884 | isa = PBXReferenceProxy; 885 | fileType = archive.ar; 886 | path = "libthird-party.a"; 887 | remoteRef = 2DF0FFE42056DD460020B375 /* PBXContainerItemProxy */; 888 | sourceTree = BUILT_PRODUCTS_DIR; 889 | }; 890 | 2DF0FFE72056DD460020B375 /* libdouble-conversion.a */ = { 891 | isa = PBXReferenceProxy; 892 | fileType = archive.ar; 893 | path = "libdouble-conversion.a"; 894 | remoteRef = 2DF0FFE62056DD460020B375 /* PBXContainerItemProxy */; 895 | sourceTree = BUILT_PRODUCTS_DIR; 896 | }; 897 | 2DF0FFE92056DD460020B375 /* libdouble-conversion.a */ = { 898 | isa = PBXReferenceProxy; 899 | fileType = archive.ar; 900 | path = "libdouble-conversion.a"; 901 | remoteRef = 2DF0FFE82056DD460020B375 /* PBXContainerItemProxy */; 902 | sourceTree = BUILT_PRODUCTS_DIR; 903 | }; 904 | 2DF0FFEB2056DD460020B375 /* libprivatedata.a */ = { 905 | isa = PBXReferenceProxy; 906 | fileType = archive.ar; 907 | path = libprivatedata.a; 908 | remoteRef = 2DF0FFEA2056DD460020B375 /* PBXContainerItemProxy */; 909 | sourceTree = BUILT_PRODUCTS_DIR; 910 | }; 911 | 2DF0FFED2056DD460020B375 /* libprivatedata-tvOS.a */ = { 912 | isa = PBXReferenceProxy; 913 | fileType = archive.ar; 914 | path = "libprivatedata-tvOS.a"; 915 | remoteRef = 2DF0FFEC2056DD460020B375 /* PBXContainerItemProxy */; 916 | sourceTree = BUILT_PRODUCTS_DIR; 917 | }; 918 | 3DAD3E841DF850E9000B6D8A /* libRCTImage-tvOS.a */ = { 919 | isa = PBXReferenceProxy; 920 | fileType = archive.ar; 921 | path = "libRCTImage-tvOS.a"; 922 | remoteRef = 3DAD3E831DF850E9000B6D8A /* PBXContainerItemProxy */; 923 | sourceTree = BUILT_PRODUCTS_DIR; 924 | }; 925 | 3DAD3E881DF850E9000B6D8A /* libRCTLinking-tvOS.a */ = { 926 | isa = PBXReferenceProxy; 927 | fileType = archive.ar; 928 | path = "libRCTLinking-tvOS.a"; 929 | remoteRef = 3DAD3E871DF850E9000B6D8A /* PBXContainerItemProxy */; 930 | sourceTree = BUILT_PRODUCTS_DIR; 931 | }; 932 | 3DAD3E8C1DF850E9000B6D8A /* libRCTNetwork-tvOS.a */ = { 933 | isa = PBXReferenceProxy; 934 | fileType = archive.ar; 935 | path = "libRCTNetwork-tvOS.a"; 936 | remoteRef = 3DAD3E8B1DF850E9000B6D8A /* PBXContainerItemProxy */; 937 | sourceTree = BUILT_PRODUCTS_DIR; 938 | }; 939 | 3DAD3E901DF850E9000B6D8A /* libRCTSettings-tvOS.a */ = { 940 | isa = PBXReferenceProxy; 941 | fileType = archive.ar; 942 | path = "libRCTSettings-tvOS.a"; 943 | remoteRef = 3DAD3E8F1DF850E9000B6D8A /* PBXContainerItemProxy */; 944 | sourceTree = BUILT_PRODUCTS_DIR; 945 | }; 946 | 3DAD3E941DF850E9000B6D8A /* libRCTText-tvOS.a */ = { 947 | isa = PBXReferenceProxy; 948 | fileType = archive.ar; 949 | path = "libRCTText-tvOS.a"; 950 | remoteRef = 3DAD3E931DF850E9000B6D8A /* PBXContainerItemProxy */; 951 | sourceTree = BUILT_PRODUCTS_DIR; 952 | }; 953 | 3DAD3E991DF850E9000B6D8A /* libRCTWebSocket-tvOS.a */ = { 954 | isa = PBXReferenceProxy; 955 | fileType = archive.ar; 956 | path = "libRCTWebSocket-tvOS.a"; 957 | remoteRef = 3DAD3E981DF850E9000B6D8A /* PBXContainerItemProxy */; 958 | sourceTree = BUILT_PRODUCTS_DIR; 959 | }; 960 | 3DAD3EA31DF850E9000B6D8A /* libReact.a */ = { 961 | isa = PBXReferenceProxy; 962 | fileType = archive.ar; 963 | path = libReact.a; 964 | remoteRef = 3DAD3EA21DF850E9000B6D8A /* PBXContainerItemProxy */; 965 | sourceTree = BUILT_PRODUCTS_DIR; 966 | }; 967 | 3DAD3EA51DF850E9000B6D8A /* libyoga.a */ = { 968 | isa = PBXReferenceProxy; 969 | fileType = archive.ar; 970 | path = libyoga.a; 971 | remoteRef = 3DAD3EA41DF850E9000B6D8A /* PBXContainerItemProxy */; 972 | sourceTree = BUILT_PRODUCTS_DIR; 973 | }; 974 | 3DAD3EA71DF850E9000B6D8A /* libyoga.a */ = { 975 | isa = PBXReferenceProxy; 976 | fileType = archive.ar; 977 | path = libyoga.a; 978 | remoteRef = 3DAD3EA61DF850E9000B6D8A /* PBXContainerItemProxy */; 979 | sourceTree = BUILT_PRODUCTS_DIR; 980 | }; 981 | 3DAD3EA91DF850E9000B6D8A /* libcxxreact.a */ = { 982 | isa = PBXReferenceProxy; 983 | fileType = archive.ar; 984 | path = libcxxreact.a; 985 | remoteRef = 3DAD3EA81DF850E9000B6D8A /* PBXContainerItemProxy */; 986 | sourceTree = BUILT_PRODUCTS_DIR; 987 | }; 988 | 3DAD3EAB1DF850E9000B6D8A /* libcxxreact.a */ = { 989 | isa = PBXReferenceProxy; 990 | fileType = archive.ar; 991 | path = libcxxreact.a; 992 | remoteRef = 3DAD3EAA1DF850E9000B6D8A /* PBXContainerItemProxy */; 993 | sourceTree = BUILT_PRODUCTS_DIR; 994 | }; 995 | 3DAD3EAD1DF850E9000B6D8A /* libjschelpers.a */ = { 996 | isa = PBXReferenceProxy; 997 | fileType = archive.ar; 998 | path = libjschelpers.a; 999 | remoteRef = 3DAD3EAC1DF850E9000B6D8A /* PBXContainerItemProxy */; 1000 | sourceTree = BUILT_PRODUCTS_DIR; 1001 | }; 1002 | 3DAD3EAF1DF850E9000B6D8A /* libjschelpers.a */ = { 1003 | isa = PBXReferenceProxy; 1004 | fileType = archive.ar; 1005 | path = libjschelpers.a; 1006 | remoteRef = 3DAD3EAE1DF850E9000B6D8A /* PBXContainerItemProxy */; 1007 | sourceTree = BUILT_PRODUCTS_DIR; 1008 | }; 1009 | 5E9157331DD0AC6500FF2AA8 /* libRCTAnimation.a */ = { 1010 | isa = PBXReferenceProxy; 1011 | fileType = archive.ar; 1012 | path = libRCTAnimation.a; 1013 | remoteRef = 5E9157321DD0AC6500FF2AA8 /* PBXContainerItemProxy */; 1014 | sourceTree = BUILT_PRODUCTS_DIR; 1015 | }; 1016 | 5E9157351DD0AC6500FF2AA8 /* libRCTAnimation.a */ = { 1017 | isa = PBXReferenceProxy; 1018 | fileType = archive.ar; 1019 | path = libRCTAnimation.a; 1020 | remoteRef = 5E9157341DD0AC6500FF2AA8 /* PBXContainerItemProxy */; 1021 | sourceTree = BUILT_PRODUCTS_DIR; 1022 | }; 1023 | 78C398B91ACF4ADC00677621 /* libRCTLinking.a */ = { 1024 | isa = PBXReferenceProxy; 1025 | fileType = archive.ar; 1026 | path = libRCTLinking.a; 1027 | remoteRef = 78C398B81ACF4ADC00677621 /* PBXContainerItemProxy */; 1028 | sourceTree = BUILT_PRODUCTS_DIR; 1029 | }; 1030 | 832341B51AAA6A8300B99B32 /* libRCTText.a */ = { 1031 | isa = PBXReferenceProxy; 1032 | fileType = archive.ar; 1033 | path = libRCTText.a; 1034 | remoteRef = 832341B41AAA6A8300B99B32 /* PBXContainerItemProxy */; 1035 | sourceTree = BUILT_PRODUCTS_DIR; 1036 | }; 1037 | ADBDB9271DFEBF0700ED6528 /* libRCTBlob.a */ = { 1038 | isa = PBXReferenceProxy; 1039 | fileType = archive.ar; 1040 | path = libRCTBlob.a; 1041 | remoteRef = ADBDB9261DFEBF0700ED6528 /* PBXContainerItemProxy */; 1042 | sourceTree = BUILT_PRODUCTS_DIR; 1043 | }; 1044 | /* End PBXReferenceProxy section */ 1045 | 1046 | /* Begin PBXResourcesBuildPhase section */ 1047 | 00E356EC1AD99517003FC87E /* Resources */ = { 1048 | isa = PBXResourcesBuildPhase; 1049 | buildActionMask = 2147483647; 1050 | files = ( 1051 | ); 1052 | runOnlyForDeploymentPostprocessing = 0; 1053 | }; 1054 | 13B07F8E1A680F5B00A75B9A /* Resources */ = { 1055 | isa = PBXResourcesBuildPhase; 1056 | buildActionMask = 2147483647; 1057 | files = ( 1058 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */, 1059 | 13B07FBD1A68108700A75B9A /* LaunchScreen.xib in Resources */, 1060 | ); 1061 | runOnlyForDeploymentPostprocessing = 0; 1062 | }; 1063 | 2D02E4791E0B4A5D006451C7 /* Resources */ = { 1064 | isa = PBXResourcesBuildPhase; 1065 | buildActionMask = 2147483647; 1066 | files = ( 1067 | 2D02E4BD1E0B4A84006451C7 /* Images.xcassets in Resources */, 1068 | ); 1069 | runOnlyForDeploymentPostprocessing = 0; 1070 | }; 1071 | 2D02E48E1E0B4A5D006451C7 /* Resources */ = { 1072 | isa = PBXResourcesBuildPhase; 1073 | buildActionMask = 2147483647; 1074 | files = ( 1075 | ); 1076 | runOnlyForDeploymentPostprocessing = 0; 1077 | }; 1078 | /* End PBXResourcesBuildPhase section */ 1079 | 1080 | /* Begin PBXShellScriptBuildPhase section */ 1081 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */ = { 1082 | isa = PBXShellScriptBuildPhase; 1083 | buildActionMask = 2147483647; 1084 | files = ( 1085 | ); 1086 | inputPaths = ( 1087 | ); 1088 | name = "Bundle React Native code and images"; 1089 | outputPaths = ( 1090 | ); 1091 | runOnlyForDeploymentPostprocessing = 0; 1092 | shellPath = /bin/sh; 1093 | shellScript = "export NODE_BINARY=node\n../node_modules/react-native/scripts/react-native-xcode.sh"; 1094 | }; 1095 | 2D02E4CB1E0B4B27006451C7 /* Bundle React Native Code And Images */ = { 1096 | isa = PBXShellScriptBuildPhase; 1097 | buildActionMask = 2147483647; 1098 | files = ( 1099 | ); 1100 | inputPaths = ( 1101 | ); 1102 | name = "Bundle React Native Code And Images"; 1103 | outputPaths = ( 1104 | ); 1105 | runOnlyForDeploymentPostprocessing = 0; 1106 | shellPath = /bin/sh; 1107 | shellScript = "export NODE_BINARY=node\n../node_modules/react-native/scripts/react-native-xcode.sh"; 1108 | }; 1109 | /* End PBXShellScriptBuildPhase section */ 1110 | 1111 | /* Begin PBXSourcesBuildPhase section */ 1112 | 00E356EA1AD99517003FC87E /* Sources */ = { 1113 | isa = PBXSourcesBuildPhase; 1114 | buildActionMask = 2147483647; 1115 | files = ( 1116 | 00E356F31AD99517003FC87E /* ModalDemoTests.m in Sources */, 1117 | ); 1118 | runOnlyForDeploymentPostprocessing = 0; 1119 | }; 1120 | 13B07F871A680F5B00A75B9A /* Sources */ = { 1121 | isa = PBXSourcesBuildPhase; 1122 | buildActionMask = 2147483647; 1123 | files = ( 1124 | 13B07FBC1A68108700A75B9A /* AppDelegate.m in Sources */, 1125 | 13B07FC11A68108700A75B9A /* main.m in Sources */, 1126 | ); 1127 | runOnlyForDeploymentPostprocessing = 0; 1128 | }; 1129 | 2D02E4771E0B4A5D006451C7 /* Sources */ = { 1130 | isa = PBXSourcesBuildPhase; 1131 | buildActionMask = 2147483647; 1132 | files = ( 1133 | 2D02E4BF1E0B4AB3006451C7 /* main.m in Sources */, 1134 | 2D02E4BC1E0B4A80006451C7 /* AppDelegate.m in Sources */, 1135 | ); 1136 | runOnlyForDeploymentPostprocessing = 0; 1137 | }; 1138 | 2D02E48C1E0B4A5D006451C7 /* Sources */ = { 1139 | isa = PBXSourcesBuildPhase; 1140 | buildActionMask = 2147483647; 1141 | files = ( 1142 | 2DCD954D1E0B4F2C00145EB5 /* ModalDemoTests.m in Sources */, 1143 | ); 1144 | runOnlyForDeploymentPostprocessing = 0; 1145 | }; 1146 | /* End PBXSourcesBuildPhase section */ 1147 | 1148 | /* Begin PBXTargetDependency section */ 1149 | 00E356F51AD99517003FC87E /* PBXTargetDependency */ = { 1150 | isa = PBXTargetDependency; 1151 | target = 13B07F861A680F5B00A75B9A /* ModalDemo */; 1152 | targetProxy = 00E356F41AD99517003FC87E /* PBXContainerItemProxy */; 1153 | }; 1154 | 2D02E4921E0B4A5D006451C7 /* PBXTargetDependency */ = { 1155 | isa = PBXTargetDependency; 1156 | target = 2D02E47A1E0B4A5D006451C7 /* ModalDemo-tvOS */; 1157 | targetProxy = 2D02E4911E0B4A5D006451C7 /* PBXContainerItemProxy */; 1158 | }; 1159 | /* End PBXTargetDependency section */ 1160 | 1161 | /* Begin PBXVariantGroup section */ 1162 | 13B07FB11A68108700A75B9A /* LaunchScreen.xib */ = { 1163 | isa = PBXVariantGroup; 1164 | children = ( 1165 | 13B07FB21A68108700A75B9A /* Base */, 1166 | ); 1167 | name = LaunchScreen.xib; 1168 | path = ModalDemo; 1169 | sourceTree = ""; 1170 | }; 1171 | /* End PBXVariantGroup section */ 1172 | 1173 | /* Begin XCBuildConfiguration section */ 1174 | 00E356F61AD99517003FC87E /* Debug */ = { 1175 | isa = XCBuildConfiguration; 1176 | buildSettings = { 1177 | BUNDLE_LOADER = "$(TEST_HOST)"; 1178 | GCC_PREPROCESSOR_DEFINITIONS = ( 1179 | "DEBUG=1", 1180 | "$(inherited)", 1181 | ); 1182 | INFOPLIST_FILE = ModalDemoTests/Info.plist; 1183 | IPHONEOS_DEPLOYMENT_TARGET = 9.0; 1184 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks"; 1185 | OTHER_LDFLAGS = ( 1186 | "-ObjC", 1187 | "-lc++", 1188 | ); 1189 | PRODUCT_BUNDLE_IDENTIFIER = "org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)"; 1190 | PRODUCT_NAME = "$(TARGET_NAME)"; 1191 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/ModalDemo.app/ModalDemo"; 1192 | }; 1193 | name = Debug; 1194 | }; 1195 | 00E356F71AD99517003FC87E /* Release */ = { 1196 | isa = XCBuildConfiguration; 1197 | buildSettings = { 1198 | BUNDLE_LOADER = "$(TEST_HOST)"; 1199 | COPY_PHASE_STRIP = NO; 1200 | INFOPLIST_FILE = ModalDemoTests/Info.plist; 1201 | IPHONEOS_DEPLOYMENT_TARGET = 9.0; 1202 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks"; 1203 | OTHER_LDFLAGS = ( 1204 | "-ObjC", 1205 | "-lc++", 1206 | ); 1207 | PRODUCT_BUNDLE_IDENTIFIER = "org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)"; 1208 | PRODUCT_NAME = "$(TARGET_NAME)"; 1209 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/ModalDemo.app/ModalDemo"; 1210 | }; 1211 | name = Release; 1212 | }; 1213 | 13B07F941A680F5B00A75B9A /* Debug */ = { 1214 | isa = XCBuildConfiguration; 1215 | buildSettings = { 1216 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 1217 | CURRENT_PROJECT_VERSION = 1; 1218 | DEAD_CODE_STRIPPING = NO; 1219 | INFOPLIST_FILE = ModalDemo/Info.plist; 1220 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 1221 | OTHER_LDFLAGS = ( 1222 | "$(inherited)", 1223 | "-ObjC", 1224 | "-lc++", 1225 | ); 1226 | PRODUCT_BUNDLE_IDENTIFIER = "org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)"; 1227 | PRODUCT_NAME = ModalDemo; 1228 | VERSIONING_SYSTEM = "apple-generic"; 1229 | }; 1230 | name = Debug; 1231 | }; 1232 | 13B07F951A680F5B00A75B9A /* Release */ = { 1233 | isa = XCBuildConfiguration; 1234 | buildSettings = { 1235 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 1236 | CURRENT_PROJECT_VERSION = 1; 1237 | INFOPLIST_FILE = ModalDemo/Info.plist; 1238 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 1239 | OTHER_LDFLAGS = ( 1240 | "$(inherited)", 1241 | "-ObjC", 1242 | "-lc++", 1243 | ); 1244 | PRODUCT_BUNDLE_IDENTIFIER = "org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)"; 1245 | PRODUCT_NAME = ModalDemo; 1246 | VERSIONING_SYSTEM = "apple-generic"; 1247 | }; 1248 | name = Release; 1249 | }; 1250 | 2D02E4971E0B4A5E006451C7 /* Debug */ = { 1251 | isa = XCBuildConfiguration; 1252 | buildSettings = { 1253 | ASSETCATALOG_COMPILER_APPICON_NAME = "App Icon & Top Shelf Image"; 1254 | ASSETCATALOG_COMPILER_LAUNCHIMAGE_NAME = LaunchImage; 1255 | CLANG_ANALYZER_NONNULL = YES; 1256 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 1257 | CLANG_WARN_INFINITE_RECURSION = YES; 1258 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1259 | DEBUG_INFORMATION_FORMAT = dwarf; 1260 | ENABLE_TESTABILITY = YES; 1261 | GCC_NO_COMMON_BLOCKS = YES; 1262 | INFOPLIST_FILE = "ModalDemo-tvOS/Info.plist"; 1263 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 1264 | OTHER_LDFLAGS = ( 1265 | "-ObjC", 1266 | "-lc++", 1267 | ); 1268 | PRODUCT_BUNDLE_IDENTIFIER = "com.facebook.REACT.ModalDemo-tvOS"; 1269 | PRODUCT_NAME = "$(TARGET_NAME)"; 1270 | SDKROOT = appletvos; 1271 | TARGETED_DEVICE_FAMILY = 3; 1272 | TVOS_DEPLOYMENT_TARGET = 9.2; 1273 | }; 1274 | name = Debug; 1275 | }; 1276 | 2D02E4981E0B4A5E006451C7 /* Release */ = { 1277 | isa = XCBuildConfiguration; 1278 | buildSettings = { 1279 | ASSETCATALOG_COMPILER_APPICON_NAME = "App Icon & Top Shelf Image"; 1280 | ASSETCATALOG_COMPILER_LAUNCHIMAGE_NAME = LaunchImage; 1281 | CLANG_ANALYZER_NONNULL = YES; 1282 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 1283 | CLANG_WARN_INFINITE_RECURSION = YES; 1284 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1285 | COPY_PHASE_STRIP = NO; 1286 | DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; 1287 | GCC_NO_COMMON_BLOCKS = YES; 1288 | INFOPLIST_FILE = "ModalDemo-tvOS/Info.plist"; 1289 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 1290 | OTHER_LDFLAGS = ( 1291 | "-ObjC", 1292 | "-lc++", 1293 | ); 1294 | PRODUCT_BUNDLE_IDENTIFIER = "com.facebook.REACT.ModalDemo-tvOS"; 1295 | PRODUCT_NAME = "$(TARGET_NAME)"; 1296 | SDKROOT = appletvos; 1297 | TARGETED_DEVICE_FAMILY = 3; 1298 | TVOS_DEPLOYMENT_TARGET = 9.2; 1299 | }; 1300 | name = Release; 1301 | }; 1302 | 2D02E4991E0B4A5E006451C7 /* Debug */ = { 1303 | isa = XCBuildConfiguration; 1304 | buildSettings = { 1305 | BUNDLE_LOADER = "$(TEST_HOST)"; 1306 | CLANG_ANALYZER_NONNULL = YES; 1307 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 1308 | CLANG_WARN_INFINITE_RECURSION = YES; 1309 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1310 | DEBUG_INFORMATION_FORMAT = dwarf; 1311 | ENABLE_TESTABILITY = YES; 1312 | GCC_NO_COMMON_BLOCKS = YES; 1313 | INFOPLIST_FILE = "ModalDemo-tvOSTests/Info.plist"; 1314 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks"; 1315 | OTHER_LDFLAGS = ( 1316 | "-ObjC", 1317 | "-lc++", 1318 | ); 1319 | PRODUCT_BUNDLE_IDENTIFIER = "com.facebook.REACT.ModalDemo-tvOSTests"; 1320 | PRODUCT_NAME = "$(TARGET_NAME)"; 1321 | SDKROOT = appletvos; 1322 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/ModalDemo-tvOS.app/ModalDemo-tvOS"; 1323 | TVOS_DEPLOYMENT_TARGET = 10.1; 1324 | }; 1325 | name = Debug; 1326 | }; 1327 | 2D02E49A1E0B4A5E006451C7 /* Release */ = { 1328 | isa = XCBuildConfiguration; 1329 | buildSettings = { 1330 | BUNDLE_LOADER = "$(TEST_HOST)"; 1331 | CLANG_ANALYZER_NONNULL = YES; 1332 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 1333 | CLANG_WARN_INFINITE_RECURSION = YES; 1334 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1335 | COPY_PHASE_STRIP = NO; 1336 | DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; 1337 | GCC_NO_COMMON_BLOCKS = YES; 1338 | INFOPLIST_FILE = "ModalDemo-tvOSTests/Info.plist"; 1339 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks"; 1340 | OTHER_LDFLAGS = ( 1341 | "-ObjC", 1342 | "-lc++", 1343 | ); 1344 | PRODUCT_BUNDLE_IDENTIFIER = "com.facebook.REACT.ModalDemo-tvOSTests"; 1345 | PRODUCT_NAME = "$(TARGET_NAME)"; 1346 | SDKROOT = appletvos; 1347 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/ModalDemo-tvOS.app/ModalDemo-tvOS"; 1348 | TVOS_DEPLOYMENT_TARGET = 10.1; 1349 | }; 1350 | name = Release; 1351 | }; 1352 | 83CBBA201A601CBA00E9B192 /* Debug */ = { 1353 | isa = XCBuildConfiguration; 1354 | buildSettings = { 1355 | ALWAYS_SEARCH_USER_PATHS = NO; 1356 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; 1357 | CLANG_CXX_LIBRARY = "libc++"; 1358 | CLANG_ENABLE_MODULES = YES; 1359 | CLANG_ENABLE_OBJC_ARC = YES; 1360 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; 1361 | CLANG_WARN_BOOL_CONVERSION = YES; 1362 | CLANG_WARN_COMMA = YES; 1363 | CLANG_WARN_CONSTANT_CONVERSION = YES; 1364 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; 1365 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 1366 | CLANG_WARN_EMPTY_BODY = YES; 1367 | CLANG_WARN_ENUM_CONVERSION = YES; 1368 | CLANG_WARN_INFINITE_RECURSION = YES; 1369 | CLANG_WARN_INT_CONVERSION = YES; 1370 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; 1371 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; 1372 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; 1373 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 1374 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; 1375 | CLANG_WARN_STRICT_PROTOTYPES = YES; 1376 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1377 | CLANG_WARN_UNREACHABLE_CODE = YES; 1378 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 1379 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; 1380 | COPY_PHASE_STRIP = NO; 1381 | ENABLE_STRICT_OBJC_MSGSEND = YES; 1382 | ENABLE_TESTABILITY = YES; 1383 | GCC_C_LANGUAGE_STANDARD = gnu99; 1384 | GCC_DYNAMIC_NO_PIC = NO; 1385 | GCC_NO_COMMON_BLOCKS = YES; 1386 | GCC_OPTIMIZATION_LEVEL = 0; 1387 | GCC_PREPROCESSOR_DEFINITIONS = ( 1388 | "DEBUG=1", 1389 | "$(inherited)", 1390 | ); 1391 | GCC_SYMBOLS_PRIVATE_EXTERN = NO; 1392 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 1393 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 1394 | GCC_WARN_UNDECLARED_SELECTOR = YES; 1395 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 1396 | GCC_WARN_UNUSED_FUNCTION = YES; 1397 | GCC_WARN_UNUSED_VARIABLE = YES; 1398 | IPHONEOS_DEPLOYMENT_TARGET = 9.0; 1399 | MTL_ENABLE_DEBUG_INFO = YES; 1400 | ONLY_ACTIVE_ARCH = YES; 1401 | SDKROOT = iphoneos; 1402 | }; 1403 | name = Debug; 1404 | }; 1405 | 83CBBA211A601CBA00E9B192 /* Release */ = { 1406 | isa = XCBuildConfiguration; 1407 | buildSettings = { 1408 | ALWAYS_SEARCH_USER_PATHS = NO; 1409 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; 1410 | CLANG_CXX_LIBRARY = "libc++"; 1411 | CLANG_ENABLE_MODULES = YES; 1412 | CLANG_ENABLE_OBJC_ARC = YES; 1413 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; 1414 | CLANG_WARN_BOOL_CONVERSION = YES; 1415 | CLANG_WARN_COMMA = YES; 1416 | CLANG_WARN_CONSTANT_CONVERSION = YES; 1417 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; 1418 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 1419 | CLANG_WARN_EMPTY_BODY = YES; 1420 | CLANG_WARN_ENUM_CONVERSION = YES; 1421 | CLANG_WARN_INFINITE_RECURSION = YES; 1422 | CLANG_WARN_INT_CONVERSION = YES; 1423 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; 1424 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; 1425 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; 1426 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 1427 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; 1428 | CLANG_WARN_STRICT_PROTOTYPES = YES; 1429 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 1430 | CLANG_WARN_UNREACHABLE_CODE = YES; 1431 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 1432 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; 1433 | COPY_PHASE_STRIP = YES; 1434 | ENABLE_NS_ASSERTIONS = NO; 1435 | ENABLE_STRICT_OBJC_MSGSEND = YES; 1436 | GCC_C_LANGUAGE_STANDARD = gnu99; 1437 | GCC_NO_COMMON_BLOCKS = YES; 1438 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 1439 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 1440 | GCC_WARN_UNDECLARED_SELECTOR = YES; 1441 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 1442 | GCC_WARN_UNUSED_FUNCTION = YES; 1443 | GCC_WARN_UNUSED_VARIABLE = YES; 1444 | IPHONEOS_DEPLOYMENT_TARGET = 9.0; 1445 | MTL_ENABLE_DEBUG_INFO = NO; 1446 | SDKROOT = iphoneos; 1447 | VALIDATE_PRODUCT = YES; 1448 | }; 1449 | name = Release; 1450 | }; 1451 | /* End XCBuildConfiguration section */ 1452 | 1453 | /* Begin XCConfigurationList section */ 1454 | 00E357021AD99517003FC87E /* Build configuration list for PBXNativeTarget "ModalDemoTests" */ = { 1455 | isa = XCConfigurationList; 1456 | buildConfigurations = ( 1457 | 00E356F61AD99517003FC87E /* Debug */, 1458 | 00E356F71AD99517003FC87E /* Release */, 1459 | ); 1460 | defaultConfigurationIsVisible = 0; 1461 | defaultConfigurationName = Release; 1462 | }; 1463 | 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "ModalDemo" */ = { 1464 | isa = XCConfigurationList; 1465 | buildConfigurations = ( 1466 | 13B07F941A680F5B00A75B9A /* Debug */, 1467 | 13B07F951A680F5B00A75B9A /* Release */, 1468 | ); 1469 | defaultConfigurationIsVisible = 0; 1470 | defaultConfigurationName = Release; 1471 | }; 1472 | 2D02E4BA1E0B4A5E006451C7 /* Build configuration list for PBXNativeTarget "ModalDemo-tvOS" */ = { 1473 | isa = XCConfigurationList; 1474 | buildConfigurations = ( 1475 | 2D02E4971E0B4A5E006451C7 /* Debug */, 1476 | 2D02E4981E0B4A5E006451C7 /* Release */, 1477 | ); 1478 | defaultConfigurationIsVisible = 0; 1479 | defaultConfigurationName = Release; 1480 | }; 1481 | 2D02E4BB1E0B4A5E006451C7 /* Build configuration list for PBXNativeTarget "ModalDemo-tvOSTests" */ = { 1482 | isa = XCConfigurationList; 1483 | buildConfigurations = ( 1484 | 2D02E4991E0B4A5E006451C7 /* Debug */, 1485 | 2D02E49A1E0B4A5E006451C7 /* Release */, 1486 | ); 1487 | defaultConfigurationIsVisible = 0; 1488 | defaultConfigurationName = Release; 1489 | }; 1490 | 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "ModalDemo" */ = { 1491 | isa = XCConfigurationList; 1492 | buildConfigurations = ( 1493 | 83CBBA201A601CBA00E9B192 /* Debug */, 1494 | 83CBBA211A601CBA00E9B192 /* Release */, 1495 | ); 1496 | defaultConfigurationIsVisible = 0; 1497 | defaultConfigurationName = Release; 1498 | }; 1499 | /* End XCConfigurationList section */ 1500 | }; 1501 | rootObject = 83CBB9F71A601CBA00E9B192 /* Project object */; 1502 | } 1503 | -------------------------------------------------------------------------------- /ios/ModalDemo.xcodeproj/xcshareddata/xcschemes/ModalDemo-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 | -------------------------------------------------------------------------------- /ios/ModalDemo.xcodeproj/xcshareddata/xcschemes/ModalDemo.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 | -------------------------------------------------------------------------------- /ios/ModalDemo/AppDelegate.h: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) Facebook, Inc. and its affiliates. 3 | * 4 | * This source code is licensed under the MIT license found in the 5 | * LICENSE file in the root directory of this source tree. 6 | */ 7 | 8 | #import 9 | #import 10 | 11 | @interface AppDelegate : UIResponder 12 | 13 | @property (nonatomic, strong) UIWindow *window; 14 | 15 | @end 16 | -------------------------------------------------------------------------------- /ios/ModalDemo/AppDelegate.m: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) Facebook, Inc. and its affiliates. 3 | * 4 | * This source code is licensed under the MIT license found in the 5 | * LICENSE file in the root directory of this source tree. 6 | */ 7 | 8 | #import "AppDelegate.h" 9 | 10 | #import 11 | #import 12 | #import 13 | 14 | @implementation AppDelegate 15 | 16 | - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 17 | { 18 | RCTBridge *bridge = [[RCTBridge alloc] initWithDelegate:self launchOptions:launchOptions]; 19 | RCTRootView *rootView = [[RCTRootView alloc] initWithBridge:bridge 20 | moduleName:@"ModalDemo" 21 | initialProperties:nil]; 22 | 23 | rootView.backgroundColor = [[UIColor alloc] initWithRed:1.0f green:1.0f blue:1.0f alpha:1]; 24 | 25 | self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds]; 26 | UIViewController *rootViewController = [UIViewController new]; 27 | rootViewController.view = rootView; 28 | self.window.rootViewController = rootViewController; 29 | [self.window makeKeyAndVisible]; 30 | return YES; 31 | } 32 | 33 | - (NSURL *)sourceURLForBridge:(RCTBridge *)bridge 34 | { 35 | #if DEBUG 36 | return [[RCTBundleURLProvider sharedSettings] jsBundleURLForBundleRoot:@"index" fallbackResource:nil]; 37 | #else 38 | return [[NSBundle mainBundle] URLForResource:@"main" withExtension:@"jsbundle"]; 39 | #endif 40 | } 41 | 42 | @end 43 | -------------------------------------------------------------------------------- /ios/ModalDemo/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 | -------------------------------------------------------------------------------- /ios/ModalDemo/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 | } -------------------------------------------------------------------------------- /ios/ModalDemo/Images.xcassets/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "info" : { 3 | "version" : 1, 4 | "author" : "xcode" 5 | } 6 | } 7 | -------------------------------------------------------------------------------- /ios/ModalDemo/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleDisplayName 8 | ModalDemo 9 | CFBundleExecutable 10 | $(EXECUTABLE_NAME) 11 | CFBundleIdentifier 12 | $(PRODUCT_BUNDLE_IDENTIFIER) 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 | NSLocationWhenInUseUsageDescription 28 | 29 | UILaunchStoryboardName 30 | LaunchScreen 31 | UIRequiredDeviceCapabilities 32 | 33 | armv7 34 | 35 | UISupportedInterfaceOrientations 36 | 37 | UIInterfaceOrientationPortrait 38 | UIInterfaceOrientationLandscapeLeft 39 | UIInterfaceOrientationLandscapeRight 40 | 41 | UIViewControllerBasedStatusBarAppearance 42 | 43 | NSLocationWhenInUseUsageDescription 44 | 45 | NSAppTransportSecurity 46 | 47 | 48 | NSAllowsArbitraryLoads 49 | 50 | NSExceptionDomains 51 | 52 | localhost 53 | 54 | NSExceptionAllowsInsecureHTTPLoads 55 | 56 | 57 | 58 | 59 | 60 | 61 | -------------------------------------------------------------------------------- /ios/ModalDemo/main.m: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) Facebook, Inc. and its affiliates. 3 | * 4 | * This source code is licensed under the MIT license found in the 5 | * LICENSE file in the root directory of this source tree. 6 | */ 7 | 8 | #import 9 | 10 | #import "AppDelegate.h" 11 | 12 | int main(int argc, char * argv[]) { 13 | @autoreleasepool { 14 | return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class])); 15 | } 16 | } 17 | -------------------------------------------------------------------------------- /ios/ModalDemoTests/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | en 7 | CFBundleExecutable 8 | $(EXECUTABLE_NAME) 9 | CFBundleIdentifier 10 | $(PRODUCT_BUNDLE_IDENTIFIER) 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 | -------------------------------------------------------------------------------- /ios/ModalDemoTests/ModalDemoTests.m: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) Facebook, Inc. and its affiliates. 3 | * 4 | * This source code is licensed under the MIT license found in the 5 | * LICENSE file in the root directory of this source tree. 6 | */ 7 | 8 | #import 9 | #import 10 | 11 | #import 12 | #import 13 | 14 | #define TIMEOUT_SECONDS 600 15 | #define TEXT_TO_LOOK_FOR @"Welcome to React Native!" 16 | 17 | @interface ModalDemoTests : XCTestCase 18 | 19 | @end 20 | 21 | @implementation ModalDemoTests 22 | 23 | - (BOOL)findSubviewInView:(UIView *)view matching:(BOOL(^)(UIView *view))test 24 | { 25 | if (test(view)) { 26 | return YES; 27 | } 28 | for (UIView *subview in [view subviews]) { 29 | if ([self findSubviewInView:subview matching:test]) { 30 | return YES; 31 | } 32 | } 33 | return NO; 34 | } 35 | 36 | - (void)testRendersWelcomeScreen 37 | { 38 | UIViewController *vc = [[[RCTSharedApplication() delegate] window] rootViewController]; 39 | NSDate *date = [NSDate dateWithTimeIntervalSinceNow:TIMEOUT_SECONDS]; 40 | BOOL foundElement = NO; 41 | 42 | __block NSString *redboxError = nil; 43 | RCTSetLogFunction(^(RCTLogLevel level, RCTLogSource source, NSString *fileName, NSNumber *lineNumber, NSString *message) { 44 | if (level >= RCTLogLevelError) { 45 | redboxError = message; 46 | } 47 | }); 48 | 49 | while ([date timeIntervalSinceNow] > 0 && !foundElement && !redboxError) { 50 | [[NSRunLoop mainRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]]; 51 | [[NSRunLoop mainRunLoop] runMode:NSRunLoopCommonModes beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]]; 52 | 53 | foundElement = [self findSubviewInView:vc.view matching:^BOOL(UIView *view) { 54 | if ([view.accessibilityLabel isEqualToString:TEXT_TO_LOOK_FOR]) { 55 | return YES; 56 | } 57 | return NO; 58 | }]; 59 | } 60 | 61 | RCTSetLogFunction(RCTDefaultLogFunction); 62 | 63 | XCTAssertNil(redboxError, @"RedBox error: %@", redboxError); 64 | XCTAssertTrue(foundElement, @"Couldn't find element with text '%@' in %d seconds", TEXT_TO_LOOK_FOR, TIMEOUT_SECONDS); 65 | } 66 | 67 | 68 | @end 69 | -------------------------------------------------------------------------------- /metro.config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Metro configuration for React Native 3 | * https://github.com/facebook/react-native 4 | * 5 | * @format 6 | */ 7 | 8 | module.exports = { 9 | transformer: { 10 | getTransformOptions: async () => ({ 11 | transform: { 12 | experimentalImportSupport: false, 13 | inlineRequires: false, 14 | }, 15 | }), 16 | }, 17 | }; 18 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ModalDemo", 3 | "version": "0.0.1", 4 | "private": true, 5 | "scripts": { 6 | "start": "node node_modules/react-native/local-cli/cli.js start", 7 | "test": "jest" 8 | }, 9 | "dependencies": { 10 | "prop-types": "^15.7.2", 11 | "react": "16.8.3", 12 | "react-native": "0.59.5", 13 | "react-native-root-siblings": "^3.2.0" 14 | }, 15 | "devDependencies": { 16 | "@babel/core": "^7.4.4", 17 | "@babel/runtime": "^7.4.4", 18 | "babel-jest": "^24.8.0", 19 | "jest": "^24.8.0", 20 | "metro-react-native-babel-preset": "^0.54.0", 21 | "react-test-renderer": "16.8.3" 22 | }, 23 | "jest": { 24 | "preset": "react-native" 25 | } 26 | } 27 | --------------------------------------------------------------------------------