Skip to content

Commit 8985195

Browse files
ci(release): publish latest release
1 parent 168fd79 commit 8985195

File tree

85 files changed

+1720
-2638
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

85 files changed

+1720
-2638
lines changed

CODEOWNERS

-1
This file was deleted.

RELEASE

+5-19
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,5 @@
1-
IPFS hash of the deployment:
2-
- CIDv0: `QmeHTibtaHFy2CsJfYCkWUkz6iRTJPbPQEnAa7syJ8TJuV`
3-
- CIDv1: `bafybeihm5hjaazzxpktgygpmi4gvtkhdfsozzxhuxmyjnbq5jiwuxskfna`
4-
5-
The latest release is always mirrored at [app.uniswap.org](https://app.uniswap.org).
6-
7-
You can also access the Uniswap Interface from an IPFS gateway.
8-
**BEWARE**: The Uniswap interface uses [`localStorage`](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage) to remember your settings, such as which tokens you have imported.
9-
**You should always use an IPFS gateway that enforces origin separation**, or our hosted deployment of the latest release at [app.uniswap.org](https://app.uniswap.org).
10-
Your Uniswap settings are never remembered across different URLs.
11-
12-
IPFS gateways:
13-
- https://bafybeihm5hjaazzxpktgygpmi4gvtkhdfsozzxhuxmyjnbq5jiwuxskfna.ipfs.dweb.link/
14-
- https://bafybeihm5hjaazzxpktgygpmi4gvtkhdfsozzxhuxmyjnbq5jiwuxskfna.ipfs.cf-ipfs.com/
15-
- [ipfs://QmeHTibtaHFy2CsJfYCkWUkz6iRTJPbPQEnAa7syJ8TJuV/](ipfs://QmeHTibtaHFy2CsJfYCkWUkz6iRTJPbPQEnAa7syJ8TJuV/)
16-
17-
### 5.30.1 (2024-05-31)
18-
19-
1+
Excited to share some new updates! Here’s what’s new:
2+
3+
Expanded support for NFTs — In addition to mainnet Ethereum, you are now able to see NFTs across all other networks that we currently support, including: Polygon, Arbitrum, Optimism, Base, BSC, and Blast.
4+
5+
Improved Unicons — We gave your wallet’s unique Unicon a makeover. Check out the rest of your accounts to see your upgraded icons.

VERSION

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
web/5.30.1
1+
mobile/1.28.2

apps/mobile/README.md

+12-1
Original file line numberDiff line numberDiff line change
@@ -176,7 +176,18 @@ These are some tools you might want to familiarize yourself with to understand t
176176

177177
## Migrations
178178

179-
We use `redux-persist` to persist the Redux state between user sessions. Most of this state is shared between the mobile app and the extension. Please review the [Wallet Migrations README](../../packages/wallet/src/state//README.md) for details on how to write migrations when you add or remove anything from the Redux state structure.
179+
We use `redux-persist` to persist Redux state between user sessions. When the Redux state schema is altered, a migration may be needed to transfer the existing persisted state to the new Redux schema. Failing to define a migration results in the app defaulting to the persisted schema, which will very likely cause `undefined` errors because the code has references to Redux state properties that were dropped in favor the persisted schema.
180+
181+
### When to define a migration
182+
183+
Anytime a required property is added or any property is renamed or deleted to/from Redux state. Migrations are not necessary when optional properties are added to an existing slice. Make sure to always add new required properties to the `schema.ts` file as well.
184+
185+
### How to migrate
186+
187+
1. Increment the `version` of `persistConfig` defined within `store.ts`
188+
2. Create a migration function within `migrations.ts`. The migration key should be the same as the `version` defined in the previous step
189+
3. Write a test for your migration within `migrations.test.ts`
190+
4. Create a new schema within `schema.ts` and ensure it is being exported by the `getSchema` function at the bottom of the file
180191

181192
## Troubleshooting
182193

apps/mobile/android/app/build.gradle

+2-2
Original file line numberDiff line numberDiff line change
@@ -136,12 +136,12 @@ android {
136136
}
137137
beta {
138138
applicationIdSuffix ".beta"
139-
versionName "1.28"
139+
versionName "1.28.2"
140140
dimension "variant"
141141
}
142142
prod {
143143
dimension "variant"
144-
versionName "1.28"
144+
versionName "1.28.2"
145145
}
146146
}
147147

apps/mobile/ios/Uniswap.xcodeproj/project.pbxproj

+8-8
Original file line numberDiff line numberDiff line change
@@ -2536,7 +2536,7 @@
25362536
"@executable_path/Frameworks",
25372537
"@executable_path/../../Frameworks",
25382538
);
2539-
MARKETING_VERSION = 1.28;
2539+
MARKETING_VERSION = 1.28.2;
25402540
MTL_FAST_MATH = YES;
25412541
OTHER_SWIFT_FLAGS = "$(inherited) -D EXPO_CONFIGURATION_RELEASE";
25422542
PRODUCT_BUNDLE_IDENTIFIER = com.uniswap.mobile.widgets;
@@ -2628,7 +2628,7 @@
26282628
"@executable_path/Frameworks",
26292629
"@executable_path/../../Frameworks",
26302630
);
2631-
MARKETING_VERSION = 1.28;
2631+
MARKETING_VERSION = 1.28.2;
26322632
MTL_FAST_MATH = YES;
26332633
OTHER_SWIFT_FLAGS = "$(inherited) -D EXPO_CONFIGURATION_RELEASE";
26342634
PRODUCT_BUNDLE_IDENTIFIER = com.uniswap.mobile.beta.widgets;
@@ -2713,7 +2713,7 @@
27132713
"@executable_path/Frameworks",
27142714
"@executable_path/../../Frameworks",
27152715
);
2716-
MARKETING_VERSION = 1.28;
2716+
MARKETING_VERSION = 1.28.2;
27172717
MTL_FAST_MATH = YES;
27182718
OTHER_SWIFT_FLAGS = "$(inherited) -D EXPO_CONFIGURATION_RELEASE";
27192719
PRODUCT_BUNDLE_IDENTIFIER = com.uniswap.mobile.WidgetIntentExtension;
@@ -2799,7 +2799,7 @@
27992799
"@executable_path/Frameworks",
28002800
"@executable_path/../../Frameworks",
28012801
);
2802-
MARKETING_VERSION = 1.28;
2802+
MARKETING_VERSION = 1.28.2;
28032803
MTL_FAST_MATH = YES;
28042804
OTHER_SWIFT_FLAGS = "$(inherited) -D EXPO_CONFIGURATION_RELEASE";
28052805
PRODUCT_BUNDLE_IDENTIFIER = com.uniswap.mobile.beta.WidgetIntentExtension;
@@ -2873,7 +2873,7 @@
28732873
"$(inherited)",
28742874
"@executable_path/Frameworks",
28752875
);
2876-
MARKETING_VERSION = 1.28;
2876+
MARKETING_VERSION = 1.28.2;
28772877
OTHER_LDFLAGS = (
28782878
"$(inherited)",
28792879
"-ObjC",
@@ -3103,7 +3103,7 @@
31033103
"@executable_path/Frameworks",
31043104
"@executable_path/../../Frameworks",
31053105
);
3106-
MARKETING_VERSION = 1.28;
3106+
MARKETING_VERSION = 1.28.2;
31073107
MTL_FAST_MATH = YES;
31083108
OTHER_SWIFT_FLAGS = "$(inherited) -D EXPO_CONFIGURATION_RELEASE";
31093109
PRODUCT_BUNDLE_IDENTIFIER = com.uniswap.mobile.OneSignalNotificationServiceExtension;
@@ -3207,7 +3207,7 @@
32073207
"$(inherited)",
32083208
"@executable_path/Frameworks",
32093209
);
3210-
MARKETING_VERSION = 1.28;
3210+
MARKETING_VERSION = 1.28.2;
32113211
OTHER_LDFLAGS = (
32123212
"$(inherited)",
32133213
"-ObjC",
@@ -3278,7 +3278,7 @@
32783278
"@executable_path/Frameworks",
32793279
"@executable_path/../../Frameworks",
32803280
);
3281-
MARKETING_VERSION = 1.28;
3281+
MARKETING_VERSION = 1.28.2;
32823282
MTL_FAST_MATH = YES;
32833283
OTHER_SWIFT_FLAGS = "$(inherited) -D EXPO_CONFIGURATION_RELEASE";
32843284
PRODUCT_BUNDLE_IDENTIFIER = com.uniswap.mobile.beta.OneSignalNotificationServiceExtension;

