Skip to content

chore: upgrade to react native 81 and expo 54#6875

Draft
Rohit3523 wants to merge 180 commits intodevelopfrom
react-native-81
Draft

chore: upgrade to react native 81 and expo 54#6875
Rohit3523 wants to merge 180 commits intodevelopfrom
react-native-81

Conversation

@Rohit3523
Copy link
Contributor

@Rohit3523 Rohit3523 commented Jan 3, 2026

Proposed changes

Depends on:
Merge Plan

  1. chore: upgrade reanimated to v4 #6720
  2. feat: Migrate to react-native-true-sheet #6970

Pending:
Update typescript to version 5.8.3
Remove UIDesignRequiresCompatibility from info.plist

Issue(s)

https://rocketchat.atlassian.net/browse/CORE-1578
Closes #6874
Closes #6803

How to test or reproduce

Screenshots

Types of changes

  • Bugfix (non-breaking change which fixes an issue)
  • Improvement (non-breaking change which improves a current function)
  • New feature (non-breaking change which adds functionality)
  • Documentation update (if none of the other choices apply)

Checklist

  • I have read the CONTRIBUTING doc
  • I have signed the CLA
  • Lint and unit tests pass locally with my changes
  • I have added tests that prove my fix is effective or that my feature works (if applicable)
  • I have added necessary documentation (if applicable)
  • Any dependent changes have been merged and published in downstream modules

Further comments

Summary by CodeRabbit

  • Chores

    • Upgraded React (18 → 19), React Native (0.79 → 0.81) and Expo stack; bumped Android SDK/build tools, Kotlin and Gradle wrapper.
    • Updated many native and JS dependencies for compatibility and enabled the new React Native architecture.
  • New Features

    • Added Expo/React Native plugins for date/time picker and web browser support.
  • Tests

    • Improved UI test selectors and stability for several flows.

@coderabbitai
Copy link
Contributor

coderabbitai bot commented Jan 3, 2026

Important

Review skipped

Draft detected.

Please check the settings in the CodeRabbit UI or the .coderabbit.yaml file in this repository. To trigger a single review, invoke the @coderabbitai review command.

⚙️ Run configuration

Configuration used: Organization UI

Review profile: CHILL

Plan: Pro

Run ID: 075a5bb2-1408-4e3c-91e5-4ab741849a4d

You can disable this status message by setting the reviews.review_status to false in the CodeRabbit configuration file.

Use the checkbox below for a quick retry:

  • 🔍 Trigger review

Walkthrough

This PR performs a comprehensive upgrade of the Android build toolchain, iOS project configuration, and project dependencies. Updates include Android SDK/build tools (35→36), Kotlin version, Gradle wrapper, React Native (0.79→0.81), Expo (53→54), React (18→19), iOS Xcode project configuration, and introduction of new build flags and Expo module integrations.

Changes

Cohort / File(s) Summary
Android Build Configuration
android/build.gradle, android/gradle.properties, android/gradle/wrapper/gradle-wrapper.properties
Updated compileSdkVersion, targetSdkVersion, and buildToolsVersion from 35/35.0.0 to 36/36.0.0; Kotlin bumped 2.0.21→2.1.20; Gradle wrapper updated 8.13→8.14.3; added edgeToEdgeEnabled=false property
Android Gradle Scripts
android/gradlew, android/gradlew.bat
Modified CLASSPATH handling and Gradle JAR invocation—shifted from explicit main class to direct -jar execution; MIT license header added to .bat file
Android Manifests & Application
android/app/src/debug/AndroidManifest.xml, android/app/src/main/AndroidManifest.xml, android/app/src/main/java/chat/rocket/reactnative/MainApplication.kt
Removed debug manifest; added configurable android:usesCleartextTraffic="${usesCleartextTraffic}" to main manifest; replaced SoLoader initialization with new loadReactNative() entry point in MainApplication
Project Configuration
app.json, babel.config.js, metro.config.js
Added Expo plugins (datetimepicker, web-browser); swapped Babel plugin from react-native-reanimated to react-native-worklets; refactored metro config to use defaultConfig retrieval for sourceExts
iOS Plist Files
ios/NotificationService/Info.plist, ios/RocketChatRN/Info.plist, ios/ShareRocketChatRN/Info.plist
Added RCTNewArchEnabled=true; repositioned IS_OFFICIAL and bugsnag configurations; minor key reordering
iOS Xcode Project
ios/RocketChatRN.xcodeproj/project.pbxproj
Added RNCAsyncStorage resource bundle references; integrated React runtimeexecutor headers; wired Expo module provisioning (ExpoModulesProvider.swift, .xcode.env configs) across multiple targets (RocketChatRN, Rocket.Chat, NotificationService, Rocket.Chat Watch)
Dependencies
package.json
Major version upgrades: React 18→19, React Native 0.79→0.81, Expo 53→54; updated all Expo and React Native ecosystem packages; bumped @expo/vector-icons 14.x→15.x; @react-native-async-storage migrated to 2.2.0; updated babel and tooling versions
Package Patches
patches/@rocket.chat+message-parser+0.31.31.patch, patches/expo-file-system+19.0.21.patch, patches/expo-image+2.3.2.patch, patches/react-native+0.81.5.patch
Removed messageParser.js conditional export; renamed FileSystemModule→FileSystemLegacyModule; removed default value from useAppleWebpCodec prop; extended excluded_info_plist for xcframework/framework exclusions

