From a1c03578cb6cdd56033c2bbd3e68881017f7a857 Mon Sep 17 00:00:00 2001 From: D N <4661784+retyui@users.noreply.github.com> Date: Tue, 3 Dec 2024 12:09:00 +0100 Subject: [PATCH] feat: Add Kotlin autolinker for React Native 0.63.x+ Issue: https://github.com/wix/react-native-navigation/issues/7821 --- autolink/postlink/activityLinker.js | 59 +++++++++++++++----------- autolink/postlink/applicationLinker.js | 45 +++++++++++++++----- autolink/postlink/path.js | 5 +-- 3 files changed, 70 insertions(+), 39 deletions(-) diff --git a/autolink/postlink/activityLinker.js b/autolink/postlink/activityLinker.js index 47215492be1..5b592fb6b02 100644 --- a/autolink/postlink/activityLinker.js +++ b/autolink/postlink/activityLinker.js @@ -54,16 +54,20 @@ class ActivityLinker { } _removeGetMainComponentName(contents) { - var match = /\/\*\*\s*\n([^\*]|(\*(?!\/)))*\*\/\s*@Override\s*protected\s*String\s*getMainComponentName\s*\(\)\s*{\s*return.+\s*\}/.exec( - contents - ); - if (match) { - debugn(' Removing getMainComponentName function'); - return contents.replace( - /\/\*\*\s*\n([^\*]|(\*(?!\/)))*\*\/\s*@Override\s*protected\s*String\s*getMainComponentName\s*\(\)\s*{\s*return.+\s*\}/, - '' - ); + var javaRegex = /\/\*\*\s*\n([^\*]|(\*(?!\/)))*\*\/\s*@Override\s*protected\s*String\s*getMainComponentName\s*\(\)\s*{\s*return.+\s*\}/; + var javaMatch = javaRegex.exec(contents); + if (javaMatch) { + debugn(' [Java] Removing getMainComponentName function'); + return contents.replace(javaRegex, ''); + } + + var kotlinRegex = /\/\*\*\s*\n([^\*]|(\*(?!\/)))*\*\/\s*override\sfun\sgetMainComponentName\(\):\sString\s=\s.+/; + var kotlinMatch = kotlinRegex.exec(contents); + if (kotlinMatch) { + debugn(' [Kotlin] Removing getMainComponentName function'); + return contents.replace(kotlinRegex, ''); } + warnn(' getMainComponentName function was not found.'); return contents; } @@ -78,6 +82,7 @@ class ActivityLinker { debugn(' Extending NavigationActivity'); return activityContent .replace(/extends\s+ReactActivity\s*/, 'extends NavigationActivity ') + .replace(/ReactActivity\(\)/, 'NavigationActivity()') .replace( /public\s+MainActivityDelegate\s*\(\s*ReactActivity\s+activity,\s*String\s+mainComponentName\s*\)/, 'public MainActivityDelegate(NavigationActivity activity, String mainComponentName)' @@ -94,30 +99,34 @@ class ActivityLinker { } _doesActivityExtendReactActivity(activityContent) { - return /public\s+class\s+MainActivity\s+extends\s+ReactActivity\s*/.test(activityContent); + return ( + /public\s+class\s+MainActivity\s+extends\s+ReactActivity\s*/.test(activityContent) || + /class\sMainActivity\s?:\s?ReactActivity\(\)/.test(activityContent) + ); } _hasAlreadyExtendNavigationActivity(activityContent) { - return /public\s+class\s+MainActivity\s+extends\s+NavigationActivity\s*/.test(activityContent); + return ( + /public\s+class\s+MainActivity\s+extends\s+NavigationActivity\s*/.test(activityContent) || + /class\sMainActivity\s:\sNavigationActivity\(\)\s\{/.test(activityContent) + ); } _removeCreateReactActivityDelegate(activityContent) { - if (this._hasCreateReactActivityDelegate(activityContent)) { - debugn(' Removing createReactActivityDelegate function'); - return activityContent.replace( - /\/\*\*\s*\n([^\*]|(\*(?!\/)))*\*\/\s*@Override\s*protected\s*ReactActivityDelegate\s*createReactActivityDelegate\s*\(\)\s*{\s*return((.|\r|\s)*?)}/, - '' - ); - } else { - warnn(' createReactActivityDelegate is already not defined in MainActivity'); - return activityContent; + var javaRegex = /\/\*\*\s*\n([^\*]|(\*(?!\/)))*\*\/\s*@Override\s*protected\s*ReactActivityDelegate\s*createReactActivityDelegate\s*\(\)\s*{\s*return((.|\r|\s)*?)}/; + if (javaRegex.test(activityContent)) { + debugn(' [Java] Removing createReactActivityDelegate function'); + return activityContent.replace(javaRegex, ''); } - } - _hasCreateReactActivityDelegate(activityContent) { - return /\/\*\*\s*\n([^\*]|(\*(?!\/)))*\*\/\s*@Override\s*protected\s*ReactActivityDelegate\s*createReactActivityDelegate\s*\(\)\s*{\s*return((.|\r|\s)*?)}/.test( - activityContent - ); + var kotlinRegex = /\/\*\*\s*\n([^\*]|(\*(?!\/)))*\*\/\s*override\sfun\screateReactActivityDelegate\(\):\sReactActivityDelegate\s=\s+DefaultReactActivityDelegate\(this,\smainComponentName,\sfabricEnabled\)/; + if (kotlinRegex.test(activityContent)) { + debugn(' [Kotlin] Removing createReactActivityDelegate function'); + return activityContent.replace(kotlinRegex, ''); + } + + warnn(' createReactActivityDelegate is already not defined in MainActivity'); + return activityContent; } } diff --git a/autolink/postlink/applicationLinker.js b/autolink/postlink/applicationLinker.js index 5d65b320e06..7b2b7cb1a03 100644 --- a/autolink/postlink/applicationLinker.js +++ b/autolink/postlink/applicationLinker.js @@ -71,6 +71,7 @@ class ApplicationLinker { /extends\s+Application\s+implements\s+ReactApplication/gi, 'extends NavigationApplication' ) + .replace(/:\sApplication\(\),\sReactApplication/, ': NavigationApplication()') .replace( 'import com.facebook.react.ReactApplication;', 'import com.reactnativenavigation.NavigationApplication;' @@ -88,13 +89,18 @@ class ApplicationLinker { } _doesExtendApplication(applicationContent) { - return /\s+MainApplication\s+extends\s+Application\s+implements\s+ReactApplication\s+/.test( - applicationContent + return ( + /\s+MainApplication\s+extends\s+Application\s+implements\s+ReactApplication\s+/.test( + applicationContent + ) || /class\sMainApplication\s:\sNavigationApplication\(\)/.test(applicationContent) ); } _hasAlreadyLinkedApplication(applicationContent) { - return /\s+extends\s+NavigationApplication\s+/.test(applicationContent); + return ( + /\s+extends\s+NavigationApplication\s+/.test(applicationContent) || + /class\sMainApplication\s:\sNavigationApplication\(\)/.test(applicationContent) + ); } _extendNavigationHost(applicationContent) { @@ -114,7 +120,8 @@ class ApplicationLinker { } else if (this._doesExtendDefaultReactNativeHost(applicationContent)) { debugn(' Changing host implementation to NavigationReactNativeHost'); return applicationContent - .replace('new DefaultReactNativeHost(this)', 'new NavigationReactNativeHost(this)') + .replace('new DefaultReactNativeHost(this)', 'new NavigationReactNativeHost(this)') // Java + .replace('DefaultReactNativeHost(this)', 'NavigationReactNativeHost(this)') // Kotlin .replace( 'import com.facebook.react.defaults.DefaultReactNativeHost;', 'import com.facebook.react.defaults.DefaultReactNativeHost;\nimport com.reactnativenavigation.react.NavigationReactNativeHost;' @@ -129,27 +136,43 @@ class ApplicationLinker { } _doesExtendDefaultReactNativeHost(applicationContent) { - return /\s*new DefaultReactNativeHost\(this\)\s*/.test(applicationContent); + return ( + /\s*new DefaultReactNativeHost\(this\)\s*/.test(applicationContent) || + /DefaultReactNativeHost\(this\)/.test(applicationContent) + ); } _hasAlreadyLinkedNavigationHost(applicationContent) { - return /\s*new NavigationReactNativeHost\(this\)\s*/.test(applicationContent); + return ( + /\s*new NavigationReactNativeHost\(this\)\s*/.test(applicationContent) || + /NavigationReactNativeHost\(this\)/.test(applicationContent) + ); } _removeSOLoaderInit(applicationContent) { if (this._isSOLoaderInitCalled(applicationContent)) { debugn(' Removing call to SOLoader.init()'); - return applicationContent.replace( - /SoLoader.init\(\s*this\s*,\s*[/* native exopackage */]*\s*false\s*\);/, - '' - ); + return applicationContent + .replace( + // Java + /SoLoader.init\(\s*this\s*,\s*[/* native exopackage */]*\s*false\s*\);/, + '' + ) + .replace( + // Kotlin + /SoLoader\.init\(this,\sfalse\)/, + '' + ); } warnn(' SOLoader.init() is not called, skipping.'); return applicationContent; } _isSOLoaderInitCalled(applicationContent) { - return /SoLoader.init\(this,\s*[/* native exopackage */]*\s*false\);/.test(applicationContent); + return ( + /SoLoader.init\(this,\s*[/* native exopackage */]*\s*false\);/.test(applicationContent) || + /SoLoader\.init\(this,\sfalse\)/.test(applicationContent) + ); } } diff --git a/autolink/postlink/path.js b/autolink/postlink/path.js index 3b5055b0098..96b481490cd 100644 --- a/autolink/postlink/path.js +++ b/autolink/postlink/path.js @@ -3,9 +3,8 @@ var ignoreFolders = { ignore: ['node_modules/**', '**/build/**', '**/Build/**', '**/DerivedData/**', '**/*-tvOS*/**'], }; -exports.mainActivityJava = glob.sync('**/MainActivity.java', ignoreFolders)[0]; -exports.mainActivityKotlin = glob.sync('**/MainActivity.kt', ignoreFolders)[0]; -var mainApplicationJava = glob.sync('**/MainApplication.java', ignoreFolders)[0]; +exports.mainActivityJava = glob.sync('**/MainActivity.{java,kt}', ignoreFolders)[0]; +var mainApplicationJava = glob.sync('**/MainApplication.{java,kt}', ignoreFolders)[0]; exports.mainApplicationJava = mainApplicationJava; exports.rootGradle = mainApplicationJava.replace(/android\/app\/.*\.java/, 'android/build.gradle');