From 74a6409b76c33125f706a412c2fc3b4a1b3104d4 Mon Sep 17 00:00:00 2001 From: Loren Posen Date: Wed, 15 Oct 2025 10:55:53 -0700 Subject: [PATCH 1/8] feat: add back icon assets for multiple resolutions and platforms --- src/inbox/assets/back-icon.png | Bin 0 -> 207 bytes src/inbox/assets/back-icon@1x.android.png | Bin 0 -> 100 bytes src/inbox/assets/back-icon@1x.ios.png | Bin 0 -> 290 bytes src/inbox/assets/back-icon@2x.android.png | Bin 0 -> 134 bytes src/inbox/assets/back-icon@2x.ios.png | Bin 0 -> 405 bytes src/inbox/assets/back-icon@3x.android.png | Bin 0 -> 167 bytes src/inbox/assets/back-icon@3x.ios.png | Bin 0 -> 761 bytes src/inbox/assets/back-icon@4x.android.png | Bin 0 -> 207 bytes src/inbox/assets/back-icon@4x.ios.png | Bin 0 -> 809 bytes 9 files changed, 0 insertions(+), 0 deletions(-) create mode 100644 src/inbox/assets/back-icon.png create mode 100644 src/inbox/assets/back-icon@1x.android.png create mode 100644 src/inbox/assets/back-icon@1x.ios.png create mode 100644 src/inbox/assets/back-icon@2x.android.png create mode 100644 src/inbox/assets/back-icon@2x.ios.png create mode 100644 src/inbox/assets/back-icon@3x.android.png create mode 100644 src/inbox/assets/back-icon@3x.ios.png create mode 100644 src/inbox/assets/back-icon@4x.android.png create mode 100644 src/inbox/assets/back-icon@4x.ios.png diff --git a/src/inbox/assets/back-icon.png b/src/inbox/assets/back-icon.png new file mode 100644 index 0000000000000000000000000000000000000000..17e52e8550e5668f7117bcb755beb70c3a21c9e9 GIT binary patch literal 207 zcmeAS@N?(olHy`uVBq!ia0vp^2_VeD0wg^q?%xcgrg^$JhEy=Vy>(TP*-+rv1Kz7M z%I*XNJ!(VLzdH4Ge@o8ai4)e^3vK%P+bZq#{MQci z?P`C0d3kxcKa^1~`zt>B;jiLd|K_aKfAEz>edFmDy)D6SUw`+o@R*?B*u=sqq~bB* zgRIJrKN^lr3+tD}bFTI}~XjubmVU5CWQ(LZ;AgeuH{an^LB{Ts5E|^v% literal 0 HcmV?d00001 diff --git a/src/inbox/assets/back-icon@1x.android.png b/src/inbox/assets/back-icon@1x.android.png new file mode 100644 index 0000000000000000000000000000000000000000..083db295f474b9903408258c71818c2c49151d35 GIT binary patch literal 100 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM0wlfaz7_*1ZBG}+kP60RiDL^GIXr{@&3`D~ x%$9kI*}bEvW6`UVO)xQz+sFfYmle_k4J{c^Y-u@of;y~NbrRr@}xDqOoe#7(!ujI zW)IKPtpOg9x^ZGiV^G!WUnY384Jwg^1?qzoV?Vqi%~F-r131d0cdTWi1E3pVgr)a^ zbA?Lk#0WF#3+Yvg4uEz<4l(HmchxGX6;goJ^ovoWl9~bgne?yCQ=RAlXaeYC>D}3~ oOlpASO$!ea%F`=`*pIb34xO&@x>?fx0{{R307*qoM6N<$f~pOBr2qf` literal 0 HcmV?d00001 diff --git a/src/inbox/assets/back-icon@2x.android.png b/src/inbox/assets/back-icon@2x.android.png new file mode 100644 index 0000000000000000000000000000000000000000..6de0a1cbb365dfd5d9274890d243ca2321f32235 GIT binary patch literal 134 zcmeAS@N?(olHy`uVBq!ia0vp^1|ZDA0wn)(8}a}tUr!gukP61P(+)Bo5D;KluXOkB z|6f;`Uoa>nrTyP5;4+Jm;dJDxe#>oP@r$y*#>rHDd13s5eN)8##R;WH*ZD4M;?i1i i;i&z7`QNqd4J-@3aX4v+{p181%;4$j=d#Wzp$P!F!7nZV literal 0 HcmV?d00001 diff --git a/src/inbox/assets/back-icon@2x.ios.png b/src/inbox/assets/back-icon@2x.ios.png new file mode 100644 index 0000000000000000000000000000000000000000..63de0e303aae4f3f09bbf5834290131501232f64 GIT binary patch literal 405 zcmV;G0c!q$3ng)KwR%cav2NeVj#yi|twB8nJ$ zhJ9XSgC81to-du2Xkr{|@HcXhBf~opLkgz`JWJYh7b%=;afNgn{8BUDujC*_hPNS> z7&ls+kwoI8u%elk7pg@(StZ8P4^;l-*^fZJ1`5~OHBd;D;qqM+{Bs~b$&5^qEx(K8 zx9Nf9ZEAjO1n+^rUo`oHziD{zE6kAXuYyr+hHU>u3T%dKe|C&IvG2#fso;0*pv4!B zOc_R!8Jq}55TlLyCq{c8swLA2SGn}>CsPgp4HtE6hRq5700000NkvXXu0mjfokp-j literal 0 HcmV?d00001 diff --git a/src/inbox/assets/back-icon@3x.android.png b/src/inbox/assets/back-icon@3x.android.png new file mode 100644 index 0000000000000000000000000000000000000000..15a983a67d97c9a6c39d91550a528c37c53a9e3e GIT binary patch literal 167 zcmeAS@N?(olHy`uVBq!ia0vp^9w5xY0wn)GsXhaw@;qG}Ln;{G-aN{9*nr0&aBl6p z15q2fw>DO<$f$_t-m`#N(R0!@qfL6}is!BA-aP5mqcux*S)AYDzOuXc@WF!zU-mWw z!RNIrdq3B7GJgpV3USMNx+#Bt3lxOZSlm}!I$z&?|8f=2ODy@*gQAaCs?P@6%;4$j K=d#Wzp$P!X&_+4{ literal 0 HcmV?d00001 diff --git a/src/inbox/assets/back-icon@3x.ios.png b/src/inbox/assets/back-icon@3x.ios.png new file mode 100644 index 0000000000000000000000000000000000000000..2320fd74d1781dc8a74cc131d8c3bb17fc52bae4 GIT binary patch literal 761 zcmVAUlM5g1Q_DYVitMm-$7Tx z*ro&sTZ*z6>QNY&<*{MBLwj|YS(XqXD^V764Y9I1Okis$Okg8b+mr|wUq@l0HcEq; zVTpm+Kw?IyI+FyL&Ez%>QDU|-7*wFaOtGZEY$rQ2K-HNzTzn^q>8HU=v}CyWZt?*4 zR&^#h0``%e>7l`lw-g9CKz639sxv7OaEQQsKxaB>Fyky00*;WF4yw+iM!+!&6ScM) z%veiJ{LO>gk?HFakuuxmtTUZbi;r*1=ySHXQyoauVV}3fsCFdksL$Qv?Q2bBWbLdnCsmVuHOp65Sf{(`X7f(3UB_ENM%STv@ zvIOdikIG_Hb<}I(pkKw$XH%o9lBnz0zU=BVRS7|_2-Hp7L=H8o0zzIAXX+Lzml{=$ zMBTymK^Jb-%uZCPG}qfJds0JQ2XB$6bg6-+U-Ef=(TP*-+rv1Kz7M z%I*XNJ!(VLzdH4Ge@o8ai4)e^3vK%P+bZq#{MQci z?P`C0d3kxcKa^1~`zt>B;jiLd|K_aKfAEz>edFmDy)D6SUw`+o@R*?B*u=sqq~bB* zgRIJrKN^lr3+tD}bFTI}~XjubmVU5CWQ(LZ;AgeuH{an^LB{Ts5E|^v% literal 0 HcmV?d00001 diff --git a/src/inbox/assets/back-icon@4x.ios.png b/src/inbox/assets/back-icon@4x.ios.png new file mode 100644 index 0000000000000000000000000000000000000000..b26634e594135d181375c3316b2ce059a31c6665 GIT binary patch literal 809 zcmV+^1J?YBP)YD1miR7mE{DV$Qg{||{bo~dA%(lFKsoPR23s4@v0mrvvjQGomaY7ADLDuqH1 zvGK4`4{>#jmgPlOeM+>Mf_`%D1%nu+I@RZN_76>V5WO1tOIiKYhsr9>jYgGKs=`*N zv}jyyRB5p)?goPB+{p5Jb-|H(#biuZo$8V=bj@m<&Aw*Mzvy;mD@cB;LBZVYApQ(D zdpPq|r)sw;OT$xM%b9U3xD%GW6CGq63GPHqLE>$f#-ZTp)T|)+jjzs91<7wcE?A); z`OV+XYNgUb`*8ccu~0$s2Ow%Rm0<*BXJiDW8UkwI7}mD<7*>7|TUr^wfilifZVG;4 n@$E^mUzjZzWv9I6RnQJ~H00000NkvXXu0mjf?i-Uu literal 0 HcmV?d00001 From 4308017aeb447334ccaac64e356b64b7a09cf4d9 Mon Sep 17 00:00:00 2001 From: Loren Posen Date: Wed, 15 Oct 2025 10:56:22 -0700 Subject: [PATCH 2/8] refactor: add HeaderBackButton component and integrate it into IterableInboxMessageDisplay --- src/inbox/components/HeaderBackButton.tsx | 39 +++++++++++++++++++ .../IterableInboxMessageDisplay.tsx | 12 +++++- 2 files changed, 50 insertions(+), 1 deletion(-) create mode 100644 src/inbox/components/HeaderBackButton.tsx diff --git a/src/inbox/components/HeaderBackButton.tsx b/src/inbox/components/HeaderBackButton.tsx new file mode 100644 index 000000000..e46948050 --- /dev/null +++ b/src/inbox/components/HeaderBackButton.tsx @@ -0,0 +1,39 @@ +import { + StyleSheet, + Text, + TouchableWithoutFeedback, + View, + type TouchableWithoutFeedbackProps, +} from 'react-native'; +import { Icon } from 'react-native-vector-icons/Icon'; +import { ITERABLE_INBOX_COLORS } from '../constants/colors'; + +const styles = StyleSheet.create({ + + returnButton: { + alignItems: 'center', + flexDirection: 'row', + }, + + returnButtonIcon: { + color: ITERABLE_INBOX_COLORS.BUTTON_PRIMARY_TEXT, + fontSize: 40, + paddingLeft: 0, + }, + + returnButtonText: { + color: ITERABLE_INBOX_COLORS.BUTTON_PRIMARY_TEXT, + fontSize: 20, + }, +}); + +export const HeaderBackButton = (props: TouchableWithoutFeedbackProps) => { + return ( + + + + Inbox + + + ); +}; diff --git a/src/inbox/components/IterableInboxMessageDisplay.tsx b/src/inbox/components/IterableInboxMessageDisplay.tsx index 7e6798c73..6799dafea 100644 --- a/src/inbox/components/IterableInboxMessageDisplay.tsx +++ b/src/inbox/components/IterableInboxMessageDisplay.tsx @@ -23,9 +23,9 @@ import { IterableInAppCloseSource, IterableInAppLocation, } from '../../inApp'; - import { ITERABLE_INBOX_COLORS } from '../constants'; import { type IterableInboxRowViewModel } from '../types'; +import { HeaderBackButton } from './HeaderBackButton'; /** * Props for the IterableInboxMessageDisplay component. @@ -222,6 +222,16 @@ export const IterableInboxMessageDisplay = ({ + { + returnToInbox(); + Iterable.trackInAppClose( + rowViewModel.inAppMessage, + IterableInAppLocation.inbox, + IterableInAppCloseSource.back + ); + }} + /> { returnToInbox(); From 2c9a954bf9f0656540f5f3bb3b2268cc5dcb46b3 Mon Sep 17 00:00:00 2001 From: Loren Posen Date: Wed, 15 Oct 2025 13:24:19 -0700 Subject: [PATCH 3/8] refactor: create HeaderBackButton and replace back-arrow --- src/inbox/components/HeaderBackButton.tsx | 75 ++++++++++++++++--- .../IterableInboxMessageDisplay.tsx | 39 +--------- 2 files changed, 68 insertions(+), 46 deletions(-) diff --git a/src/inbox/components/HeaderBackButton.tsx b/src/inbox/components/HeaderBackButton.tsx index e46948050..15687ac4e 100644 --- a/src/inbox/components/HeaderBackButton.tsx +++ b/src/inbox/components/HeaderBackButton.tsx @@ -1,38 +1,93 @@ import { + Image, + PixelRatio, + Platform, StyleSheet, Text, TouchableWithoutFeedback, View, + type ImageSourcePropType, type TouchableWithoutFeedbackProps, } from 'react-native'; -import { Icon } from 'react-native-vector-icons/Icon'; import { ITERABLE_INBOX_COLORS } from '../constants/colors'; +// Base64 encoded back arrow icons +// [Original image](https://github.com/react-navigation/react-navigation/blob/main/packages/elements/src/assets/back-icon%404x.ios.png) +const backArrowLarge = + ''; + +export const ICON_SIZE = Platform.OS === 'ios' ? 21 : 24; +export const ICON_MARGIN = Platform.OS === 'ios' ? 8 : 3; +const ICON_COLOR = ITERABLE_INBOX_COLORS.BUTTON_PRIMARY_TEXT; + const styles = StyleSheet.create({ + icon: { + height: ICON_SIZE, + margin: ICON_MARGIN, + width: ICON_SIZE, + }, returnButton: { alignItems: 'center', + display: 'flex', flexDirection: 'row', }, - returnButtonIcon: { - color: ITERABLE_INBOX_COLORS.BUTTON_PRIMARY_TEXT, - fontSize: 40, - paddingLeft: 0, - }, - returnButtonText: { color: ITERABLE_INBOX_COLORS.BUTTON_PRIMARY_TEXT, fontSize: 20, }, }); -export const HeaderBackButton = (props: TouchableWithoutFeedbackProps) => { +/** + * Props for the HeaderBackButton component. + */ +export interface HeaderBackButtonProps extends TouchableWithoutFeedbackProps { + /** + * The text to display next to the back arrow. + */ + label?: string; + /** + * The URI of the image to display. + * + * This defaults to a base64 encoded version of [the back arrow used in react-navigation/elements](https://github.com/react-navigation/react-navigation/blob/main/packages/elements/src/assets/back-icon%404x.ios.png) + */ + imageUri?: string; + /** + * The source of the image to display. + */ + imageSource?: ImageSourcePropType; +} + +/** + * A back arrow button used in a header + * + * @returns A button with a back arrow + */ +export const HeaderBackButton = ({ + label, + imageUri = backArrowLarge, + imageSource = { + uri: imageUri, + width: PixelRatio.getPixelSizeForLayoutSize(ICON_SIZE), + height: PixelRatio.getPixelSizeForLayoutSize(ICON_SIZE), + }, + ...props +}: HeaderBackButtonProps) => { return ( - - Inbox + + {label && {label}} ); diff --git a/src/inbox/components/IterableInboxMessageDisplay.tsx b/src/inbox/components/IterableInboxMessageDisplay.tsx index 6799dafea..d42306a04 100644 --- a/src/inbox/components/IterableInboxMessageDisplay.tsx +++ b/src/inbox/components/IterableInboxMessageDisplay.tsx @@ -1,13 +1,12 @@ import { useEffect, useState } from 'react'; import { Linking, + Platform, ScrollView, StyleSheet, Text, - TouchableWithoutFeedback, View, } from 'react-native'; -import Icon from 'react-native-vector-icons/Ionicons'; import { WebView, type WebViewMessageEvent } from 'react-native-webview'; import { @@ -86,6 +85,7 @@ export const IterableInboxMessageDisplay = ({ header: { flexDirection: 'row', + height: Platform.OS === 'ios' ? 44 : 56, justifyContent: 'center', width: '100%', }, @@ -119,11 +119,6 @@ export const IterableInboxMessageDisplay = ({ fontWeight: 'bold', }, - returnButton: { - alignItems: 'center', - flexDirection: 'row', - }, - returnButtonContainer: { alignItems: 'center', flexDirection: 'row', @@ -133,17 +128,6 @@ export const IterableInboxMessageDisplay = ({ width: '25%', ...(isPortrait ? {} : { marginLeft: 80 }), }, - - returnButtonIcon: { - color: ITERABLE_INBOX_COLORS.BUTTON_PRIMARY_TEXT, - fontSize: 40, - paddingLeft: 0, - }, - - returnButtonText: { - color: ITERABLE_INBOX_COLORS.BUTTON_PRIMARY_TEXT, - fontSize: 20, - }, }); const JS = ` @@ -223,6 +207,7 @@ export const IterableInboxMessageDisplay = ({ { returnToInbox(); Iterable.trackInAppClose( @@ -232,24 +217,6 @@ export const IterableInboxMessageDisplay = ({ ); }} /> - { - returnToInbox(); - Iterable.trackInAppClose( - rowViewModel.inAppMessage, - IterableInAppLocation.inbox, - IterableInAppCloseSource.back - ); - }} - > - - - Inbox - - From 9bf11789bc9c7fc0eeaa8750964a9dba4bea2a87 Mon Sep 17 00:00:00 2001 From: Loren Posen Date: Wed, 15 Oct 2025 13:24:39 -0700 Subject: [PATCH 4/8] chore: add prettier-eslint and update dependencies in package.json and yarn.lock --- package.json | 1 + yarn.lock | 253 ++++++++++++++++++++++++++++++++++++++++++++++++++- 2 files changed, 249 insertions(+), 5 deletions(-) diff --git a/package.json b/package.json index ef832a4ad..c3e1dfc52 100644 --- a/package.json +++ b/package.json @@ -89,6 +89,7 @@ "eslint-plugin-tsdoc": "^0.3.0", "jest": "^29.7.0", "prettier": "^3.0.3", + "prettier-eslint": "^16.4.2", "react": "19.0.0", "react-native": "0.79.3", "react-native-builder-bob": "^0.40.4", diff --git a/yarn.lock b/yarn.lock index d9961ac53..ea98be24c 100644 --- a/yarn.lock +++ b/yarn.lock @@ -3142,6 +3142,7 @@ __metadata: eslint-plugin-tsdoc: ^0.3.0 jest: ^29.7.0 prettier: ^3.0.3 + prettier-eslint: ^16.4.2 react: 19.0.0 react-native: 0.79.3 react-native-builder-bob: ^0.40.4 @@ -4750,6 +4751,24 @@ __metadata: languageName: node linkType: hard +"@typescript-eslint/parser@npm:^6.21.0": + version: 6.21.0 + resolution: "@typescript-eslint/parser@npm:6.21.0" + dependencies: + "@typescript-eslint/scope-manager": 6.21.0 + "@typescript-eslint/types": 6.21.0 + "@typescript-eslint/typescript-estree": 6.21.0 + "@typescript-eslint/visitor-keys": 6.21.0 + debug: ^4.3.4 + peerDependencies: + eslint: ^7.0.0 || ^8.0.0 + peerDependenciesMeta: + typescript: + optional: true + checksum: 162fe3a867eeeffda7328bce32dae45b52283c68c8cb23258fb9f44971f761991af61f71b8c9fe1aa389e93dfe6386f8509c1273d870736c507d76dd40647b68 + languageName: node + linkType: hard + "@typescript-eslint/parser@npm:^7.1.1": version: 7.18.0 resolution: "@typescript-eslint/parser@npm:7.18.0" @@ -4796,6 +4815,16 @@ __metadata: languageName: node linkType: hard +"@typescript-eslint/scope-manager@npm:6.21.0": + version: 6.21.0 + resolution: "@typescript-eslint/scope-manager@npm:6.21.0" + dependencies: + "@typescript-eslint/types": 6.21.0 + "@typescript-eslint/visitor-keys": 6.21.0 + checksum: 71028b757da9694528c4c3294a96cc80bc7d396e383a405eab3bc224cda7341b88e0fc292120b35d3f31f47beac69f7083196c70616434072fbcd3d3e62d3376 + languageName: node + linkType: hard + "@typescript-eslint/scope-manager@npm:7.18.0": version: 7.18.0 resolution: "@typescript-eslint/scope-manager@npm:7.18.0" @@ -4855,6 +4884,13 @@ __metadata: languageName: node linkType: hard +"@typescript-eslint/types@npm:6.21.0": + version: 6.21.0 + resolution: "@typescript-eslint/types@npm:6.21.0" + checksum: 9501b47d7403417af95fc1fb72b2038c5ac46feac0e1598a46bcb43e56a606c387e9dcd8a2a0abe174c91b509f2d2a8078b093786219eb9a01ab2fbf9ee7b684 + languageName: node + linkType: hard + "@typescript-eslint/types@npm:7.18.0": version: 7.18.0 resolution: "@typescript-eslint/types@npm:7.18.0" @@ -4887,6 +4923,25 @@ __metadata: languageName: node linkType: hard +"@typescript-eslint/typescript-estree@npm:6.21.0": + version: 6.21.0 + resolution: "@typescript-eslint/typescript-estree@npm:6.21.0" + dependencies: + "@typescript-eslint/types": 6.21.0 + "@typescript-eslint/visitor-keys": 6.21.0 + debug: ^4.3.4 + globby: ^11.1.0 + is-glob: ^4.0.3 + minimatch: 9.0.3 + semver: ^7.5.4 + ts-api-utils: ^1.0.1 + peerDependenciesMeta: + typescript: + optional: true + checksum: dec02dc107c4a541e14fb0c96148f3764b92117c3b635db3a577b5a56fc48df7a556fa853fb82b07c0663b4bf2c484c9f245c28ba3e17e5cb0918ea4cab2ea21 + languageName: node + linkType: hard + "@typescript-eslint/typescript-estree@npm:7.18.0": version: 7.18.0 resolution: "@typescript-eslint/typescript-estree@npm:7.18.0" @@ -4981,6 +5036,16 @@ __metadata: languageName: node linkType: hard +"@typescript-eslint/visitor-keys@npm:6.21.0": + version: 6.21.0 + resolution: "@typescript-eslint/visitor-keys@npm:6.21.0" + dependencies: + "@typescript-eslint/types": 6.21.0 + eslint-visitor-keys: ^3.4.1 + checksum: 67c7e6003d5af042d8703d11538fca9d76899f0119130b373402819ae43f0bc90d18656aa7add25a24427ccf1a0efd0804157ba83b0d4e145f06107d7d1b7433 + languageName: node + linkType: hard + "@typescript-eslint/visitor-keys@npm:7.18.0": version: 7.18.0 resolution: "@typescript-eslint/visitor-keys@npm:7.18.0" @@ -5186,6 +5251,13 @@ __metadata: languageName: node linkType: hard +"ansi-regex@npm:^2.0.0": + version: 2.1.1 + resolution: "ansi-regex@npm:2.1.1" + checksum: 190abd03e4ff86794f338a31795d262c1dfe8c91f7e01d04f13f646f1dcb16c5800818f886047876f1272f065570ab86b24b99089f8b68a0e11ff19aed4ca8f1 + languageName: node + linkType: hard + "ansi-regex@npm:^4.1.0": version: 4.1.1 resolution: "ansi-regex@npm:4.1.1" @@ -5207,6 +5279,13 @@ __metadata: languageName: node linkType: hard +"ansi-styles@npm:^2.2.1": + version: 2.2.1 + resolution: "ansi-styles@npm:2.2.1" + checksum: ebc0e00381f2a29000d1dac8466a640ce11943cef3bda3cd0020dc042e31e1058ab59bf6169cd794a54c3a7338a61ebc404b7c91e004092dd20e028c432c9c2c + languageName: node + linkType: hard + "ansi-styles@npm:^3.2.0, ansi-styles@npm:^3.2.1": version: 3.2.1 resolution: "ansi-styles@npm:3.2.1" @@ -5916,6 +5995,19 @@ __metadata: languageName: node linkType: hard +"chalk@npm:^1.1.3": + version: 1.1.3 + resolution: "chalk@npm:1.1.3" + dependencies: + ansi-styles: ^2.2.1 + escape-string-regexp: ^1.0.2 + has-ansi: ^2.0.0 + strip-ansi: ^3.0.0 + supports-color: ^2.0.0 + checksum: 9d2ea6b98fc2b7878829eec223abcf404622db6c48396a9b9257f6d0ead2acf18231ae368d6a664a83f272b0679158da12e97b5229f794939e555cc574478acd + languageName: node + linkType: hard + "chalk@npm:^2.4.2": version: 2.4.2 resolution: "chalk@npm:2.4.2" @@ -6222,6 +6314,13 @@ __metadata: languageName: node linkType: hard +"common-tags@npm:^1.8.2": + version: 1.8.2 + resolution: "common-tags@npm:1.8.2" + checksum: 767a6255a84bbc47df49a60ab583053bb29a7d9687066a18500a516188a062c4e4cd52de341f22de0b07062e699b1b8fe3cfa1cb55b241cb9301aeb4f45b4dff + languageName: node + linkType: hard + "compare-func@npm:^2.0.0": version: 2.0.0 resolution: "compare-func@npm:2.0.0" @@ -6937,6 +7036,13 @@ __metadata: languageName: node linkType: hard +"dlv@npm:^1.1.3": + version: 1.1.3 + resolution: "dlv@npm:1.1.3" + checksum: d7381bca22ed11933a1ccf376db7a94bee2c57aa61e490f680124fa2d1cd27e94eba641d9f45be57caab4f9a6579de0983466f620a2cd6230d7ec93312105ae7 + languageName: node + linkType: hard + "doctrine@npm:^2.1.0": version: 2.1.0 resolution: "doctrine@npm:2.1.0" @@ -7293,7 +7399,7 @@ __metadata: languageName: node linkType: hard -"escape-string-regexp@npm:^1.0.5": +"escape-string-regexp@npm:^1.0.2, escape-string-regexp@npm:^1.0.5": version: 1.0.5 resolution: "escape-string-regexp@npm:1.0.5" checksum: 6092fda75c63b110c706b6a9bfde8a612ad595b628f0bd2147eea1d3406723020810e591effc7db1da91d80a71a737a313567c5abb3813e8d9c71f4aa595b410 @@ -7510,7 +7616,7 @@ __metadata: languageName: node linkType: hard -"eslint-scope@npm:^7.2.2": +"eslint-scope@npm:^7.1.1, eslint-scope@npm:^7.2.2": version: 7.2.2 resolution: "eslint-scope@npm:7.2.2" dependencies: @@ -7534,7 +7640,7 @@ __metadata: languageName: node linkType: hard -"eslint@npm:^8.51.0": +"eslint@npm:^8.51.0, eslint@npm:^8.57.1": version: 8.57.1 resolution: "eslint@npm:8.57.1" dependencies: @@ -7582,7 +7688,7 @@ __metadata: languageName: node linkType: hard -"espree@npm:^9.6.0, espree@npm:^9.6.1": +"espree@npm:^9.3.1, espree@npm:^9.6.0, espree@npm:^9.6.1": version: 9.6.1 resolution: "espree@npm:9.6.1" dependencies: @@ -7603,7 +7709,7 @@ __metadata: languageName: node linkType: hard -"esquery@npm:^1.4.2": +"esquery@npm:^1.4.0, esquery@npm:^1.4.2": version: 1.6.0 resolution: "esquery@npm:1.6.0" dependencies: @@ -8419,6 +8525,15 @@ __metadata: languageName: node linkType: hard +"has-ansi@npm:^2.0.0": + version: 2.0.0 + resolution: "has-ansi@npm:2.0.0" + dependencies: + ansi-regex: ^2.0.0 + checksum: 1b51daa0214440db171ff359d0a2d17bc20061164c57e76234f614c91dbd2a79ddd68dfc8ee73629366f7be45a6df5f2ea9de83f52e1ca24433f2cc78c35d8ec + languageName: node + linkType: hard + "has-bigints@npm:^1.0.1, has-bigints@npm:^1.0.2": version: 1.0.2 resolution: "has-bigints@npm:1.0.2" @@ -10427,6 +10542,23 @@ __metadata: languageName: node linkType: hard +"loglevel-colored-level-prefix@npm:^1.0.0": + version: 1.0.0 + resolution: "loglevel-colored-level-prefix@npm:1.0.0" + dependencies: + chalk: ^1.1.3 + loglevel: ^1.4.1 + checksum: 146aa7d0ea900d6d8523e945b2265be240e4c7c4752dae678983764dd756c44194684af1ee8ea721feff4c4f8c5771544a02a6cd8b269a663cffe9b4fcf955f1 + languageName: node + linkType: hard + +"loglevel@npm:^1.4.1": + version: 1.9.2 + resolution: "loglevel@npm:1.9.2" + checksum: 896c67b90a507bfcfc1e9a4daa7bf789a441dd70d95cd13b998d6dd46233a3bfadfb8fadb07250432bbfb53bf61e95f2520f9b11f9d3175cc460e5c251eca0af + languageName: node + linkType: hard + "loose-envify@npm:^1.0.0, loose-envify@npm:^1.4.0": version: 1.4.0 resolution: "loose-envify@npm:1.4.0" @@ -11164,6 +11296,15 @@ __metadata: languageName: node linkType: hard +"minimatch@npm:9.0.3": + version: 9.0.3 + resolution: "minimatch@npm:9.0.3" + dependencies: + brace-expansion: ^2.0.1 + checksum: 253487976bf485b612f16bf57463520a14f512662e592e95c571afdab1442a6a6864b6c88f248ce6fc4ff0b6de04ac7aa6c8bb51e868e99d1d65eb0658a708b5 + languageName: node + linkType: hard + "minimatch@npm:^3.0.4, minimatch@npm:^3.0.5, minimatch@npm:^3.1.1, minimatch@npm:^3.1.2": version: 3.1.2 resolution: "minimatch@npm:3.1.2" @@ -12097,6 +12238,34 @@ __metadata: languageName: node linkType: hard +"prettier-eslint@npm:^16.4.2": + version: 16.4.2 + resolution: "prettier-eslint@npm:16.4.2" + dependencies: + "@typescript-eslint/parser": ^6.21.0 + common-tags: ^1.8.2 + dlv: ^1.1.3 + eslint: ^8.57.1 + indent-string: ^4.0.0 + lodash.merge: ^4.6.2 + loglevel-colored-level-prefix: ^1.0.0 + prettier: ^3.5.3 + pretty-format: ^29.7.0 + require-relative: ^0.8.7 + tslib: ^2.8.1 + vue-eslint-parser: ^9.4.3 + peerDependencies: + prettier-plugin-svelte: ^3.0.0 + svelte-eslint-parser: "*" + peerDependenciesMeta: + prettier-plugin-svelte: + optional: true + svelte-eslint-parser: + optional: true + checksum: ad420f2d3b6f0c055e0eefed2f32876e4ac29d5c0202778ae531438224c7d07b67dcfb64054bc61a0cc88f231988198f229395361a9b2112ad048d08b6d5bc80 + languageName: node + linkType: hard + "prettier-linter-helpers@npm:^1.0.0": version: 1.0.0 resolution: "prettier-linter-helpers@npm:1.0.0" @@ -12115,6 +12284,15 @@ __metadata: languageName: node linkType: hard +"prettier@npm:^3.5.3": + version: 3.6.2 + resolution: "prettier@npm:3.6.2" + bin: + prettier: bin/prettier.cjs + checksum: 0206f5f437892e8858f298af8850bf9d0ef1c22e21107a213ba56bfb9c2387a2020bfda244a20161d8e3dad40c6b04101609a55d370dece53d0a31893b64f861 + languageName: node + linkType: hard + "pretty-format@npm:30.2.0, pretty-format@npm:^30.0.5": version: 30.2.0 resolution: "pretty-format@npm:30.2.0" @@ -12999,6 +13177,13 @@ __metadata: languageName: node linkType: hard +"require-relative@npm:^0.8.7": + version: 0.8.7 + resolution: "require-relative@npm:0.8.7" + checksum: f1c3be06977823bba43600344d9ea6fbf8a55bdb81ec76533126849ab4024e6c31c6666f37fa4b5cfeda9c41dee89b8e19597cac02bdefaab42255c6708661ab + languageName: node + linkType: hard + "reselect@npm:^4.1.7": version: 4.1.8 resolution: "reselect@npm:4.1.8" @@ -13274,6 +13459,15 @@ __metadata: languageName: node linkType: hard +"semver@npm:^7.3.6": + version: 7.7.3 + resolution: "semver@npm:7.7.3" + bin: + semver: bin/semver.js + checksum: f013a3ee4607857bcd3503b6ac1d80165f7f8ea94f5d55e2d3e33df82fce487aa3313b987abf9b39e0793c83c9fc67b76c36c067625141a9f6f704ae0ea18db2 + languageName: node + linkType: hard + "semver@npm:^7.6.3": version: 7.7.2 resolution: "semver@npm:7.7.2" @@ -13818,6 +14012,15 @@ __metadata: languageName: node linkType: hard +"strip-ansi@npm:^3.0.0": + version: 3.0.1 + resolution: "strip-ansi@npm:3.0.1" + dependencies: + ansi-regex: ^2.0.0 + checksum: 9b974de611ce5075c70629c00fa98c46144043db92ae17748fb780f706f7a789e9989fd10597b7c2053ae8d1513fd707816a91f1879b2f71e6ac0b6a863db465 + languageName: node + linkType: hard + "strip-ansi@npm:^5.0.0": version: 5.2.0 resolution: "strip-ansi@npm:5.2.0" @@ -13903,6 +14106,13 @@ __metadata: languageName: node linkType: hard +"supports-color@npm:^2.0.0": + version: 2.0.0 + resolution: "supports-color@npm:2.0.0" + checksum: 602538c5812b9006404370b5a4b885d3e2a1f6567d314f8b4a41974ffe7d08e525bf92ae0f9c7030e3b4c78e4e34ace55d6a67a74f1571bc205959f5972f88f0 + languageName: node + linkType: hard + "supports-color@npm:^5.3.0": version: 5.5.0 resolution: "supports-color@npm:5.5.0" @@ -14077,6 +14287,15 @@ __metadata: languageName: node linkType: hard +"ts-api-utils@npm:^1.0.1": + version: 1.4.3 + resolution: "ts-api-utils@npm:1.4.3" + peerDependencies: + typescript: ">=4.2.0" + checksum: ea00dee382d19066b2a3d8929f1089888b05fec797e32e7a7004938eda1dccf2e77274ee2afcd4166f53fab9b8d7ee90ebb225a3183f9ba8817d636f688a148d + languageName: node + linkType: hard + "ts-api-utils@npm:^1.3.0": version: 1.4.0 resolution: "ts-api-utils@npm:1.4.0" @@ -14100,6 +14319,13 @@ __metadata: languageName: node linkType: hard +"tslib@npm:^2.8.1": + version: 2.8.1 + resolution: "tslib@npm:2.8.1" + checksum: e4aba30e632b8c8902b47587fd13345e2827fa639e7c3121074d5ee0880723282411a8838f830b55100cbe4517672f84a2472667d355b81e8af165a55dc6203a + languageName: node + linkType: hard + "tsutils@npm:^3.21.0": version: 3.21.0 resolution: "tsutils@npm:3.21.0" @@ -14630,6 +14856,23 @@ __metadata: languageName: node linkType: hard +"vue-eslint-parser@npm:^9.4.3": + version: 9.4.3 + resolution: "vue-eslint-parser@npm:9.4.3" + dependencies: + debug: ^4.3.4 + eslint-scope: ^7.1.1 + eslint-visitor-keys: ^3.3.0 + espree: ^9.3.1 + esquery: ^1.4.0 + lodash: ^4.17.21 + semver: ^7.3.6 + peerDependencies: + eslint: ">=6.0.0" + checksum: 8d5b7ef7c5ee264ca2ba78da4b95ac7a66175a458d153a35e92cd7c55b794db0f2c31a8fdd40021bab4496f2f64ab80d7dbb6dccff4103beb4564c439a88fa42 + languageName: node + linkType: hard + "walker@npm:^1.0.7, walker@npm:^1.0.8": version: 1.0.8 resolution: "walker@npm:1.0.8" From 0e1bb0411c0a97ba335b80ad0699645ecbebfbb6 Mon Sep 17 00:00:00 2001 From: Loren Posen Date: Wed, 15 Oct 2025 13:25:17 -0700 Subject: [PATCH 5/8] chore: remove unused back icon assets for various resolutions and platforms --- src/inbox/assets/back-icon.png | Bin 207 -> 0 bytes src/inbox/assets/back-icon@1x.android.png | Bin 100 -> 0 bytes src/inbox/assets/back-icon@1x.ios.png | Bin 290 -> 0 bytes src/inbox/assets/back-icon@2x.android.png | Bin 134 -> 0 bytes src/inbox/assets/back-icon@2x.ios.png | Bin 405 -> 0 bytes src/inbox/assets/back-icon@3x.android.png | Bin 167 -> 0 bytes src/inbox/assets/back-icon@3x.ios.png | Bin 761 -> 0 bytes src/inbox/assets/back-icon@4x.android.png | Bin 207 -> 0 bytes src/inbox/assets/back-icon@4x.ios.png | Bin 809 -> 0 bytes 9 files changed, 0 insertions(+), 0 deletions(-) delete mode 100644 src/inbox/assets/back-icon.png delete mode 100644 src/inbox/assets/back-icon@1x.android.png delete mode 100644 src/inbox/assets/back-icon@1x.ios.png delete mode 100644 src/inbox/assets/back-icon@2x.android.png delete mode 100644 src/inbox/assets/back-icon@2x.ios.png delete mode 100644 src/inbox/assets/back-icon@3x.android.png delete mode 100644 src/inbox/assets/back-icon@3x.ios.png delete mode 100644 src/inbox/assets/back-icon@4x.android.png delete mode 100644 src/inbox/assets/back-icon@4x.ios.png diff --git a/src/inbox/assets/back-icon.png b/src/inbox/assets/back-icon.png deleted file mode 100644 index 17e52e8550e5668f7117bcb755beb70c3a21c9e9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 207 zcmeAS@N?(olHy`uVBq!ia0vp^2_VeD0wg^q?%xcgrg^$JhEy=Vy>(TP*-+rv1Kz7M z%I*XNJ!(VLzdH4Ge@o8ai4)e^3vK%P+bZq#{MQci z?P`C0d3kxcKa^1~`zt>B;jiLd|K_aKfAEz>edFmDy)D6SUw`+o@R*?B*u=sqq~bB* zgRIJrKN^lr3+tD}bFTI}~XjubmVU5CWQ(LZ;AgeuH{an^LB{Ts5E|^v% diff --git a/src/inbox/assets/back-icon@1x.android.png b/src/inbox/assets/back-icon@1x.android.png deleted file mode 100644 index 083db295f474b9903408258c71818c2c49151d35..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 100 zcmeAS@N?(olHy`uVBq!ia0vp^5+KaM0wlfaz7_*1ZBG}+kP60RiDL^GIXr{@&3`D~ x%$9kI*}bEvW6`UVO)xQz+sFfYmle_k4J{c^Y-u@of;y~NbrRr@}xDqOoe#7(!ujI zW)IKPtpOg9x^ZGiV^G!WUnY384Jwg^1?qzoV?Vqi%~F-r131d0cdTWi1E3pVgr)a^ zbA?Lk#0WF#3+Yvg4uEz<4l(HmchxGX6;goJ^ovoWl9~bgne?yCQ=RAlXaeYC>D}3~ oOlpASO$!ea%F`=`*pIb34xO&@x>?fx0{{R307*qoM6N<$f~pOBr2qf` diff --git a/src/inbox/assets/back-icon@2x.android.png b/src/inbox/assets/back-icon@2x.android.png deleted file mode 100644 index 6de0a1cbb365dfd5d9274890d243ca2321f32235..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 134 zcmeAS@N?(olHy`uVBq!ia0vp^1|ZDA0wn)(8}a}tUr!gukP61P(+)Bo5D;KluXOkB z|6f;`Uoa>nrTyP5;4+Jm;dJDxe#>oP@r$y*#>rHDd13s5eN)8##R;WH*ZD4M;?i1i i;i&z7`QNqd4J-@3aX4v+{p181%;4$j=d#Wzp$P!F!7nZV diff --git a/src/inbox/assets/back-icon@2x.ios.png b/src/inbox/assets/back-icon@2x.ios.png deleted file mode 100644 index 63de0e303aae4f3f09bbf5834290131501232f64..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 405 zcmV;G0c!q$3ng)KwR%cav2NeVj#yi|twB8nJ$ zhJ9XSgC81to-du2Xkr{|@HcXhBf~opLkgz`JWJYh7b%=;afNgn{8BUDujC*_hPNS> z7&ls+kwoI8u%elk7pg@(StZ8P4^;l-*^fZJ1`5~OHBd;D;qqM+{Bs~b$&5^qEx(K8 zx9Nf9ZEAjO1n+^rUo`oHziD{zE6kAXuYyr+hHU>u3T%dKe|C&IvG2#fso;0*pv4!B zOc_R!8Jq}55TlLyCq{c8swLA2SGn}>CsPgp4HtE6hRq5700000NkvXXu0mjfokp-j diff --git a/src/inbox/assets/back-icon@3x.android.png b/src/inbox/assets/back-icon@3x.android.png deleted file mode 100644 index 15a983a67d97c9a6c39d91550a528c37c53a9e3e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 167 zcmeAS@N?(olHy`uVBq!ia0vp^9w5xY0wn)GsXhaw@;qG}Ln;{G-aN{9*nr0&aBl6p z15q2fw>DO<$f$_t-m`#N(R0!@qfL6}is!BA-aP5mqcux*S)AYDzOuXc@WF!zU-mWw z!RNIrdq3B7GJgpV3USMNx+#Bt3lxOZSlm}!I$z&?|8f=2ODy@*gQAaCs?P@6%;4$j K=d#Wzp$P!X&_+4{ diff --git a/src/inbox/assets/back-icon@3x.ios.png b/src/inbox/assets/back-icon@3x.ios.png deleted file mode 100644 index 2320fd74d1781dc8a74cc131d8c3bb17fc52bae4..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 761 zcmVAUlM5g1Q_DYVitMm-$7Tx z*ro&sTZ*z6>QNY&<*{MBLwj|YS(XqXD^V764Y9I1Okis$Okg8b+mr|wUq@l0HcEq; zVTpm+Kw?IyI+FyL&Ez%>QDU|-7*wFaOtGZEY$rQ2K-HNzTzn^q>8HU=v}CyWZt?*4 zR&^#h0``%e>7l`lw-g9CKz639sxv7OaEQQsKxaB>Fyky00*;WF4yw+iM!+!&6ScM) z%veiJ{LO>gk?HFakuuxmtTUZbi;r*1=ySHXQyoauVV}3fsCFdksL$Qv?Q2bBWbLdnCsmVuHOp65Sf{(`X7f(3UB_ENM%STv@ zvIOdikIG_Hb<}I(pkKw$XH%o9lBnz0zU=BVRS7|_2-Hp7L=H8o0zzIAXX+Lzml{=$ zMBTymK^Jb-%uZCPG}qfJds0JQ2XB$6bg6-+U-Ef=(TP*-+rv1Kz7M z%I*XNJ!(VLzdH4Ge@o8ai4)e^3vK%P+bZq#{MQci z?P`C0d3kxcKa^1~`zt>B;jiLd|K_aKfAEz>edFmDy)D6SUw`+o@R*?B*u=sqq~bB* zgRIJrKN^lr3+tD}bFTI}~XjubmVU5CWQ(LZ;AgeuH{an^LB{Ts5E|^v% diff --git a/src/inbox/assets/back-icon@4x.ios.png b/src/inbox/assets/back-icon@4x.ios.png deleted file mode 100644 index b26634e594135d181375c3316b2ce059a31c6665..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 809 zcmV+^1J?YBP)YD1miR7mE{DV$Qg{||{bo~dA%(lFKsoPR23s4@v0mrvvjQGomaY7ADLDuqH1 zvGK4`4{>#jmgPlOeM+>Mf_`%D1%nu+I@RZN_76>V5WO1tOIiKYhsr9>jYgGKs=`*N zv}jyyRB5p)?goPB+{p5Jb-|H(#biuZo$8V=bj@m<&Aw*Mzvy;mD@cB;LBZVYApQ(D zdpPq|r)sw;OT$xM%b9U3xD%GW6CGq63GPHqLE>$f#-ZTp)T|)+jjzs91<7wcE?A); z`OV+XYNgUb`*8ccu~0$s2Ow%Rm0<*BXJiDW8UkwI7}mD<7*>7|TUr^wfilifZVG;4 n@$E^mUzjZzWv9I6RnQJ~H00000NkvXXu0mjf?i-Uu From 4fe7c80dc9ee1a3dc43f1f89bcfbc3b77b0620cb Mon Sep 17 00:00:00 2001 From: Loren Posen Date: Wed, 15 Oct 2025 14:14:35 -0700 Subject: [PATCH 6/8] feat: replace icon references with base64 encoded images --- example/src/components/App/App.constants.ts | 18 +++++++++++--- example/src/components/App/App.utils.tsx | 27 ++++++++++++++++++--- example/src/components/App/Main.tsx | 2 +- 3 files changed, 39 insertions(+), 8 deletions(-) diff --git a/example/src/components/App/App.constants.ts b/example/src/components/App/App.constants.ts index f84c390cb..65bf314e6 100644 --- a/example/src/components/App/App.constants.ts +++ b/example/src/components/App/App.constants.ts @@ -1,7 +1,19 @@ import { Route } from '../../constants'; +// Taken from https://github.com/ionic-team/ionicons/blob/main/src/svg/cash-outline.svg +export const cashIcon = + ''; + +// Taken from https://github.com/ionic-team/ionicons/blob/main/src/svg/person-outline.svg +export const personIcon = + ''; + +// Taken from https://github.com/ionic-team/ionicons/blob/main/src/svg/mail-outline.svg +export const mailIcon = + ''; + export const routeIcon = { - [Route.Commerce]: 'cash-outline', - [Route.Inbox]: 'mail-outline', - [Route.User]: 'person-outline', + [Route.Commerce]: cashIcon, + [Route.Inbox]: mailIcon, + [Route.User]: personIcon, }; diff --git a/example/src/components/App/App.utils.tsx b/example/src/components/App/App.utils.tsx index 2dcde5374..6a3990472 100644 --- a/example/src/components/App/App.utils.tsx +++ b/example/src/components/App/App.utils.tsx @@ -1,5 +1,24 @@ -import Icon from 'react-native-vector-icons/Ionicons'; +import { Image, View } from 'react-native'; +import type { Route } from '../../constants/routes'; -export const getIcon = (name: string, props: Record) => ( - -); +export const getIcon = (name: Route, props: Record) => { + const { color, size = 25 } = props; + + return ( + + + + ); +}; diff --git a/example/src/components/App/Main.tsx b/example/src/components/App/Main.tsx index 55b0d74e2..b944801fe 100644 --- a/example/src/components/App/Main.tsx +++ b/example/src/components/App/Main.tsx @@ -25,7 +25,7 @@ export const Main = () => { screenOptions={({ route }) => { const iconName = routeIcon[route.name]; return { - tabBarIcon: (props) => getIcon(iconName, props), + tabBarIcon: (props) => getIcon(iconName as Route, props), tabBarActiveTintColor: colors.brandPurple, tabBarInactiveTintColor: colors.textSecondary, headerShown: false, From 6ece6e031147935ecd50f46e65be519d306f00ab Mon Sep 17 00:00:00 2001 From: Loren Posen Date: Wed, 15 Oct 2025 14:15:19 -0700 Subject: [PATCH 7/8] chore: remove react-native-vector-icons dependency and related configurations --- README.md | 1 - example/android/app/build.gradle | 2 - example/package.json | 1 - jest.config.js | 2 +- package.json | 3 -- yarn.lock | 68 +------------------------------- 6 files changed, 3 insertions(+), 74 deletions(-) diff --git a/README.md b/README.md index fb5a88a63..9f8ead8e9 100644 --- a/README.md +++ b/README.md @@ -43,7 +43,6 @@ Iterable's React Native SDK relies on: _UI Components require additional peer dependencies_ - [React Navigation 6+](https://github.com/react-navigation/react-navigation) - [React Native Safe Area Context 4+](https://github.com/th3rdwave/react-native-safe-area-context) - - [React Native Vector Icons 10+](https://github.com/oblador/react-native-vector-icons) - [React Native WebView 13+](https://github.com/react-native-webview/react-native-webview) - **iOS** diff --git a/example/android/app/build.gradle b/example/android/app/build.gradle index 060f74075..a060fdcce 100644 --- a/example/android/app/build.gradle +++ b/example/android/app/build.gradle @@ -117,5 +117,3 @@ dependencies { implementation jscFlavor } } - -apply from: file("../../node_modules/react-native-vector-icons/fonts.gradle") diff --git a/example/package.json b/example/package.json index 40ca703a4..20dddd07b 100644 --- a/example/package.json +++ b/example/package.json @@ -19,7 +19,6 @@ "react-native-gesture-handler": "^2.24.0", "react-native-safe-area-context": "^5.1.0", "react-native-screens": "^4.9.1", - "react-native-vector-icons": "^10.2.0", "react-native-webview": "^13.13.1" }, "devDependencies": { diff --git a/jest.config.js b/jest.config.js index 482f53a77..c5b75e86c 100644 --- a/jest.config.js +++ b/jest.config.js @@ -6,7 +6,7 @@ module.exports = { ], testMatch: ['/src/**/*.(test|spec).[jt]s?(x)'], transformIgnorePatterns: [ - 'node_modules/(?!(react-native|@react-native|@react-navigation|react-native-screens|react-native-safe-area-context|react-native-gesture-handler|react-native-webview|react-native-vector-icons)/)', + 'node_modules/(?!(react-native|@react-native|@react-navigation|react-native-screens|react-native-safe-area-context|react-native-gesture-handler|react-native-webview)/)', ], collectCoverageFrom: [ 'src/**/*.{cjs,js,jsx,mjs,ts,tsx}', diff --git a/package.json b/package.json index c3e1dfc52..bb356a6f5 100644 --- a/package.json +++ b/package.json @@ -77,7 +77,6 @@ "@testing-library/react-native": "^13.3.3", "@types/jest": "^29.5.5", "@types/react": "^19.0.0", - "@types/react-native-vector-icons": "^6.4.18", "@typescript-eslint/eslint-plugin": "^8.13.0", "@typescript-eslint/parser": "^8.13.0", "commitlint": "^19.6.1", @@ -96,7 +95,6 @@ "react-native-gesture-handler": "^2.26.0", "react-native-safe-area-context": "^5.4.0", "react-native-screens": "^4.10.0", - "react-native-vector-icons": "^10.2.0", "react-native-webview": "^13.14.1", "react-test-renderer": "19.0.0", "release-it": "^17.10.0", @@ -114,7 +112,6 @@ "react": "*", "react-native": "*", "react-native-safe-area-context": "*", - "react-native-vector-icons": "*", "react-native-webview": "*" }, "peerDependenciesMeta": { diff --git a/yarn.lock b/yarn.lock index ea98be24c..ec4f21c4c 100644 --- a/yarn.lock +++ b/yarn.lock @@ -3108,7 +3108,6 @@ __metadata: react-native-gesture-handler: ^2.24.0 react-native-safe-area-context: ^5.1.0 react-native-screens: ^4.9.1 - react-native-vector-icons: ^10.2.0 react-native-webview: ^13.13.1 react-test-renderer: 19.0.0 languageName: unknown @@ -3130,7 +3129,6 @@ __metadata: "@testing-library/react-native": ^13.3.3 "@types/jest": ^29.5.5 "@types/react": ^19.0.0 - "@types/react-native-vector-icons": ^6.4.18 "@typescript-eslint/eslint-plugin": ^8.13.0 "@typescript-eslint/parser": ^8.13.0 commitlint: ^19.6.1 @@ -3149,7 +3147,6 @@ __metadata: react-native-gesture-handler: ^2.26.0 react-native-safe-area-context: ^5.4.0 react-native-screens: ^4.10.0 - react-native-vector-icons: ^10.2.0 react-native-webview: ^13.14.1 react-test-renderer: 19.0.0 release-it: ^17.10.0 @@ -3163,7 +3160,6 @@ __metadata: react: "*" react-native: "*" react-native-safe-area-context: "*" - react-native-vector-icons: "*" react-native-webview: "*" peerDependenciesMeta: expo: @@ -4615,25 +4611,6 @@ __metadata: languageName: node linkType: hard -"@types/react-native-vector-icons@npm:^6.4.18": - version: 6.4.18 - resolution: "@types/react-native-vector-icons@npm:6.4.18" - dependencies: - "@types/react": "*" - "@types/react-native": ^0.70 - checksum: 1ef458cb5e7a37f41eb400e3153940b1b152e4df76a7c06c7a47c712dbfe46e14b9999f04dde1bd074f338f850e161c6c925174ddea33386b74f8112c940065b - languageName: node - linkType: hard - -"@types/react-native@npm:^0.70": - version: 0.70.19 - resolution: "@types/react-native@npm:0.70.19" - dependencies: - "@types/react": "*" - checksum: 79b504fa56340631079e7c20ea0d9412ec14147b76d0ce189f4403936f529ef1e6fd031383afab117846c5ae039123bcf3afc948bae4432269c6780282726f71 - languageName: node - linkType: hard - "@types/react-test-renderer@npm:^19.0.0": version: 19.1.0 resolution: "@types/react-test-renderer@npm:19.1.0" @@ -6172,17 +6149,6 @@ __metadata: languageName: node linkType: hard -"cliui@npm:^7.0.2": - version: 7.0.4 - resolution: "cliui@npm:7.0.4" - dependencies: - string-width: ^4.2.0 - strip-ansi: ^6.0.0 - wrap-ansi: ^7.0.0 - checksum: ce2e8f578a4813806788ac399b9e866297740eecd4ad1823c27fd344d78b22c5f8597d548adbcc46f0573e43e21e751f39446c5a5e804a12aace402b7a315d7f - languageName: node - linkType: hard - "cliui@npm:^8.0.1": version: 8.0.1 resolution: "cliui@npm:8.0.1" @@ -12370,7 +12336,7 @@ __metadata: languageName: node linkType: hard -"prop-types@npm:^15.7.2, prop-types@npm:^15.8.1": +"prop-types@npm:^15.8.1": version: 15.8.1 resolution: "prop-types@npm:15.8.1" dependencies: @@ -12760,21 +12726,6 @@ __metadata: languageName: node linkType: hard -"react-native-vector-icons@npm:^10.2.0": - version: 10.2.0 - resolution: "react-native-vector-icons@npm:10.2.0" - dependencies: - prop-types: ^15.7.2 - yargs: ^16.1.1 - bin: - fa-upgrade.sh: bin/fa-upgrade.sh - fa5-upgrade: bin/fa5-upgrade.sh - fa6-upgrade: bin/fa6-upgrade.sh - generate-icon: bin/generate-icon.js - checksum: fda930df4e63f12533268f5b339ebe4c77c691eae43503328466b3087ed868a06a4593fd246e75ac6b5ec955543eec35608c7922191bdcc3b3a94ed7f3575ef0 - languageName: node - linkType: hard - "react-native-webview@npm:^13.13.1": version: 13.14.1 resolution: "react-native-webview@npm:13.14.1" @@ -15200,7 +15151,7 @@ __metadata: languageName: node linkType: hard -"yargs-parser@npm:^20.2.2, yargs-parser@npm:^20.2.9": +"yargs-parser@npm:^20.2.9": version: 20.2.9 resolution: "yargs-parser@npm:20.2.9" checksum: 8bb69015f2b0ff9e17b2c8e6bfe224ab463dd00ca211eece72a4cd8a906224d2703fb8a326d36fdd0e68701e201b2a60ed7cf81ce0fd9b3799f9fe7745977ae3 @@ -15226,21 +15177,6 @@ __metadata: languageName: node linkType: hard -"yargs@npm:^16.1.1": - version: 16.2.0 - resolution: "yargs@npm:16.2.0" - dependencies: - cliui: ^7.0.2 - escalade: ^3.1.1 - get-caller-file: ^2.0.5 - require-directory: ^2.1.1 - string-width: ^4.2.0 - y18n: ^5.0.5 - yargs-parser: ^20.2.2 - checksum: b14afbb51e3251a204d81937c86a7e9d4bdbf9a2bcee38226c900d00f522969ab675703bee2a6f99f8e20103f608382936034e64d921b74df82b63c07c5e8f59 - languageName: node - linkType: hard - "yargs@npm:^17.0.0, yargs@npm:^17.3.1, yargs@npm:^17.5.1, yargs@npm:^17.6.2": version: 17.7.2 resolution: "yargs@npm:17.7.2" From c1f3dec93b6ececc78e2d566b8d7e7d44131ad7d Mon Sep 17 00:00:00 2001 From: Loren Posen Date: Wed, 15 Oct 2025 14:29:20 -0700 Subject: [PATCH 8/8] test: add comprehensive tests for HeaderBackButton component --- .eslintrc.js | 1 + .../components/HeaderBackButton.test.tsx | 243 ++++++++++++++++++ 2 files changed, 244 insertions(+) create mode 100644 src/inbox/components/HeaderBackButton.test.tsx diff --git a/.eslintrc.js b/.eslintrc.js index e0f808c23..bad66d857 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -48,4 +48,5 @@ module.exports = { }, }, ], + ignorePatterns: ['coverage/**/*', 'lib/**/*', 'docs/**/*'], }; diff --git a/src/inbox/components/HeaderBackButton.test.tsx b/src/inbox/components/HeaderBackButton.test.tsx new file mode 100644 index 000000000..76917e0c0 --- /dev/null +++ b/src/inbox/components/HeaderBackButton.test.tsx @@ -0,0 +1,243 @@ +/* eslint-disable @typescript-eslint/no-explicit-any */ +import { fireEvent, render } from '@testing-library/react-native'; +import { PixelRatio } from 'react-native'; +import { HeaderBackButton, ICON_MARGIN, ICON_SIZE } from './HeaderBackButton'; + +describe('HeaderBackButton', () => { + beforeEach(() => { + jest.clearAllMocks(); + }); + + describe('Rendering', () => { + it('should render without crashing', () => { + const { getByTestId } = render(); + expect(getByTestId('back-button')).toBeTruthy(); + }); + + it('should render with default back arrow image', () => { + const { UNSAFE_getByType } = render(); + const image = UNSAFE_getByType('Image' as any); + expect(image).toBeTruthy(); + expect(image.props.source).toMatchObject({ + uri: expect.stringContaining('data:image/png;base64'), + width: PixelRatio.getPixelSizeForLayoutSize(ICON_SIZE), + height: PixelRatio.getPixelSizeForLayoutSize(ICON_SIZE), + }); + }); + + it('should render without label by default', () => { + const { queryByText } = render(); + expect(queryByText(/./)).toBeNull(); + }); + + it('should render with label when provided', () => { + const label = 'Back'; + const { getByText } = render(); + expect(getByText(label)).toBeTruthy(); + }); + + it('should render with custom label text', () => { + const customLabel = 'Go Back to Home'; + const { getByText } = render(); + expect(getByText(customLabel)).toBeTruthy(); + }); + }); + + describe('Custom Image Props', () => { + it('should render with custom imageUri', () => { + const customUri = 'https://example.com/custom-back-icon.png'; + const { UNSAFE_getByType } = render( + + ); + const image = UNSAFE_getByType('Image' as any); + expect(image.props.source).toMatchObject({ + uri: customUri, + }); + }); + + it('should render with custom imageSource', () => { + const customSource = { uri: 'https://example.com/icon.png' }; + const { UNSAFE_getByType } = render( + + ); + const image = UNSAFE_getByType('Image' as any); + expect(image.props.source).toEqual(customSource); + }); + + it('should prioritize imageSource over imageUri when both are provided', () => { + const customUri = 'https://example.com/custom-back-icon.png'; + const customSource = { uri: 'https://example.com/icon.png' }; + const { UNSAFE_getByType } = render( + + ); + const image = UNSAFE_getByType('Image' as any); + expect(image.props.source).toEqual(customSource); + }); + }); + + describe('Image Properties', () => { + it('should render image with correct properties', () => { + const { UNSAFE_getByType } = render(); + const image = UNSAFE_getByType('Image' as any); + + expect(image.props.resizeMode).toBe('contain'); + expect(image.props.fadeDuration).toBe(0); + expect(image.props.height).toBe(ICON_SIZE); + expect(image.props.width).toBe(ICON_SIZE); + expect(image.props.resizeMethod).toBe('scale'); + expect(image.props.tintColor).toBeTruthy(); + }); + + it('should apply correct style to image', () => { + const { UNSAFE_getByType } = render(); + const image = UNSAFE_getByType('Image' as any); + + expect(image.props.style).toMatchObject({ + height: ICON_SIZE, + margin: ICON_MARGIN, + width: ICON_SIZE, + }); + }); + }); + + describe('Touch Interaction', () => { + it('should call onPress when button is pressed', () => { + const onPressMock = jest.fn(); + const { getByTestId } = render( + + ); + + fireEvent.press(getByTestId('back-button')); + expect(onPressMock).toHaveBeenCalledTimes(1); + }); + + it('should call onPressIn when touch starts', () => { + const onPressInMock = jest.fn(); + const { getByTestId } = render( + + ); + + fireEvent(getByTestId('back-button'), 'pressIn'); + expect(onPressInMock).toHaveBeenCalledTimes(1); + }); + + it('should call onPressOut when touch ends', () => { + const onPressOutMock = jest.fn(); + const { getByTestId } = render( + + ); + + fireEvent(getByTestId('back-button'), 'pressOut'); + expect(onPressOutMock).toHaveBeenCalledTimes(1); + }); + + it('should not trigger onPress when disabled', () => { + const onPressMock = jest.fn(); + const { getByTestId } = render( + + ); + + fireEvent.press(getByTestId('back-button')); + expect(onPressMock).not.toHaveBeenCalled(); + }); + }); + + describe('Platform-specific behavior', () => { + it('should export correct icon size constant', () => { + // ICON_SIZE is evaluated at module load time based on Platform.OS + expect(ICON_SIZE).toBeDefined(); + expect([21, 24]).toContain(ICON_SIZE); + }); + + it('should export correct icon margin constant', () => { + // ICON_MARGIN is evaluated at module load time based on Platform.OS + expect(ICON_MARGIN).toBeDefined(); + expect([3, 8]).toContain(ICON_MARGIN); + }); + + it('should use consistent icon size in image props', () => { + const { UNSAFE_getByType } = render(); + const image = UNSAFE_getByType('Image' as any); + + expect(image.props.height).toBe(ICON_SIZE); + expect(image.props.width).toBe(ICON_SIZE); + }); + }); + + describe('Accessibility', () => { + it('should accept accessibility props', () => { + const { getByTestId } = render( + + ); + + const button = getByTestId('back-button'); + expect(button.props.accessible).toBe(true); + expect(button.props.accessibilityLabel).toBe('Navigate back'); + expect(button.props.accessibilityHint).toBe('Returns to previous screen'); + }); + }); + + describe('Component Structure', () => { + it('should render View with correct flex direction', () => { + const { UNSAFE_getAllByType } = render(); + const views = UNSAFE_getAllByType('View' as any); + + // Find the view with returnButton style + const returnButtonView = views.find( + (view) => + view.props.style?.flexDirection === 'row' && + view.props.style?.alignItems === 'center' + ); + expect(returnButtonView).toBeTruthy(); + }); + + it('should render label text with correct style when provided', () => { + const { getByText } = render(); + const labelElement = getByText('Back'); + + expect(labelElement.props.style).toMatchObject({ + fontSize: 20, + }); + }); + }); + + describe('Edge Cases', () => { + it('should handle empty string label', () => { + const { queryByText } = render(); + // Empty string should not render text + expect(queryByText('')).toBeNull(); + }); + + it('should handle multiple props correctly', () => { + const onPressMock = jest.fn(); + const label = 'Custom Back'; + const customUri = 'https://example.com/icon.png'; + + const { getByText, getByTestId } = render( + + ); + + expect(getByText(label)).toBeTruthy(); + expect(getByTestId('back-button').props.accessible).toBe(true); + + fireEvent.press(getByTestId('back-button')); + expect(onPressMock).toHaveBeenCalledTimes(1); + }); + }); +});