Estimated code review effort

🎯 4 (Complex) | ⏱️ ~60 minutes

Possibly related PRs

Suggested reviewers

  • diegolmello
  • OtavioStasiak

Poem

🐰 Versions dance in grand array,
From eighteen up to nineteen's way,
Build tools climb from thirty-five,
Gradle, Kotlin—all alive!
New Arch awaits on iOS shores,
While Android opens wider doors! 🚀

🚥 Pre-merge checks | ✅ 4 | ❌ 1

❌ Failed checks (1 warning)

Check name Status Explanation Resolution
Docstring Coverage ⚠️ Warning Docstring coverage is 0.00% which is insufficient. The required threshold is 80.00%. You can run @coderabbitai generate docstrings to improve docstring coverage.
✅ Passed checks (4 passed)
Check name Status Explanation
Title check ✅ Passed The PR title 'chore: upgrade to react native 81 and expo 54' accurately and concisely summarizes the main change—upgrading React Native and Expo versions across the entire project.
Linked Issues check ✅ Passed The PR addresses issue #6874 by upgrading React Native from 0.79 to 0.81, which resolves the reported iOS Switch layout issue on iOS 26 that was attributed to RN 0.79.
Out of Scope Changes check ✅ Passed All changes are directly related to the upgrade objectives: dependency updates (package.json, gradle files), configuration adjustments (build tools, Expo plugins, Babel), and necessary patches for compatibility with RN 81 and Expo 54.
Description Check ✅ Passed Check skipped - CodeRabbit’s high-level summary is enabled.

✏️ Tip: You can configure your own custom pre-merge checks in the settings.


Thanks for using CodeRabbit! It's free for OSS, and your support helps us grow. If you like it, consider giving us a shout-out.

❤️ Share

Comment @coderabbitai help to get the list of available commands and usage tips.

@Rohit3523 Rohit3523 temporarily deployed to approve_e2e_testing January 4, 2026 08:54 — with GitHub Actions Inactive
@Rohit3523 Rohit3523 temporarily deployed to approve_e2e_testing January 4, 2026 08:58 — with GitHub Actions Inactive
@Rohit3523 Rohit3523 had a problem deploying to official_android_build March 24, 2026 17:54 — with GitHub Actions Error
@Rohit3523 Rohit3523 had a problem deploying to experimental_android_build March 24, 2026 17:54 — with GitHub Actions Error
@Rohit3523 Rohit3523 had a problem deploying to experimental_ios_build March 24, 2026 17:54 — with GitHub Actions Error
@Rohit3523 Rohit3523 had a problem deploying to official_android_build March 24, 2026 19:23 — with GitHub Actions Error
@Rohit3523 Rohit3523 had a problem deploying to experimental_android_build March 24, 2026 19:23 — with GitHub Actions Error
@Rohit3523 Rohit3523 had a problem deploying to experimental_ios_build March 24, 2026 19:23 — with GitHub Actions Error
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 3

Caution

Some comments are outside the diff and can’t be posted inline due to platform limitations.

⚠️ Outside diff range comments (1)
app/containers/Button/index.tsx (1)

9-20: ⚠️ Potential issue | 🟠 Major

Prevent enabled from bypassing disabled/loading safeguards.

Because IButtonProps extends RectButtonProps (line 9) and {...otherProps} is spread last (line 97), a caller can pass enabled={true} to override enabled={!isDisabled} (line 93). This allows re-enabling presses while loading or disabled is true.

Apply the proposed fix to exclude problematic props from the interface and ensure explicit props take precedence:

