brick-module 0.1.3 → 0.1.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -44,6 +44,7 @@ Pod::Spec.new do |s|
44
44
  "OTHER_CPLUSPLUSFLAGS" => "-DFOLLY_NO_CONFIG -DFOLLY_MOBILE=1 -DFOLLY_USE_LIBCPP=1",
45
45
  "CLANG_CXX_LANGUAGE_STANDARD" => "c++17"
46
46
  })
47
+ s.dependency 'React'
47
48
  s.dependency "React-Codegen"
48
49
  s.dependency "RCT-Folly"
49
50
  s.dependency "RCTRequired"
@@ -119,17 +119,26 @@ ext.applyBrickModules = { settings ->
119
119
  def packageJson = readPackageJson(new File(moduleDir, "package.json"))
120
120
  if (!isBrickModule(packageJson)) return
121
121
 
122
+ // Check if autoRegister is enabled (default: true)
123
+ def autoRegister = packageJson.brickModule.autoRegister != false
124
+ if (!autoRegister) return
125
+
122
126
  def androidDir = new File(moduleDir, "android")
123
127
  if (androidDir.exists()) {
124
- settings.include(":${depName}")
125
- settings.project(":${depName}").projectDir = androidDir
128
+ // Convert npm package name to valid Gradle project name
129
+ // Replace @ with empty and / with _
130
+ def gradleProjectName = depName.replaceAll("^@", "").replaceAll("/", "_")
131
+
132
+ settings.include(":${gradleProjectName}")
133
+ settings.project(":${gradleProjectName}").projectDir = androidDir
126
134
  foundModules.add(depName)
127
135
 
128
136
  // Store module data for project configuration
129
137
  if (packageJson.brickModule.android?.package) {
130
138
  brickModulesData[depName] = [
131
139
  moduleName: packageJson.brickModule.android.moduleName ?: depName,
132
- packageName: packageJson.brickModule.android.package
140
+ packageName: packageJson.brickModule.android.package,
141
+ gradleProjectName: gradleProjectName
133
142
  ]
134
143
  }
135
144
  }
@@ -145,8 +154,8 @@ ext.applyBrickModules = { settings ->
145
154
 
146
155
  // Include generated module
147
156
  if (brickDir.exists()) {
148
- settings.include(":brick-generated")
149
- settings.project(":brick-generated").projectDir = brickDir
157
+ settings.include(":brick-codegen")
158
+ settings.project(":brick-codegen").projectDir = brickDir
150
159
  }
151
160
 
152
161
  if (!foundModules.isEmpty()) {
@@ -184,8 +193,8 @@ def configureBrickModules(project) {
184
193
  project.android.sourceSets.main.java.srcDir "${project.buildDir}/generated/source/brick-provider"
185
194
 
186
195
  // Create provider generation task
187
- project.tasks.create('generateBrickModuleProvider') {
188
- description = 'Generates BrickModuleProvider class for automatic module registration'
196
+ project.tasks.create('generateBrickModulesList') {
197
+ description = 'Generates BrickModulesList class for module injection in MainActivity'
189
198
  group = 'brick-module'
190
199
 
191
200
  doLast {
@@ -194,7 +203,7 @@ def configureBrickModules(project) {
194
203
  }
195
204
 
196
205
  // Hook into build process
197
- project.preBuild.dependsOn('generateBrickModuleProvider')
206
+ project.preBuild.dependsOn('generateBrickModulesList')
198
207
  project.preBuild.doFirst {
199
208
  def workingDir = projectRoot
200
209
 
@@ -223,8 +232,39 @@ def configureBrickModules(project) {
223
232
  // Add dependencies
224
233
  project.dependencies {
225
234
  // Add brick module dependencies
235
+ brickLog("🔍 Processing subprojects for dependencies...")
236
+ brickLog("📋 Available brick modules: ${brickModulesData.keySet()}")
237
+
238
+ brickLog("🔍 Available subprojects: ${rootProject.subprojects.collect { it.name }}")
239
+
226
240
  rootProject.subprojects.each { subproject ->
227
- if (subproject.name.startsWith('brick-') || brickModulesData.containsKey(subproject.name)) {
241
+ // Try to find matching brick module by normalizing names
242
+ def foundModule = null
243
+ def normalizedSubprojectName = subproject.name.replaceAll('[-_]', '') // Remove both - and _
244
+
245
+ brickModulesData.each { moduleKey, moduleData ->
246
+ def normalizedModuleKey = moduleKey.replaceAll('[@/-]', '')
247
+
248
+ if (normalizedSubprojectName == normalizedModuleKey) {
249
+ foundModule = moduleKey
250
+ return
251
+ }
252
+ }
253
+
254
+ if (foundModule) {
255
+ brickLog("✅ Adding brick module dependency: ${foundModule} (matched with ${subproject.name})")
256
+ add('implementation', subproject)
257
+ } else if (brickModulesData.containsKey(subproject.name)) {
258
+ brickLog("✅ Adding brick module dependency: ${subproject.name}")
259
+ add('implementation', subproject)
260
+ }
261
+
262
+ if (subproject.name == 'brick-module') {
263
+ brickLog("✅ Adding brick-module dependency")
264
+ add('implementation', subproject)
265
+ }
266
+ if (subproject.name == 'brick-codegen') {
267
+ brickLog("✅ Adding brick-codegen dependency")
228
268
  add('implementation', subproject)
229
269
  }
230
270
  }
@@ -236,12 +276,11 @@ def configureBrickModules(project) {
236
276
  }
237
277
  }
238
278
 
239
- // Generate BrickModuleProvider.kt
279
+ // Generate BrickModulesList.kt
240
280
  def generateProvider(project, brickModulesData) {
241
281
  def modules = brickModulesData.collect { name, data ->
242
282
  [
243
- name: data.moduleName,
244
- className: data.moduleName + "Module",
283
+ moduleName: data.moduleName,
245
284
  packageName: data.packageName
246
285
  ]
247
286
  }
@@ -249,31 +288,62 @@ def generateProvider(project, brickModulesData) {
249
288
  def outputDir = project.file("${project.buildDir}/generated/source/brick-provider/com/brickmodule")
250
289
  outputDir.mkdirs()
251
290
 
252
- def imports = modules.collect { "import ${it.packageName}.${it.className}" }.join("\n")
253
- def moduleInstances = modules.collect { " ${it.name}Module()" }.join(",\n")
254
- def moduleNames = modules.collect { " \"${it.name}\"" }.join(",\n")
291
+ def imports = modules.collect { "import ${it.packageName}.${it.moduleName}" }.join("\n")
292
+ def moduleInstances = modules.collect { " ${it.moduleName}(context)" }.join(",\n")
293
+ def moduleNames = modules.collect { " \"${it.moduleName}\"" }.join(",\n")
255
294
 
256
- new File(outputDir, "BrickModuleProvider.kt").text = """// Auto-generated by brick_modules.gradle
295
+ new File(outputDir, "BrickModulesList.kt").text = """// Auto-generated by brick_modules.gradle
257
296
  package com.brickmodule
258
297
 
259
- import com.brickmodule.BrickModuleRegistry
298
+ import com.facebook.react.bridge.ReactContext
260
299
  ${imports}
261
300
 
262
- object BrickModuleProvider {
263
- fun registerAll() {
264
- val modules = listOf<Any>(
301
+ /**
302
+ * Auto-generated helper for brick modules
303
+ * Use this class to get auto-linked modules in your MainActivity
304
+ *
305
+ * Example usage:
306
+ * ```kotlin
307
+ * override fun getBrickModules(context: ReactContext): List<BrickModuleBase> {
308
+ * return BrickModulesList(context).modules
309
+ * }
310
+ * ```
311
+ */
312
+ class BrickModulesList(private val context: ReactContext) {
313
+ /**
314
+ * Get all auto-linked brick modules
315
+ *
316
+ * @return List of auto-linked module instances
317
+ */
318
+ val modules: List<BrickModuleBase> by lazy {
319
+ listOf<BrickModuleBase>(
265
320
  ${moduleInstances}
266
321
  )
267
-
268
- if (modules.isNotEmpty()) {
269
- BrickModuleRegistry.register(modules)
270
- println("🧱 BrickModuleProvider: Registered \${modules.size} modules")
271
- }
272
322
  }
273
323
 
274
- fun getAvailableModules(): List<String> = listOf(
324
+ /**
325
+ * Get available module names (for debugging)
326
+ *
327
+ * @return List of module names
328
+ */
329
+ val availableModules: List<String> = listOf(
275
330
  ${moduleNames}
276
331
  )
332
+
333
+ companion object {
334
+ /**
335
+ * Legacy static method for backward compatibility
336
+ * @deprecated Use BrickModulesList(context).modules instead
337
+ */
338
+ @Deprecated(
339
+ message = "Use BrickModulesList(context).modules instead",
340
+ replaceWith = ReplaceWith("BrickModulesList(context).modules")
341
+ )
342
+ @JvmStatic
343
+ fun getAutolinkedModules(context: ReactContext): MutableList<Any> {
344
+ return BrickModulesList(context).modules.toMutableList() as MutableList<Any>
345
+ }
346
+ }
277
347
  }
278
348
  """
279
349
  }
@@ -1,6 +1,6 @@
1
1
  buildscript {
2
2
  // Buildscript is evaluated before everything else so we can't use getExtOrDefault
3
- def kotlin_version = rootProject.ext.has("kotlinVersion") ? rootProject.ext.get("kotlinVersion") : project.properties["HotUpdater_kotlinVersion"]
3
+ def kotlin_version = rootProject.ext.has("kotlinVersion") ? rootProject.ext.get("kotlinVersion") : project.properties["BrickModule_kotlinVersion"]
4
4
 
5
5
  repositories {
6
6
  google()
@@ -8,7 +8,7 @@ buildscript {
8
8
  }
9
9
 
10
10
  dependencies {
11
- classpath "com.android.tools.build:gradle:7.2.1"
11
+ classpath "com.android.tools.build:gradle:8.2.0"
12
12
  // noinspection DifferentKotlinGradleVersion
13
13
  classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
14
14
 
@@ -103,10 +103,10 @@ dependencies {
103
103
  //noinspection GradleDynamicVersion
104
104
  if (project.ext.shouldConsumeReactNativeFromMavenCentral()) {
105
105
  // noinspection GradleDynamicVersion
106
- implementation 'com.facebook.react:react-android:+'
106
+ api 'com.facebook.react:react-android:+'
107
107
  } else {
108
108
  // noinspection GradleDynamicVersion
109
- implementation 'com.facebook.react:react-native:+'
109
+ api 'com.facebook.react:react-native:+'
110
110
  }
111
111
  implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
112
112
  implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.4"
@@ -0,0 +1,7 @@
1
+ distributionBase=GRADLE_USER_HOME
2
+ distributionPath=wrapper/dists
3
+ distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-bin.zip
4
+ networkTimeout=10000
5
+ validateDistributionUrl=true
6
+ zipStoreBase=GRADLE_USER_HOME
7
+ zipStorePath=wrapper/dists
@@ -0,0 +1,252 @@
1
+ #!/bin/sh
2
+
3
+ #
4
+ # Copyright © 2015-2021 the original 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
+ # SPDX-License-Identifier: Apache-2.0
19
+ #
20
+
21
+ ##############################################################################
22
+ #
23
+ # Gradle start up script for POSIX generated by Gradle.
24
+ #
25
+ # Important for running:
26
+ #
27
+ # (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
28
+ # noncompliant, but you have some other compliant shell such as ksh or
29
+ # bash, then to run this script, type that shell name before the whole
30
+ # command line, like:
31
+ #
32
+ # ksh Gradle
33
+ #
34
+ # Busybox and similar reduced shells will NOT work, because this script
35
+ # requires all of these POSIX shell features:
36
+ # * functions;
37
+ # * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
38
+ # «${var#prefix}», «${var%suffix}», and «$( cmd )»;
39
+ # * compound commands having a testable exit status, especially «case»;
40
+ # * various built-in commands including «command», «set», and «ulimit».
41
+ #
42
+ # Important for patching:
43
+ #
44
+ # (2) This script targets any POSIX shell, so it avoids extensions provided
45
+ # by Bash, Ksh, etc; in particular arrays are avoided.
46
+ #
47
+ # The "traditional" practice of packing multiple parameters into a
48
+ # space-separated string is a well documented source of bugs and security
49
+ # problems, so this is (mostly) avoided, by progressively accumulating
50
+ # options in "$@", and eventually passing that to Java.
51
+ #
52
+ # Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
53
+ # and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
54
+ # see the in-line comments for details.
55
+ #
56
+ # There are tweaks for specific operating systems such as AIX, CygWin,
57
+ # Darwin, MinGW, and NonStop.
58
+ #
59
+ # (3) This script is generated from the Groovy template
60
+ # https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
61
+ # within the Gradle project.
62
+ #
63
+ # You can find Gradle at https://github.com/gradle/gradle/.
64
+ #
65
+ ##############################################################################
66
+
67
+ # Attempt to set APP_HOME
68
+
69
+ # Resolve links: $0 may be a link
70
+ app_path=$0
71
+
72
+ # Need this for daisy-chained symlinks.
73
+ while
74
+ APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
75
+ [ -h "$app_path" ]
76
+ do
77
+ ls=$( ls -ld "$app_path" )
78
+ link=${ls#*' -> '}
79
+ case $link in #(
80
+ /*) app_path=$link ;; #(
81
+ *) app_path=$APP_HOME$link ;;
82
+ esac
83
+ done
84
+
85
+ # This is normally unused
86
+ # shellcheck disable=SC2034
87
+ APP_BASE_NAME=${0##*/}
88
+ # Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
89
+ APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s
90
+ ' "$PWD" ) || exit
91
+
92
+ # Use the maximum available, or set MAX_FD != -1 to use that value.
93
+ MAX_FD=maximum
94
+
95
+ warn () {
96
+ echo "$*"
97
+ } >&2
98
+
99
+ die () {
100
+ echo
101
+ echo "$*"
102
+ echo
103
+ exit 1
104
+ } >&2
105
+
106
+ # OS specific support (must be 'true' or 'false').
107
+ cygwin=false
108
+ msys=false
109
+ darwin=false
110
+ nonstop=false
111
+ case "$( uname )" in #(
112
+ CYGWIN* ) cygwin=true ;; #(
113
+ Darwin* ) darwin=true ;; #(
114
+ MSYS* | MINGW* ) msys=true ;; #(
115
+ NONSTOP* ) nonstop=true ;;
116
+ esac
117
+
118
+ CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
119
+
120
+
121
+ # Determine the Java command to use to start the JVM.
122
+ if [ -n "$JAVA_HOME" ] ; then
123
+ if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
124
+ # IBM's JDK on AIX uses strange locations for the executables
125
+ JAVACMD=$JAVA_HOME/jre/sh/java
126
+ else
127
+ JAVACMD=$JAVA_HOME/bin/java
128
+ fi
129
+ if [ ! -x "$JAVACMD" ] ; then
130
+ die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
131
+
132
+ Please set the JAVA_HOME variable in your environment to match the
133
+ location of your Java installation."
134
+ fi
135
+ else
136
+ JAVACMD=java
137
+ if ! command -v java >/dev/null 2>&1
138
+ then
139
+ die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
140
+
141
+ Please set the JAVA_HOME variable in your environment to match the
142
+ location of your Java installation."
143
+ fi
144
+ fi
145
+
146
+ # Increase the maximum file descriptors if we can.
147
+ if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
148
+ case $MAX_FD in #(
149
+ max*)
150
+ # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
151
+ # shellcheck disable=SC2039,SC3045
152
+ MAX_FD=$( ulimit -H -n ) ||
153
+ warn "Could not query maximum file descriptor limit"
154
+ esac
155
+ case $MAX_FD in #(
156
+ '' | soft) :;; #(
157
+ *)
158
+ # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
159
+ # shellcheck disable=SC2039,SC3045
160
+ ulimit -n "$MAX_FD" ||
161
+ warn "Could not set maximum file descriptor limit to $MAX_FD"
162
+ esac
163
+ fi
164
+
165
+ # Collect all arguments for the java command, stacking in reverse order:
166
+ # * args from the command line
167
+ # * the main class name
168
+ # * -classpath
169
+ # * -D...appname settings
170
+ # * --module-path (only if needed)
171
+ # * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
172
+
173
+ # For Cygwin or MSYS, switch paths to Windows format before running java
174
+ if "$cygwin" || "$msys" ; then
175
+ APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
176
+ CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
177
+
178
+ JAVACMD=$( cygpath --unix "$JAVACMD" )
179
+
180
+ # Now convert the arguments - kludge to limit ourselves to /bin/sh
181
+ for arg do
182
+ if
183
+ case $arg in #(
184
+ -*) false ;; # don't mess with options #(
185
+ /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
186
+ [ -e "$t" ] ;; #(
187
+ *) false ;;
188
+ esac
189
+ then
190
+ arg=$( cygpath --path --ignore --mixed "$arg" )
191
+ fi
192
+ # Roll the args list around exactly as many times as the number of
193
+ # args, so each arg winds up back in the position where it started, but
194
+ # possibly modified.
195
+ #
196
+ # NB: a `for` loop captures its iteration list before it begins, so
197
+ # changing the positional parameters here affects neither the number of
198
+ # iterations, nor the values presented in `arg`.
199
+ shift # remove old arg
200
+ set -- "$@" "$arg" # push replacement arg
201
+ done
202
+ fi
203
+
204
+
205
+ # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
206
+ DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
207
+
208
+ # Collect all arguments for the java command:
209
+ # * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
210
+ # and any embedded shellness will be escaped.
211
+ # * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
212
+ # treated as '${Hostname}' itself on the command line.
213
+
214
+ set -- \
215
+ "-Dorg.gradle.appname=$APP_BASE_NAME" \
216
+ -classpath "$CLASSPATH" \
217
+ org.gradle.wrapper.GradleWrapperMain \
218
+ "$@"
219
+
220
+ # Stop when "xargs" is not available.
221
+ if ! command -v xargs >/dev/null 2>&1
222
+ then
223
+ die "xargs is not available"
224
+ fi
225
+
226
+ # Use "xargs" to parse quoted args.
227
+ #
228
+ # With -n1 it outputs one arg per line, with the quotes and backslashes removed.
229
+ #
230
+ # In Bash we could simply go:
231
+ #
232
+ # readarray ARGS < <( xargs -n1 <<<"$var" ) &&
233
+ # set -- "${ARGS[@]}" "$@"
234
+ #
235
+ # but POSIX shell has neither arrays nor command substitution, so instead we
236
+ # post-process each arg (as a line of input to sed) to backslash-escape any
237
+ # character that might be a shell metacharacter, then use eval to reverse
238
+ # that process (while maintaining the separation between arguments), and wrap
239
+ # the whole thing up as a single "set" statement.
240
+ #
241
+ # This will of course break if any of these variables contains a newline or
242
+ # an unmatched quote.
243
+ #
244
+
245
+ eval "set -- $(
246
+ printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
247
+ xargs -n1 |
248
+ sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
249
+ tr '\n' ' '
250
+ )" '"$@"'
251
+
252
+ exec "$JAVACMD" "$@"
@@ -0,0 +1,94 @@
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
+ @rem SPDX-License-Identifier: Apache-2.0
17
+ @rem
18
+
19
+ @if "%DEBUG%"=="" @echo off
20
+ @rem ##########################################################################
21
+ @rem
22
+ @rem Gradle startup script for Windows
23
+ @rem
24
+ @rem ##########################################################################
25
+
26
+ @rem Set local scope for the variables with windows NT shell
27
+ if "%OS%"=="Windows_NT" setlocal
28
+
29
+ set DIRNAME=%~dp0
30
+ if "%DIRNAME%"=="" set DIRNAME=.
31
+ @rem This is normally unused
32
+ set APP_BASE_NAME=%~n0
33
+ set APP_HOME=%DIRNAME%
34
+
35
+ @rem Resolve any "." and ".." in APP_HOME to make it shorter.
36
+ for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
37
+
38
+ @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
39
+ set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
40
+
41
+ @rem Find java.exe
42
+ if defined JAVA_HOME goto findJavaFromJavaHome
43
+
44
+ set JAVA_EXE=java.exe
45
+ %JAVA_EXE% -version >NUL 2>&1
46
+ if %ERRORLEVEL% equ 0 goto execute
47
+
48
+ echo. 1>&2
49
+ echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2
50
+ echo. 1>&2
51
+ echo Please set the JAVA_HOME variable in your environment to match the 1>&2
52
+ echo location of your Java installation. 1>&2
53
+
54
+ goto fail
55
+
56
+ :findJavaFromJavaHome
57
+ set JAVA_HOME=%JAVA_HOME:"=%
58
+ set JAVA_EXE=%JAVA_HOME%/bin/java.exe
59
+
60
+ if exist "%JAVA_EXE%" goto execute
61
+
62
+ echo. 1>&2
63
+ echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2
64
+ echo. 1>&2
65
+ echo Please set the JAVA_HOME variable in your environment to match the 1>&2
66
+ echo location of your Java installation. 1>&2
67
+
68
+ goto fail
69
+
70
+ :execute
71
+ @rem Setup the command line
72
+
73
+ set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
74
+
75
+
76
+ @rem Execute Gradle
77
+ "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
78
+
79
+ :end
80
+ @rem End local scope for the variables with windows NT shell
81
+ if %ERRORLEVEL% equ 0 goto mainEnd
82
+
83
+ :fail
84
+ rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
85
+ rem the _cmd.exe /c_ return code!
86
+ set EXIT_CODE=%ERRORLEVEL%
87
+ if %EXIT_CODE% equ 0 set EXIT_CODE=1
88
+ if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
89
+ exit /b %EXIT_CODE%
90
+
91
+ :mainEnd
92
+ if "%OS%"=="Windows_NT" endlocal
93
+
94
+ :omega
@@ -1,23 +1,75 @@
1
1
  package com.brickmodule
2
2
 
3
+ import com.facebook.react.bridge.ReactContext
4
+
3
5
  /**
4
- * Minimal interface that all Brick modules must implement
5
- * Contains only essential properties for module identification
6
+ * Minimal interface that all Brick modules must implement Contains only essential properties for
7
+ * module identification
6
8
  */
7
9
  interface BrickModuleBase {
10
+ /** The name of the module (required for registration) */
11
+ val moduleName: String
12
+
8
13
  /**
9
- * The name of the module (required for registration)
14
+ * Returns the constants exposed by this module
15
+ * Override this method to provide module-specific constants
16
+ * @return Map of constant name to value
10
17
  */
11
- val moduleName: String
18
+ fun getConstants(): Map<String, Any> = emptyMap()
12
19
  }
13
20
 
14
21
  /**
15
- * Error types for Brick modules
22
+ * Abstract base class for Brick modules that provides internal implementation details Module
23
+ * implementations should extend this class instead of implementing BrickModuleBase directly
16
24
  */
25
+ abstract class BrickModuleSpec(private val reactContext: ReactContext) : BrickModuleBase {
26
+
27
+ /**
28
+ * Get the ReactContext instance
29
+ * @return ReactContext for this module
30
+ */
31
+ protected fun getReactContext(): ReactContext = reactContext
32
+
33
+ /**
34
+ * Send an event to JavaScript This method handles event validation and routing through the
35
+ * BrickHost
36
+ *
37
+ * @param eventName The name of the event (without module prefix)
38
+ * @param data The event data to send
39
+ */
40
+ protected fun sendEvent(eventName: String, data: Any?) {
41
+ val context = getReactContext()
42
+
43
+ // Format event name with module prefix
44
+ val fullEventName = "${moduleName}_$eventName"
45
+
46
+ // Get the BrickHost from ReactContext
47
+ val activity = context.currentActivity
48
+ if (activity !is BrickModuleRegistrar) {
49
+ println(
50
+ "❌ $moduleName: Current activity does not implement BrickHost, cannot send event"
51
+ )
52
+ return
53
+ }
54
+
55
+ // Send event via BrickHost's moduleRegistry
56
+ try {
57
+ activity.getModuleRegistry().sendEvent(context, fullEventName, data)
58
+ } catch (e: Exception) {
59
+ println("❌ $moduleName: Failed to send event $fullEventName: ${e.message}")
60
+ }
61
+ }
62
+ }
63
+
64
+ /** Error types for Brick modules */
17
65
  open class BrickModuleError(message: String, val errorCode: String) : Exception(message) {
18
66
  class TypeMismatch(message: String) : BrickModuleError("Type mismatch: $message", "TYPE_ERROR")
19
- class ExecutionError(message: String) : BrickModuleError("Execution error: $message", "EXECUTION_ERROR")
20
- class InvalidDefinition(message: String) : BrickModuleError("Invalid definition: $message", "DEFINITION_ERROR")
21
- class MethodNotFound(message: String) : BrickModuleError("Method not found: $message", "METHOD_NOT_FOUND")
22
- class ModuleNotFound(message: String) : BrickModuleError("Module not found: $message", "MODULE_NOT_FOUND")
23
- }
67
+ class ExecutionError(message: String) :
68
+ BrickModuleError("Execution error: $message", "EXECUTION_ERROR")
69
+ class InvalidDefinition(message: String) :
70
+ BrickModuleError("Invalid definition: $message", "DEFINITION_ERROR")
71
+ class MethodNotFound(message: String) :
72
+ BrickModuleError("Method not found: $message", "METHOD_NOT_FOUND")
73
+ class ModuleNotFound(message: String) :
74
+ BrickModuleError("Module not found: $message", "MODULE_NOT_FOUND")
75
+ }