Search code examples
react-nativefirebase-cloud-messaging

Android app crashes when FCM received both on background and foreground


As soon as app receives an FCM it crashes, regardless if its open or running in the background. I don`t know where the issue might be (my first rn experiment) but i assume its something to do with the setup im having?

android\build.gradle

buildscript {
    ext {
        buildToolsVersion = "33.0.0"
        minSdkVersion = 21
        compileSdkVersion = 33
        targetSdkVersion = 33
        firebaseVersion = "9.18.0"
        firebaseMessagingVersion = "18.8.0" 
        // We use NDK 23 which has both M1 support and is the side-by-side NDK version from AGP.
        ndkVersion = "23.1.7779620"
    }
    repositories {
        google()
        mavenCentral()
    }
    dependencies {
        classpath("com.android.tools.build:gradle:7.3.1")
        classpath("com.facebook.react:react-native-gradle-plugin")
        classpath("com.google.gms:google-services:4.3.15")
    }
}

android\app\src\main\java\com\usapp\MainActivity.java

    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  package="com.usapp">

  <uses-permission android:name="android.permission.INTERNET" />

  <application
    android:name=".MainApplication"
    android:label="@string/app_name"
    android:icon="@mipmap/ic_launcher"
    android:roundIcon="@mipmap/ic_launcher_round"
    android:allowBackup="false"
    android:theme="@style/AppTheme">
    <activity
      android:name=".MainActivity"
      android:label="@string/app_name"
      android:screenOrientation="portrait"
      android:configChanges="keyboardHidden|orientation|screenSize"
      android:launchMode="singleTask"
      android:windowSoftInputMode="adjustResize"
      android:exported="true">
      <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <category android:name="android.intent.category.LAUNCHER" />
      </intent-filter>
    </activity>
    <!-- [START firebase_service] -->
    <service
      android:name=".java.MyFirebaseMessagingService"
      android:exported="false">
      <intent-filter>
        <action android:name="com.google.firebase.MESSAGING_EVENT" />
      </intent-filter>
    </service>
    <!-- [END firebase_service] -->
  </application>
</manifest>

Solution

  • I reverted a few changes from 3 days ago and it seems to be okay now. Here is the code I hope offer help for some. Me personally have no experiance in gradle, java and all that so im just tapping around in the dark.

    AndroidManifest.xml

    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="com.usapp">
    
        <uses-permission android:name="android.permission.INTERNET" />
    
        <application
          android:name=".MainApplication"
          android:label="@string/app_name"
          android:icon="@mipmap/ic_launcher"
          android:roundIcon="@mipmap/ic_launcher_round"
          android:allowBackup="false"
          android:theme="@style/AppTheme">
          <activity
            android:name=".MainActivity"
            android:label="@string/app_name"
            android:screenOrientation="portrait"
            android:configChanges="keyboardHidden|orientation|screenSize"
            android:launchMode="singleTask"
            android:windowSoftInputMode="adjustResize"
            android:exported="true">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
          </activity>
        </application>
    </manifest>
    

    MainApplication.java

      package com.usapp;
        
        import android.app.Application;
        import com.facebook.react.PackageList;
        import com.facebook.react.ReactApplication;
        import com.facebook.react.ReactNativeHost;
        import com.facebook.react.ReactPackage;
        import com.facebook.react.defaults.DefaultNewArchitectureEntryPoint;
        import com.facebook.react.defaults.DefaultReactNativeHost;
        import com.facebook.soloader.SoLoader;
        import java.util.List;
        
        public class MainApplication extends Application implements ReactApplication {
        
          private final ReactNativeHost mReactNativeHost =
              new DefaultReactNativeHost(this) {
                @Override
                public boolean getUseDeveloperSupport() {
                  return BuildConfig.DEBUG;
                }
        
                @Override
                protected List<ReactPackage> getPackages() {
                  @SuppressWarnings("UnnecessaryLocalVariable")
                  List<ReactPackage> packages = new PackageList(this).getPackages();
                  // Packages that cannot be autolinked yet can be added manually here, for example:
                  // packages.add(new MyReactNativePackage());
                  return packages;
                }
        
                @Override
                protected String getJSMainModuleName() {
                  return "index";
                }
        
                @Override
                protected boolean isNewArchEnabled() {
                  return BuildConfig.IS_NEW_ARCHITECTURE_ENABLED;
                }
        
                @Override
                protected Boolean isHermesEnabled() {
                  return BuildConfig.IS_HERMES_ENABLED;
                }
              };
              
        
          @Override
          public ReactNativeHost getReactNativeHost() {
            return mReactNativeHost;
          }
        
          @Override
          public void onCreate() {
            super.onCreate();
            SoLoader.init(this, /* native exopackage */ false);
            if (BuildConfig.IS_NEW_ARCHITECTURE_ENABLED) {
              // If you opted-in for the New Architecture, we load the native entry point for this app.
              DefaultNewArchitectureEntryPoint.load();
            }
            ReactNativeFlipper.initializeFlipper(this, getReactNativeHost().getReactInstanceManager());
          }
        }
    
    MainActivity.java
    
    package com.usapp;
    
    import android.os.Bundle;
    import com.facebook.react.ReactActivity;
    import com.facebook.react.ReactActivityDelegate;
    import com.facebook.react.defaults.DefaultNewArchitectureEntryPoint;
    import com.facebook.react.defaults.DefaultReactActivityDelegate;
    
    public class MainActivity extends ReactActivity {
    
    @Override
      protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(null);
      }
    
      /**
       * Returns the name of the main component registered from JavaScript. This is used to schedule
       * rendering of the component.
       */
      @Override
      protected String getMainComponentName() {
        return "usapp";
      }
    
      /**
       * Returns the instance of the {@link ReactActivityDelegate}. Here we use a util class {@link
       * DefaultReactActivityDelegate} which allows you to easily enable Fabric and Concurrent React
       * (aka React 18) with two boolean flags.
       */
      @Override
      protected ReactActivityDelegate createReactActivityDelegate() {
        return new DefaultReactActivityDelegate(
            this,
            getMainComponentName(),
            // If you opted-in for the New Architecture, we enable the Fabric Renderer.
            DefaultNewArchitectureEntryPoint.getFabricEnabled(), // fabricEnabled
            // If you opted-in for the New Architecture, we enable Concurrent React (i.e. React 18).
            DefaultNewArchitectureEntryPoint.getConcurrentReactEnabled() // concurrentRootEnabled
            );
      }
    }
    

    app/build.gradle

    apply plugin: "com.android.application"
    apply plugin: "com.facebook.react"
    apply plugin: 'com.google.gms.google-services'
    apply from: "../../node_modules/react-native-vector-icons/fonts.gradle"
    
    
    import com.android.build.OutputFile
    
    /**
     * This is the configuration block to customize your React Native Android app.
     * By default you don't need to apply any configuration, just uncomment the lines you need.
     */
    react {
        /* Folders */
        //   The root of your project, i.e. where "package.json" lives. Default is '..'
        // root = file("../")
        //   The folder where the react-native NPM package is. Default is ../node_modules/react-native
        // reactNativeDir = file("../node_modules/react-native")
        //   The folder where the react-native Codegen package is. Default is ../node_modules/react-native-codegen
        // codegenDir = file("../node_modules/react-native-codegen")
        //   The cli.js file which is the React Native CLI entrypoint. Default is ../node_modules/react-native/cli.js
        // cliFile = file("../node_modules/react-native/cli.js")
    
        /* Variants */
        //   The list of variants to that are debuggable. For those we're going to
        //   skip the bundling of the JS bundle and the assets. By default is just 'debug'.
        //   If you add flavors like lite, prod, etc. you'll have to list your debuggableVariants.
        // debuggableVariants = ["liteDebug", "prodDebug"]
    
        /* Bundling */
        //   A list containing the node command and its flags. Default is just 'node'.
        // nodeExecutableAndArgs = ["node"]
        //
        //   The command to run when bundling. By default is 'bundle'
        // bundleCommand = "ram-bundle"
        //
        //   The path to the CLI configuration file. Default is empty.
        // bundleConfig = file(../rn-cli.config.js)
        //
        //   The name of the generated asset file containing your JS bundle
        // bundleAssetName = "MyApplication.android.bundle"
        //
        //   The entry file for bundle generation. Default is 'index.android.js' or 'index.js'
        // entryFile = file("../js/MyApplication.android.js")
        //
        //   A list of extra flags to pass to the 'bundle' commands.
        //   See https://github.com/react-native-community/cli/blob/main/docs/commands.md#bundle
        // extraPackagerArgs = []
    
        /* Hermes Commands */
        //   The hermes compiler command to run. By default it is 'hermesc'
        // hermesCommand = "$rootDir/my-custom-hermesc/bin/hermesc"
        //
        //   The list of flags to pass to the Hermes compiler. By default is "-O", "-output-source-map"
        // hermesFlags = ["-O", "-output-source-map"]
    }
    
    /**
     * Set this to true to create four separate APKs instead of one,
     * one for each native architecture. This is useful if you don't
     * use App Bundles (https://developer.android.com/guide/app-bundle/)
     * and want to have separate APKs to upload to the Play Store.
     */
    def enableSeparateBuildPerCPUArchitecture = false
    
    /**
     * Set this to true to Run Proguard on Release builds to minify the Java bytecode.
     */
    def enableProguardInReleaseBuilds = false
    
    /**
     * The preferred build flavor of JavaScriptCore (JSC)
     *
     * For example, to use the international variant, you can use:
     * `def jscFlavor = 'org.webkit:android-jsc-intl:+'`
     *
     * The international variant includes ICU i18n library and necessary data
     * allowing to use e.g. `Date.toLocaleString` and `String.localeCompare` that
     * give correct results when using with locales other than en-US. Note that
     * this variant is about 6MiB larger per architecture than default.
     */
    def jscFlavor = 'org.webkit:android-jsc:+'
    
    /**
     * Private function to get the list of Native Architectures you want to build.
     * This reads the value from reactNativeArchitectures in your gradle.properties
     * file and works together with the --active-arch-only flag of react-native run-android.
     */
    def reactNativeArchitectures() {
        def value = project.getProperties().get("reactNativeArchitectures")
        return value ? value.split(",") : ["armeabi-v7a", "x86", "x86_64", "arm64-v8a"]
    }
    
    android {
        ndkVersion rootProject.ext.ndkVersion
    
        compileSdkVersion rootProject.ext.compileSdkVersion
    
        namespace "com.usapp"
        defaultConfig {
            applicationId "com.usapp"
            minSdkVersion rootProject.ext.minSdkVersion
            targetSdkVersion rootProject.ext.targetSdkVersion
            versionCode 1
            versionName "1.0"
        }
    
        splits {
            abi {
                reset()
                enable enableSeparateBuildPerCPUArchitecture
                universalApk false  // If true, also generate a universal APK
                include (*reactNativeArchitectures())
            }
        }
        signingConfigs {
            debug {
                storeFile file('debug.keystore')
                storePassword 'android'
                keyAlias 'androiddebugkey'
                keyPassword 'android'
            }
        }
        buildTypes {
            debug {
                signingConfig signingConfigs.debug
            }
            release {
                // Caution! In production, you need to generate your own keystore file.
                // see https://reactnative.dev/docs/signed-apk-android.
                signingConfig signingConfigs.debug
                minifyEnabled enableProguardInReleaseBuilds
                proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro"
            }
        }
    
        // applicationVariants are e.g. debug, release
        applicationVariants.all { variant ->
            variant.outputs.each { output ->
                // For each separate APK per architecture, set a unique version code as described here:
                // https://developer.android.com/studio/build/configure-apk-splits.html
                // Example: versionCode 1 will generate 1001 for armeabi-v7a, 1002 for x86, etc.
                def versionCodes = ["armeabi-v7a": 1, "x86": 2, "arm64-v8a": 3, "x86_64": 4]
                def abi = output.getFilter(OutputFile.ABI)
                if (abi != null) {  // null for the universal-debug, universal-release variants
                    output.versionCodeOverride =
                            defaultConfig.versionCode * 1000 + versionCodes.get(abi)
                }
    
            }
        }
    }
    
    dependencies {
        // The version of react-native is set by the React Native Gradle Plugin
        implementation("com.facebook.react:react-android")
    implementation 'com.facebook.fresco:animated-gif:2.5.0'
    
    
        implementation("androidx.swiperefreshlayout:swiperefreshlayout:1.0.0")
    
        debugImplementation("com.facebook.flipper:flipper:${FLIPPER_VERSION}")
        debugImplementation("com.facebook.flipper:flipper-network-plugin:${FLIPPER_VERSION}") {
            exclude group:'com.squareup.okhttp3', module:'okhttp'
        }
    
        debugImplementation("com.facebook.flipper:flipper-fresco-plugin:${FLIPPER_VERSION}")
        if (hermesEnabled.toBoolean()) {
            implementation("com.facebook.react:hermes-android")
        } else {
            implementation jscFlavor
        }
    }
    
    apply from: file("../../node_modules/@react-native-community/cli-platform-android/native_modules.gradle"); applyNativeModulesAppBuildGradle(project)
    

    android/build.gradle

    // Top-level build file where you can add configuration options common to all sub-projects/modules.
    
    buildscript {
        ext {
            buildToolsVersion = "33.0.0"
            minSdkVersion = 21
            compileSdkVersion = 33
            targetSdkVersion = 33
    
            // We use NDK 23 which has both M1 support and is the side-by-side NDK version from AGP.
            ndkVersion = "23.1.7779620"
        }
        repositories {
            google()
            mavenCentral()
        }
        dependencies {
            classpath("com.android.tools.build:gradle:7.3.1")
            classpath("com.facebook.react:react-native-gradle-plugin")
            classpath 'com.google.gms:google-services:4.3.15'
        }
    }