Proposed fix
-interface IButtonProps extends RectButtonProps {
+interface IButtonProps extends Omit<RectButtonProps, 'enabled' | 'onPress' | 'style'> {
 	title: string;
 	onPress: () => void;
 	type?: 'primary' | 'secondary';
@@
 	return (
 		<RectButton
+			{...otherProps}
 			onPress={onPress}
 			enabled={!isDisabled}
 			style={containerStyle}
 			accessibilityLabel={title}
 			accessibilityRole='button'
-			{...otherProps}>
+		>
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@app/containers/Button/index.tsx` around lines 9 - 20, The component allows
callers to override the derived enabled state because IButtonProps extends
RectButtonProps and {...otherProps} is spread after setting enabled, so callers
can pass enabled to bypass disabled/loading; fix by excluding enabled from the
inherited props and/or filtering it out before the spread: change the props type
to extend Omit<RectButtonProps, 'enabled'> (i.e., IButtonProps extends
Omit<RectButtonProps, 'enabled'>) or destructure enabled out of otherProps
(const { enabled, ...rest } = otherProps) and spread rest instead of otherProps,
and ensure the component always sets enabled={!isDisabled} (or similar) when
rendering the RectButton so explicit props cannot override it.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Inline comments:
In `@app/containers/AudioPlayer/Seek.tsx`:
- Around line 76-80: The scale animation is being restarted on every gesture
update because withTiming(1.3, ...) is called inside the .onUpdate handler; move
the scale-up call into the gesture's .onStart handler so it runs once when the
gesture begins (e.g., call scale.value = withTiming(1.3, { duration: 150 }) in
.onStart), keep translateX.value and clamp logic inside .onUpdate using contextX
and maxWidth, and ensure you restore scale (e.g., scale.value = withTiming(1) or
appropriate reset) in .onEnd/.onFinalize so the knob returns to normal after the
gesture.

In `@app/containers/UIKit/Overflow.tsx`:
- Around line 44-57: The module-scoped touchable map (touchable) used in
Overflow leaks refs and can mis-anchor Popover when blockId repeats or is empty;
replace it with an instance-local ref inside the Overflow component (e.g.,
remove touchable and create const touchableRef = useRef<View | null>(null)
inside the Overflow function), attach that ref to the touchable element, and
remove all accesses to the global touchable[blockId] so the ref lifecycle is
tied to the component instance (update any code using touchableRef variable name
or blockId-dependent lookup to use the local touchableRef and, if needed, handle
blockId changes with useEffect).

In `@app/views/SecurityPrivacyView.tsx`:
- Around line 106-112: The "Send_crash_report" List.Item is using the wrong
accessibility label variable; replace the additionalAccessibilityLabel value
currently set to analyticsEventsState with crashReportState so VoiceOver reports
the correct enabled/disabled state for the crash report toggle. Update the
List.Item (title 'Send_crash_report') to use crashReportState for
additionalAccessibilityLabel and keep the existing Switch props
(value={crashReportState}, onValueChange={toggleCrashReport}, testID) unchanged.

---

Outside diff comments:
In `@app/containers/Button/index.tsx`:
- Around line 9-20: The component allows callers to override the derived enabled
state because IButtonProps extends RectButtonProps and {...otherProps} is spread
after setting enabled, so callers can pass enabled to bypass disabled/loading;
fix by excluding enabled from the inherited props and/or filtering it out before
the spread: change the props type to extend Omit<RectButtonProps, 'enabled'>
(i.e., IButtonProps extends Omit<RectButtonProps, 'enabled'>) or destructure
enabled out of otherProps (const { enabled, ...rest } = otherProps) and spread
rest instead of otherProps, and ensure the component always sets
enabled={!isDisabled} (or similar) when rendering the RectButton so explicit
props cannot override it.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: Organization UI

Review profile: CHILL

Plan: Pro

Run ID: bddc7b3f-32f5-4a46-9548-0793d81b218b

📥 Commits

Reviewing files that changed from the base of the PR and between 691fb63 and ee7133d.

⛔ Files ignored due to path filters (25)
  • app/containers/Avatar/__snapshots__/Avatar.test.tsx.snap is excluded by !**/*.snap
  • app/containers/Button/__snapshots__/Button.test.tsx.snap is excluded by !**/*.snap
  • app/containers/Chip/__snapshots__/Chip.test.tsx.snap is excluded by !**/*.snap
  • app/containers/DirectoryItem/__snapshots__/DirectoryItem.test.tsx.snap is excluded by !**/*.snap
  • app/containers/Header/components/HeaderButton/__snapshots__/HeaderButtons.test.tsx.snap is excluded by !**/*.snap
  • app/containers/InAppNotification/__snapshots__/NotifierComponent.test.tsx.snap is excluded by !**/*.snap
  • app/containers/List/__snapshots__/List.test.tsx.snap is excluded by !**/*.snap
  • app/containers/LoginServices/__snapshots__/LoginServices.test.tsx.snap is excluded by !**/*.snap
  • app/containers/MessageComposer/__snapshots__/MessageComposer.test.tsx.snap is excluded by !**/*.snap
  • app/containers/ReactionsList/__snapshots__/ReactionsList.test.tsx.snap is excluded by !**/*.snap
  • app/containers/RoomHeader/__snapshots__/RoomHeader.test.tsx.snap is excluded by !**/*.snap
  • app/containers/RoomItem/__snapshots__/RoomItem.test.tsx.snap is excluded by !**/*.snap
  • app/containers/SearchBox/__snapshots__/SearchBox.test.tsx.snap is excluded by !**/*.snap
  • app/containers/ServerItem/__snapshots__/ServerItem.test.tsx.snap is excluded by !**/*.snap
  • app/containers/TextInput/__snapshots__/TextInput.test.tsx.snap is excluded by !**/*.snap
  • app/containers/UIKit/__snapshots__/UiKitMessage.test.tsx.snap is excluded by !**/*.snap
  • app/containers/UIKit/__snapshots__/UiKitModal.test.tsx.snap is excluded by !**/*.snap
  • app/containers/markdown/__snapshots__/Markdown.test.tsx.snap is excluded by !**/*.snap
  • app/containers/message/__snapshots__/Message.test.tsx.snap is excluded by !**/*.snap
  • app/views/CannedResponsesListView/__snapshots__/CannedResponseItem.test.tsx.snap is excluded by !**/*.snap
  • app/views/CreateChannelView/RoomSettings/__snapshots__/SwitchItem.test.tsx.snap is excluded by !**/*.snap
  • app/views/DiscussionsView/__snapshots__/Item.test.tsx.snap is excluded by !**/*.snap
  • app/views/NewServerView/components/ServersHistoryItem/__snapshots__/ServersHistoryItem.test.tsx.snap is excluded by !**/*.snap
  • app/views/RoomView/LoadMore/__snapshots__/LoadMore.test.tsx.snap is excluded by !**/*.snap
  • app/views/ThreadMessagesView/__snapshots__/Item.test.tsx.snap is excluded by !**/*.snap
📒 Files selected for processing (33)
  • .maestro/tests/accessibilityAndAppearance/ToastsAndDialogs.yml
  • .maestro/tests/assorted/join-from-directory.yaml
  • .maestro/tests/assorted/profile.yaml
  • .maestro/tests/room/room-actions.yaml
  • android/app/src/main/java/chat/rocket/reactnative/MainApplication.kt
  • android/app/src/main/java/chat/rocket/reactnative/networking/SSLPinningTurboModule.java
  • app/containers/ActionSheet/Handle.tsx
  • app/containers/AudioPlayer/Seek.tsx
  • app/containers/Button/index.tsx
  • app/containers/List/ListItem.tsx
  • app/containers/MessageComposer/components/RecordAudio/RecordAudio.tsx
  • app/containers/MessageComposer/hooks/useEmojiKeyboard.tsx
  • app/containers/RoomItem/Actions.tsx
  • app/containers/RoomItem/Touchable.tsx
  • app/containers/RoomItem/interfaces.ts
  • app/containers/UIKit/Overflow.tsx
  • app/lib/encryption/encryption.ts
  • app/lib/methods/handleMediaDownload.ts
  • app/lib/methods/helpers/fileDownload.ts
  • app/lib/methods/helpers/fileUpload/Upload.android.ts
  • app/lib/methods/helpers/sslPinning.ts
  • app/lib/methods/sendFileMessage/utils.ts
  • app/views/AttachmentView.tsx
  • app/views/CreateDiscussionView/index.tsx
  • app/views/DirectoryView/Options.tsx
  • app/views/RoomView/List/components/List.tsx
  • app/views/SecurityPrivacyView.tsx
  • app/views/ShareListView/index.tsx
  • app/views/ShareView/Thumbs.tsx
  • app/views/UserNotificationPreferencesView/index.tsx
  • app/views/UserPreferencesView/index.tsx
  • babel.config.js
  • ios/NotificationService/Info.plist
✅ Files skipped from review due to trivial changes (10)
  • app/containers/MessageComposer/components/RecordAudio/RecordAudio.tsx
  • app/lib/encryption/encryption.ts
  • app/views/AttachmentView.tsx
  • app/lib/methods/helpers/fileDownload.ts
  • .maestro/tests/accessibilityAndAppearance/ToastsAndDialogs.yml
  • app/lib/methods/handleMediaDownload.ts
  • app/containers/ActionSheet/Handle.tsx
  • app/views/UserPreferencesView/index.tsx
  • app/views/UserNotificationPreferencesView/index.tsx
  • app/containers/RoomItem/interfaces.ts
🚧 Files skipped from review as they are similar to previous changes (2)
  • babel.config.js
  • android/app/src/main/java/chat/rocket/reactnative/MainApplication.kt
📜 Review details
⏰ Context from checks skipped due to timeout of 90000ms. You can increase the timeout in your CodeRabbit configuration to a maximum of 15 minutes (900000ms). (4)
  • GitHub Check: E2E Run Android (12) / Android Tests
  • GitHub Check: E2E Run Android (5) / Android Tests
  • GitHub Check: E2E Run Android (11) / Android Tests
  • GitHub Check: E2E Build iOS / ios-build
🧰 Additional context used
🧠 Learnings (7)
📓 Common learnings
Learnt from: Rohit3523
Repo: RocketChat/Rocket.Chat.ReactNative PR: 6930
File: package.json:101-101
Timestamp: 2026-02-05T13:55:06.688Z
Learning: The RocketChat/Rocket.Chat.ReactNative repository uses a fork of react-native-image-crop-picker (RocketChat/react-native-image-crop-picker) with custom Android edge-to-edge fixes, not the upstream ivpusic/react-native-image-crop-picker package. Dependencies should reference commit pins from the RocketChat fork.
📚 Learning: 2026-03-17T19:15:26.536Z
Learnt from: Rohit3523
Repo: RocketChat/Rocket.Chat.ReactNative PR: 6970
File: .maestro/tests/room/share-message.yaml:77-79
Timestamp: 2026-03-17T19:15:26.536Z
Learning: In YAML test files under .maestro/tests/room, use tapping the empty area (e.g., tapOn: point: 5%,10%) to dismiss both the bottom sheet and keyboard when needed. Do not rely on action-sheet-handle alone if the keyboard also needs to be dismissed in the same step. This pattern is acceptable for tests where a single tap should close both UI elements.

Applied to files:

  • .maestro/tests/room/room-actions.yaml
📚 Learning: 2026-03-05T14:28:10.004Z
Learnt from: Rohit3523
Repo: RocketChat/Rocket.Chat.ReactNative PR: 6997
File: .maestro/tests/room/message-markdown-click.yaml:28-39
Timestamp: 2026-03-05T14:28:10.004Z
Learning: In Maestro YAML selector fields (text, id) within the Rocket.Chat React Native repository, use the contains pattern '.*keyword.*' (leading and trailing '.*') for matching text. The pattern '.*keyword*.' is incorrect and will fail to match cases where the keyword appears at the end of the element's text. This guideline applies to all Maestro YAML selector fields across the codebase.

Applied to files:

  • .maestro/tests/room/room-actions.yaml
  • .maestro/tests/assorted/profile.yaml
  • .maestro/tests/assorted/join-from-directory.yaml
📚 Learning: 2026-03-17T19:15:30.463Z
Learnt from: Rohit3523
Repo: RocketChat/Rocket.Chat.ReactNative PR: 6970
File: .maestro/tests/room/share-message.yaml:77-79
Timestamp: 2026-03-17T19:15:30.463Z
Learning: In `.maestro/tests/room/share-message.yaml` (Rocket.Chat React Native), the `tapOn: point: 5%,10%` step is intentional: it taps the empty area above the bottom sheet and keyboard to dismiss both simultaneously. Using `action-sheet-handle` instead would only close the sheet but not the keyboard. This pattern is acceptable when both need to be dismissed together in a single step.

Applied to files:

  • .maestro/tests/assorted/profile.yaml
📚 Learning: 2025-12-17T15:56:22.578Z
Learnt from: OtavioStasiak
Repo: RocketChat/Rocket.Chat.ReactNative PR: 6499
File: app/containers/ServerItem/index.tsx:34-36
Timestamp: 2025-12-17T15:56:22.578Z
Learning: In the Rocket.Chat React Native codebase, for radio button components on iOS, include the selection state ("Selected"/"Unselected") in the accessibilityLabel instead of using accessibilityState={{ checked: hasCheck }}, because iOS VoiceOver has known issues with accessibilityRole="radio" + accessibilityState that prevent correct state announcement.

Applied to files:

  • app/views/CreateDiscussionView/index.tsx
  • app/containers/List/ListItem.tsx
📚 Learning: 2026-03-10T15:21:45.098Z
Learnt from: Rohit3523
Repo: RocketChat/Rocket.Chat.ReactNative PR: 7046
File: app/containers/InAppNotification/NotifierComponent.stories.tsx:46-75
Timestamp: 2026-03-10T15:21:45.098Z
Learning: In `app/containers/InAppNotification/NotifierComponent.tsx` (React Native, Rocket.Chat), `NotifierComponent` is exported as a Redux-connected component via `connect(mapStateToProps)`. The `isMasterDetail` prop is automatically injected from `state.app.isMasterDetail` and does not need to be passed explicitly at call sites or in Storybook stories that use the default (connected) export.

Applied to files:

  • app/containers/UIKit/Overflow.tsx
  • app/views/ShareListView/index.tsx
📚 Learning: 2026-03-04T20:13:17.288Z
Learnt from: divyanshu-patil
Repo: RocketChat/Rocket.Chat.ReactNative PR: 6957
File: ios/RocketChat Watch App/Views/MessageComposerView.swift:37-55
Timestamp: 2026-03-04T20:13:17.288Z
Learning: In the WatchOS app (ios/RocketChat Watch App) for Rocket.Chat React Native, using SwiftUI `Button` inside a `ScrollView` on WatchOS causes accidental message sends because button tap targets can be triggered during scroll gestures. `Text` with `.onTapGesture` is the preferred pattern for tappable items in scroll views on WatchOS. To preserve accessibility, add `.accessibilityAddTraits(.isButton)` and `.accessibilityLabel()` to the `Text` element instead.

Applied to files:

  • app/containers/List/ListItem.tsx
🔇 Additional comments (20)
app/containers/List/ListItem.tsx (2)

23-25: Good E2E/iOS gating strategy.

This guard is consistent with the codebase pattern and keeps behavior narrowly scoped to iOS E2E runs.


177-177: Conditional accessible usage looks correct.

Switching to accessible={!shouldDisableAccessibility} is a clean way to avoid iOS E2E accessibility interference without affecting normal app behavior.

ios/NotificationService/Info.plist (2)

25-26: LGTM!

The IS_OFFICIAL key reordering is a cosmetic change with no functional impact.


36-37: LGTM — New Architecture flag aligns with RN 0.81 upgrade.

Enabling RCTNewArchEnabled is appropriate for React Native 0.81, where the New Architecture is the default. Adding this flag consistently across all iOS targets (main app, share extension, notification service) ensures uniform build configuration.

Note: If the NotificationService extension is purely native Swift/ObjC code that doesn't use React Native directly, this flag may have no runtime effect but maintains consistency for any shared React Native dependencies.

.maestro/tests/room/room-actions.yaml (1)

124-127: Change from extendedWaitUntil to scrollUntilVisible looks appropriate.

Using scrollUntilVisible after the swipe gesture ensures the 'Star' option is brought into view, which is more robust if the action sheet content varies in length.

However, similar action sheet interactions for 'Unstar' (lines 174-177) and 'Pin' (lines 229-232) still use extendedWaitUntil after the swipe. If the 'Star' option requires scrolling, the same may apply to these other actions depending on their position in the sheet.

[approve_code_changes, request_verification]

#!/bin/bash
# Description: Check for similar patterns in this file and other Maestro tests 
# to verify if other action sheet interactions should also use scrollUntilVisible.

# Find all swipe followed by extendedWaitUntil patterns in this file
echo "=== Patterns in room-actions.yaml ==="
rg -n -A 5 'swipe:' .maestro/tests/room/room-actions.yaml

# Check other Maestro tests for similar action-sheet patterns
echo -e "\n=== Similar patterns in other Maestro test files ==="
rg -n -B 2 -A 5 'action-sheet-handle' .maestro/tests/ --glob '*.yaml' | head -100
.maestro/tests/assorted/profile.yaml (1)

99-100: LGTM!

Good refactor to use the shared hide-keyboard.yaml helper instead of brittle text-matching tapOn commands for dismissing the keyboard after username and nickname input. This improves maintainability and aligns with the project's pattern of centralizing common Maestro interactions.

Also applies to: 115-116

.maestro/tests/assorted/join-from-directory.yaml (1)

100-103: Good move to stable id selectors for directory filters.

Line 100/103 and Line 148/151 now target deterministic element ids, which should reduce Maestro flakiness across locales and UI state changes.

Also applies to: 148-151

app/views/DirectoryView/Options.tsx (2)

50-50: List.Radio testID wiring is correct and matches E2E usage.

Line 50 generates the ids used by the updated Maestro flow (directory-switch-users / directory-switch-teams).


74-74: Nice addition of explicit testID on the global users switch.

Line 74 improves automation targeting for this control and keeps behavior unchanged.

app/views/SecurityPrivacyView.tsx (1)

96-110: LGTM on testID relocation.

Moving testID from List.Item to the nested Switch component is consistent with the PR pattern. E2E tests targeting security-privacy-view-analytics-events and security-privacy-view-crash-report will now interact with the toggle controls directly.

app/views/CreateDiscussionView/index.tsx (1)

172-177: Remove concern about E2E test relocation—testID is newly added, not moved.

The testID='room-actions-encrypt' appears only on the Switch component and does not exist elsewhere in the codebase. This is a new testID addition, not a relocation from the List.Item. Since no existing E2E or Maestro tests reference this testID, there are no tests to update.

			> Likely an incorrect or invalid review comment.
app/containers/RoomItem/Actions.tsx (1)

12-12: LGTM — scheduleOnRN migration is correct.

The migration from runOnJS(triggerHideAnimation)(value) to scheduleOnRN(triggerHideAnimation, value) correctly maintains the same behavior for scheduling JS-thread work from the worklet context. The toValue parameter is passed appropriately for both RTL and LTR swipe threshold transitions.

Also applies to: 84-91

app/views/RoomView/List/components/List.tsx (1)

3-4: LGTM — Scroll handler migration is correct.

The scheduleOnRN(setVisible, boolean) pattern correctly schedules the React state update from the animated scroll handler worklet context.

Also applies to: 26-34

app/containers/RoomItem/Touchable.tsx (1)

9-9: LGTM — Gesture callback migration is correct.

Both long press and pan gesture end callbacks are properly migrated:

  • scheduleOnRN(handleLongPress) correctly handles the no-argument case
  • scheduleOnRN(handleRelease, event) correctly passes the gesture event containing translationX needed for the release logic

Also applies to: 173-189

app/containers/MessageComposer/hooks/useEmojiKeyboard.tsx (1)

4-6: LGTM — Emoji keyboard state synchronization is correctly migrated.

The useAnimatedReaction callbacks properly schedule React state updates via scheduleOnRN(setState, value). The boolean currentValue is passed correctly to synchronize showEmojiKeyboard and showEmojiSearchbar states.

Also applies to: 159-159, 175-175

app/containers/AudioPlayer/Seek.tsx (2)

67-85: LGTM — Gesture API migration from PanGestureHandler to Gesture.Pan() is correct.

The migration correctly maps:

  • Context capture via contextX shared value
  • onUpdate for translation updates with clamping
  • onEnd scheduling onChangeTime with milliseconds (matching expo-av expectations)
  • GestureDetector wrapper replacing PanGestureHandler

Also applies to: 125-127


12-12: The scheduleOnRN API usage is compatible and correctly implemented.

The function signature matches the expected pattern (function reference + arguments). However, be aware there are reported iOS crashes in audio worklet contexts (see react-native-worklets documentation). Test thoroughly on iOS devices to ensure no crashes occur when seeking completes in your audio player implementation.

app/containers/Button/index.tsx (1)

3-3: Import migration looks good.

No concerns on this segment.

app/views/ShareListView/index.tsx (1)

104-104: The code is safe as written. The share extension saves all media files locally to the app's cache directory (iOS) or app group container (Android) and passes only file:// URIs to the React component. According to the Expo legacy FileSystem API documentation, getInfoAsync(uri) returns the size field by default for file:// URIs without requiring the { size: true } option. The { size: true } option is only required for remote URIs like MediaLibrary (ph://) URIs, which do not occur in this code path.

			> Likely an incorrect or invalid review comment.
app/views/ShareView/Thumbs.tsx (1)

96-113: Nice simplification using shared Touch wrapper.

This removes platform branching in Thumb while keeping the interaction flow intact in this component.

Comment on lines +76 to +80
.onUpdate(event => {
const newX = contextX.value + event.translationX;
translateX.value = clamp(newX, 0, maxWidth.value);
scale.value = withTiming(1.3, { duration: 150 });
})
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue | 🟡 Minor

Scale animation triggers repeatedly on every pan update.

The withTiming(1.3, ...) call inside .onUpdate() executes on every gesture update event (potentially 60+ times per second), restarting the 150ms animation each time. This is inefficient and may cause visual jitter.

Move the scale-up animation to .onStart() where it runs once when the gesture begins.

🔧 Proposed fix
 .onStart(() => {
   contextX.value = translateX.value;
+  scale.value = withTiming(1.3, { duration: 150 });
 })
 .onUpdate(event => {
   const newX = contextX.value + event.translationX;
   translateX.value = clamp(newX, 0, maxWidth.value);
-  scale.value = withTiming(1.3, { duration: 150 });
 })
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@app/containers/AudioPlayer/Seek.tsx` around lines 76 - 80, The scale
animation is being restarted on every gesture update because withTiming(1.3,
...) is called inside the .onUpdate handler; move the scale-up call into the
gesture's .onStart handler so it runs once when the gesture begins (e.g., call
scale.value = withTiming(1.3, { duration: 150 }) in .onStart), keep
translateX.value and clamp logic inside .onUpdate using contextX and maxWidth,
and ensure you restore scale (e.g., scale.value = withTiming(1) or appropriate
reset) in .onEnd/.onFinalize so the knob returns to normal after the gesture.

Comment on lines +44 to +57
const touchable: { [key: string]: React.RefObject<View | null> } = {};

export const Overflow = ({ element, loading, action, parser }: IOverflow) => {
const { theme } = useTheme();
const options = element?.options || [];
const blockId = element?.blockId || '';
const [show, onShow] = useState(false);

if (!touchable[blockId]) {
touchable[blockId] = React.createRef();
}

const touchableRef = touchable[blockId] as React.RefObject<any>;

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue | 🟠 Major

Avoid module-scoped ref registry for anchors.

The global touchable map can leak refs over time and can collide when blockId repeats/empties, which may anchor Popover to the wrong element. Prefer an instance-local ref.

💡 Suggested fix
-import React, { useState } from 'react';
-import { FlatList, StyleSheet, Text, type View } from 'react-native';
+import React, { useRef, useState } from 'react';
+import { FlatList, StyleSheet, Text } from 'react-native';
@@
-const touchable: { [key: string]: React.RefObject<View | null> } = {};
-
 export const Overflow = ({ element, loading, action, parser }: IOverflow) => {
 	const { theme } = useTheme();
 	const options = element?.options || [];
-	const blockId = element?.blockId || '';
 	const [show, onShow] = useState(false);
-
-	if (!touchable[blockId]) {
-		touchable[blockId] = React.createRef();
-	}
-
-	const touchableRef = touchable[blockId] as React.RefObject<any>;
+	const touchableRef = useRef<React.ElementRef<typeof Touch>>(null);
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@app/containers/UIKit/Overflow.tsx` around lines 44 - 57, The module-scoped
touchable map (touchable) used in Overflow leaks refs and can mis-anchor Popover
when blockId repeats or is empty; replace it with an instance-local ref inside
the Overflow component (e.g., remove touchable and create const touchableRef =
useRef<View | null>(null) inside the Overflow function), attach that ref to the
touchable element, and remove all accesses to the global touchable[blockId] so
the ref lifecycle is tied to the component instance (update any code using
touchableRef variable name or blockId-dependent lookup to use the local
touchableRef and, if needed, handle blockId changes with useEffect).

Comment on lines 106 to 112
<List.Item
title='Send_crash_report'
testID='security-privacy-view-crash-report'
right={() => <Switch value={crashReportState} onValueChange={toggleCrashReport} />}
right={() => (
<Switch value={crashReportState} onValueChange={toggleCrashReport} testID='security-privacy-view-crash-report' />
)}
additionalAccessibilityLabel={analyticsEventsState}
/>
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue | 🟡 Minor

Pre-existing bug: wrong accessibility label for crash report toggle.

Line 111 uses analyticsEventsState for the "Send_crash_report" item's additionalAccessibilityLabel, but it should use crashReportState. This causes VoiceOver to announce the wrong enabled/disabled state for this toggle.

Since you're modifying adjacent code, consider fixing this while you're here.

🐛 Proposed fix
 				<List.Item
 					title='Send_crash_report'
 					right={() => (
 						<Switch value={crashReportState} onValueChange={toggleCrashReport} testID='security-privacy-view-crash-report' />
 					)}
-					additionalAccessibilityLabel={analyticsEventsState}
+					additionalAccessibilityLabel={crashReportState}
 				/>
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
<List.Item
title='Send_crash_report'
testID='security-privacy-view-crash-report'
right={() => <Switch value={crashReportState} onValueChange={toggleCrashReport} />}
right={() => (
<Switch value={crashReportState} onValueChange={toggleCrashReport} testID='security-privacy-view-crash-report' />
)}
additionalAccessibilityLabel={analyticsEventsState}
/>
<List.Item
title='Send_crash_report'
right={() => (
<Switch value={crashReportState} onValueChange={toggleCrashReport} testID='security-privacy-view-crash-report' />
)}
additionalAccessibilityLabel={crashReportState}
/>
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@app/views/SecurityPrivacyView.tsx` around lines 106 - 112, The
"Send_crash_report" List.Item is using the wrong accessibility label variable;
replace the additionalAccessibilityLabel value currently set to
analyticsEventsState with crashReportState so VoiceOver reports the correct
enabled/disabled state for the crash report toggle. Update the List.Item (title
'Send_crash_report') to use crashReportState for additionalAccessibilityLabel
and keep the existing Switch props (value={crashReportState},
onValueChange={toggleCrashReport}, testID) unchanged.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Projects

None yet

Development

Successfully merging this pull request may close these issues.

bug(iOS): switch having layout problem after iOS 26 update

3 participants