apps/mobile/jest-setup.js

+11
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,17 @@ import { localizeMock as mockRNLocalize } from 'react-native-localize/mock'
99
import { AppearanceSettingType } from 'wallet/src/features/appearance/slice'
1010
import { mockLocalizationContext } from 'wallet/src/test/mocks/utils'
1111

12+
// avoids polluting console in test runs, while keeping important log levels
13+
global.console = {
14+
...console,
15+
// uncomment to ignore a specific log level
16+
log: jest.fn(),
17+
debug: jest.fn(),
18+
info: jest.fn(),
19+
// warn: jest.fn(),
20+
// error: jest.fn(),
21+
}
22+
1223
// Mock Sentry crash reporting
1324
jest.mock('@sentry/react-native', () => ({
1425
init: () => jest.fn(),

apps/mobile/src/app/migrations.test.ts

+20-5
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,6 @@ import {
104104
} from 'wallet/src/features/wallet/accounts/types'
105105
import { initialWalletState, SwapProtectionSetting } from 'wallet/src/features/wallet/slice'
106106
import { createMigrate } from 'wallet/src/state/createMigrate'
107-
import { getAllKeysOfNestedObject } from 'wallet/src/state/testUtils'
108107
import {
109108
fiatPurchaseTransactionInfo,
110109
signerMnemonicAccount,
@@ -125,6 +124,26 @@ const fiatOnRampTxDetailsFailed = transactionDetails({
125124
}),
126125
})
127126

127+
// helps with object assignment
128+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
129+
const getAllKeysOfNestedObject = (obj: any, prefix = ''): string[] => {
130+
const keys = Object.keys(obj)
131+
if (!keys.length && prefix !== '') {
132+
return [prefix.slice(0, -1)]
133+
}
134+
return keys.reduce<string[]>((res, el) => {
135+
if (Array.isArray(obj[el])) {
136+
return [...res]
137+
}
138+
139+
if (typeof obj[el] === 'object' && obj[el] !== null) {
140+
return [...res, ...getAllKeysOfNestedObject(obj[el], prefix + el + '.')]
141+
}
142+
143+
return [...res, prefix + el]
144+
}, [])
145+
}
146+
128147
describe('Redux state migrations', () => {
129148
it('is able to perform all migrations starting from the initial schema', async () => {
130149
const initialSchemaStub = {
@@ -185,10 +204,6 @@ describe('Redux state migrations', () => {
185204
},
186205
}
187206

188-
if (!migratedSchema) {
189-
throw new Error('Migrated schema is undefined')
190-
}
191-
192207
const migratedSchemaKeys = new Set(getAllKeysOfNestedObject(migratedSchema))
193208
const latestSchemaKeys = new Set(getAllKeysOfNestedObject(getSchema()))
194209
const initialStateKeys = new Set(getAllKeysOfNestedObject(initialState))

apps/mobile/src/app/migrations.ts

+6-4
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,6 @@ import {
1919
} from 'wallet/src/features/transactions/types'
2020
import { Account, AccountType } from 'wallet/src/features/wallet/accounts/types'
2121
import { SwapProtectionSetting } from 'wallet/src/features/wallet/slice'
22-
import { removeWalletIsUnlockedState } from 'wallet/src/state/sharedMigrations'
2322

2423
export const OLD_DEMO_ACCOUNT_ADDRESS = '0xdd0E380579dF30E38524F9477808d9eE37E2dEa6'
2524

@@ -877,7 +876,10 @@ export const migrations = {
877876
return newState
878877
},
879878

880-
63: removeWalletIsUnlockedState,
881-
}
879+
63: function removeWalletIsUnlockedState(state: any) {
880+
const newState = { ...state }
881+
delete newState.wallet.isUnlocked
882882

883-
export const MOBILE_STATE_VERSION = 63
883+
return newState
884+
},
885+
}

apps/mobile/src/app/store.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ import { isRejectedWithValue } from '@reduxjs/toolkit'
33
import * as Sentry from '@sentry/react'
44
import { MMKV } from 'react-native-mmkv'
55
import { Storage, persistReducer, persistStore } from 'redux-persist'
6-
import { MOBILE_STATE_VERSION, migrations } from 'src/app/migrations'
6+
import { migrations } from 'src/app/migrations'
77
import { isNonJestDev } from 'utilities/src/environment'
88
import { logger } from 'utilities/src/logger/logger'
99
import { fiatOnRampAggregatorApi, fiatOnRampApi } from 'wallet/src/features/fiatOnRamp/api'
@@ -66,7 +66,7 @@ export const persistConfig = {
6666
key: 'root',
6767
storage: reduxStorage,
6868
whitelist,
69-
version: MOBILE_STATE_VERSION,
69+
version: 63,
7070
migrate: createMigrate(migrations),
7171
}
7272

apps/mobile/src/components/WalletConnect/ModalWithOverlay/ModalWithOverlay.tsx

+11-5
Original file line numberDiff line numberDiff line change
@@ -11,9 +11,11 @@ import {
1111
NativeScrollEvent,
1212
NativeSyntheticEvent,
1313
ScrollView,
14+
StyleProp,
1415
View,
16+
ViewStyle,
1517
} from 'react-native'
16-
import { useDerivedValue } from 'react-native-reanimated'
18+
import { AnimatedStyle, useDerivedValue } from 'react-native-reanimated'
1719
import { ScrollDownOverlay } from 'src/components/WalletConnect/ModalWithOverlay/ScrollDownOverlay'
1820
import { Button, Flex, useDeviceInsets } from 'ui/src'
1921
import { spacing } from 'ui/src/theme'
@@ -30,6 +32,7 @@ type ModalWithOverlayProps = PropsWithChildren<
3032
onReject: () => void
3133
onConfirm: () => void
3234
disableConfirm?: boolean
35+
contentContainerStyle?: StyleProp<AnimatedStyle<StyleProp<ViewStyle>>>
3336
}
3437
>
3538

@@ -48,6 +51,7 @@ export function ModalWithOverlay({
4851
onReject,
4952
onConfirm,
5053
disableConfirm,
54+
contentContainerStyle,
5155
...bottomSheetModalProps
5256
}: ModalWithOverlayProps): JSX.Element {
5357
const scrollViewRef = useRef<ScrollView>(null)
@@ -114,10 +118,12 @@ export function ModalWithOverlay({
114118
<BottomSheetModal overrideInnerContainer {...bottomSheetModalProps}>
115119
<BottomSheetScrollView
116120
ref={scrollViewRef}
117-
contentContainerStyle={{
118-
paddingHorizontal: spacing.spacing24,
119-
paddingTop: spacing.spacing36,
120-
}}
121+
contentContainerStyle={
122+
contentContainerStyle ?? {
123+
paddingHorizontal: spacing.spacing24,
124+
paddingTop: spacing.spacing36,
125+
}
126+
}
121127
showsVerticalScrollIndicator={false}
122128
onLayout={handleScrollViewLayout}
123129
onScroll={handleScroll}>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
1+
import { useBottomSheetInternal } from '@gorhom/bottom-sheet'
2+
import { useTranslation } from 'react-i18next'
3+
import Animated, { useAnimatedStyle } from 'react-native-reanimated'
4+
import { ModalWithOverlay } from 'src/components/WalletConnect/ModalWithOverlay/ModalWithOverlay'
5+
import { RequestDetailsContent } from 'src/components/WalletConnect/RequestModal/RequestDetails'
6+
import { useUwuLinkContractAllowlist } from 'src/components/WalletConnect/ScanSheet/util'
7+
import { SignRequest } from 'src/features/walletConnect/walletConnectSlice'
8+
import { Flex, useIsDarkMode } from 'ui/src'
9+
import { spacing } from 'ui/src/theme'
10+
import { ModalName } from 'uniswap/src/features/telemetry/constants'
11+
import { RemoteImage } from 'wallet/src/features/images/RemoteImage'
12+
13+
type Props = {
14+
onClose: () => void
15+
onConfirm: () => void
16+
onReject: () => void
17+
request: SignRequest
18+
}
19+
20+
export function KidSuperCheckinModal({
21+
onClose,
22+
onConfirm,
23+
onReject,
24+
request,
25+
}: Props): JSX.Element {
26+
const { t } = useTranslation()
27+
28+
return (
29+
<ModalWithOverlay
30+
confirmationButtonText={t('common.button.checkin')}
31+
contentContainerStyle={{
32+
paddingHorizontal: spacing.spacing24,
33+
paddingTop: spacing.spacing8,
34+
}}
35+
name={ModalName.UwULinkErc20SendModal}
36+
scrollDownButtonText={t('walletConnect.request.button.scrollDown')}
37+
onClose={onClose}
38+
onConfirm={onConfirm}
39+
onReject={onReject}>
40+
<KidSuperCheckinModalContent request={request} />
41+
</ModalWithOverlay>
42+
)
43+
}
44+
45+
function useUniswapCafeLogo(): string | undefined {
46+
const isDarkMode = useIsDarkMode()
47+
const uwuLinkContractAllowlist = useUwuLinkContractAllowlist()
48+
const logos = uwuLinkContractAllowlist.tokenRecipients.find(
49+
(recipient) => recipient.name === 'Uniswap Cafe'
50+
)?.logo
51+
52+
if (!logos) {
53+
return
54+
}
55+
56+
return isDarkMode ? logos.dark : logos.light
57+
}
58+
59+
function KidSuperCheckinModalContent({ request }: { request: SignRequest }): JSX.Element {
60+
const { animatedFooterHeight } = useBottomSheetInternal()
61+
const bottomSpacerStyle = useAnimatedStyle(() => ({
62+
height: animatedFooterHeight.value,
63+
}))
64+
65+
const logo = useUniswapCafeLogo()
66+
67+
return (
68+
<Flex centered gap="$spacing12" justifyContent="space-between" pb="$spacing12">
69+
<Flex centered gap="$spacing20">
70+
{logo && <RemoteImage height={50} uri={logo} width={200} />}
71+
<Flex
72+
centered
73+
borderColor="$surface3"
74+
borderRadius="$rounded20"
75+
borderWidth={1}
76+
gap="$spacing12"
77+
px="$spacing24"
78+
py="$spacing24">
79+
<RequestDetailsContent request={request} />
80+
</Flex>
81+
</Flex>
82+
<Animated.View style={bottomSpacerStyle} />
83+
</Flex>
84+
)
85+
}

apps/mobile/src/components/WalletConnect/RequestModal/RequestDetails.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -194,7 +194,7 @@ function isSignTypedDataRequest(request: WalletConnectRequest): request is SignR
194194
return request.type === EthMethod.SignTypedData || request.type === EthMethod.SignTypedDataV4
195195
}
196196

197-
function RequestDetailsContent({ request }: Props): JSX.Element {
197+
export function RequestDetailsContent({ request }: Props): JSX.Element {
198198
const { t } = useTranslation()
199199

200200
if (isSignTypedDataRequest(request)) {

0 commit comments

Comments
 (0)