@apps-in-toss/native-modules 0.0.0-dev.1752049503789 → 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (75) hide show
  1. package/dist/bridges-meta.json +36 -13
  2. package/dist/index.cjs +256 -120
  3. package/dist/index.d.cts +697 -229
  4. package/dist/index.d.ts +697 -229
  5. package/dist/index.js +233 -103
  6. package/package.json +7 -8
  7. package/src/AppsInTossModule/constants.ts +6 -0
  8. package/src/AppsInTossModule/native-event-emitter/appsInTossEvent.ts +13 -0
  9. package/src/AppsInTossModule/native-event-emitter/contactsViral.ts +140 -0
  10. package/src/AppsInTossModule/native-event-emitter/event-plugins/EntryMessageExitedEvent.ts +10 -0
  11. package/src/AppsInTossModule/native-event-emitter/event-plugins/UpdateLocationEvent.ts +60 -0
  12. package/src/AppsInTossModule/native-event-emitter/index.ts +5 -0
  13. package/src/AppsInTossModule/native-event-emitter/internal/AppBridgeCallbackEvent.ts +45 -0
  14. package/src/AppsInTossModule/native-event-emitter/internal/VisibilityChangedByTransparentServiceWebEvent.ts +50 -0
  15. package/src/AppsInTossModule/native-event-emitter/internal/appBridge.spec.ts +135 -0
  16. package/src/AppsInTossModule/native-event-emitter/internal/appBridge.ts +79 -0
  17. package/src/AppsInTossModule/native-event-emitter/internal/onVisibilityChangedByTransparentServiceWeb.ts +20 -0
  18. package/src/AppsInTossModule/native-event-emitter/nativeEventEmitter.ts +35 -0
  19. package/src/AppsInTossModule/native-event-emitter/startUpdateLocation.ts +98 -0
  20. package/src/AppsInTossModule/native-event-emitter/types.ts +4 -0
  21. package/src/AppsInTossModule/native-modules/AppsInTossModule.ts +89 -0
  22. package/src/AppsInTossModule/native-modules/ads/googleAdMob.ts +690 -0
  23. package/src/AppsInTossModule/native-modules/ads/types.ts +106 -0
  24. package/src/AppsInTossModule/native-modules/appLogin.ts +29 -0
  25. package/src/AppsInTossModule/native-modules/checkoutPayment.ts +80 -0
  26. package/src/AppsInTossModule/native-modules/eventLog.spec.ts +300 -0
  27. package/src/AppsInTossModule/native-modules/eventLog.ts +77 -0
  28. package/src/AppsInTossModule/native-modules/fetchAlbumPhotos.ts +88 -0
  29. package/src/AppsInTossModule/native-modules/fetchContacts.ts +121 -0
  30. package/src/AppsInTossModule/native-modules/getClipboardText.ts +47 -0
  31. package/src/AppsInTossModule/native-modules/getCurrentLocation.ts +65 -0
  32. package/src/AppsInTossModule/native-modules/getDeviceId.ts +33 -0
  33. package/src/AppsInTossModule/native-modules/getGameCenterGameProfile.ts +68 -0
  34. package/src/AppsInTossModule/native-modules/getOperationalEnvironment.ts +37 -0
  35. package/src/AppsInTossModule/native-modules/getPermission.ts +58 -0
  36. package/src/AppsInTossModule/native-modules/getTossAppVersion.ts +33 -0
  37. package/src/AppsInTossModule/native-modules/getTossShareLink.ts +39 -0
  38. package/src/AppsInTossModule/native-modules/iap.ts +213 -0
  39. package/src/AppsInTossModule/native-modules/index.ts +86 -0
  40. package/src/AppsInTossModule/native-modules/isMinVersionSupported.spec.ts +190 -0
  41. package/src/AppsInTossModule/native-modules/isMinVersionSupported.ts +68 -0
  42. package/src/AppsInTossModule/native-modules/openCamera.ts +81 -0
  43. package/src/AppsInTossModule/native-modules/openGameCenterLeaderboard.ts +44 -0
  44. package/src/AppsInTossModule/native-modules/openPermissionDialog.ts +54 -0
  45. package/src/AppsInTossModule/native-modules/requestPermission.ts +63 -0
  46. package/src/AppsInTossModule/native-modules/saveBase64Data.ts +57 -0
  47. package/src/AppsInTossModule/native-modules/setClipboardText.ts +39 -0
  48. package/src/AppsInTossModule/native-modules/setDeviceOrientation.ts +74 -0
  49. package/src/AppsInTossModule/native-modules/storage.ts +100 -0
  50. package/src/AppsInTossModule/native-modules/submitGameCenterLeaderBoardScore.ts +74 -0
  51. package/src/AppsInTossModule/native-modules/tossCore.ts +29 -0
  52. package/src/BedrockModule/native-modules/core/BedrockCoreModule.ts +8 -0
  53. package/src/BedrockModule/native-modules/index.ts +4 -0
  54. package/src/BedrockModule/native-modules/natives/BedrockModule.ts +20 -0
  55. package/src/BedrockModule/native-modules/natives/closeView.ts +25 -0
  56. package/src/BedrockModule/native-modules/natives/generateHapticFeedback/index.ts +27 -0
  57. package/src/BedrockModule/native-modules/natives/generateHapticFeedback/types.ts +38 -0
  58. package/src/BedrockModule/native-modules/natives/getLocale.ts +46 -0
  59. package/src/BedrockModule/native-modules/natives/getNetworkStatus/index.ts +59 -0
  60. package/src/BedrockModule/native-modules/natives/getNetworkStatus/types.ts +1 -0
  61. package/src/BedrockModule/native-modules/natives/getPlatformOS.ts +37 -0
  62. package/src/BedrockModule/native-modules/natives/getSchemeUri.ts +27 -0
  63. package/src/BedrockModule/native-modules/natives/index.ts +11 -0
  64. package/src/BedrockModule/native-modules/natives/openURL.ts +40 -0
  65. package/src/BedrockModule/native-modules/natives/setIosSwipeGestureEnabled.ts +43 -0
  66. package/src/BedrockModule/native-modules/natives/setScreenAwakeMode.ts +66 -0
  67. package/src/BedrockModule/native-modules/natives/setSecureScreen.ts +31 -0
  68. package/src/BedrockModule/native-modules/natives/share.ts +36 -0
  69. package/src/async-bridges.ts +3 -0
  70. package/src/event-bridges.ts +2 -0
  71. package/src/index.ts +16 -0
  72. package/src/types.ts +108 -0
  73. package/src/utils/compareVersion.spec.ts +176 -0
  74. package/src/utils/compareVersion.ts +104 -0
  75. package/src/utils/generateUUID.ts +5 -0
