├── LICENCE
├── README.md
├── example-app
├── .buckconfig
├── .gitattributes
├── .gitignore
├── App.js
├── __tests__
│ └── App.js
├── android
│ ├── app
│ │ ├── BUCK
│ │ ├── build.gradle
│ │ ├── build_defs.bzl
│ │ ├── debug.keystore
│ │ ├── proguard-rules.pro
│ │ └── src
│ │ │ ├── debug
│ │ │ ├── AndroidManifest.xml
│ │ │ └── java
│ │ │ │ └── com
│ │ │ │ └── exampleapp
│ │ │ │ └── ReactNativeFlipper.java
│ │ │ └── main
│ │ │ ├── AndroidManifest.xml
│ │ │ ├── java
│ │ │ └── com
│ │ │ │ └── exampleapp
│ │ │ │ ├── MainActivity.java
│ │ │ │ ├── MainApplication.java
│ │ │ │ └── generated
│ │ │ │ └── BasePackageList.java
│ │ │ └── res
│ │ │ ├── drawable
│ │ │ ├── splashscreen.xml
│ │ │ └── splashscreen_image.png
│ │ │ ├── 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
│ │ │ ├── colors.xml
│ │ │ ├── strings.xml
│ │ │ └── styles.xml
│ ├── build.gradle
│ ├── gradle.properties
│ ├── gradle
│ │ └── wrapper
│ │ │ ├── gradle-wrapper.jar
│ │ │ └── gradle-wrapper.properties
│ ├── gradlew
│ ├── gradlew.bat
│ └── settings.gradle
├── app.json
├── app
│ ├── components
│ │ └── dynamic-component.js
│ ├── screens
│ │ └── home-page.js
│ ├── store
│ │ ├── actions.js
│ │ ├── reducers.js
│ │ └── store.js
│ └── utils.js
├── babel.config.js
├── index.js
├── ios
│ ├── Podfile
│ ├── exampleapp.xcodeproj
│ │ ├── project.pbxproj
│ │ └── xcshareddata
│ │ │ └── xcschemes
│ │ │ └── exampleapp.xcscheme
│ └── exampleapp
│ │ ├── AppDelegate.h
│ │ ├── AppDelegate.m
│ │ ├── Base.lproj
│ │ └── LaunchScreen.xib
│ │ ├── Images.xcassets
│ │ ├── AppIcon.appiconset
│ │ │ └── Contents.json
│ │ ├── Contents.json
│ │ ├── SplashScreen.imageset
│ │ │ ├── Contents.json
│ │ │ └── splashscreen.png
│ │ └── SplashScreenBackground.imageset
│ │ │ ├── Contents.json
│ │ │ └── background.png
│ │ ├── Info.plist
│ │ ├── SplashScreen.storyboard
│ │ ├── Supporting
│ │ └── Expo.plist
│ │ └── main.m
├── metro.config.js
├── package.json
├── packages.js
└── yarn.lock
├── remote-components-demo.gif
└── remote-components
├── .gitignore
├── babel.config.js
├── components
├── counter.js
├── new-page.js
├── next-page-button.js
├── todo-status.js
└── todo-status2.js
├── package.json
├── rollup.config.js
└── yarn.lock
/LICENCE:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2021 Sarath KCM
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in
13 | all copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21 | THE SOFTWARE.
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ## Lazy loading React Native components from a server
2 |
3 | Blog post - https://sarathkcm.dev/blog/lazy-loading-react-native-components-from-a-server
4 |
5 | 
6 |
--------------------------------------------------------------------------------
/example-app/.buckconfig:
--------------------------------------------------------------------------------
1 |
2 | [android]
3 | target = Google Inc.:Google APIs:23
4 |
5 | [maven_repositories]
6 | central = https://repo1.maven.org/maven2
7 |
--------------------------------------------------------------------------------
/example-app/.gitattributes:
--------------------------------------------------------------------------------
1 | *.pbxproj -text
2 |
--------------------------------------------------------------------------------
/example-app/.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 | *.hprof
33 |
34 | # node.js
35 | #
36 | node_modules/
37 | npm-debug.log
38 | yarn-error.log
39 |
40 | # BUCK
41 | buck-out/
42 | \.buckd/
43 | *.keystore
44 | !debug.keystore
45 |
46 | # fastlane
47 | #
48 | # It is recommended to not store the screenshots in the git repo. Instead, use fastlane to re-generate the
49 | # screenshots whenever they are needed.
50 | # For more information about the recommended setup visit:
51 | # https://docs.fastlane.tools/best-practices/source-control/
52 |
53 | */fastlane/report.xml
54 | */fastlane/Preview.html
55 | */fastlane/screenshots
56 |
57 | # Bundle artifacts
58 | *.jsbundle
59 |
60 | # CocoaPods
61 | /ios/Pods/
62 |
63 | # Expo
64 | .expo/*
65 | web-build/
66 |
--------------------------------------------------------------------------------
/example-app/App.js:
--------------------------------------------------------------------------------
1 | import { StatusBar } from 'expo-status-bar';
2 | import React from 'react';
3 | import { StyleSheet } from 'react-native';
4 | import { NavigationContainer } from '@react-navigation/native';
5 | import { createStackNavigator } from '@react-navigation/stack';
6 | import { SafeAreaProvider } from 'react-native-safe-area-context';
7 | import HomePage from './app/screens/home-page';
8 | import configureStore from "./app/store/store"
9 | import { Provider } from 'react-redux';
10 | import DynamicComponent from './app/components/dynamic-component';
11 |
12 | const Stack = createStackNavigator();
13 | const store = configureStore();
14 |
15 | export default function App() {
16 | return (
17 |
18 |
19 |
20 |
21 |
22 |
23 | {() => }
24 |
25 |
26 |
27 |
28 |
29 |
30 | );
31 | }
32 |
33 | const styles = StyleSheet.create({
34 | container: {
35 | flex: 1,
36 | backgroundColor: '#fff',
37 | alignItems: 'center',
38 | justifyContent: 'center',
39 | },
40 | });
41 |
--------------------------------------------------------------------------------
/example-app/__tests__/App.js:
--------------------------------------------------------------------------------
1 | import 'react-native';
2 | import React from 'react';
3 | import App from '../App';
4 |
5 | // Note: test renderer must be required after react-native.
6 | import renderer from 'react-test-renderer';
7 |
8 | it('renders correctly', () => {
9 | renderer.create();
10 | });
11 |
--------------------------------------------------------------------------------
/example-app/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.exampleapp",
39 | )
40 |
41 | android_resource(
42 | name = "res",
43 | package = "com.exampleapp",
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 |
--------------------------------------------------------------------------------
/example-app/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. If none specified and
19 | * // "index.android.js" exists, it will be used. Otherwise "index.js" is
20 | * // default. Can be overridden with ENTRY_FILE environment variable.
21 | * entryFile: "index.android.js",
22 | *
23 | * // https://reactnative.dev/docs/performance#enable-the-ram-format
24 | * bundleCommand: "ram-bundle",
25 | *
26 | * // whether to bundle JS and assets in debug mode
27 | * bundleInDebug: false,
28 | *
29 | * // whether to bundle JS and assets in release mode
30 | * bundleInRelease: true,
31 | *
32 | * // whether to bundle JS and assets in another build variant (if configured).
33 | * // See http://tools.android.com/tech-docs/new-build-system/user-guide#TOC-Build-Variants
34 | * // The configuration property can be in the following formats
35 | * // 'bundleIn${productFlavor}${buildType}'
36 | * // 'bundleIn${buildType}'
37 | * // bundleInFreeDebug: true,
38 | * // bundleInPaidRelease: true,
39 | * // bundleInBeta: true,
40 | *
41 | * // whether to disable dev mode in custom build variants (by default only disabled in release)
42 | * // for example: to disable dev mode in the staging build type (if configured)
43 | * devDisabledInStaging: true,
44 | * // The configuration property can be in the following formats
45 | * // 'devDisabledIn${productFlavor}${buildType}'
46 | * // 'devDisabledIn${buildType}'
47 | *
48 | * // the root of your project, i.e. where "package.json" lives
49 | * root: "../../",
50 | *
51 | * // where to put the JS bundle asset in debug mode
52 | * jsBundleDirDebug: "$buildDir/intermediates/assets/debug",
53 | *
54 | * // where to put the JS bundle asset in release mode
55 | * jsBundleDirRelease: "$buildDir/intermediates/assets/release",
56 | *
57 | * // where to put drawable resources / React Native assets, e.g. the ones you use via
58 | * // require('./image.png')), in debug mode
59 | * resourcesDirDebug: "$buildDir/intermediates/res/merged/debug",
60 | *
61 | * // where to put drawable resources / React Native assets, e.g. the ones you use via
62 | * // require('./image.png')), in release mode
63 | * resourcesDirRelease: "$buildDir/intermediates/res/merged/release",
64 | *
65 | * // by default the gradle tasks are skipped if none of the JS files or assets change; this means
66 | * // that we don't look at files in android/ or ios/ to determine whether the tasks are up to
67 | * // date; if you have any other folders that you want to ignore for performance reasons (gradle
68 | * // indexes the entire tree), add them here. Alternatively, if you have JS files in android/
69 | * // for example, you might want to remove it from here.
70 | * inputExcludes: ["android/**", "ios/**"],
71 | *
72 | * // override which node gets called and with what additional arguments
73 | * nodeExecutableAndArgs: ["node"],
74 | *
75 | * // supply additional arguments to the packager
76 | * extraPackagerArgs: []
77 | * ]
78 | */
79 |
80 | project.ext.react = [
81 | enableHermes: false
82 | ]
83 |
84 | apply from: '../../node_modules/react-native-unimodules/gradle.groovy'
85 | apply from: "../../node_modules/react-native/react.gradle"
86 | apply from: "../../node_modules/expo-constants/scripts/get-app-config-android.gradle"
87 | apply from: "../../node_modules/expo-updates/scripts/create-manifest-android.gradle"
88 |
89 | /**
90 | * Set this to true to create two separate APKs instead of one:
91 | * - An APK that only works on ARM devices
92 | * - An APK that only works on x86 devices
93 | * The advantage is the size of the APK is reduced by about 4MB.
94 | * Upload all the APKs to the Play Store and people will download
95 | * the correct one based on the CPU architecture of their device.
96 | */
97 | def enableSeparateBuildPerCPUArchitecture = false
98 |
99 | /**
100 | * Run Proguard to shrink the Java bytecode in release builds.
101 | */
102 | def enableProguardInReleaseBuilds = false
103 |
104 | /**
105 | * The preferred build flavor of JavaScriptCore.
106 | *
107 | * For example, to use the international variant, you can use:
108 | * `def jscFlavor = 'org.webkit:android-jsc-intl:+'`
109 | *
110 | * The international variant includes ICU i18n library and necessary data
111 | * allowing to use e.g. `Date.toLocaleString` and `String.localeCompare` that
112 | * give correct results when using with locales other than en-US. Note that
113 | * this variant is about 6MiB larger per architecture than default.
114 | */
115 | def jscFlavor = 'org.webkit:android-jsc:+'
116 |
117 | /**
118 | * Whether to enable the Hermes VM.
119 | *
120 | * This should be set on project.ext.react and mirrored here. If it is not set
121 | * on project.ext.react, JavaScript will not be compiled to Hermes Bytecode
122 | * and the benefits of using Hermes will therefore be sharply reduced.
123 | */
124 | def enableHermes = project.ext.react.get("enableHermes", false);
125 |
126 | android {
127 | compileSdkVersion rootProject.ext.compileSdkVersion
128 |
129 | compileOptions {
130 | sourceCompatibility JavaVersion.VERSION_1_8
131 | targetCompatibility JavaVersion.VERSION_1_8
132 | }
133 |
134 | defaultConfig {
135 | applicationId "com.exampleapp"
136 | minSdkVersion rootProject.ext.minSdkVersion
137 | targetSdkVersion rootProject.ext.targetSdkVersion
138 | versionCode 1
139 | versionName "1.0"
140 | }
141 | splits {
142 | abi {
143 | reset()
144 | enable enableSeparateBuildPerCPUArchitecture
145 | universalApk false // If true, also generate a universal APK
146 | include "armeabi-v7a", "x86", "arm64-v8a", "x86_64"
147 | }
148 | }
149 | signingConfigs {
150 | debug {
151 | storeFile file('debug.keystore')
152 | storePassword 'android'
153 | keyAlias 'androiddebugkey'
154 | keyPassword 'android'
155 | }
156 | }
157 | buildTypes {
158 | debug {
159 | signingConfig signingConfigs.debug
160 | }
161 | release {
162 | // Caution! In production, you need to generate your own keystore file.
163 | // see https://reactnative.dev/docs/signed-apk-android.
164 | signingConfig signingConfigs.debug
165 | minifyEnabled enableProguardInReleaseBuilds
166 | proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro"
167 | }
168 | }
169 |
170 | // applicationVariants are e.g. debug, release
171 | applicationVariants.all { variant ->
172 | variant.outputs.each { output ->
173 | // For each separate APK per architecture, set a unique version code as described here:
174 | // https://developer.android.com/studio/build/configure-apk-splits.html
175 | def versionCodes = ["armeabi-v7a": 1, "x86": 2, "arm64-v8a": 3, "x86_64": 4]
176 | def abi = output.getFilter(OutputFile.ABI)
177 | if (abi != null) { // null for the universal-debug, universal-release variants
178 | output.versionCodeOverride =
179 | versionCodes.get(abi) * 1048576 + defaultConfig.versionCode
180 | }
181 |
182 | }
183 | }
184 | }
185 |
186 | dependencies {
187 | implementation fileTree(dir: "libs", include: ["*.jar"])
188 | //noinspection GradleDynamicVersion
189 | implementation "com.facebook.react:react-native:+" // From node_modules
190 | implementation "androidx.swiperefreshlayout:swiperefreshlayout:1.0.0"
191 | debugImplementation("com.facebook.flipper:flipper:${FLIPPER_VERSION}") {
192 | exclude group:'com.facebook.fbjni'
193 | }
194 | debugImplementation("com.facebook.flipper:flipper-network-plugin:${FLIPPER_VERSION}") {
195 | exclude group:'com.facebook.flipper'
196 | exclude group:'com.squareup.okhttp3', module:'okhttp'
197 | }
198 | debugImplementation("com.facebook.flipper:flipper-fresco-plugin:${FLIPPER_VERSION}") {
199 | exclude group:'com.facebook.flipper'
200 | }
201 | addUnimodulesDependencies()
202 |
203 | if (enableHermes) {
204 | def hermesPath = "../../node_modules/hermes-engine/android/";
205 | debugImplementation files(hermesPath + "hermes-debug.aar")
206 | releaseImplementation files(hermesPath + "hermes-release.aar")
207 | } else {
208 | implementation jscFlavor
209 | }
210 | }
211 |
212 | // Run this once to be able to run the application with BUCK
213 | // puts all compile dependencies into folder libs for BUCK to use
214 | task copyDownloadableDepsToLibs(type: Copy) {
215 | from configurations.compile
216 | into 'libs'
217 | }
218 |
219 | apply from: file("../../node_modules/@react-native-community/cli-platform-android/native_modules.gradle"); applyNativeModulesAppBuildGradle(project)
220 |
--------------------------------------------------------------------------------
/example-app/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 |
--------------------------------------------------------------------------------
/example-app/android/app/debug.keystore:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/debug.keystore
--------------------------------------------------------------------------------
/example-app/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 |
--------------------------------------------------------------------------------
/example-app/android/app/src/debug/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
3 |
4 |
5 |
6 |
7 |
8 |
--------------------------------------------------------------------------------
/example-app/android/app/src/debug/java/com/exampleapp/ReactNativeFlipper.java:
--------------------------------------------------------------------------------
1 | /**
2 | * Copyright (c) Facebook, Inc. and its affiliates.
3 | *
4 | *
This source code is licensed under the MIT license found in the LICENSE file in the root
5 | * directory of this source tree.
6 | */
7 | package com.exampleapp;
8 |
9 | import android.content.Context;
10 | import com.facebook.flipper.android.AndroidFlipperClient;
11 | import com.facebook.flipper.android.utils.FlipperUtils;
12 | import com.facebook.flipper.core.FlipperClient;
13 | import com.facebook.flipper.plugins.crashreporter.CrashReporterPlugin;
14 | import com.facebook.flipper.plugins.databases.DatabasesFlipperPlugin;
15 | import com.facebook.flipper.plugins.fresco.FrescoFlipperPlugin;
16 | import com.facebook.flipper.plugins.inspector.DescriptorMapping;
17 | import com.facebook.flipper.plugins.inspector.InspectorFlipperPlugin;
18 | import com.facebook.flipper.plugins.network.FlipperOkhttpInterceptor;
19 | import com.facebook.flipper.plugins.network.NetworkFlipperPlugin;
20 | import com.facebook.flipper.plugins.react.ReactFlipperPlugin;
21 | import com.facebook.flipper.plugins.sharedpreferences.SharedPreferencesFlipperPlugin;
22 | import com.facebook.react.ReactInstanceManager;
23 | import com.facebook.react.bridge.ReactContext;
24 | import com.facebook.react.modules.network.NetworkingModule;
25 | import okhttp3.OkHttpClient;
26 |
27 | public class ReactNativeFlipper {
28 | public static void initializeFlipper(Context context, ReactInstanceManager reactInstanceManager) {
29 | if (FlipperUtils.shouldEnableFlipper(context)) {
30 | final FlipperClient client = AndroidFlipperClient.getInstance(context);
31 | client.addPlugin(new InspectorFlipperPlugin(context, DescriptorMapping.withDefaults()));
32 | client.addPlugin(new ReactFlipperPlugin());
33 | client.addPlugin(new DatabasesFlipperPlugin(context));
34 | client.addPlugin(new SharedPreferencesFlipperPlugin(context));
35 | client.addPlugin(CrashReporterPlugin.getInstance());
36 | NetworkFlipperPlugin networkFlipperPlugin = new NetworkFlipperPlugin();
37 | NetworkingModule.setCustomClientBuilder(
38 | new NetworkingModule.CustomClientBuilder() {
39 | @Override
40 | public void apply(OkHttpClient.Builder builder) {
41 | builder.addNetworkInterceptor(new FlipperOkhttpInterceptor(networkFlipperPlugin));
42 | }
43 | });
44 | client.addPlugin(networkFlipperPlugin);
45 | client.start();
46 | // Fresco Plugin needs to ensure that ImagePipelineFactory is initialized
47 | // Hence we run if after all native modules have been initialized
48 | ReactContext reactContext = reactInstanceManager.getCurrentReactContext();
49 | if (reactContext == null) {
50 | reactInstanceManager.addReactInstanceEventListener(
51 | new ReactInstanceManager.ReactInstanceEventListener() {
52 | @Override
53 | public void onReactContextInitialized(ReactContext reactContext) {
54 | reactInstanceManager.removeReactInstanceEventListener(this);
55 | reactContext.runOnNativeModulesQueueThread(
56 | new Runnable() {
57 | @Override
58 | public void run() {
59 | client.addPlugin(new FrescoFlipperPlugin());
60 | }
61 | });
62 | }
63 | });
64 | } else {
65 | client.addPlugin(new FrescoFlipperPlugin());
66 | }
67 | }
68 | }
69 | }
--------------------------------------------------------------------------------
/example-app/android/app/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
32 |
33 |
34 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
--------------------------------------------------------------------------------
/example-app/android/app/src/main/java/com/exampleapp/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.exampleapp;
2 |
3 | import android.os.Bundle;
4 |
5 | import com.facebook.react.ReactActivity;
6 | import com.facebook.react.ReactActivityDelegate;
7 | import com.facebook.react.ReactRootView;
8 | import com.swmansion.gesturehandler.react.RNGestureHandlerEnabledRootView;
9 |
10 | import expo.modules.splashscreen.singletons.SplashScreen;
11 | import expo.modules.splashscreen.SplashScreenImageResizeMode;
12 |
13 | public class MainActivity extends ReactActivity {
14 | @Override
15 | protected void onCreate(Bundle savedInstanceState) {
16 | super.onCreate(null);
17 | // SplashScreen.show(...) has to be called after super.onCreate(...)
18 | // Below line is handled by '@expo/configure-splash-screen' command and it's discouraged to modify it manually
19 | SplashScreen.show(this, SplashScreenImageResizeMode.CONTAIN, ReactRootView.class, false);
20 | }
21 |
22 |
23 | /**
24 | * Returns the name of the main component registered from JavaScript.
25 | * This is used to schedule rendering of the component.
26 | */
27 | @Override
28 | protected String getMainComponentName() {
29 | return "main";
30 | }
31 |
32 | @Override
33 | protected ReactActivityDelegate createReactActivityDelegate() {
34 | return new ReactActivityDelegate(this, getMainComponentName()) {
35 | @Override
36 | protected ReactRootView createRootView() {
37 | return new RNGestureHandlerEnabledRootView(MainActivity.this);
38 | }
39 | };
40 | }
41 | }
42 |
--------------------------------------------------------------------------------
/example-app/android/app/src/main/java/com/exampleapp/MainApplication.java:
--------------------------------------------------------------------------------
1 | package com.exampleapp;
2 |
3 | import android.app.Application;
4 | import android.content.Context;
5 | import android.net.Uri;
6 |
7 | import com.facebook.react.PackageList;
8 | import com.facebook.react.ReactApplication;
9 | import com.facebook.react.ReactInstanceManager;
10 | import com.facebook.react.ReactNativeHost;
11 | import com.facebook.react.ReactPackage;
12 | import com.facebook.react.shell.MainReactPackage;
13 | import com.facebook.soloader.SoLoader;
14 | import com.exampleapp.generated.BasePackageList;
15 |
16 | import org.unimodules.adapters.react.ReactAdapterPackage;
17 | import org.unimodules.adapters.react.ModuleRegistryAdapter;
18 | import org.unimodules.adapters.react.ReactModuleRegistryProvider;
19 | import org.unimodules.core.interfaces.Package;
20 | import org.unimodules.core.interfaces.SingletonModule;
21 | import expo.modules.constants.ConstantsPackage;
22 | import expo.modules.permissions.PermissionsPackage;
23 | import expo.modules.filesystem.FileSystemPackage;
24 | import expo.modules.updates.UpdatesController;
25 |
26 | import java.lang.reflect.InvocationTargetException;
27 | import java.util.Arrays;
28 | import java.util.List;
29 | import javax.annotation.Nullable;
30 |
31 | public class MainApplication extends Application implements ReactApplication {
32 | private final ReactModuleRegistryProvider mModuleRegistryProvider = new ReactModuleRegistryProvider(
33 | new BasePackageList().getPackageList()
34 | );
35 |
36 | private final ReactNativeHost mReactNativeHost = new ReactNativeHost(this) {
37 | @Override
38 | public boolean getUseDeveloperSupport() {
39 | return BuildConfig.DEBUG;
40 | }
41 |
42 | @Override
43 | protected List getPackages() {
44 | List packages = new PackageList(this).getPackages();
45 | packages.add(new ModuleRegistryAdapter(mModuleRegistryProvider));
46 | return packages;
47 | }
48 |
49 | @Override
50 | protected String getJSMainModuleName() {
51 | return "index";
52 | }
53 |
54 | @Override
55 | protected @Nullable String getJSBundleFile() {
56 | if (BuildConfig.DEBUG) {
57 | return super.getJSBundleFile();
58 | } else {
59 | return UpdatesController.getInstance().getLaunchAssetFile();
60 | }
61 | }
62 |
63 | @Override
64 | protected @Nullable String getBundleAssetName() {
65 | if (BuildConfig.DEBUG) {
66 | return super.getBundleAssetName();
67 | } else {
68 | return UpdatesController.getInstance().getBundleAssetName();
69 | }
70 | }
71 | };
72 |
73 | @Override
74 | public ReactNativeHost getReactNativeHost() {
75 | return mReactNativeHost;
76 | }
77 |
78 | @Override
79 | public void onCreate() {
80 | super.onCreate();
81 | SoLoader.init(this, /* native exopackage */ false);
82 |
83 | if (!BuildConfig.DEBUG) {
84 | UpdatesController.initialize(this);
85 | }
86 |
87 | initializeFlipper(this, getReactNativeHost().getReactInstanceManager());
88 | }
89 |
90 | /**
91 | * Loads Flipper in React Native templates. Call this in the onCreate method with something like
92 | * initializeFlipper(this, getReactNativeHost().getReactInstanceManager());
93 | *
94 | * @param context
95 | * @param reactInstanceManager
96 | */
97 | private static void initializeFlipper(
98 | Context context, ReactInstanceManager reactInstanceManager) {
99 | if (BuildConfig.DEBUG) {
100 | try {
101 | /*
102 | We use reflection here to pick up the class that initializes Flipper,
103 | since Flipper library is not available in release mode
104 | */
105 | Class> aClass = Class.forName("com.exampleapp.ReactNativeFlipper");
106 | aClass
107 | .getMethod("initializeFlipper", Context.class, ReactInstanceManager.class)
108 | .invoke(null, context, reactInstanceManager);
109 | } catch (ClassNotFoundException e) {
110 | e.printStackTrace();
111 | } catch (NoSuchMethodException e) {
112 | e.printStackTrace();
113 | } catch (IllegalAccessException e) {
114 | e.printStackTrace();
115 | } catch (InvocationTargetException e) {
116 | e.printStackTrace();
117 | }
118 | }
119 | }
120 | }
121 |
--------------------------------------------------------------------------------
/example-app/android/app/src/main/java/com/exampleapp/generated/BasePackageList.java:
--------------------------------------------------------------------------------
1 | package com.exampleapp.generated;
2 |
3 | import java.util.Arrays;
4 | import java.util.List;
5 | import org.unimodules.core.interfaces.Package;
6 |
7 | public class BasePackageList {
8 | public List getPackageList() {
9 | return Arrays.asList(
10 | new expo.modules.application.ApplicationPackage(),
11 | new expo.modules.constants.ConstantsPackage(),
12 | new expo.modules.errorrecovery.ErrorRecoveryPackage(),
13 | new expo.modules.filesystem.FileSystemPackage(),
14 | new expo.modules.font.FontLoaderPackage(),
15 | new expo.modules.imageloader.ImageLoaderPackage(),
16 | new expo.modules.keepawake.KeepAwakePackage(),
17 | new expo.modules.lineargradient.LinearGradientPackage(),
18 | new expo.modules.location.LocationPackage(),
19 | new expo.modules.permissions.PermissionsPackage(),
20 | new expo.modules.securestore.SecureStorePackage(),
21 | new expo.modules.splashscreen.SplashScreenPackage(),
22 | new expo.modules.sqlite.SQLitePackage(),
23 | new expo.modules.updates.UpdatesPackage()
24 | );
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/drawable/splashscreen.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 |
6 |
7 |
8 |
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/drawable/splashscreen_image.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/drawable/splashscreen_image.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/values/colors.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | #FFFFFF
5 |
6 |
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | example-app
3 |
4 |
--------------------------------------------------------------------------------
/example-app/android/app/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
7 |
12 |
13 |
--------------------------------------------------------------------------------
/example-app/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 = "29.0.2"
6 | minSdkVersion = 21
7 | compileSdkVersion = 29
8 | targetSdkVersion = 29
9 | }
10 | repositories {
11 | google()
12 | jcenter()
13 | }
14 | dependencies {
15 | classpath("com.android.tools.build:gradle:3.5.3")
16 |
17 | // NOTE: Do not place your application dependencies here; they belong
18 | // in the individual module build.gradle files
19 | }
20 | }
21 |
22 | allprojects {
23 | repositories {
24 | mavenLocal()
25 | maven {
26 | // All of React Native (JS, Obj-C sources, Android binaries) is installed from npm
27 | url("$rootDir/../node_modules/react-native/android")
28 | }
29 | maven {
30 | // Android JSC is installed from npm
31 | url("$rootDir/../node_modules/jsc-android/dist")
32 | }
33 |
34 | google()
35 | jcenter()
36 | maven { url 'https://www.jitpack.io' }
37 | }
38 | }
39 |
--------------------------------------------------------------------------------
/example-app/android/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 |
3 | # IDE (e.g. Android Studio) users:
4 | # Gradle settings configured through the IDE *will override*
5 | # any settings specified in this file.
6 |
7 | # For more details on how to configure your build environment visit
8 | # http://www.gradle.org/docs/current/userguide/build_environment.html
9 |
10 | # Specifies the JVM arguments used for the daemon process.
11 | # The setting is particularly useful for tweaking memory settings.
12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m
13 | org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
14 |
15 | # When configured, Gradle will run in incubating parallel mode.
16 | # This option should only be used with decoupled projects. More details, visit
17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
18 | # org.gradle.parallel=true
19 |
20 | # AndroidX package structure to make it clearer which packages are bundled with the
21 | # Android operating system, and which are packaged with your app's APK
22 | # https://developer.android.com/topic/libraries/support-library/androidx-rn
23 | android.useAndroidX=true
24 |
25 | # Automatically convert third-party libraries to use AndroidX
26 | android.enableJetifier=true
27 |
28 | # Version of flipper SDK to use with React Native
29 | FLIPPER_VERSION=0.54.0
--------------------------------------------------------------------------------
/example-app/android/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/android/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/example-app/android/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | distributionBase=GRADLE_USER_HOME
2 | distributionPath=wrapper/dists
3 | distributionUrl=https\://services.gradle.org/distributions/gradle-6.3-all.zip
4 | zipStoreBase=GRADLE_USER_HOME
5 | zipStorePath=wrapper/dists
6 |
--------------------------------------------------------------------------------
/example-app/android/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env sh
2 |
3 | #
4 | # Copyright 2015 the original author or authors.
5 | #
6 | # Licensed under the Apache License, Version 2.0 (the "License");
7 | # you may not use this file except in compliance with the License.
8 | # You may obtain a copy of the License at
9 | #
10 | # https://www.apache.org/licenses/LICENSE-2.0
11 | #
12 | # Unless required by applicable law or agreed to in writing, software
13 | # distributed under the License is distributed on an "AS IS" BASIS,
14 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 | # See the License for the specific language governing permissions and
16 | # limitations under the License.
17 | #
18 |
19 | ##############################################################################
20 | ##
21 | ## Gradle start up script for UN*X
22 | ##
23 | ##############################################################################
24 |
25 | # Attempt to set APP_HOME
26 | # Resolve links: $0 may be a link
27 | PRG="$0"
28 | # Need this for relative symlinks.
29 | while [ -h "$PRG" ] ; do
30 | ls=`ls -ld "$PRG"`
31 | link=`expr "$ls" : '.*-> \(.*\)$'`
32 | if expr "$link" : '/.*' > /dev/null; then
33 | PRG="$link"
34 | else
35 | PRG=`dirname "$PRG"`"/$link"
36 | fi
37 | done
38 | SAVED="`pwd`"
39 | cd "`dirname \"$PRG\"`/" >/dev/null
40 | APP_HOME="`pwd -P`"
41 | cd "$SAVED" >/dev/null
42 |
43 | APP_NAME="Gradle"
44 | APP_BASE_NAME=`basename "$0"`
45 |
46 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
47 | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
48 |
49 | # Use the maximum available, or set MAX_FD != -1 to use that value.
50 | MAX_FD="maximum"
51 |
52 | warn () {
53 | echo "$*"
54 | }
55 |
56 | die () {
57 | echo
58 | echo "$*"
59 | echo
60 | exit 1
61 | }
62 |
63 | # OS specific support (must be 'true' or 'false').
64 | cygwin=false
65 | msys=false
66 | darwin=false
67 | nonstop=false
68 | case "`uname`" in
69 | CYGWIN* )
70 | cygwin=true
71 | ;;
72 | Darwin* )
73 | darwin=true
74 | ;;
75 | MINGW* )
76 | msys=true
77 | ;;
78 | NONSTOP* )
79 | nonstop=true
80 | ;;
81 | esac
82 |
83 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
84 |
85 | # Determine the Java command to use to start the JVM.
86 | if [ -n "$JAVA_HOME" ] ; then
87 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
88 | # IBM's JDK on AIX uses strange locations for the executables
89 | JAVACMD="$JAVA_HOME/jre/sh/java"
90 | else
91 | JAVACMD="$JAVA_HOME/bin/java"
92 | fi
93 | if [ ! -x "$JAVACMD" ] ; then
94 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
95 |
96 | Please set the JAVA_HOME variable in your environment to match the
97 | location of your Java installation."
98 | fi
99 | else
100 | JAVACMD="java"
101 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
102 |
103 | Please set the JAVA_HOME variable in your environment to match the
104 | location of your Java installation."
105 | fi
106 |
107 | # Increase the maximum file descriptors if we can.
108 | if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
109 | MAX_FD_LIMIT=`ulimit -H -n`
110 | if [ $? -eq 0 ] ; then
111 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
112 | MAX_FD="$MAX_FD_LIMIT"
113 | fi
114 | ulimit -n $MAX_FD
115 | if [ $? -ne 0 ] ; then
116 | warn "Could not set maximum file descriptor limit: $MAX_FD"
117 | fi
118 | else
119 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
120 | fi
121 | fi
122 |
123 | # For Darwin, add options to specify how the application appears in the dock
124 | if $darwin; then
125 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
126 | fi
127 |
128 | # For Cygwin or MSYS, switch paths to Windows format before running java
129 | if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then
130 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
131 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
132 | JAVACMD=`cygpath --unix "$JAVACMD"`
133 |
134 | # We build the pattern for arguments to be converted via cygpath
135 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
136 | SEP=""
137 | for dir in $ROOTDIRSRAW ; do
138 | ROOTDIRS="$ROOTDIRS$SEP$dir"
139 | SEP="|"
140 | done
141 | OURCYGPATTERN="(^($ROOTDIRS))"
142 | # Add a user-defined pattern to the cygpath arguments
143 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
144 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
145 | fi
146 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
147 | i=0
148 | for arg in "$@" ; do
149 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
150 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
151 |
152 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
153 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
154 | else
155 | eval `echo args$i`="\"$arg\""
156 | fi
157 | i=`expr $i + 1`
158 | done
159 | case $i in
160 | 0) set -- ;;
161 | 1) set -- "$args0" ;;
162 | 2) set -- "$args0" "$args1" ;;
163 | 3) set -- "$args0" "$args1" "$args2" ;;
164 | 4) set -- "$args0" "$args1" "$args2" "$args3" ;;
165 | 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
166 | 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
167 | 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
168 | 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
169 | 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
170 | esac
171 | fi
172 |
173 | # Escape application args
174 | save () {
175 | for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
176 | echo " "
177 | }
178 | APP_ARGS=`save "$@"`
179 |
180 | # Collect all arguments for the java command, following the shell quoting and substitution rules
181 | eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
182 |
183 | exec "$JAVACMD" "$@"
184 |
--------------------------------------------------------------------------------
/example-app/android/gradlew.bat:
--------------------------------------------------------------------------------
1 | @rem
2 | @rem Copyright 2015 the original author or authors.
3 | @rem
4 | @rem Licensed under the Apache License, Version 2.0 (the "License");
5 | @rem you may not use this file except in compliance with the License.
6 | @rem You may obtain a copy of the License at
7 | @rem
8 | @rem https://www.apache.org/licenses/LICENSE-2.0
9 | @rem
10 | @rem Unless required by applicable law or agreed to in writing, software
11 | @rem distributed under the License is distributed on an "AS IS" BASIS,
12 | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 | @rem See the License for the specific language governing permissions and
14 | @rem limitations under the License.
15 | @rem
16 |
17 | @if "%DEBUG%" == "" @echo off
18 | @rem ##########################################################################
19 | @rem
20 | @rem Gradle startup script for Windows
21 | @rem
22 | @rem ##########################################################################
23 |
24 | @rem Set local scope for the variables with windows NT shell
25 | if "%OS%"=="Windows_NT" setlocal
26 |
27 | set DIRNAME=%~dp0
28 | if "%DIRNAME%" == "" set DIRNAME=.
29 | set APP_BASE_NAME=%~n0
30 | set APP_HOME=%DIRNAME%
31 |
32 | @rem Resolve any "." and ".." in APP_HOME to make it shorter.
33 | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
34 |
35 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
36 | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
37 |
38 | @rem Find java.exe
39 | if defined JAVA_HOME goto findJavaFromJavaHome
40 |
41 | set JAVA_EXE=java.exe
42 | %JAVA_EXE% -version >NUL 2>&1
43 | if "%ERRORLEVEL%" == "0" goto init
44 |
45 | echo.
46 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
47 | echo.
48 | echo Please set the JAVA_HOME variable in your environment to match the
49 | echo location of your Java installation.
50 |
51 | goto fail
52 |
53 | :findJavaFromJavaHome
54 | set JAVA_HOME=%JAVA_HOME:"=%
55 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
56 |
57 | if exist "%JAVA_EXE%" goto init
58 |
59 | echo.
60 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
61 | echo.
62 | echo Please set the JAVA_HOME variable in your environment to match the
63 | echo location of your Java installation.
64 |
65 | goto fail
66 |
67 | :init
68 | @rem Get command-line arguments, handling Windows variants
69 |
70 | if not "%OS%" == "Windows_NT" goto win9xME_args
71 |
72 | :win9xME_args
73 | @rem Slurp the command line arguments.
74 | set CMD_LINE_ARGS=
75 | set _SKIP=2
76 |
77 | :win9xME_args_slurp
78 | if "x%~1" == "x" goto execute
79 |
80 | set CMD_LINE_ARGS=%*
81 |
82 | :execute
83 | @rem Setup the command line
84 |
85 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
86 |
87 | @rem Execute Gradle
88 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
89 |
90 | :end
91 | @rem End local scope for the variables with windows NT shell
92 | if "%ERRORLEVEL%"=="0" goto mainEnd
93 |
94 | :fail
95 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
96 | rem the _cmd.exe /c_ return code!
97 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
98 | exit /b 1
99 |
100 | :mainEnd
101 | if "%OS%"=="Windows_NT" endlocal
102 |
103 | :omega
104 |
--------------------------------------------------------------------------------
/example-app/android/settings.gradle:
--------------------------------------------------------------------------------
1 | rootProject.name = 'exampleapp'
2 |
3 | apply from: '../node_modules/react-native-unimodules/gradle.groovy'
4 | includeUnimodulesProjects()
5 |
6 | apply from: file("../node_modules/@react-native-community/cli-platform-android/native_modules.gradle");
7 | applyNativeModulesSettingsGradle(settings)
8 |
9 | include ':app'
10 |
--------------------------------------------------------------------------------
/example-app/app.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "example-app",
3 | "displayName": "example-app",
4 | "expo": {
5 | "name": "example-app",
6 | "slug": "example-app",
7 | "version": "1.0.0",
8 | "assetBundlePatterns": [
9 | "**/*"
10 | ]
11 | }
12 | }
13 |
--------------------------------------------------------------------------------
/example-app/app/components/dynamic-component.js:
--------------------------------------------------------------------------------
1 | import React, { useMemo, Suspense } from 'react';
2 | import { Text, View } from 'react-native';
3 | import { fetchComponent } from "../utils";
4 |
5 | const DynamicComponent = ({ __id, children, ...props }) => {
6 | const Component = useMemo(() => {
7 | return React.lazy(async () => fetchComponent(__id))
8 | }, [__id]);
9 |
10 | return (
11 | Loading...}>
12 | {children}
13 |
14 | )
15 | };
16 |
17 | export default React.memo(DynamicComponent);
--------------------------------------------------------------------------------
/example-app/app/screens/home-page.js:
--------------------------------------------------------------------------------
1 | import React, { useState } from 'react'
2 | import { StyleSheet, Switch, TouchableOpacity, View, ScrollView } from 'react-native'
3 | import { Text, Button, Input } from 'react-native-elements'
4 | import DynamicComponent from '../components/dynamic-component';
5 | import { addTodo, toggleTodo } from '../store/actions';
6 | import { useDispatch, useSelector } from 'react-redux';
7 | import Icon from "react-native-vector-icons/MaterialCommunityIcons";
8 |
9 | const HomePage = ({ navigation }) => {
10 | const [showDynamicComponents, setShowDynamicComponents] = useState(false);
11 | const [todoText, setTodoText] = useState("");
12 |
13 | const todos = useSelector((state) => state.todos);
14 | const dispatch = useDispatch();
15 | const addTodoToStore = () => todoText.trim() && dispatch(addTodo({ title: todoText }));
16 | const toggleTodoToStore = (id, completed) => dispatch(toggleTodo({ id, completed }));
17 |
18 | const renderDynamicComponentsSwitch = () => (
19 |
20 | Toggle remote components
21 |
28 |
29 | );
30 |
31 | const renderDynamicComponents = () => (
32 | <>
33 | {showDynamicComponents && (
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 | )}
47 | >
48 | );
49 |
50 | const renderTodoSection = () => (
51 | <>
52 | Todos
53 |
54 | }
56 | placeholder="Enter Todo"
57 | containerStyle={{ flex: 1, marginTop: 8 }}
58 | onChangeText={text => setTodoText(text)}
59 | value={todoText}
60 | />
61 |
70 |
71 | {todos.map(todo => (
72 | toggleTodoToStore(todo.id, !todo.completed)}
75 | style={{ flexDirection: "row", justifyContent: "flex-start", alignItems: "baseline", marginVertical: 5 }}
76 | >
77 |
78 | {todo.title}
79 |
80 | ))}
81 |
82 | >
83 | )
84 | return (
85 |
86 | {renderTodoSection()}
87 | {renderDynamicComponentsSwitch()}
88 | {renderDynamicComponents()}
89 |
90 | )
91 | }
92 |
93 | export default HomePage
94 |
95 | const styles = StyleSheet.create({})
96 |
--------------------------------------------------------------------------------
/example-app/app/store/actions.js:
--------------------------------------------------------------------------------
1 |
2 | import { nanoid } from "nanoid";
3 |
4 | export const ADD_TODO = "store/add-todo";
5 | export const TOGGLE_TODO = "store/toggle-todo";
6 |
7 | export const addTodo = ({ title, completed = false }) => ({
8 | type: ADD_TODO, payload: { id: nanoid(), title, completed }
9 | });
10 |
11 | export const toggleTodo = ({ id, completed }) => ({
12 | type: TOGGLE_TODO, payload: { id, completed }
13 | });
--------------------------------------------------------------------------------
/example-app/app/store/reducers.js:
--------------------------------------------------------------------------------
1 | import { ADD_TODO, TOGGLE_TODO } from "./actions";
2 |
3 | export default function reducer(state = [], action) {
4 | switch (action.type) {
5 | case ADD_TODO:
6 | return [...state, action.payload]
7 | case TOGGLE_TODO:
8 | return state?.map(a => {
9 | if (a.id === action.payload.id) {
10 | return { ...a, completed: action.payload.completed }
11 | }
12 | return a;
13 | });
14 | default:
15 | return state;
16 | }
17 | }
--------------------------------------------------------------------------------
/example-app/app/store/store.js:
--------------------------------------------------------------------------------
1 | import { createStore, combineReducers } from 'redux';
2 | import todoReducer from './reducers';
3 | const rootReducer = combineReducers(
4 | { todos: todoReducer }
5 | );
6 | const configureStore = () => {
7 | return createStore(rootReducer);
8 | }
9 | export default configureStore;
--------------------------------------------------------------------------------
/example-app/app/utils.js:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { Text } from "react-native";
3 | import packages from "../packages"
4 |
5 | function getParsedModule(code, moduleName, packages) {
6 | const _this = Object.create(packages);
7 | function require(name) {
8 | if (!(name in _this) && moduleName === name) {
9 | let module = { exports: {} };
10 | _this[name] = () => module;
11 | let wrapper = Function("require, exports, module", code);
12 | wrapper(require, module.exports, module);
13 | } else if (!(name in _this)) {
14 | throw `Module '${name}' not found`
15 | }
16 | return (_this[name]()).exports;
17 | }
18 |
19 | return require(moduleName);
20 | }
21 |
22 | export async function fetchComponent(id) {
23 | try {
24 | const text = await fetch(`http://10.0.2.2:8080/${id}.js?time=${Date.now()}`).then(a => {
25 | if (!a.ok) {
26 | throw new Error('Network response was not ok');
27 | }
28 | return a.text()
29 | });
30 | return { default: getParsedModule(text, id, packages) };
31 | } catch (error) {
32 | console.log(error)
33 | return { default() { return Failed to Render } }
34 | }
35 | }
--------------------------------------------------------------------------------
/example-app/babel.config.js:
--------------------------------------------------------------------------------
1 | module.exports = function(api) {
2 | api.cache(true);
3 | return {
4 | presets: ['babel-preset-expo'],
5 | };
6 | };
7 |
--------------------------------------------------------------------------------
/example-app/index.js:
--------------------------------------------------------------------------------
1 | import { registerRootComponent } from 'expo';
2 | import 'react-native-gesture-handler';
3 | import 'react-native-get-random-values';
4 | import App from './App';
5 |
6 | // registerRootComponent calls AppRegistry.registerComponent('main', () => App);
7 | // It also ensures that whether you load the app in the Expo client or in a native build,
8 | // the environment is set up appropriately
9 | registerRootComponent(App);
10 |
--------------------------------------------------------------------------------
/example-app/ios/Podfile:
--------------------------------------------------------------------------------
1 | require_relative '../node_modules/react-native/scripts/react_native_pods'
2 | require_relative '../node_modules/react-native-unimodules/cocoapods.rb'
3 | require_relative '../node_modules/@react-native-community/cli-platform-ios/native_modules'
4 |
5 | platform :ios, '11.0'
6 |
7 | target 'exampleapp' do
8 | use_unimodules!
9 | config = use_native_modules!
10 |
11 | use_react_native!(:path => config["reactNativePath"])
12 |
13 | # Uncomment the code below to enable Flipper.
14 | #
15 | # You should not install Flipper in CI environments when creating release
16 | # builds, this will lead to significantly slower build times.
17 | #
18 | # Note that if you have use_frameworks! enabled, Flipper will not work.
19 | #
20 | # use_flipper!
21 | # post_install do |installer|
22 | # flipper_post_install(installer)
23 | # end
24 | end
25 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp.xcodeproj/project.pbxproj:
--------------------------------------------------------------------------------
1 | // !$*UTF8*$!
2 | {
3 | archiveVersion = 1;
4 | classes = {
5 | };
6 | objectVersion = 46;
7 | objects = {
8 |
9 | /* Begin PBXBuildFile section */
10 | 13B07FBC1A68108700A75B9A /* AppDelegate.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB01A68108700A75B9A /* AppDelegate.m */; };
11 | 13B07FBD1A68108700A75B9A /* LaunchScreen.xib in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB11A68108700A75B9A /* LaunchScreen.xib */; };
12 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB51A68108700A75B9A /* Images.xcassets */; };
13 | 13B07FC11A68108700A75B9A /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB71A68108700A75B9A /* main.m */; };
14 | 3E461D99554A48A4959DE609 /* SplashScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = AA286B85B6C04FC6940260E9 /* SplashScreen.storyboard */; };
15 | 96905EF65AED1B983A6B3ABC /* libPods-exampleapp.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 58EEBF8E8E6FB1BC6CAF49B5 /* libPods-exampleapp.a */; };
16 | BB2F792D24A3F905000567C9 /* Expo.plist in Resources */ = {isa = PBXBuildFile; fileRef = BB2F792C24A3F905000567C9 /* Expo.plist */; };
17 | /* End PBXBuildFile section */
18 |
19 | /* Begin PBXFileReference section */
20 | 008F07F21AC5B25A0029DE68 /* main.jsbundle */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; path = main.jsbundle; sourceTree = ""; };
21 | 13B07F961A680F5B00A75B9A /* exampleapp.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = exampleapp.app; sourceTree = BUILT_PRODUCTS_DIR; };
22 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = AppDelegate.h; path = exampleapp/AppDelegate.h; sourceTree = ""; };
23 | 13B07FB01A68108700A75B9A /* AppDelegate.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = AppDelegate.m; path = exampleapp/AppDelegate.m; sourceTree = ""; };
24 | 13B07FB21A68108700A75B9A /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = Base.lproj/LaunchScreen.xib; sourceTree = ""; };
25 | 13B07FB51A68108700A75B9A /* Images.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Images.xcassets; path = exampleapp/Images.xcassets; sourceTree = ""; };
26 | 13B07FB61A68108700A75B9A /* Info.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = Info.plist; path = exampleapp/Info.plist; sourceTree = ""; };
27 | 13B07FB71A68108700A75B9A /* main.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = main.m; path = exampleapp/main.m; sourceTree = ""; };
28 | 58EEBF8E8E6FB1BC6CAF49B5 /* libPods-exampleapp.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-exampleapp.a"; sourceTree = BUILT_PRODUCTS_DIR; };
29 | 6C2E3173556A471DD304B334 /* Pods-exampleapp.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-exampleapp.debug.xcconfig"; path = "Target Support Files/Pods-exampleapp/Pods-exampleapp.debug.xcconfig"; sourceTree = ""; };
30 | 7A4D352CD337FB3A3BF06240 /* Pods-exampleapp.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-exampleapp.release.xcconfig"; path = "Target Support Files/Pods-exampleapp/Pods-exampleapp.release.xcconfig"; sourceTree = ""; };
31 | AA286B85B6C04FC6940260E9 /* SplashScreen.storyboard */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.storyboard; name = SplashScreen.storyboard; path = exampleapp/SplashScreen.storyboard; sourceTree = ""; };
32 | BB2F792C24A3F905000567C9 /* Expo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; path = Expo.plist; sourceTree = ""; };
33 | ED297162215061F000B7C4FE /* JavaScriptCore.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = JavaScriptCore.framework; path = System/Library/Frameworks/JavaScriptCore.framework; sourceTree = SDKROOT; };
34 | 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; };
35 | /* End PBXFileReference section */
36 |
37 | /* Begin PBXFrameworksBuildPhase section */
38 | 13B07F8C1A680F5B00A75B9A /* Frameworks */ = {
39 | isa = PBXFrameworksBuildPhase;
40 | buildActionMask = 2147483647;
41 | files = (
42 | 96905EF65AED1B983A6B3ABC /* libPods-exampleapp.a in Frameworks */,
43 | );
44 | runOnlyForDeploymentPostprocessing = 0;
45 | };
46 | /* End PBXFrameworksBuildPhase section */
47 |
48 | /* Begin PBXGroup section */
49 | 13B07FAE1A68108700A75B9A /* exampleapp */ = {
50 | isa = PBXGroup;
51 | children = (
52 | BB2F792B24A3F905000567C9 /* Supporting */,
53 | 008F07F21AC5B25A0029DE68 /* main.jsbundle */,
54 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */,
55 | 13B07FB01A68108700A75B9A /* AppDelegate.m */,
56 | 13B07FB51A68108700A75B9A /* Images.xcassets */,
57 | 13B07FB61A68108700A75B9A /* Info.plist */,
58 | 13B07FB11A68108700A75B9A /* LaunchScreen.xib */,
59 | 13B07FB71A68108700A75B9A /* main.m */,
60 | AA286B85B6C04FC6940260E9 /* SplashScreen.storyboard */,
61 | );
62 | name = exampleapp;
63 | sourceTree = "";
64 | };
65 | 2D16E6871FA4F8E400B85C8A /* Frameworks */ = {
66 | isa = PBXGroup;
67 | children = (
68 | ED297162215061F000B7C4FE /* JavaScriptCore.framework */,
69 | ED2971642150620600B7C4FE /* JavaScriptCore.framework */,
70 | 58EEBF8E8E6FB1BC6CAF49B5 /* libPods-exampleapp.a */,
71 | );
72 | name = Frameworks;
73 | sourceTree = "";
74 | };
75 | 832341AE1AAA6A7D00B99B32 /* Libraries */ = {
76 | isa = PBXGroup;
77 | children = (
78 | );
79 | name = Libraries;
80 | sourceTree = "";
81 | };
82 | 83CBB9F61A601CBA00E9B192 = {
83 | isa = PBXGroup;
84 | children = (
85 | 13B07FAE1A68108700A75B9A /* exampleapp */,
86 | 832341AE1AAA6A7D00B99B32 /* Libraries */,
87 | 83CBBA001A601CBA00E9B192 /* Products */,
88 | 2D16E6871FA4F8E400B85C8A /* Frameworks */,
89 | D65327D7A22EEC0BE12398D9 /* Pods */,
90 | );
91 | indentWidth = 2;
92 | sourceTree = "";
93 | tabWidth = 2;
94 | usesTabs = 0;
95 | };
96 | 83CBBA001A601CBA00E9B192 /* Products */ = {
97 | isa = PBXGroup;
98 | children = (
99 | 13B07F961A680F5B00A75B9A /* exampleapp.app */,
100 | );
101 | name = Products;
102 | sourceTree = "";
103 | };
104 | BB2F792B24A3F905000567C9 /* Supporting */ = {
105 | isa = PBXGroup;
106 | children = (
107 | BB2F792C24A3F905000567C9 /* Expo.plist */,
108 | );
109 | name = Supporting;
110 | path = exampleapp/Supporting;
111 | sourceTree = "";
112 | };
113 | D65327D7A22EEC0BE12398D9 /* Pods */ = {
114 | isa = PBXGroup;
115 | children = (
116 | 6C2E3173556A471DD304B334 /* Pods-exampleapp.debug.xcconfig */,
117 | 7A4D352CD337FB3A3BF06240 /* Pods-exampleapp.release.xcconfig */,
118 | );
119 | path = Pods;
120 | sourceTree = "";
121 | };
122 | /* End PBXGroup section */
123 |
124 | /* Begin PBXNativeTarget section */
125 | 13B07F861A680F5B00A75B9A /* exampleapp */ = {
126 | isa = PBXNativeTarget;
127 | buildConfigurationList = 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "exampleapp" */;
128 | buildPhases = (
129 | 08A4A3CD28434E44B6B9DE2E /* [CP] Check Pods Manifest.lock */,
130 | FD10A7F022414F080027D42C /* Start Packager */,
131 | 13B07F871A680F5B00A75B9A /* Sources */,
132 | 13B07F8C1A680F5B00A75B9A /* Frameworks */,
133 | 13B07F8E1A680F5B00A75B9A /* Resources */,
134 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */,
135 | 800E24972A6A228C8D4807E9 /* [CP] Copy Pods Resources */,
136 | );
137 | buildRules = (
138 | );
139 | dependencies = (
140 | );
141 | name = exampleapp;
142 | productName = exampleapp;
143 | productReference = 13B07F961A680F5B00A75B9A /* exampleapp.app */;
144 | productType = "com.apple.product-type.application";
145 | };
146 | /* End PBXNativeTarget section */
147 |
148 | /* Begin PBXProject section */
149 | 83CBB9F71A601CBA00E9B192 /* Project object */ = {
150 | isa = PBXProject;
151 | attributes = {
152 | LastUpgradeCheck = 1130;
153 | TargetAttributes = {
154 | 13B07F861A680F5B00A75B9A = {
155 | LastSwiftMigration = 1120;
156 | };
157 | };
158 | };
159 | buildConfigurationList = 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "exampleapp" */;
160 | compatibilityVersion = "Xcode 3.2";
161 | developmentRegion = en;
162 | hasScannedForEncodings = 0;
163 | knownRegions = (
164 | en,
165 | Base,
166 | );
167 | mainGroup = 83CBB9F61A601CBA00E9B192;
168 | productRefGroup = 83CBBA001A601CBA00E9B192 /* Products */;
169 | projectDirPath = "";
170 | projectRoot = "";
171 | targets = (
172 | 13B07F861A680F5B00A75B9A /* exampleapp */,
173 | );
174 | };
175 | /* End PBXProject section */
176 |
177 | /* Begin PBXResourcesBuildPhase section */
178 | 13B07F8E1A680F5B00A75B9A /* Resources */ = {
179 | isa = PBXResourcesBuildPhase;
180 | buildActionMask = 2147483647;
181 | files = (
182 | BB2F792D24A3F905000567C9 /* Expo.plist in Resources */,
183 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */,
184 | 13B07FBD1A68108700A75B9A /* LaunchScreen.xib in Resources */,
185 | 3E461D99554A48A4959DE609 /* SplashScreen.storyboard in Resources */,
186 | );
187 | runOnlyForDeploymentPostprocessing = 0;
188 | };
189 | /* End PBXResourcesBuildPhase section */
190 |
191 | /* Begin PBXShellScriptBuildPhase section */
192 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */ = {
193 | isa = PBXShellScriptBuildPhase;
194 | buildActionMask = 2147483647;
195 | files = (
196 | );
197 | inputPaths = (
198 | );
199 | name = "Bundle React Native code and images";
200 | outputPaths = (
201 | );
202 | runOnlyForDeploymentPostprocessing = 0;
203 | shellPath = /bin/sh;
204 | shellScript = "export NODE_BINARY=node\n../node_modules/react-native/scripts/react-native-xcode.sh\n../node_modules/expo-constants/scripts/get-app-config-ios.sh\n../node_modules/expo-updates/scripts/create-manifest-ios.sh\n";
205 | };
206 | 08A4A3CD28434E44B6B9DE2E /* [CP] Check Pods Manifest.lock */ = {
207 | isa = PBXShellScriptBuildPhase;
208 | buildActionMask = 2147483647;
209 | files = (
210 | );
211 | inputFileListPaths = (
212 | );
213 | inputPaths = (
214 | "${PODS_PODFILE_DIR_PATH}/Podfile.lock",
215 | "${PODS_ROOT}/Manifest.lock",
216 | );
217 | name = "[CP] Check Pods Manifest.lock";
218 | outputFileListPaths = (
219 | );
220 | outputPaths = (
221 | "$(DERIVED_FILE_DIR)/Pods-exampleapp-checkManifestLockResult.txt",
222 | );
223 | runOnlyForDeploymentPostprocessing = 0;
224 | shellPath = /bin/sh;
225 | shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
226 | showEnvVarsInLog = 0;
227 | };
228 | 800E24972A6A228C8D4807E9 /* [CP] Copy Pods Resources */ = {
229 | isa = PBXShellScriptBuildPhase;
230 | buildActionMask = 2147483647;
231 | files = (
232 | );
233 | inputPaths = (
234 | "${PODS_ROOT}/Target Support Files/Pods-exampleapp/Pods-exampleapp-resources.sh",
235 | "${PODS_CONFIGURATION_BUILD_DIR}/React-Core/AccessibilityResources.bundle",
236 | );
237 | name = "[CP] Copy Pods Resources";
238 | outputPaths = (
239 | "${TARGET_BUILD_DIR}/${UNLOCALIZED_RESOURCES_FOLDER_PATH}/AccessibilityResources.bundle",
240 | );
241 | runOnlyForDeploymentPostprocessing = 0;
242 | shellPath = /bin/sh;
243 | shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-exampleapp/Pods-exampleapp-resources.sh\"\n";
244 | showEnvVarsInLog = 0;
245 | };
246 | FD10A7F022414F080027D42C /* Start Packager */ = {
247 | isa = PBXShellScriptBuildPhase;
248 | buildActionMask = 2147483647;
249 | files = (
250 | );
251 | inputFileListPaths = (
252 | );
253 | inputPaths = (
254 | );
255 | name = "Start Packager";
256 | outputFileListPaths = (
257 | );
258 | outputPaths = (
259 | );
260 | runOnlyForDeploymentPostprocessing = 0;
261 | shellPath = /bin/sh;
262 | shellScript = "export RCT_METRO_PORT=\"${RCT_METRO_PORT:=8081}\"\necho \"export RCT_METRO_PORT=${RCT_METRO_PORT}\" > \"${SRCROOT}/../node_modules/react-native/scripts/.packager.env\"\nif [ -z \"${RCT_NO_LAUNCH_PACKAGER+xxx}\" ] ; then\n if nc -w 5 -z localhost ${RCT_METRO_PORT} ; then\n if ! curl -s \"http://localhost:${RCT_METRO_PORT}/status\" | grep -q \"packager-status:running\" ; then\n echo \"Port ${RCT_METRO_PORT} already in use, packager is either not running or not running correctly\"\n exit 2\n fi\n else\n open \"$SRCROOT/../node_modules/react-native/scripts/launchPackager.command\" || echo \"Can't start packager automatically\"\n fi\nfi\n";
263 | showEnvVarsInLog = 0;
264 | };
265 | /* End PBXShellScriptBuildPhase section */
266 |
267 | /* Begin PBXSourcesBuildPhase section */
268 | 13B07F871A680F5B00A75B9A /* Sources */ = {
269 | isa = PBXSourcesBuildPhase;
270 | buildActionMask = 2147483647;
271 | files = (
272 | 13B07FBC1A68108700A75B9A /* AppDelegate.m in Sources */,
273 | 13B07FC11A68108700A75B9A /* main.m in Sources */,
274 | );
275 | runOnlyForDeploymentPostprocessing = 0;
276 | };
277 | /* End PBXSourcesBuildPhase section */
278 |
279 | /* Begin PBXVariantGroup section */
280 | 13B07FB11A68108700A75B9A /* LaunchScreen.xib */ = {
281 | isa = PBXVariantGroup;
282 | children = (
283 | 13B07FB21A68108700A75B9A /* Base */,
284 | );
285 | name = LaunchScreen.xib;
286 | path = exampleapp;
287 | sourceTree = "";
288 | };
289 | /* End PBXVariantGroup section */
290 |
291 | /* Begin XCBuildConfiguration section */
292 | 13B07F941A680F5B00A75B9A /* Debug */ = {
293 | isa = XCBuildConfiguration;
294 | baseConfigurationReference = 6C2E3173556A471DD304B334 /* Pods-exampleapp.debug.xcconfig */;
295 | buildSettings = {
296 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
297 | CLANG_ENABLE_MODULES = YES;
298 | CURRENT_PROJECT_VERSION = 1;
299 | ENABLE_BITCODE = NO;
300 | "EXCLUDED_ARCHS[sdk=iphonesimulator*]" = arm64;
301 | GCC_PREPROCESSOR_DEFINITIONS = (
302 | "$(inherited)",
303 | "FB_SONARKIT_ENABLED=1",
304 | );
305 | INFOPLIST_FILE = exampleapp/Info.plist;
306 | IPHONEOS_DEPLOYMENT_TARGET = 11.0;
307 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
308 | OTHER_LDFLAGS = (
309 | "$(inherited)",
310 | "-ObjC",
311 | "-lc++",
312 | );
313 | PRODUCT_BUNDLE_IDENTIFIER = org.name.exampleapp;
314 | PRODUCT_NAME = exampleapp;
315 | SWIFT_OPTIMIZATION_LEVEL = "-Onone";
316 | SWIFT_VERSION = 5.0;
317 | VERSIONING_SYSTEM = "apple-generic";
318 | };
319 | name = Debug;
320 | };
321 | 13B07F951A680F5B00A75B9A /* Release */ = {
322 | isa = XCBuildConfiguration;
323 | baseConfigurationReference = 7A4D352CD337FB3A3BF06240 /* Pods-exampleapp.release.xcconfig */;
324 | buildSettings = {
325 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
326 | CLANG_ENABLE_MODULES = YES;
327 | CURRENT_PROJECT_VERSION = 1;
328 | "EXCLUDED_ARCHS[sdk=iphonesimulator*]" = arm64;
329 | INFOPLIST_FILE = exampleapp/Info.plist;
330 | IPHONEOS_DEPLOYMENT_TARGET = 11.0;
331 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
332 | OTHER_LDFLAGS = (
333 | "$(inherited)",
334 | "-ObjC",
335 | "-lc++",
336 | );
337 | PRODUCT_BUNDLE_IDENTIFIER = org.name.exampleapp;
338 | PRODUCT_NAME = exampleapp;
339 | SWIFT_VERSION = 5.0;
340 | VERSIONING_SYSTEM = "apple-generic";
341 | };
342 | name = Release;
343 | };
344 | 83CBBA201A601CBA00E9B192 /* Debug */ = {
345 | isa = XCBuildConfiguration;
346 | buildSettings = {
347 | ALWAYS_SEARCH_USER_PATHS = NO;
348 | CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES;
349 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
350 | CLANG_CXX_LIBRARY = "libc++";
351 | CLANG_ENABLE_MODULES = YES;
352 | CLANG_ENABLE_OBJC_ARC = YES;
353 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
354 | CLANG_WARN_BOOL_CONVERSION = YES;
355 | CLANG_WARN_COMMA = YES;
356 | CLANG_WARN_CONSTANT_CONVERSION = YES;
357 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
358 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
359 | CLANG_WARN_EMPTY_BODY = YES;
360 | CLANG_WARN_ENUM_CONVERSION = YES;
361 | CLANG_WARN_INFINITE_RECURSION = YES;
362 | CLANG_WARN_INT_CONVERSION = YES;
363 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
364 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
365 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
366 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
367 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
368 | CLANG_WARN_STRICT_PROTOTYPES = YES;
369 | CLANG_WARN_SUSPICIOUS_MOVE = YES;
370 | CLANG_WARN_UNREACHABLE_CODE = YES;
371 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
372 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
373 | COPY_PHASE_STRIP = NO;
374 | ENABLE_STRICT_OBJC_MSGSEND = YES;
375 | ENABLE_TESTABILITY = YES;
376 | GCC_C_LANGUAGE_STANDARD = gnu99;
377 | GCC_DYNAMIC_NO_PIC = NO;
378 | GCC_NO_COMMON_BLOCKS = YES;
379 | GCC_OPTIMIZATION_LEVEL = 0;
380 | GCC_PREPROCESSOR_DEFINITIONS = (
381 | "DEBUG=1",
382 | "$(inherited)",
383 | );
384 | GCC_SYMBOLS_PRIVATE_EXTERN = NO;
385 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
386 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
387 | GCC_WARN_UNDECLARED_SELECTOR = YES;
388 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
389 | GCC_WARN_UNUSED_FUNCTION = YES;
390 | GCC_WARN_UNUSED_VARIABLE = YES;
391 | IPHONEOS_DEPLOYMENT_TARGET = 10.0;
392 | LD_RUNPATH_SEARCH_PATHS = "/usr/lib/swift $(inherited)";
393 | LIBRARY_SEARCH_PATHS = (
394 | "\"$(TOOLCHAIN_DIR)/usr/lib/swift/$(PLATFORM_NAME)\"",
395 | "\"$(TOOLCHAIN_DIR)/usr/lib/swift-5.0/$(PLATFORM_NAME)\"",
396 | "\"$(inherited)\"",
397 | );
398 | MTL_ENABLE_DEBUG_INFO = YES;
399 | ONLY_ACTIVE_ARCH = YES;
400 | SDKROOT = iphoneos;
401 | };
402 | name = Debug;
403 | };
404 | 83CBBA211A601CBA00E9B192 /* Release */ = {
405 | isa = XCBuildConfiguration;
406 | buildSettings = {
407 | ALWAYS_SEARCH_USER_PATHS = NO;
408 | CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES;
409 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
410 | CLANG_CXX_LIBRARY = "libc++";
411 | CLANG_ENABLE_MODULES = YES;
412 | CLANG_ENABLE_OBJC_ARC = YES;
413 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
414 | CLANG_WARN_BOOL_CONVERSION = YES;
415 | CLANG_WARN_COMMA = YES;
416 | CLANG_WARN_CONSTANT_CONVERSION = YES;
417 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
418 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
419 | CLANG_WARN_EMPTY_BODY = YES;
420 | CLANG_WARN_ENUM_CONVERSION = YES;
421 | CLANG_WARN_INFINITE_RECURSION = YES;
422 | CLANG_WARN_INT_CONVERSION = YES;
423 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
424 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
425 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
426 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
427 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
428 | CLANG_WARN_STRICT_PROTOTYPES = YES;
429 | CLANG_WARN_SUSPICIOUS_MOVE = YES;
430 | CLANG_WARN_UNREACHABLE_CODE = YES;
431 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
432 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
433 | COPY_PHASE_STRIP = YES;
434 | ENABLE_NS_ASSERTIONS = NO;
435 | ENABLE_STRICT_OBJC_MSGSEND = YES;
436 | GCC_C_LANGUAGE_STANDARD = gnu99;
437 | GCC_NO_COMMON_BLOCKS = YES;
438 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
439 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
440 | GCC_WARN_UNDECLARED_SELECTOR = YES;
441 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
442 | GCC_WARN_UNUSED_FUNCTION = YES;
443 | GCC_WARN_UNUSED_VARIABLE = YES;
444 | IPHONEOS_DEPLOYMENT_TARGET = 10.0;
445 | LD_RUNPATH_SEARCH_PATHS = "/usr/lib/swift $(inherited)";
446 | LIBRARY_SEARCH_PATHS = (
447 | "\"$(TOOLCHAIN_DIR)/usr/lib/swift/$(PLATFORM_NAME)\"",
448 | "\"$(TOOLCHAIN_DIR)/usr/lib/swift-5.0/$(PLATFORM_NAME)\"",
449 | "\"$(inherited)\"",
450 | );
451 | MTL_ENABLE_DEBUG_INFO = NO;
452 | SDKROOT = iphoneos;
453 | VALIDATE_PRODUCT = YES;
454 | };
455 | name = Release;
456 | };
457 | /* End XCBuildConfiguration section */
458 |
459 | /* Begin XCConfigurationList section */
460 | 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "exampleapp" */ = {
461 | isa = XCConfigurationList;
462 | buildConfigurations = (
463 | 13B07F941A680F5B00A75B9A /* Debug */,
464 | 13B07F951A680F5B00A75B9A /* Release */,
465 | );
466 | defaultConfigurationIsVisible = 0;
467 | defaultConfigurationName = Release;
468 | };
469 | 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "exampleapp" */ = {
470 | isa = XCConfigurationList;
471 | buildConfigurations = (
472 | 83CBBA201A601CBA00E9B192 /* Debug */,
473 | 83CBBA211A601CBA00E9B192 /* Release */,
474 | );
475 | defaultConfigurationIsVisible = 0;
476 | defaultConfigurationName = Release;
477 | };
478 | /* End XCConfigurationList section */
479 | };
480 | rootObject = 83CBB9F71A601CBA00E9B192 /* Project object */;
481 | }
482 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp.xcodeproj/xcshareddata/xcschemes/exampleapp.xcscheme:
--------------------------------------------------------------------------------
1 |
2 |
5 |
8 |
9 |
15 |
21 |
22 |
23 |
24 |
25 |
30 |
31 |
33 |
39 |
40 |
41 |
42 |
43 |
53 |
55 |
61 |
62 |
63 |
64 |
70 |
72 |
78 |
79 |
80 |
81 |
83 |
84 |
87 |
88 |
89 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/AppDelegate.h:
--------------------------------------------------------------------------------
1 | #import
2 | #import
3 | #import
4 | #import
5 |
6 | #import
7 |
8 | @interface AppDelegate : UMAppDelegateWrapper
9 |
10 | @end
11 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/AppDelegate.m:
--------------------------------------------------------------------------------
1 | #import "AppDelegate.h"
2 |
3 | #import
4 | #import
5 | #import
6 | #import
7 |
8 | #import
9 | #import
10 | #import
11 | #import
12 | #import
13 |
14 | #if defined(FB_SONARKIT_ENABLED) && __has_include()
15 | #import
16 | #import
17 | #import
18 | #import
19 | #import
20 | #import
21 |
22 | static void InitializeFlipper(UIApplication *application) {
23 | FlipperClient *client = [FlipperClient sharedClient];
24 | SKDescriptorMapper *layoutDescriptorMapper = [[SKDescriptorMapper alloc] initWithDefaults];
25 | [client addPlugin:[[FlipperKitLayoutPlugin alloc] initWithRootNode:application withDescriptorMapper:layoutDescriptorMapper]];
26 | [client addPlugin:[[FKUserDefaultsPlugin alloc] initWithSuiteName:nil]];
27 | [client addPlugin:[FlipperKitReactPlugin new]];
28 | [client addPlugin:[[FlipperKitNetworkPlugin alloc] initWithNetworkAdapter:[SKIOSNetworkAdapter new]]];
29 | [client start];
30 | }
31 | #endif
32 |
33 | @interface AppDelegate ()
34 |
35 | @property (nonatomic, strong) UMModuleRegistryAdapter *moduleRegistryAdapter;
36 | @property (nonatomic, strong) NSDictionary *launchOptions;
37 |
38 | @end
39 |
40 | @implementation AppDelegate
41 |
42 | - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
43 | {
44 | #if defined(FB_SONARKIT_ENABLED) && __has_include()
45 | InitializeFlipper(application);
46 | #endif
47 |
48 | self.moduleRegistryAdapter = [[UMModuleRegistryAdapter alloc] initWithModuleRegistryProvider:[[UMModuleRegistryProvider alloc] init]];
49 | self.launchOptions = launchOptions;
50 | self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
51 | #ifdef DEBUG
52 | [self initializeReactNativeApp];
53 | #else
54 | EXUpdatesAppController *controller = [EXUpdatesAppController sharedInstance];
55 | controller.delegate = self;
56 | [controller startAndShowLaunchScreen:self.window];
57 | #endif
58 |
59 | [super application:application didFinishLaunchingWithOptions:launchOptions];
60 |
61 | return YES;
62 | }
63 |
64 | - (RCTBridge *)initializeReactNativeApp
65 | {
66 | RCTBridge *bridge = [[RCTBridge alloc] initWithDelegate:self launchOptions:self.launchOptions];
67 | RCTRootView *rootView = [[RCTRootView alloc] initWithBridge:bridge moduleName:@"main" initialProperties:nil];
68 | rootView.backgroundColor = [[UIColor alloc] initWithRed:1.0f green:1.0f blue:1.0f alpha:1];
69 |
70 | UIViewController *rootViewController = [UIViewController new];
71 | rootViewController.view = rootView;
72 | self.window.rootViewController = rootViewController;
73 | [self.window makeKeyAndVisible];
74 |
75 | return bridge;
76 | }
77 |
78 | - (NSArray> *)extraModulesForBridge:(RCTBridge *)bridge
79 | {
80 | NSArray> *extraModules = [_moduleRegistryAdapter extraModulesForBridge:bridge];
81 | // If you'd like to export some custom RCTBridgeModules that are not Expo modules, add them here!
82 | return extraModules;
83 | }
84 |
85 | - (NSURL *)sourceURLForBridge:(RCTBridge *)bridge {
86 | #ifdef DEBUG
87 | return [[RCTBundleURLProvider sharedSettings] jsBundleURLForBundleRoot:@"index" fallbackResource:nil];
88 | #else
89 | return [[EXUpdatesAppController sharedInstance] launchAssetUrl];
90 | #endif
91 | }
92 |
93 | - (void)appController:(EXUpdatesAppController *)appController didStartWithSuccess:(BOOL)success {
94 | appController.bridge = [self initializeReactNativeApp];
95 | EXSplashScreenService *splashScreenService = (EXSplashScreenService *)[UMModuleRegistryProvider getSingletonModuleForClass:[EXSplashScreenService class]];
96 | [splashScreenService showSplashScreenFor:self.window.rootViewController];
97 | }
98 |
99 | // Linking API
100 | - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url options:(NSDictionary *)options {
101 | return [RCTLinkingManager application:application openURL:url options:options];
102 | }
103 |
104 | // Universal Links
105 | - (BOOL)application:(UIApplication *)application continueUserActivity:(nonnull NSUserActivity *)userActivity restorationHandler:(nonnull void (^)(NSArray> * _Nullable))restorationHandler {
106 | return [RCTLinkingManager application:application
107 | continueUserActivity:userActivity
108 | restorationHandler:restorationHandler];
109 | }
110 |
111 | @end
112 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/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 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/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 | }
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/Images.xcassets/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "info" : {
3 | "version" : 1,
4 | "author" : "xcode"
5 | }
6 | }
7 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/Images.xcassets/SplashScreen.imageset/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "images": [
3 | {
4 | "idiom": "universal",
5 | "filename": "splashscreen.png",
6 | "scale": "1x"
7 | },
8 | {
9 | "idiom": "universal",
10 | "scale": "2x"
11 | },
12 | {
13 | "idiom": "universal",
14 | "scale": "3x"
15 | }
16 | ],
17 | "info": {
18 | "version": 1,
19 | "author": "xcode"
20 | }
21 | }
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/Images.xcassets/SplashScreen.imageset/splashscreen.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/ios/exampleapp/Images.xcassets/SplashScreen.imageset/splashscreen.png
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/Images.xcassets/SplashScreenBackground.imageset/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "images": [
3 | {
4 | "idiom": "universal",
5 | "filename": "background.png",
6 | "scale": "1x"
7 | },
8 | {
9 | "idiom": "universal",
10 | "scale": "2x"
11 | },
12 | {
13 | "idiom": "universal",
14 | "scale": "3x"
15 | }
16 | ],
17 | "info": {
18 | "version": 1,
19 | "author": "xcode"
20 | }
21 | }
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/Images.xcassets/SplashScreenBackground.imageset/background.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/example-app/ios/exampleapp/Images.xcassets/SplashScreenBackground.imageset/background.png
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/Info.plist:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | CFBundleDevelopmentRegion
6 | en
7 | CFBundleDisplayName
8 | example-app
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 | NSAppTransportSecurity
28 |
29 | NSAllowsArbitraryLoads
30 |
31 | NSExceptionDomains
32 |
33 | localhost
34 |
35 | NSExceptionAllowsInsecureHTTPLoads
36 |
37 |
38 |
39 |
40 | NSLocationWhenInUseUsageDescription
41 |
42 | UILaunchStoryboardName
43 | SplashScreen
44 | UIRequiredDeviceCapabilities
45 |
46 | armv7
47 |
48 | UISupportedInterfaceOrientations
49 |
50 | UIInterfaceOrientationPortrait
51 | UIInterfaceOrientationLandscapeLeft
52 | UIInterfaceOrientationLandscapeRight
53 |
54 | UIViewControllerBasedStatusBarAppearance
55 |
56 | UIStatusBarStyle
57 | UIStatusBarStyleDefault
58 |
59 |
60 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/SplashScreen.storyboard:
--------------------------------------------------------------------------------
1 |
2 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
31 |
39 |
40 |
41 |
42 |
53 |
54 |
55 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |
74 |
75 |
76 |
77 |
78 |
79 |
80 |
81 |
82 |
83 |
84 |
85 |
86 |
87 |
88 |
89 |
90 |
91 |
92 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/Supporting/Expo.plist:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | EXUpdatesSDKVersion
6 | YOUR-APP-SDK-VERSION-HERE
7 | EXUpdatesURL
8 | YOUR-APP-URL-HERE
9 |
10 |
11 |
--------------------------------------------------------------------------------
/example-app/ios/exampleapp/main.m:
--------------------------------------------------------------------------------
1 | #import
2 |
3 | #import "AppDelegate.h"
4 |
5 | int main(int argc, char * argv[]) {
6 | @autoreleasepool {
7 | return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
8 | }
9 | }
10 |
11 |
--------------------------------------------------------------------------------
/example-app/metro.config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | transformer: {
3 | assetPlugins: ['expo-asset/tools/hashAssetFiles'],
4 | },
5 | };
6 |
--------------------------------------------------------------------------------
/example-app/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "main": "index.js",
3 | "scripts": {
4 | "android": "react-native run-android",
5 | "ios": "react-native run-ios",
6 | "web": "expo start --web",
7 | "start": "react-native start",
8 | "test": "jest"
9 | },
10 | "dependencies": {
11 | "@react-native-community/masked-view": "^0.1.10",
12 | "@react-navigation/native": "^5.9.3",
13 | "@react-navigation/stack": "^5.14.3",
14 | "expo": "~40.0.0",
15 | "expo-splash-screen": "~0.8.0",
16 | "expo-status-bar": "~1.0.3",
17 | "expo-updates": "~0.4.0",
18 | "nanoid": "^3.1.20",
19 | "react": "16.13.1",
20 | "react-dom": "16.13.1",
21 | "react-native": "~0.63.4",
22 | "react-native-elements": "^3.3.0",
23 | "react-native-gesture-handler": "^1.10.3",
24 | "react-native-get-random-values": "^1.6.0",
25 | "react-native-reanimated": "^2.0.0",
26 | "react-native-safe-area-context": "^3.2.0",
27 | "react-native-screens": "^2.18.1",
28 | "react-native-unimodules": "~0.12.0",
29 | "react-native-vector-icons": "^8.1.0",
30 | "react-native-web": "~0.13.12",
31 | "react-redux": "^7.2.2",
32 | "redux": "^4.0.5"
33 | },
34 | "devDependencies": {
35 | "@babel/core": "~7.9.0",
36 | "babel-jest": "~25.2.6",
37 | "jest": "~25.2.6",
38 | "react-test-renderer": "~16.13.1"
39 | },
40 | "jest": {
41 | "preset": "react-native"
42 | },
43 | "private": true
44 | }
45 |
--------------------------------------------------------------------------------
/example-app/packages.js:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 | import * as ReactNative from "react-native";
3 | import * as ReactRedux from "react-redux";
4 | import * as ReactNativeElements from "react-native-elements";
5 | import * as ReactNavigationNative from "@react-navigation/native";
6 | import * as Icons from "react-native-vector-icons";
7 | const Packages = {
8 | "react": () => React,
9 | "react-native": () => ReactNative,
10 | "react-redux": () => ReactRedux,
11 | "react-native-elements": () => ReactNativeElements,
12 | "@react-navigation/native": () => ReactNavigationNative,
13 | "react-native-vector-icons": () => Icons
14 | }
15 |
16 |
17 | const fromPairs = (pairs) => Object.assign({}, ...pairs.map(([k, v]) => ({ [k]: v })));
18 | const AllPackages = fromPairs(
19 | Object.keys(Packages).map(k => [k, () => ({ exports: Packages[k]() })])
20 | );
21 |
22 | export default AllPackages
--------------------------------------------------------------------------------
/remote-components-demo.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/sarathkcm/react-native-remote-components/587767e14ecbe1a1a4b0bde49900ae19b7d7ac74/remote-components-demo.gif
--------------------------------------------------------------------------------
/remote-components/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | dist
--------------------------------------------------------------------------------
/remote-components/babel.config.js:
--------------------------------------------------------------------------------
1 | const presets = [
2 | [
3 | "@babel/preset-env", { modules: false }],
4 | "@babel/preset-react"
5 | ]
6 | const plugins = []
7 |
8 | plugins.push(["@babel/plugin-proposal-class-properties"])
9 |
10 | module.exports = {
11 | presets,
12 | plugins
13 | }
--------------------------------------------------------------------------------
/remote-components/components/counter.js:
--------------------------------------------------------------------------------
1 | import React, { useState } from 'react'
2 | import { StyleSheet, View } from 'react-native'
3 | import { Button, Card, Text } from 'react-native-elements';
4 |
5 | const Counter = () => {
6 | const [count, setCount] = useState(0)
7 | return (
8 |
9 |
10 |
13 |
14 | )
15 | }
16 |
17 | export default Counter
18 |
19 | const styles = StyleSheet.create({})
20 |
--------------------------------------------------------------------------------
/remote-components/components/new-page.js:
--------------------------------------------------------------------------------
1 | import { useNavigation } from '@react-navigation/native'
2 | import React from 'react'
3 | import { StyleSheet, View } from 'react-native'
4 | import { Text, Button } from 'react-native-elements';
5 | import TodoStatus from "./todo-status";
6 | import TodoStatus2 from "./todo-status2";
7 |
8 | const NewPage = () => {
9 | const navigation = useNavigation();
10 | return (
11 |
12 |
13 | This entire page in navigation is a remote component and it comes with other server components embedded in it as well
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 | )
22 | }
23 |
24 | export default NewPage
25 |
26 | const styles = StyleSheet.create({})
27 |
--------------------------------------------------------------------------------
/remote-components/components/next-page-button.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import { StyleSheet, View } from 'react-native'
3 | import { useNavigation } from '@react-navigation/native'
4 | import { Button, Card, Text } from 'react-native-elements'
5 | import Icons from "react-native-vector-icons";
6 |
7 | const NextPageButton = () => {
8 | const navigation = useNavigation();
9 | return (
10 |
11 |
12 |
18 |
19 | )
20 | }
21 |
22 | export default NextPageButton
23 |
24 | const styles = StyleSheet.create({})
25 |
--------------------------------------------------------------------------------
/remote-components/components/todo-status.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import { View } from 'react-native'
3 | import { Button, Card, Text } from 'react-native-elements'
4 | import { useDispatch, useSelector } from 'react-redux';
5 |
6 | const Status = ({ flow }) => {
7 | const todos = useSelector(state => state.todos);
8 | const dispatch = useDispatch();
9 | return (
10 |
11 |
12 |
13 | • There are {todos.length} Todos
14 | • Completed {todos.filter(a => a.completed).length} Todos
15 | • Using redux hooks
16 |
17 | {flow === "vertical" && }
18 |
27 |
28 |
29 | )
30 | }
31 |
32 | export default Status
33 |
--------------------------------------------------------------------------------
/remote-components/components/todo-status2.js:
--------------------------------------------------------------------------------
1 | import React from 'react'
2 | import { View } from 'react-native'
3 | import { Button, Card, Text } from 'react-native-elements'
4 | import { connect } from 'react-redux';
5 |
6 | const Status = ({ todos, markCompleted, children, flow }) => {
7 | console.log("children", children)
8 | return (
9 |
10 |
11 |
12 | • There are {todos.length} Todos
13 | • Completed {todos.filter(a => a.completed).length} Todos
14 | • Using connect()
15 |
16 | {flow === "vertical" && }
17 |
22 | {children}
23 |
24 |
25 | )
26 | }
27 |
28 | export default connect(
29 | state => ({ todos: state.todos }),
30 | dispatch => ({
31 | markCompleted: (id) => dispatch({
32 | type: "store/toggle-todo",
33 | payload: { id: id, completed: false }
34 | })
35 | })
36 | )(Status)
37 |
--------------------------------------------------------------------------------
/remote-components/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "remote-components",
3 | "version": "1.0.0",
4 | "description": "",
5 | "scripts": {
6 | "start": "http-server ./dist",
7 | "build": "rollup --config ./rollup.config.js",
8 | "watch": "chokidar \"./components/*.js\" -c \"yarn build\""
9 | },
10 | "dependencies": {
11 | "@react-native-community/masked-view": "^0.1.10",
12 | "@react-navigation/native": "^5.9.3",
13 | "@react-navigation/stack": "^5.14.3",
14 | "expo": "~40.0.0",
15 | "expo-splash-screen": "~0.8.0",
16 | "expo-status-bar": "~1.0.3",
17 | "expo-updates": "~0.4.0",
18 | "nanoid": "^3.1.20",
19 | "react": "16.13.1",
20 | "react-dom": "16.13.1",
21 | "react-native": "~0.63.4",
22 | "react-native-elements": "^3.3.0",
23 | "react-native-gesture-handler": "^1.10.3",
24 | "react-native-get-random-values": "^1.6.0",
25 | "react-native-reanimated": "^2.0.0",
26 | "react-native-safe-area-context": "^3.2.0",
27 | "react-native-screens": "^2.18.1",
28 | "react-native-unimodules": "~0.12.0",
29 | "react-native-vector-icons": "^8.1.0",
30 | "react-native-web": "~0.13.12",
31 | "react-redux": "^7.2.2",
32 | "redux": "^4.0.5"
33 | },
34 | "devDependencies": {
35 | "@babel/plugin-proposal-class-properties": "^7.13.0",
36 | "@babel/preset-env": "^7.13.9",
37 | "@babel/preset-react": "^7.12.13",
38 | "babel-core": "^6.26.3",
39 | "babel-plugin-module-resolver": "^4.1.0",
40 | "babel-preset-env": "^1.7.0",
41 | "chokidar-cli": "^2.1.0",
42 | "http-server": "^0.12.3",
43 | "rollup": "^2.40.0",
44 | "rollup-plugin-babel": "^4.4.0",
45 | "rollup-plugin-commonjs": "^10.1.0",
46 | "rollup-plugin-node-resolve": "^5.2.0",
47 | "rollup-plugin-terser": "^7.0.2"
48 | },
49 | "author": "sarathkcm",
50 | "license": "MIT"
51 | }
52 |
--------------------------------------------------------------------------------
/remote-components/rollup.config.js:
--------------------------------------------------------------------------------
1 | import babel from 'rollup-plugin-babel'
2 | import commonjs from 'rollup-plugin-commonjs'
3 | import resolve from 'rollup-plugin-node-resolve'
4 | import { terser } from "rollup-plugin-terser";
5 |
6 | const fs = require("fs");
7 |
8 | const pkg = JSON.parse(require("fs")
9 | .readFileSync(require("path")
10 | .resolve('./package.json'), 'utf-8'));
11 |
12 | const external = Object.keys(pkg.dependencies || {});
13 |
14 | const allComponents = fs.readdirSync("./components");
15 |
16 | const allFiles = allComponents
17 | .filter(a => a.endsWith(".js"))
18 | .map(a => `./components/${a}`)
19 |
20 | const getConfig = (file) => ({
21 | input: file,
22 | output: [
23 | {
24 | dir: "dist",
25 | format: 'cjs',
26 | exports: "auto"
27 | }
28 | ],
29 | plugins: [
30 | resolve(),
31 | babel(),
32 | commonjs(),
33 | terser()
34 | ],
35 | external
36 | })
37 |
38 | export default allFiles.map(getConfig)
--------------------------------------------------------------------------------