@@ -0,0 +1,690 @@
1
+ import { noop } from 'es-toolkit';
2
+ import type { AdMobFullScreenEvent, AdMobHandlerParams, InterstitialAd, RewardedAd } from './types';
3
+ import { INTERNAL__appBridgeHandler } from '../../native-event-emitter/internal/appBridge';
4
+ import { getOperationalEnvironment } from '../getOperationalEnvironment';
5
+ import { isMinVersionSupported } from '../isMinVersionSupported';
6
+
7
+ // MARK: Interstitial AD (load)
8
+
9
+ export interface LoadAdMobInterstitialAdOptions {
10
+ /**
11
+ * 광고 단위 ID
12
+ */
13
+ adUnitId: string;
14
+ }
15
+
16
+ /**
17
+ * @public
18
+ * @category 광고
19
+ * @name LoadAdMobInterstitialAdEvent
20
+ * @description 전면 광고를 불러오는 함수에서 발생하는 이벤트 타입이에요. `loaded` 이벤트가 발생하면 광고를 성공적으로 불러온 거예요. 이때 [InterstitialAd](/react-native/reference/native-modules/광고/InterstitialAd.html) 객체가 함께 반환돼요.
21
+ */
22
+ export type LoadAdMobInterstitialAdEvent =
23
+ | AdMobFullScreenEvent
24
+ | {
25
+ type: 'loaded';
26
+ data: InterstitialAd;
27
+ };
28
+
29
+ /**
30
+ * @public
31
+ * @category 광고
32
+ * @name LoadAdMobInterstitialAdParams
33
+ * @description 전면 광고를 불러오는 함수에 필요한 옵션 객체예요.
34
+ */
35
+ export type LoadAdMobInterstitialAdParams = AdMobHandlerParams<
36
+ LoadAdMobInterstitialAdOptions,
37
+ LoadAdMobInterstitialAdEvent
38
+ >;
39
+
40
+ /**
41
+ * @public
42
+ * @category 광고
43
+ * @name loadAdMobInterstitialAd
44
+ * @description 앱 화면 전체를 덮는 전면 광고를 미리 불러와서, 광고가 필요한 시점에 바로 보여줄 수 있도록 준비하는 함수예요.
45
+ * @param {LoadAdMobInterstitialAdParams} params 광고를 불러올 때 사용할 설정 값이에요. 광고 ID와 광고의 동작에 대한 콜백을 설정할 수 있어요.
46
+ * @param {LoadAdMobInterstitialAdOptions} params.options 광고를 불러올 때 전달할 옵션 객체예요.
47
+ * @param {string} params.options.adUnitId 광고 단위 ID예요. 발급받은 전면 광고용 ID를 입력해요.
48
+ * @param {(event: LoadAdMobInterstitialAdEvent) => void} [params.onEvent] 광고 관련 이벤트가 발생했을 때 호출돼요. (예시: 광고가 닫히거나 클릭됐을 때). 자세한 이벤트 타입은 [LoadAdMobInterstitialAdEvent](/react-native/reference/native-modules/광고/LoadAdMobInterstitialAdEvent.html)를 참고하세요.
49
+ * @param {(reason: unknown) => void} [params.onError] 광고를 불러오지 못했을 때 호출돼요. (예시: 네트워크 오류나 지원하지 않는 환경일 때)
50
+ * @property {() => boolean} [isSupported] 현재 실행 중인 앱(예: 토스 앱, 개발용 샌드박스 앱 등)에서 Google AdMob 광고 기능을 지원하는지 확인하는 함수예요. 기능을 사용하기 전에 지원 여부를 확인해야 해요.
51
+ *
52
+ * @example
53
+ * ### 버튼 눌러 불러온 전면 광고 보여주기
54
+ * ```tsx
55
+ * import { GoogleAdMob } from '@apps-in-toss/framework';
56
+ * import { useFocusEffect } from '@granite-js/native/@react-navigation/native';
57
+ * import { useNavigation } from '@granite-js/react-native';
58
+ * import { useCallback, useState } from 'react';
59
+ * import { Button, Text, View } from 'react-native';
60
+ *
61
+ * const AD_UNIT_ID = '<AD_UNIT_ID>';
62
+ *
63
+ * export function GoogleAdmobInterstitialAdExample() {
64
+ * const [adLoadStatus, setAdLoadStatus] = useState<'not_loaded' | 'loaded' | 'failed'>('not_loaded');
65
+ * const navigation = useNavigation();
66
+ *
67
+ * const loadAd = useCallback(() => {
68
+ * if (GoogleAdMob.loadAdMobInterstitialAd.isSupported() !== true) {
69
+ * return;
70
+ * }
71
+
72
+ * const cleanup = GoogleAdMob.loadAdMobInterstitialAd({
73
+ * options: {
74
+ * adUnitId: AD_UNIT_ID,
75
+ * },
76
+ * onEvent: (event) => {
77
+ * switch (event.type) {
78
+ * case 'loaded':
79
+ * console.log('광고 로드 성공', event.data);
80
+ * setAdLoadStatus('loaded');
81
+ * break;
82
+ *
83
+ * case 'clicked':
84
+ * console.log('광고 클릭');
85
+ * break;
86
+ *
87
+ * case 'dismissed':
88
+ * console.log('광고 닫힘');
89
+ * navigation.navigate('/examples/google-admob-interstitial-ad-landing');
90
+ * break;
91
+ *
92
+ * case 'failedToShow':
93
+ * console.log('광고 보여주기 실패');
94
+ * break;
95
+ *
96
+ * case 'impression':
97
+ * console.log('광고 노출');
98
+ * break;
99
+ *
100
+ * case 'show':
101
+ * console.log('광고 컨텐츠 보여졌음');
102
+ * break;
103
+ * }
104
+ * },
105
+ * onError: (error) => {
106
+ * console.error('광고 불러오기 실패', error);
107
+ * },
108
+ * });
109
+ *
110
+ * return cleanup;
111
+ * }, [navigation]);
112
+ *
113
+ * const showAd = useCallback(() => {
114
+ * if (GoogleAdMob.showAdMobInterstitialAd.isSupported() !== true) {
115
+ * return;
116
+ * }
117
+ *
118
+ * GoogleAdMob.showAdMobInterstitialAd({
119
+ * options: {
120
+ * adUnitId: AD_UNIT_ID,
121
+ * },
122
+ * onEvent: (event) => {
123
+ * switch (event.type) {
124
+ * case 'requested':
125
+ * console.log('광고 보여주기 요청 완료');
126
+ * break;
127
+ * }
128
+ * },
129
+ * onError: (error) => {
130
+ * console.error('광고 보여주기 실패', error);
131
+ * },
132
+ * });
133
+ * }, []);
134
+ *
135
+ * useFocusEffect(loadAd);
136
+ *
137
+ * return (
138
+ * <View>
139
+ * <Text>
140
+ * {adLoadStatus === 'not_loaded' && '광고 로드 하지 않음 '}
141
+ * {adLoadStatus === 'loaded' && '광고 로드 완료'}
142
+ * {adLoadStatus === 'failed' && '광고 로드 실패'}
143
+ * </Text>
144
+ *
145
+ * <Button title="Show Ad" onPress={showAd} disabled={adLoadStatus !== 'loaded'} />
146
+ * </View>
147
+ * );
148
+ * }
149
+ * ```
150
+ */
151
+ export function loadAdMobInterstitialAd(params: LoadAdMobInterstitialAdParams) {
152
+ if (!loadAdMobInterstitialAd.isSupported()) {
153
+ params.onError(new Error(UNSUPPORTED_ERROR_MESSAGE));
154
+ return noop as () => void;
155
+ }
156
+
157
+ const { onEvent, onError, options } = params;
158
+
159
+ const unregisterCallbacks = INTERNAL__appBridgeHandler.invokeAppBridgeMethod('loadAdMobInterstitialAd', options, {
160
+ onAdClicked: () => {
161
+ onEvent({ type: 'clicked' });
162
+ },
163
+ onAdDismissed: () => {
164
+ onEvent({ type: 'dismissed' });
165
+ },
166
+ onAdFailedToShow: () => {
167
+ onEvent({ type: 'failedToShow' });
168
+ },
169
+ onAdImpression: () => {
170
+ onEvent({ type: 'impression' });
171
+ },
172
+ onAdShow: () => {
173
+ onEvent({ type: 'show' });
174
+ },
175
+ onSuccess: (result: InterstitialAd) => onEvent({ type: 'loaded', data: result }),
176
+ onError,
177
+ });
178
+
179
+ return unregisterCallbacks;
180
+ }
181
+
182
+ // MARK: Interstitial AD (show)
183
+
184
+ export interface ShowAdMobInterstitialAdOptions {
185
+ /**
186
+ * 광고 단위 ID
187
+ */
188
+ adUnitId: string;
189
+ }
190
+
191
+ /**
192
+ * @public
193
+ * @category 광고
194
+ * @name ShowAdMobInterstitialAdEvent
195
+ * @description 전면 광고를 보여주는 함수에서 발생하는 이벤트 타입이에요. `requested` 이벤트가 발생하면 광고 노출 요청이 Google AdMob에 성공적으로 전달된 거예요.
196
+ */
197
+ export type ShowAdMobInterstitialAdEvent = { type: 'requested' };
198
+
199
+ /**
200
+ * @public
201
+ * @category 광고
202
+ * @name ShowAdMobInterstitialAdParams
203
+ * @description 불러온 전면 광고를 보여주는 함수에 필요한 옵션 객체예요.
204
+ */
205
+ export type ShowAdMobInterstitialAdParams = AdMobHandlerParams<
206
+ ShowAdMobInterstitialAdOptions,
207
+ ShowAdMobInterstitialAdEvent
208
+ >;
209
+
210
+ /**
211
+ * @public
212
+ * @category 광고
213
+ * @name showAdMobInterstitialAd
214
+ * @description 앱 화면 전체를 덮는 전면 광고를 사용자에게 노출해요. 이 함수는 `loadAdMobInterstitialAd` 로 미리 불러온 광고를 실제로 사용자에게 노출해요.
215
+ * @param {ShowAdMobInterstitialAdParams} params 광고를 보여주기 위해 사용할 설정 값이에요. 광고 ID와과 광고의 동작에 대한 콜백을 설정할 수 있어요.
216
+ * @param {ShowAdMobInterstitialAdOptions} params.options 광고를 보여줄 때 전달할 옵션 객체예요.
217
+ * @param {string} params.options.adUnitId 광고 단위 ID예요. `loadAdMobInterstitialAd` 로 불러온 전면 광고용 ID를 입력해요.
218
+ * @param {(event: ShowAdMobInterstitialAdEvent) => void} [params.onEvent] 광고 관련 이벤트가 발생했을 때 호출돼요. (예시: 광고 노출을 요청했을 때). 자세한 이벤트 타입은 [ShowAdMobInterstitialAdEvent](/react-native/reference/native-modules/광고/ShowAdMobInterstitialAdEvent.html)를 참고하세요.
219
+ * @param {(reason: unknown) => void} [params.onError] 광고를 노출하지 못했을 때 호출돼요. (예시: 네트워크 오류나 지원하지 않는 환경일 때)
220
+ * @property {() => boolean} [isSupported] 현재 실행 중인 앱(예: 토스 앱, 개발용 샌드박스 앱 등)에서 Google AdMob 광고 기능을 지원하는지 확인하는 함수예요. 기능을 사용하기 전에 지원 여부를 확인해야 해요.
221
+ *
222
+ * @example
223
+ * ### 버튼 눌러 불러온 전면 광고 보여주기
224
+ * ```tsx
225
+ * import { GoogleAdMob } from '@apps-in-toss/framework';
226
+ * import { useFocusEffect } from '@granite-js/native/@react-navigation/native';
227
+ * import { useNavigation } from '@granite-js/react-native';
228
+ * import { useCallback, useState } from 'react';
229
+ * import { Button, Text, View } from 'react-native';
230
+ *
231
+ * const AD_UNIT_ID = '<AD_UNIT_ID>';
232
+ *
233
+ * export function GoogleAdmobInterstitialAdExample() {
234
+ * const [adLoadStatus, setAdLoadStatus] = useState<'not_loaded' | 'loaded' | 'failed'>('not_loaded');
235
+ * const navigation = useNavigation();
236
+ *
237
+ * const loadAd = useCallback(() => {
238
+ * if (GoogleAdMob.loadAdMobInterstitialAd.isSupported() !== true) {
239
+ * return;
240
+ * }
241
+
242
+ * const cleanup = GoogleAdMob.loadAdMobInterstitialAd({
243
+ * options: {
244
+ * adUnitId: AD_UNIT_ID,
245
+ * },
246
+ * onEvent: (event) => {
247
+ * switch (event.type) {
248
+ * case 'loaded':
249
+ * console.log('광고 로드 성공', event.data);
250
+ * setAdLoadStatus('loaded');
251
+ * break;
252
+ *
253
+ * case 'clicked':
254
+ * console.log('광고 클릭');
255
+ * break;
256
+ *
257
+ * case 'dismissed':
258
+ * console.log('광고 닫힘');
259
+ * navigation.navigate('/examples/google-admob-interstitial-ad-landing');
260
+ * break;
261
+ *
262
+ * case 'failedToShow':
263
+ * console.log('광고 보여주기 실패');
264
+ * break;
265
+ *
266
+ * case 'impression':
267
+ * console.log('광고 노출');
268
+ * break;
269
+ *
270
+ * case 'show':
271
+ * console.log('광고 컨텐츠 보여졌음');
272
+ * break;
273
+ * }
274
+ * },
275
+ * onError: (error) => {
276
+ * console.error('광고 불러오기 실패', error);
277
+ * },
278
+ * });
279
+ *
280
+ * return cleanup;
281
+ * }, [navigation]);
282
+ *
283
+ * const showAd = useCallback(() => {
284
+ * if (GoogleAdMob.showAdMobInterstitialAd.isSupported() !== true) {
285
+ * return;
286
+ * }
287
+ *
288
+ * GoogleAdMob.showAdMobInterstitialAd({
289
+ * options: {
290
+ * adUnitId: AD_UNIT_ID,
291
+ * },
292
+ * onEvent: (event) => {
293
+ * switch (event.type) {
294
+ * case 'requested':
295
+ * console.log('광고 보여주기 요청 완료');
296
+ * break;
297
+ * }
298
+ * },
299
+ * onError: (error) => {
300
+ * console.error('광고 보여주기 실패', error);
301
+ * },
302
+ * });
303
+ * }, []);
304
+ *
305
+ * useFocusEffect(loadAd);
306
+ *
307
+ * return (
308
+ * <View>
309
+ * <Text>
310
+ * {adLoadStatus === 'not_loaded' && '광고 로드 하지 않음 '}
311
+ * {adLoadStatus === 'loaded' && '광고 로드 완료'}
312
+ * {adLoadStatus === 'failed' && '광고 로드 실패'}
313
+ * </Text>
314
+ *
315
+ * <Button title="Show Ad" onPress={showAd} disabled={adLoadStatus !== 'loaded'} />
316
+ * </View>
317
+ * );
318
+ * }
319
+ * ```
320
+ */
321
+ export function showAdMobInterstitialAd(params: ShowAdMobInterstitialAdParams) {
322
+ if (!showAdMobInterstitialAd.isSupported()) {
323
+ params.onError(new Error(UNSUPPORTED_ERROR_MESSAGE));
324
+ return noop as () => void;
325
+ }
326
+
327
+ const { onEvent, onError, options } = params;
328
+
329
+ const unregisterCallbacks = INTERNAL__appBridgeHandler.invokeAppBridgeMethod('showAdMobInterstitialAd', options, {
330
+ onSuccess: () => onEvent({ type: 'requested' }),
331
+ onError,
332
+ });
333
+
334
+ return unregisterCallbacks;
335
+ }
336
+
337
+ // MARK: Rewarded AD (load)
338
+
339
+ export interface LoadAdMobRewardedAdOptions {
340
+ /**
341
+ * 광고 단위 ID
342
+ */
343
+ adUnitId: string;
344
+ }
345
+
346
+ /**
347
+ * @public
348
+ * @category 광고
349
+ * @name LoadAdMobRewardedAdEvent
350
+ * @description 보상형 광고를 불러오는 함수에서 발생하는 이벤트 타입이에요. `loaded` 이벤트가 발생하면 광고를 성공적으로 불러온 거예요. 이때 [RewardedAd](/react-native/reference/native-modules/광고/RewardedAd.html) 객체가 함께 반환돼요. `userEarnedReward` 이벤트는 사용자가 광고를 끝까지 시청해, 보상 조건을 충족했을 때 발생해요.
351
+ */
352
+ export type LoadAdMobRewardedAdEvent =
353
+ | AdMobFullScreenEvent
354
+ | { type: 'loaded'; data: RewardedAd }
355
+ | { type: 'userEarnedReward' };
356
+
357
+ /**
358
+ * @public
359
+ * @category 광고
360
+ * @name LoadAdMobRewardedAdParams
361
+ * @description 보상형 광고를 불러오는 함수에 필요한 옵션 객체예요.
362
+ */
363
+ export type LoadAdMobRewardedAdParams = AdMobHandlerParams<LoadAdMobRewardedAdOptions, LoadAdMobRewardedAdEvent>;
364
+
365
+ /**
366
+ * @public
367
+ * @category 광고
368
+ * @name loadAdMobRewardedAd
369
+ * @description 사용자가 광고를 끝까지 시청하면 리워드를 제공할 수 있는 보상형 광고를 미리 불러와서, 광고가 필요한 시점에 바로 보여줄 수 있도록 준비하는 함수예요.
370
+ * @param {LoadAdMobRewardedAdParams} params 광고를 불러올 때 사용할 설정 값이에요. 광고 ID와 광고의 동작에 대한 콜백을 설정할 수 있어요.
371
+ * @param {LoadAdMobRewardedAdOptions} params.options 광고를 불러올 때 전달할 옵션 객체예요.
372
+ * @param {string} params.options.adUnitId 광고 단위 ID예요. 발급받은 보상형 광고용 ID를 입력해요.
373
+ * @param {(event: LoadAdMobRewardedAdEvent) => void} [params.onEvent] 광고 관련 이벤트가 발생했을 때 호출돼요. (예시: 광고가 닫히거나 클릭됐을 때). 자세한 이벤트 타입은 [LoadAdMobRewardedAdEvent](/react-native/reference/native-modules/광고/LoadAdMobRewardedAdEvent.html)를 참고하세요.
374
+ * @param {(reason: unknown) => void} [params.onError] 광고를 불러오지 못했을 때 호출돼요. (예시: 네트워크 오류나 지원하지 않는 환경일 때)
375
+ * @property {() => boolean} [isSupported] 현재 실행 중인 앱(예: 토스 앱, 개발용 샌드박스 앱 등)에서 Google AdMob 광고 기능을 지원하는지 확인하는 함수예요. 기능을 사용하기 전에 지원 여부를 확인해야 해요.
376
+ *
377
+ * @example
378
+ *
379
+ * ### 버튼 눌러 불러온 보상형 광고 보여주기
380
+ *
381
+ * ```tsx
382
+ * import { GoogleAdMob } from '@apps-in-toss/framework';
383
+ * import { useCallback, useState } from 'react';
384
+ * import { Button, Text, View } from 'react-native';
385
+ *
386
+ * const AD_UNIT_ID = '<AD_UNIT_ID>';
387
+ *
388
+ * export function GoogleAdmobRewardedAdExample() {
389
+ * const [adLoadStatus, setAdLoadStatus] = useState<'not_loaded' | 'loaded' | 'failed'>('not_loaded');
390
+ *
391
+ * const loadAd = useCallback(() => {
392
+ * if (GoogleAdMob.loadAdMobRewardedAd.isSupported() !== true) {
393
+ * return;
394
+ * }
395
+ *
396
+ * const cleanup = GoogleAdMob.loadAdMobRewardedAd({
397
+ * options: {
398
+ * adUnitId: AD_UNIT_ID,
399
+ * },
400
+ * onEvent: (event) => {
401
+ * console.log(event.type);
402
+ * switch (event.type) {
403
+ * case 'loaded':
404
+ * console.log('광고 로드 성공', event.data);
405
+ * setAdLoadStatus('loaded');
406
+ * break;
407
+ *
408
+ * case 'clicked':
409
+ * console.log('광고 클릭');
410
+ * break;
411
+ *
412
+ * case 'dismissed':
413
+ * console.log('광고 닫힘');
414
+ * break;
415
+ *
416
+ * case 'failedToShow':
417
+ * console.log('광고 보여주기 실패');
418
+ * break;
419
+ *
420
+ * case 'impression':
421
+ * console.log('광고 노출');
422
+ * break;
423
+ *
424
+ * case 'show':
425
+ * console.log('광고 컨텐츠 보여졌음');
426
+ * break;
427
+ *
428
+ * case 'userEarnedReward':
429
+ * console.log('사용자가 광고 시청을 완료했음');
430
+ * break;
431
+ * }
432
+ * },
433
+ * onError: (error) => {
434
+ * console.error('광고 불러오기 실패', error);
435
+ * },
436
+ * });
437
+ *
438
+ * return cleanup;
439
+ * }, []);
440
+ *
441
+ * const showAd = useCallback(() => {
442
+ * if (GoogleAdMob.showAdMobRewardedAd.isSupported() !== true) {
443
+ * return;
444
+ * }
445
+ *
446
+ * GoogleAdMob.showAdMobRewardedAd({
447
+ * options: {
448
+ * adUnitId: AD_UNIT_ID,
449
+ * },
450
+ * onEvent: (event) => {
451
+ * switch (event.type) {
452
+ * case 'requested':
453
+ * console.log('광고 보여주기 요청 완료');
454
+ * setAdLoadStatus('not_loaded');
455
+ * break;
456
+ * }
457
+ * },
458
+ * onError: (error) => {
459
+ * console.error('광고 보여주기 실패', error);
460
+ * },
461
+ * });
462
+ * }, []);
463
+ *
464
+ * return (
465
+ * <View>
466
+ * <Text>
467
+ * {adLoadStatus === 'not_loaded' && '광고 로드 하지 않음 '}
468
+ * {adLoadStatus === 'loaded' && '광고 로드 완료'}
469
+ * {adLoadStatus === 'failed' && '광고 로드 실패'}
470
+ * </Text>
471
+ *
472
+ * <Button title="Load Ad" onPress={loadAd} />
473
+ * <Button title="Show Ad" onPress={showAd} disabled={adLoadStatus !== 'loaded'} />
474
+ * </View>
475
+ * );
476
+ * }
477
+ * ```
478
+ */
479
+ export function loadAdMobRewardedAd(params: LoadAdMobRewardedAdParams) {
480
+ if (!loadAdMobRewardedAd.isSupported()) {
481
+ params.onError(new Error(UNSUPPORTED_ERROR_MESSAGE));
482
+ return noop as () => void;
483
+ }
484
+
485
+ const { onEvent, onError, options } = params;
486
+
487
+ const unregisterCallbacks = INTERNAL__appBridgeHandler.invokeAppBridgeMethod('loadAdMobRewardedAd', options, {
488
+ onAdClicked: () => {
489
+ onEvent({ type: 'clicked' });
490
+ },
491
+ onAdDismissed: () => {
492
+ onEvent({ type: 'dismissed' });
493
+ },
494
+ onAdFailedToShow: () => {
495
+ onEvent({ type: 'failedToShow' });
496
+ },
497
+ onAdImpression: () => {
498
+ onEvent({ type: 'impression' });
499
+ },
500
+ onAdShow: () => {
501
+ onEvent({ type: 'show' });
502
+ },
503
+ onUserEarnedReward: () => {
504
+ onEvent({ type: 'userEarnedReward' });
505
+ },
506
+ onSuccess: (result: RewardedAd) => onEvent({ type: 'loaded', data: result }),
507
+ onError,
508
+ });
509
+
510
+ return unregisterCallbacks;
511
+ }
512
+
513
+ // MARK: Rewarded AD (show)
514
+
515
+ export interface ShowAdMobRewardedAdOptions {
516
+ /**
517
+ * 광고 단위 ID
518
+ */
519
+ adUnitId: string;
520
+ }
521
+
522
+ /**
523
+ * @public
524
+ * @category 광고
525
+ * @name ShowAdMobRewardedAdEvent
526
+ * @description 보상형 광고를 보여주는 함수에서 발생하는 이벤트 타입이에요. `requested` 이벤트가 발생하면 광고 노출 요청이 Google AdMob에 성공적으로 전달된 거예요.
527
+ */
528
+ export type ShowAdMobRewardedAdEvent = { type: 'requested' };
529
+
530
+ /**
531
+ * @public
532
+ * @category 광고
533
+ * @name ShowAdMobRewardedAdParams
534
+ * @description 불러온 보상형 광고를 보여주는 함수에 필요한 옵션 객체예요.
535
+ */
536
+ export type ShowAdMobRewardedAdParams = AdMobHandlerParams<ShowAdMobRewardedAdOptions, ShowAdMobRewardedAdEvent>;
537
+
538
+ /**
539
+ * @public
540
+ * @category 광고
541
+ * @name showAdMobRewardedAd
542
+ * @description 사용자가 광고를 끝까지 보면 리워드를 받을 수 있도록, 보상형 광고를 화면에 보여줘요. 이 함수는 `loadAdMobRewardedAd` 로 미리 불러온 광고를 실제로 사용자에게 노출해요.
543
+ * @param {ShowAdMobRewardedAdParams} params 광고를 보여주기 위해 사용할 설정 값이에요. 광고 ID와 광고의 동작에 대한 콜백을 설정할 수 있어요.
544
+ * @param {ShowAdMobRewardedAdOptions} params.options 광고를 보여줄 때 전달할 옵션 객체예요.
545
+ * @param {string} params.options.adUnitId 광고 단위 ID예요. `loadAdMobRewardedAd` 로 불러온 보상형 광고용 ID를 입력해요.
546
+ * @param {(event: ShowAdMobRewardedAdEvent) => void} [params.onEvent] 광고 관련 이벤트가 발생했을 때 호출돼요. (예시: 광고 노출을 요청했을 때). 자세한 이벤트 타입은 [ShowAdMobRewardedAdEvent](/react-native/reference/native-modules/광고/ShowAdMobRewardedAdEvent.html)를 참고하세요.
547
+ * @param {(reason: unknown) => void} [params.onError] 광고를 불러오지 못했을 때 호출돼요. (예시: 네트워크 오류나 지원하지 않는 환경일 때)
548
+ * @property {() => boolean} [isSupported] 현재 실행 중인 앱(예: 토스 앱, 개발용 샌드박스 앱 등)에서 Google AdMob 광고 기능을 지원하는지 확인하는 함수예요. 기능을 사용하기 전에 지원 여부를 확인해야 해요.
549
+ *
550
+ * @example
551
+ * ### 버튼 눌러 불러온 보상형 광고 보여주기
552
+ *
553
+ * ```tsx
554
+ * import { GoogleAdMob } from '@apps-in-toss/framework';
555
+ * import { useCallback, useState } from 'react';
556
+ * import { Button, Text, View } from 'react-native';
557
+ *
558
+ * const AD_UNIT_ID = '<AD_UNIT_ID>';
559
+ *
560
+ * export function GoogleAdmobRewardedAdExample() {
561
+ * const [adLoadStatus, setAdLoadStatus] = useState<'not_loaded' | 'loaded' | 'failed'>('not_loaded');
562
+ *
563
+ * const loadAd = useCallback(() => {
564
+ * if (GoogleAdMob.loadAdMobRewardedAd.isSupported() !== true) {
565
+ * return;
566
+ * }
567
+ *
568
+ * const cleanup = GoogleAdMob.loadAdMobRewardedAd({
569
+ * options: {
570
+ * adUnitId: AD_UNIT_ID,
571
+ * },
572
+ * onEvent: (event) => {
573
+ * console.log(event.type);
574
+ * switch (event.type) {
575
+ * case 'loaded':
576
+ * console.log('광고 로드 성공', event.data);
577
+ * setAdLoadStatus('loaded');
578
+ * break;
579
+ *
580
+ * case 'clicked':
581
+ * console.log('광고 클릭');
582
+ * break;
583
+ *
584
+ * case 'dismissed':
585
+ * console.log('광고 닫힘');
586
+ * break;
587
+ *
588
+ * case 'failedToShow':
589
+ * console.log('광고 보여주기 실패');
590
+ * break;
591
+ *
592
+ * case 'impression':
593
+ * console.log('광고 노출');
594
+ * break;
595
+ *
596
+ * case 'show':
597
+ * console.log('광고 컨텐츠 보여졌음');
598
+ * break;
599
+ *
600
+ * case 'userEarnedReward':
601
+ * console.log('사용자가 광고 시청을 완료했음');
602
+ * break;
603
+ * }
604
+ * },
605
+ * onError: (error) => {
606
+ * console.error('광고 불러오기 실패', error);
607
+ * },
608
+ * });
609
+ *
610
+ * return cleanup;
611
+ * }, []);
612
+ *
613
+ * const showAd = useCallback(() => {
614
+ * if (GoogleAdMob.showAdMobRewardedAd.isSupported() !== true) {
615
+ * return;
616
+ * }
617
+ *
618
+ * GoogleAdMob.showAdMobRewardedAd({
619
+ * options: {
620
+ * adUnitId: AD_UNIT_ID,
621
+ * },
622
+ * onEvent: (event) => {
623
+ * switch (event.type) {
624
+ * case 'requested':
625
+ * console.log('광고 보여주기 요청 완료');
626
+ * setAdLoadStatus('not_loaded');
627
+ * break;
628
+ * }
629
+ * },
630
+ * onError: (error) => {
631
+ * console.error('광고 보여주기 실패', error);
632
+ * },
633
+ * });
634
+ * }, []);
635
+ *
636
+ * return (
637
+ * <View>
638
+ * <Text>
639
+ * {adLoadStatus === 'not_loaded' && '광고 로드 하지 않음 '}
640
+ * {adLoadStatus === 'loaded' && '광고 로드 완료'}
641
+ * {adLoadStatus === 'failed' && '광고 로드 실패'}
642
+ * </Text>
643
+ *
644
+ * <Button title="Load Ad" onPress={loadAd} />
645
+ * <Button title="Show Ad" onPress={showAd} disabled={adLoadStatus !== 'loaded'} />
646
+ * </View>
647
+ * );
648
+ * }
649
+ * ```
650
+ */
651
+ export function showAdMobRewardedAd(params: ShowAdMobRewardedAdParams) {
652
+ if (!showAdMobRewardedAd.isSupported()) {
653
+ params.onError(new Error(UNSUPPORTED_ERROR_MESSAGE));
654
+ return noop as () => void;
655
+ }
656
+
657
+ const { onEvent, onError, options } = params;
658
+
659
+ const unregisterCallbacks = INTERNAL__appBridgeHandler.invokeAppBridgeMethod('showAdMobRewardedAd', options, {
660
+ onSuccess: () => onEvent({ type: 'requested' }),
661
+ onError,
662
+ });
663
+
664
+ return unregisterCallbacks;
665
+ }
666
+
667
+ // MARK: - isSupported
668
+
669
+ const ANDROID_GOOGLE_AD_MOB_SUPPORTED_VERSION = '5.209.0';
670
+ const IOS_GOOGLE_AD_MOB_SUPPORTED_VERSION = '5.209.0';
671
+ const UNSUPPORTED_ERROR_MESSAGE = 'This feature is not supported in the current environment';
672
+ const ENVIRONMENT = getOperationalEnvironment();
673
+
674
+ function createIsSupported() {
675
+ return () => {
676
+ if (ENVIRONMENT !== 'toss') {
677
+ return false;
678
+ }
679
+
680
+ return isMinVersionSupported({
681
+ android: ANDROID_GOOGLE_AD_MOB_SUPPORTED_VERSION,
682
+ ios: IOS_GOOGLE_AD_MOB_SUPPORTED_VERSION,
683
+ });
684
+ };
685
+ }
686
+
687
+ loadAdMobInterstitialAd.isSupported = createIsSupported();
688
+ loadAdMobRewardedAd.isSupported = createIsSupported();
689
+ showAdMobInterstitialAd.isSupported = createIsSupported();
690
+ showAdMobRewardedAd.isSupported = createIsSupported();