@apps-in-toss/native-modules 0.0.0-dev.1752115036458 → 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.
- package/dist/bridges-meta.json +36 -13
- package/dist/index.cjs +256 -120
- package/dist/index.d.cts +697 -229
- package/dist/index.d.ts +697 -229
- package/dist/index.js +233 -103
- package/package.json +7 -8
- package/src/AppsInTossModule/constants.ts +6 -0
- package/src/AppsInTossModule/native-event-emitter/appsInTossEvent.ts +13 -0
- package/src/AppsInTossModule/native-event-emitter/contactsViral.ts +140 -0
- package/src/AppsInTossModule/native-event-emitter/event-plugins/EntryMessageExitedEvent.ts +10 -0
- package/src/AppsInTossModule/native-event-emitter/event-plugins/UpdateLocationEvent.ts +60 -0
- package/src/AppsInTossModule/native-event-emitter/index.ts +5 -0
- package/src/AppsInTossModule/native-event-emitter/internal/AppBridgeCallbackEvent.ts +45 -0
- package/src/AppsInTossModule/native-event-emitter/internal/VisibilityChangedByTransparentServiceWebEvent.ts +50 -0
- package/src/AppsInTossModule/native-event-emitter/internal/appBridge.spec.ts +135 -0
- package/src/AppsInTossModule/native-event-emitter/internal/appBridge.ts +79 -0
- package/src/AppsInTossModule/native-event-emitter/internal/onVisibilityChangedByTransparentServiceWeb.ts +20 -0
- package/src/AppsInTossModule/native-event-emitter/nativeEventEmitter.ts +35 -0
- package/src/AppsInTossModule/native-event-emitter/startUpdateLocation.ts +98 -0
- package/src/AppsInTossModule/native-event-emitter/types.ts +4 -0
- package/src/AppsInTossModule/native-modules/AppsInTossModule.ts +89 -0
- package/src/AppsInTossModule/native-modules/ads/googleAdMob.ts +690 -0
- package/src/AppsInTossModule/native-modules/ads/types.ts +106 -0
- package/src/AppsInTossModule/native-modules/appLogin.ts +29 -0
- package/src/AppsInTossModule/native-modules/checkoutPayment.ts +80 -0
- package/src/AppsInTossModule/native-modules/eventLog.spec.ts +300 -0
- package/src/AppsInTossModule/native-modules/eventLog.ts +77 -0
- package/src/AppsInTossModule/native-modules/fetchAlbumPhotos.ts +88 -0
- package/src/AppsInTossModule/native-modules/fetchContacts.ts +121 -0
- package/src/AppsInTossModule/native-modules/getClipboardText.ts +47 -0
- package/src/AppsInTossModule/native-modules/getCurrentLocation.ts +65 -0
- package/src/AppsInTossModule/native-modules/getDeviceId.ts +33 -0
- package/src/AppsInTossModule/native-modules/getGameCenterGameProfile.ts +68 -0
- package/src/AppsInTossModule/native-modules/getOperationalEnvironment.ts +37 -0
- package/src/AppsInTossModule/native-modules/getPermission.ts +58 -0
- package/src/AppsInTossModule/native-modules/getTossAppVersion.ts +33 -0
- package/src/AppsInTossModule/native-modules/getTossShareLink.ts +39 -0
- package/src/AppsInTossModule/native-modules/iap.ts +213 -0
- package/src/AppsInTossModule/native-modules/index.ts +86 -0
- package/src/AppsInTossModule/native-modules/isMinVersionSupported.spec.ts +190 -0
- package/src/AppsInTossModule/native-modules/isMinVersionSupported.ts +68 -0
- package/src/AppsInTossModule/native-modules/openCamera.ts +81 -0
- package/src/AppsInTossModule/native-modules/openGameCenterLeaderboard.ts +44 -0
- package/src/AppsInTossModule/native-modules/openPermissionDialog.ts +54 -0
- package/src/AppsInTossModule/native-modules/requestPermission.ts +63 -0
- package/src/AppsInTossModule/native-modules/saveBase64Data.ts +57 -0
- package/src/AppsInTossModule/native-modules/setClipboardText.ts +39 -0
- package/src/AppsInTossModule/native-modules/setDeviceOrientation.ts +74 -0
- package/src/AppsInTossModule/native-modules/storage.ts +100 -0
- package/src/AppsInTossModule/native-modules/submitGameCenterLeaderBoardScore.ts +74 -0
- package/src/AppsInTossModule/native-modules/tossCore.ts +29 -0
- package/src/BedrockModule/native-modules/core/BedrockCoreModule.ts +8 -0
- package/src/BedrockModule/native-modules/index.ts +4 -0
- package/src/BedrockModule/native-modules/natives/BedrockModule.ts +20 -0
- package/src/BedrockModule/native-modules/natives/closeView.ts +25 -0
- package/src/BedrockModule/native-modules/natives/generateHapticFeedback/index.ts +27 -0
- package/src/BedrockModule/native-modules/natives/generateHapticFeedback/types.ts +38 -0
- package/src/BedrockModule/native-modules/natives/getLocale.ts +46 -0
- package/src/BedrockModule/native-modules/natives/getNetworkStatus/index.ts +59 -0
- package/src/BedrockModule/native-modules/natives/getNetworkStatus/types.ts +1 -0
- package/src/BedrockModule/native-modules/natives/getPlatformOS.ts +37 -0
- package/src/BedrockModule/native-modules/natives/getSchemeUri.ts +27 -0
- package/src/BedrockModule/native-modules/natives/index.ts +11 -0
- package/src/BedrockModule/native-modules/natives/openURL.ts +40 -0
- package/src/BedrockModule/native-modules/natives/setIosSwipeGestureEnabled.ts +43 -0
- package/src/BedrockModule/native-modules/natives/setScreenAwakeMode.ts +66 -0
- package/src/BedrockModule/native-modules/natives/setSecureScreen.ts +31 -0
- package/src/BedrockModule/native-modules/natives/share.ts +36 -0
- package/src/async-bridges.ts +3 -0
- package/src/event-bridges.ts +2 -0
- package/src/index.ts +16 -0
- package/src/types.ts +108 -0
- package/src/utils/compareVersion.spec.ts +176 -0
- package/src/utils/compareVersion.ts +104 -0
- package/src/utils/generateUUID.ts +5 -0
|
@@ -0,0 +1,213 @@
|
|
|
1
|
+
import { AppsInTossModule } from './AppsInTossModule';
|
|
2
|
+
import { isMinVersionSupported } from './isMinVersionSupported';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* @public
|
|
6
|
+
* @category 인앱결제
|
|
7
|
+
* @name IapCreateOneTimePurchaseOrderOptions
|
|
8
|
+
* @description 인앱결제 1건을 요청할 때 필요한 정보예요.
|
|
9
|
+
* @property {string} productId - 주문할 상품의 ID예요.
|
|
10
|
+
*/
|
|
11
|
+
export interface IapCreateOneTimePurchaseOrderOptions {
|
|
12
|
+
productId: string;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* @public
|
|
17
|
+
* @category 인앱결제
|
|
18
|
+
* @name IapCreateOneTimePurchaseOrderResult
|
|
19
|
+
* @description 인앱결제 1건이 완료되면 결제 세부 정보와 상품 정보를 담아 반환해요. 반환된 정보로 결제한 상품의 정보를 화면에 표시할 수 있어요.
|
|
20
|
+
* @property {string} orderId - 결제 주문 ID이에요. 결제 완료 후 [결제 상태를 조회](https://developers-apps-in-toss.toss.im/api/getIapOrderStatus.html)할 때 사용해요.
|
|
21
|
+
* @property {string} displayName - 화면에 표시할 상품 이름이에요.
|
|
22
|
+
* @property {string} displayAmount - 통화 단위가 포함된 가격 정보예요. 예를 들어 `1,000원`으로 가격과 통화가 함께 표시돼요.
|
|
23
|
+
* @property {number} amount - 상품 가격 숫자 값이에요. 화폐 단위와 쉼표를 제외한 순수 숫자예요. 예를 들어 `1000`으로 표시돼요.
|
|
24
|
+
* @property {string} currency - [ISO 4217 표준](https://ko.wikipedia.org/wiki/ISO_4217)에 따른 상품 가격 통화 단위예요. 예를 들어 원화는 `KRW`, 달러는 `USD`로 표시돼요.
|
|
25
|
+
* @property {number} fraction - 가격을 표시할 때 소수점 아래 몇 자리까지 보여줄지 정하는 값이에요. 예를 들어 달러는 소수점 둘째 자리까지 보여줘서 `2`, 원화는 소수점이 필요 없어서 `0`이에요
|
|
26
|
+
* @property {string | null} miniAppIconUrl - 미니앱 아이콘 이미지의 URL이에요. 아이콘은 앱인토스 콘솔에서 설정한 이미지예요. 콘솔에서 아이콘을 등록하지 않았다면 `null`로 반환돼요.
|
|
27
|
+
*/
|
|
28
|
+
export interface IapCreateOneTimePurchaseOrderResult {
|
|
29
|
+
orderId: string;
|
|
30
|
+
displayName: string;
|
|
31
|
+
displayAmount: string;
|
|
32
|
+
amount: number;
|
|
33
|
+
currency: string;
|
|
34
|
+
fraction: number;
|
|
35
|
+
miniAppIconUrl: string | null;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* @public
|
|
40
|
+
* @category 인앱결제
|
|
41
|
+
* @name iapCreateOneTimePurchaseOrder
|
|
42
|
+
* @description
|
|
43
|
+
* 특정 인앱결제 주문서 페이지로 이동해요. 사용자가 상품 구매 버튼을 누르는 상황 등에 사용할 수 있어요. 사용자의 결제는 이동한 페이지에서 진행돼요. 만약 결제 중에 에러가 발생하면 에러 유형에 따라 에러 페이지로 이동해요.
|
|
44
|
+
* @param {IapCreateOneTimePurchaseOrderOptions} params - 인앱결제를 생성할 때 필요한 정보예요.
|
|
45
|
+
* @param {string} params.productId - 주문할 상품의 ID예요.
|
|
46
|
+
* @returns {Promise<IapCreateOneTimePurchaseOrderResult | undefined>} 결제에 성공하면 결제 결과 객체를 반환해요. 앱 버전이 최소 지원 버전(안드로이드 5.219.0, iOS 5.219.0)보다 낮으면 인앱결제를 실행할 수 없어서 `undefined`를 반환해요.
|
|
47
|
+
*
|
|
48
|
+
* @throw {code: "INVALID_PRODUCT_ID"} - 유효하지 않은 상품 ID이거나, 해당 상품이 존재하지 않을 때 발생해요.
|
|
49
|
+
* @throw {code: "PAYMENT_PENDING"} - 사용자가 요청한 결제가 아직 승인을 기다리고 있을 때 발생해요.
|
|
50
|
+
* @throw {code: "NETWORK_ERROR"} - 서버 내부 문제로 요청을 처리할 수 없을 때 발생해요.
|
|
51
|
+
* @throw {code: "INVALID_USER_ENVIRONMENT"} - 특정 기기, 계정 또는 설정 환경에서 구매할 수 없는 상품일 때 발생해요.
|
|
52
|
+
* @throw {code: "ITEM_ALREADY_OWNED"} - 사용자가 이미 구매한 상품을 다시 구매하려고 할 때 발생해요.
|
|
53
|
+
* @throw {code: "APP_MARKET_VERIFICATION_FAILED"} - 사용자가 결제를 완료했지만, 앱스토어에서 사용자 정보 검증에 실패했을 때 발생해요. 사용자가 앱스토어에 문의해서 환불을 요청해야해요.
|
|
54
|
+
* @throw {code: "TOSS_SERVER_VERIFICATION_FAILED"} - 사용자가 결제를 완료했지만, 서버 전송에 실패해서 결제 정보를 저장할 수 없을 때 발생해요.
|
|
55
|
+
* @throw {code: "INTERNAL_ERROR"} - 서버 내부 문제로 요청을 처리할 수 없을 때 발생해요.
|
|
56
|
+
* @throw {code: "KOREAN_ACCOUNT_ONLY"} - iOS 환경에서 사용자의 계정이 한국 계정이 아닐 때 발생해요.
|
|
57
|
+
* @throw {code: "USER_CANCELED"} - 사용자가 결제를 완료하지 않고 주문서 페이지를 이탈했을 때 발생해요.
|
|
58
|
+
*
|
|
59
|
+
* @example
|
|
60
|
+
* ### 특정 인앱결제 주문서 페이지로 이동하기
|
|
61
|
+
*
|
|
62
|
+
* ```tsx
|
|
63
|
+
* import { IAP } from "@apps-in-toss/web-framework";
|
|
64
|
+
* import { Button } from "@toss-design-system/react-native";
|
|
65
|
+
*
|
|
66
|
+
* interface Props {
|
|
67
|
+
* productId: string;
|
|
68
|
+
* }
|
|
69
|
+
*
|
|
70
|
+
* function IapCreateOneTimePurchaseOrderButton({ productId }: Props) {
|
|
71
|
+
* async function handleClick() {
|
|
72
|
+
* try {
|
|
73
|
+
* await IAP.createOneTimePurchaseOrder({
|
|
74
|
+
* productId,
|
|
75
|
+
* });
|
|
76
|
+
* console.error("인앱결제에 성공했어요");
|
|
77
|
+
* } catch (error) {
|
|
78
|
+
* console.error('인앱결제에 실패했어요:', error);
|
|
79
|
+
* }
|
|
80
|
+
* }
|
|
81
|
+
*
|
|
82
|
+
* return <Button onClick={handleClick}>구매하기</Button>;
|
|
83
|
+
* }
|
|
84
|
+
* ```
|
|
85
|
+
*/
|
|
86
|
+
async function createOneTimePurchaseOrder(params: IapCreateOneTimePurchaseOrderOptions) {
|
|
87
|
+
const isSupported = isMinVersionSupported({
|
|
88
|
+
android: '5.219.0',
|
|
89
|
+
ios: '5.219.0',
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
if (!isSupported) {
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
return AppsInTossModule.iapCreateOneTimePurchaseOrder(params);
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* @public
|
|
101
|
+
* @category 인앱결제
|
|
102
|
+
* @name IapProductListItem
|
|
103
|
+
* @description 인앱결제로 구매할 수 있는 상품 하나의 정보를 담은 객체예요. 상품 목록을 화면에 표시할 때 사용해요.
|
|
104
|
+
* @property {string} sku - 상품의 고유 ID예요. [IAP.createOneTimePurchaseOrder](https://developers-apps-in-toss.toss.im/bedrock/reference/native-modules/%EC%9D%B8%EC%95%B1%20%EA%B2%B0%EC%A0%9C/createOneTimePurchaseOrder.html)를 호출할때 사용하는 `productId`와 동일한 값이에요.
|
|
105
|
+
* @property {string} displayName - 화면에 표시할 상품 이름이에요. 상품 이름은 앱인토스 콘솔에서 설정한 값이에요.
|
|
106
|
+
* @property {string} displayAmount - 통화 단위가 포함된 가격 정보예요. 예를 들어 `1,000원`으로 가격과 통화가 함께 표시돼요.
|
|
107
|
+
* @property {string} iconUrl - 상품 아이콘 이미지의 URL이에요. 아이콘은 앱인토스 콘솔에서 설정한 이미지예요.
|
|
108
|
+
* @property {string} description - 상품에 대한 설명이에요. 설명은 앱인토스 콘솔에서 설정한 값이에요.
|
|
109
|
+
*/
|
|
110
|
+
export interface IapProductListItem {
|
|
111
|
+
sku: string;
|
|
112
|
+
displayAmount: string;
|
|
113
|
+
displayName: string;
|
|
114
|
+
iconUrl: string;
|
|
115
|
+
description: string;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
/**
|
|
119
|
+
* @public
|
|
120
|
+
* @category 인앱결제
|
|
121
|
+
* @name iapGetProductItemList
|
|
122
|
+
* @description 인앱결제로 구매할 수 있는 상품 목록을 가져와요. 상품 목록 화면에 진입할 때 호출해요.
|
|
123
|
+
* @returns {Promise<{ products: IapProductListItem[] } | undefined>} 상품 목록을 포함한 객체를 반환해요. 앱 버전이 최소 지원 버전(안드로이드 5.219.0, iOS 5.219.0)보다 낮으면 `undefined`를 반환해요.
|
|
124
|
+
*
|
|
125
|
+
* @example
|
|
126
|
+
* ### 구매 가능한 인앱결제 상품목록 가져오기
|
|
127
|
+
*
|
|
128
|
+
* ```tsx
|
|
129
|
+
* import { IAP, IapProductListItem } from "@apps-in-toss/framework";
|
|
130
|
+
* import { Button, List, ListRow } from "@toss-design-system/react-native";
|
|
131
|
+
* import { useEffect, useState } from "react";
|
|
132
|
+
*
|
|
133
|
+
* function IapProductList() {
|
|
134
|
+
* const [products, setProducts] = useState<IapProductListItem[]>([]);
|
|
135
|
+
*
|
|
136
|
+
* async function buyIapProduct(productId: string) {
|
|
137
|
+
* try {
|
|
138
|
+
* await IAP.createOneTimePurchaseOrder({
|
|
139
|
+
* productId,
|
|
140
|
+
* });
|
|
141
|
+
*
|
|
142
|
+
* console.error("인앱결제에 성공했어요");
|
|
143
|
+
* } catch (error) {
|
|
144
|
+
* console.error("인앱결제에 실패했어요:", error);
|
|
145
|
+
* }
|
|
146
|
+
* }
|
|
147
|
+
*
|
|
148
|
+
* useEffect(() => {
|
|
149
|
+
* async function fetchProducts() {
|
|
150
|
+
* try {
|
|
151
|
+
* const response = await IAP.getProductItemList();
|
|
152
|
+
* setProducts(response?.products ?? []);
|
|
153
|
+
* } catch (error) {
|
|
154
|
+
* console.error("상품 목록을 가져오는 데 실패했어요:", error);
|
|
155
|
+
* }
|
|
156
|
+
* }
|
|
157
|
+
*
|
|
158
|
+
* fetchProducts();
|
|
159
|
+
* }, []);
|
|
160
|
+
*
|
|
161
|
+
* return (
|
|
162
|
+
* <List>
|
|
163
|
+
* {products.map((product) => (
|
|
164
|
+
* <ListRow
|
|
165
|
+
* key={product.sku}
|
|
166
|
+
* left={
|
|
167
|
+
* <ListRow.Image type="square" source={{ uri: product.iconUrl }} />
|
|
168
|
+
* }
|
|
169
|
+
* right={
|
|
170
|
+
* <Button size="medium" onPress={() => buyIapProduct(product.sku)}>
|
|
171
|
+
* 구매하기
|
|
172
|
+
* </Button>
|
|
173
|
+
* }
|
|
174
|
+
* contents={
|
|
175
|
+
* <ListRow.Texts
|
|
176
|
+
* type="3RowTypeA"
|
|
177
|
+
* top={product.displayName}
|
|
178
|
+
* middle={product.description}
|
|
179
|
+
* bottom={product.displayAmount}
|
|
180
|
+
* />
|
|
181
|
+
* }
|
|
182
|
+
* />
|
|
183
|
+
* ))}
|
|
184
|
+
* </List>
|
|
185
|
+
* );
|
|
186
|
+
* }
|
|
187
|
+
* ```
|
|
188
|
+
*/
|
|
189
|
+
async function getProductItemList() {
|
|
190
|
+
const isSupported = isMinVersionSupported({
|
|
191
|
+
android: '5.219.0',
|
|
192
|
+
ios: '5.219.0',
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
if (!isSupported) {
|
|
196
|
+
return;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
return AppsInTossModule.iapGetProductItemList({});
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
/**
|
|
203
|
+
* @public
|
|
204
|
+
* @category 인앱결제
|
|
205
|
+
* @name IAP
|
|
206
|
+
* @description 인앱결제 관련 기능을 모은 객체예요. 단건 인앱결제 주문서 이동과 상품 목록 조회 기능을 제공해요.
|
|
207
|
+
* @property {typeof createOneTimePurchaseOrder} [createOneTimePurchaseOrder] 특정 인앱결제 주문서 페이지로 이동해요. 자세한 내용은 [createOneTimePurchaseOrder](https://developers-apps-in-toss.toss.im/bedrock/reference/native-modules/%EC%9D%B8%EC%95%B1%20%EA%B2%B0%EC%A0%9C/getProductItemList.html) 문서를 참고하세요.
|
|
208
|
+
* @property {typeof getProductItemList} [getProductItemList] 인앱결제로 구매할 수 있는 상품 목록을 가져와요. 자세한 내용은 [getProductItemList](https://developers-apps-in-toss.toss.im/bedrock/reference/native-modules/%EC%9D%B8%EC%95%B1%20%EA%B2%B0%EC%A0%9C/createOneTimePurchaseOrder.html) 문서를 참고하세요.
|
|
209
|
+
*/
|
|
210
|
+
export const IAP = {
|
|
211
|
+
createOneTimePurchaseOrder,
|
|
212
|
+
getProductItemList,
|
|
213
|
+
};
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
/* eslint-disable import/order */
|
|
2
|
+
|
|
3
|
+
// AdMob
|
|
4
|
+
import {
|
|
5
|
+
loadAdMobInterstitialAd,
|
|
6
|
+
loadAdMobRewardedAd,
|
|
7
|
+
showAdMobInterstitialAd,
|
|
8
|
+
showAdMobRewardedAd,
|
|
9
|
+
type LoadAdMobInterstitialAdEvent,
|
|
10
|
+
type LoadAdMobInterstitialAdOptions,
|
|
11
|
+
type LoadAdMobRewardedAdEvent,
|
|
12
|
+
type LoadAdMobRewardedAdOptions,
|
|
13
|
+
type ShowAdMobInterstitialAdEvent,
|
|
14
|
+
type ShowAdMobInterstitialAdOptions,
|
|
15
|
+
type ShowAdMobRewardedAdEvent,
|
|
16
|
+
type ShowAdMobRewardedAdOptions,
|
|
17
|
+
} from './ads/googleAdMob';
|
|
18
|
+
|
|
19
|
+
// TossPay
|
|
20
|
+
import { checkoutPayment, type CheckoutPaymentOptions, type CheckoutPaymentResult } from './checkoutPayment';
|
|
21
|
+
|
|
22
|
+
export { AppsInTossModuleInstance as INTERNAL__AppsInTossModule } from './AppsInTossModule';
|
|
23
|
+
export { AppsInTossModule } from './AppsInTossModule';
|
|
24
|
+
|
|
25
|
+
export * from './appLogin';
|
|
26
|
+
export * from './eventLog';
|
|
27
|
+
export * from './fetchAlbumPhotos';
|
|
28
|
+
export * from './fetchContacts';
|
|
29
|
+
export * from './getClipboardText';
|
|
30
|
+
export * from './getCurrentLocation';
|
|
31
|
+
export * from './getDeviceId';
|
|
32
|
+
export * from './getOperationalEnvironment';
|
|
33
|
+
export * from './getTossAppVersion';
|
|
34
|
+
export * from './getTossShareLink';
|
|
35
|
+
export * from './iap';
|
|
36
|
+
export * from './isMinVersionSupported';
|
|
37
|
+
export * from './openCamera';
|
|
38
|
+
export * from './saveBase64Data';
|
|
39
|
+
export * from './setClipboardText';
|
|
40
|
+
export * from './setDeviceOrientation';
|
|
41
|
+
export * from './storage';
|
|
42
|
+
export * from './openGameCenterLeaderboard';
|
|
43
|
+
export * from './getGameCenterGameProfile';
|
|
44
|
+
export * from './submitGameCenterLeaderBoardScore';
|
|
45
|
+
export * from '../native-event-emitter/contactsViral';
|
|
46
|
+
|
|
47
|
+
export type {
|
|
48
|
+
LoadAdMobInterstitialAdEvent,
|
|
49
|
+
LoadAdMobInterstitialAdOptions,
|
|
50
|
+
LoadAdMobRewardedAdEvent,
|
|
51
|
+
LoadAdMobRewardedAdOptions,
|
|
52
|
+
ShowAdMobInterstitialAdEvent,
|
|
53
|
+
ShowAdMobInterstitialAdOptions,
|
|
54
|
+
ShowAdMobRewardedAdEvent,
|
|
55
|
+
CheckoutPaymentOptions,
|
|
56
|
+
CheckoutPaymentResult,
|
|
57
|
+
ShowAdMobRewardedAdOptions,
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* @public
|
|
62
|
+
* @category 토스페이
|
|
63
|
+
* @name TossPay
|
|
64
|
+
* @description 토스페이 결제 관련 함수를 모아둔 객체예요.
|
|
65
|
+
* @property {typeof checkoutPayment} [checkoutPayment] 토스페이 결제를 인증하는 함수예요. 자세한 내용은 [checkoutPayment](/react-native/reference/native-modules/토스페이/checkoutPayment)를 참고하세요.
|
|
66
|
+
*/
|
|
67
|
+
export const TossPay = {
|
|
68
|
+
checkoutPayment,
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* @public
|
|
73
|
+
* @category 광고
|
|
74
|
+
* @name GoogleAdMob
|
|
75
|
+
* @description Google AdMob 광고 관련 함수를 모아둔 객체예요.
|
|
76
|
+
* @property {typeof loadAdMobInterstitialAd} [loadAdMobInterstitialAd] 전면 광고를 로드하는 함수예요. 자세한 내용은 [loadAdMobInterstitialAd](/react-native/reference/native-modules/광고/loadAdMobInterstitialAd.html)를 참고하세요.
|
|
77
|
+
* @property {typeof showAdMobInterstitialAd} [showAdMobInterstitialAd] 로드한 전면 광고를 보여주는 함수예요. 자세한 내용은 [showAdMobInterstitialAd](/react-native/reference/native-modules/광고/showAdMobInterstitialAd.html)를 참고하세요.
|
|
78
|
+
* @property {typeof loadAdMobRewardedAd} [loadAdMobRewardedAd] 보상형 광고를 로드하는 함수예요. 자세한 내용은 [loadAdMobRewardedAd](/react-native/reference/native-modules/광고/loadAdMobRewardedAd.html)를 참고하세요.
|
|
79
|
+
* @property {typeof showAdMobRewardedAd} [showAdMobRewardedAd] 로드한 보상형 광고를 보여주는 함수예요. 자세한 내용은 [showAdMobRewardedAd](/react-native/reference/native-modules/광고/showAdMobRewardedAd.html)를 참고하세요.
|
|
80
|
+
*/
|
|
81
|
+
export const GoogleAdMob = {
|
|
82
|
+
loadAdMobInterstitialAd,
|
|
83
|
+
showAdMobInterstitialAd,
|
|
84
|
+
loadAdMobRewardedAd,
|
|
85
|
+
showAdMobRewardedAd,
|
|
86
|
+
};
|
|
@@ -0,0 +1,190 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
2
|
+
|
|
3
|
+
describe('isMinVersionSupported per-test mocks with dynamic import', () => {
|
|
4
|
+
beforeEach(() => {
|
|
5
|
+
// 각 테스트 전 모듈 캐시 초기화
|
|
6
|
+
vi.resetModules();
|
|
7
|
+
});
|
|
8
|
+
|
|
9
|
+
it('iOS 환경 테스트: 현재 버전이 최소 버전보다 낮을 때', async () => {
|
|
10
|
+
// doMock을 사용하여 모킹이 해당 테스트에 국한되도록 함
|
|
11
|
+
vi.doMock('./AppsInTossModule.js', () => ({
|
|
12
|
+
AppsInTossModule: {
|
|
13
|
+
tossAppVersion: '1.0.0',
|
|
14
|
+
operationalEnvironment: 'toss',
|
|
15
|
+
},
|
|
16
|
+
}));
|
|
17
|
+
|
|
18
|
+
vi.doMock('react-native', () => ({
|
|
19
|
+
Platform: { OS: 'ios' },
|
|
20
|
+
}));
|
|
21
|
+
|
|
22
|
+
const { isMinVersionSupported } = await import('./isMinVersionSupported.js');
|
|
23
|
+
const result = isMinVersionSupported({
|
|
24
|
+
android: '1.0.0',
|
|
25
|
+
ios: '2.0.0',
|
|
26
|
+
});
|
|
27
|
+
expect(result).toBe(false);
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it('Android 환경 테스트: 현재 버전이 최소 버전보다 높을 때', async () => {
|
|
31
|
+
vi.doMock('./AppsInTossModule.js', () => ({
|
|
32
|
+
AppsInTossModule: {
|
|
33
|
+
tossAppVersion: '3.0.0',
|
|
34
|
+
operationalEnvironment: 'toss',
|
|
35
|
+
},
|
|
36
|
+
}));
|
|
37
|
+
|
|
38
|
+
vi.doMock('react-native', () => ({
|
|
39
|
+
Platform: { OS: 'android' },
|
|
40
|
+
}));
|
|
41
|
+
|
|
42
|
+
const { isMinVersionSupported } = await import('./isMinVersionSupported.js');
|
|
43
|
+
const result = isMinVersionSupported({
|
|
44
|
+
android: '2.0.0',
|
|
45
|
+
ios: '1.0.0',
|
|
46
|
+
});
|
|
47
|
+
expect(result).toBe(true);
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
it('iOS 환경 테스트: 현재 버전이 최소 버전과 같을 때', async () => {
|
|
51
|
+
vi.doMock('./AppsInTossModule.js', () => ({
|
|
52
|
+
AppsInTossModule: {
|
|
53
|
+
tossAppVersion: '2.0.0',
|
|
54
|
+
operationalEnvironment: 'toss',
|
|
55
|
+
},
|
|
56
|
+
}));
|
|
57
|
+
|
|
58
|
+
vi.doMock('react-native', () => ({
|
|
59
|
+
Platform: { OS: 'ios' },
|
|
60
|
+
}));
|
|
61
|
+
|
|
62
|
+
const { isMinVersionSupported } = await import('./isMinVersionSupported.js');
|
|
63
|
+
const result = isMinVersionSupported({
|
|
64
|
+
android: '1.0.0',
|
|
65
|
+
ios: '2.0.0',
|
|
66
|
+
});
|
|
67
|
+
expect(result).toBe(true);
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
it('iOS 환경 테스트: minVersion이 "always"일 때', async () => {
|
|
71
|
+
vi.doMock('./AppsInTossModule.js', () => ({
|
|
72
|
+
AppsInTossModule: {
|
|
73
|
+
tossAppVersion: '1.0.0',
|
|
74
|
+
operationalEnvironment: 'toss',
|
|
75
|
+
},
|
|
76
|
+
}));
|
|
77
|
+
|
|
78
|
+
vi.doMock('react-native', () => ({
|
|
79
|
+
Platform: { OS: 'ios' },
|
|
80
|
+
}));
|
|
81
|
+
|
|
82
|
+
const { isMinVersionSupported } = await import('./isMinVersionSupported.js');
|
|
83
|
+
const result = isMinVersionSupported({
|
|
84
|
+
android: '1.0.0',
|
|
85
|
+
ios: 'always',
|
|
86
|
+
});
|
|
87
|
+
expect(result).toBe(true);
|
|
88
|
+
});
|
|
89
|
+
|
|
90
|
+
it('Android 환경 테스트: minVersion이 "always"일 때', async () => {
|
|
91
|
+
vi.doMock('./AppsInTossModule.js', () => ({
|
|
92
|
+
AppsInTossModule: {
|
|
93
|
+
tossAppVersion: '1.0.0',
|
|
94
|
+
operationalEnvironment: 'toss',
|
|
95
|
+
},
|
|
96
|
+
}));
|
|
97
|
+
|
|
98
|
+
vi.doMock('react-native', () => ({
|
|
99
|
+
Platform: { OS: 'android' },
|
|
100
|
+
}));
|
|
101
|
+
|
|
102
|
+
const { isMinVersionSupported } = await import('./isMinVersionSupported.js');
|
|
103
|
+
const result = isMinVersionSupported({
|
|
104
|
+
android: 'always',
|
|
105
|
+
ios: '1.0.0',
|
|
106
|
+
});
|
|
107
|
+
expect(result).toBe(true);
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
it('iOS 환경 테스트: minVersion이 "never"일 때', async () => {
|
|
111
|
+
vi.doMock('./AppsInTossModule.js', () => ({
|
|
112
|
+
AppsInTossModule: {
|
|
113
|
+
tossAppVersion: '1.0.0',
|
|
114
|
+
operationalEnvironment: 'toss',
|
|
115
|
+
},
|
|
116
|
+
}));
|
|
117
|
+
|
|
118
|
+
vi.doMock('react-native', () => ({
|
|
119
|
+
Platform: { OS: 'ios' },
|
|
120
|
+
}));
|
|
121
|
+
|
|
122
|
+
const { isMinVersionSupported } = await import('./isMinVersionSupported.js');
|
|
123
|
+
const result = isMinVersionSupported({
|
|
124
|
+
android: '1.0.0',
|
|
125
|
+
ios: 'never',
|
|
126
|
+
});
|
|
127
|
+
expect(result).toBe(false);
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
it('Android 환경 테스트: minVersion이 "never"일 때', async () => {
|
|
131
|
+
vi.doMock('./AppsInTossModule.js', () => ({
|
|
132
|
+
AppsInTossModule: {
|
|
133
|
+
tossAppVersion: '1.0.0',
|
|
134
|
+
operationalEnvironment: 'toss',
|
|
135
|
+
},
|
|
136
|
+
}));
|
|
137
|
+
|
|
138
|
+
vi.doMock('react-native', () => ({
|
|
139
|
+
Platform: { OS: 'android' },
|
|
140
|
+
}));
|
|
141
|
+
|
|
142
|
+
const { isMinVersionSupported } = await import('./isMinVersionSupported.js');
|
|
143
|
+
const result = isMinVersionSupported({
|
|
144
|
+
android: 'never',
|
|
145
|
+
ios: '1.0.0',
|
|
146
|
+
});
|
|
147
|
+
expect(result).toBe(false);
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
it('minVersion이 없을 때 false를 반환', async () => {
|
|
151
|
+
vi.doMock('./AppsInTossModule.js', () => ({
|
|
152
|
+
AppsInTossModule: {
|
|
153
|
+
tossAppVersion: '1.0.0',
|
|
154
|
+
operationalEnvironment: 'toss',
|
|
155
|
+
},
|
|
156
|
+
}));
|
|
157
|
+
|
|
158
|
+
vi.doMock('react-native', () => ({
|
|
159
|
+
Platform: { OS: 'ios' },
|
|
160
|
+
}));
|
|
161
|
+
|
|
162
|
+
const { isMinVersionSupported } = await import('./isMinVersionSupported.js');
|
|
163
|
+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
164
|
+
// @ts-expect-error: ios 값 누락 의도
|
|
165
|
+
const result = isMinVersionSupported({
|
|
166
|
+
android: '1.0.0',
|
|
167
|
+
});
|
|
168
|
+
expect(result).toBe(false);
|
|
169
|
+
});
|
|
170
|
+
|
|
171
|
+
it('sandbox 환경 테스트: 버전에 상관없이 항상 true를 반환', async () => {
|
|
172
|
+
vi.doMock('./AppsInTossModule.js', () => ({
|
|
173
|
+
AppsInTossModule: {
|
|
174
|
+
tossAppVersion: '1.0.0',
|
|
175
|
+
operationalEnvironment: 'sandbox',
|
|
176
|
+
},
|
|
177
|
+
}));
|
|
178
|
+
|
|
179
|
+
vi.doMock('react-native', () => ({
|
|
180
|
+
Platform: { OS: 'ios' },
|
|
181
|
+
}));
|
|
182
|
+
|
|
183
|
+
const { isMinVersionSupported } = await import('./isMinVersionSupported.js');
|
|
184
|
+
const result = isMinVersionSupported({
|
|
185
|
+
android: '5.0.0',
|
|
186
|
+
ios: '5.0.0',
|
|
187
|
+
});
|
|
188
|
+
expect(result).toBe(true);
|
|
189
|
+
});
|
|
190
|
+
});
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { Platform } from 'react-native';
|
|
2
|
+
import { AppsInTossModule } from './AppsInTossModule';
|
|
3
|
+
import { compareVersions } from '../../utils/compareVersion';
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* @public
|
|
7
|
+
* @category 환경 확인
|
|
8
|
+
* @kind function
|
|
9
|
+
* @name isMinVersionSupported
|
|
10
|
+
* @description
|
|
11
|
+
* 현재 토스 앱 버전이 지정한 최소 버전 이상인지 확인해요.
|
|
12
|
+
*
|
|
13
|
+
* 이 함수는 현재 실행 중인 토스 앱의 버전이 파라미터로 전달된 최소 버전 요구사항을 충족하는지 확인해요. 특정 기능이 최신 버전에서만 동작할 때, 사용자에게 앱 업데이트를 안내할 수 있어요.
|
|
14
|
+
*
|
|
15
|
+
* @param {Object} minVersions 플랫폼별 최소 버전 요구사항을 지정하는 객체예요.
|
|
16
|
+
* @param {(`${number}.${number}.${number}` | 'always' | 'never')} minVersions.android 안드로이드 플랫폼의 최소 버전 요구사항이에요.
|
|
17
|
+
* @param {(`${number}.${number}.${number}` | 'always' | 'never')} minVersions.ios iOS 플랫폼의 최소 버전 요구사항이에요.
|
|
18
|
+
* @returns {boolean} 현재 앱 버전이 최소 버전 이상이면 true, 그렇지 않으면 false를 반환해요.
|
|
19
|
+
*
|
|
20
|
+
* @example
|
|
21
|
+
* ### 앱 버전 확인하기
|
|
22
|
+
*
|
|
23
|
+
* ```tsx
|
|
24
|
+
* import { isMinVersionSupported } from '@apps-in-toss/framework';
|
|
25
|
+
* import { Text, View } from 'react-native';
|
|
26
|
+
*
|
|
27
|
+
* function VersionCheck() {
|
|
28
|
+
* const isSupported = isMinVersionSupported({
|
|
29
|
+
* android: '1.2.0',
|
|
30
|
+
* ios: '1.3.0'
|
|
31
|
+
* });
|
|
32
|
+
*
|
|
33
|
+
* return (
|
|
34
|
+
* <View>
|
|
35
|
+
* {!isSupported && (
|
|
36
|
+
* <Text>최신 버전으로 업데이트가 필요해요.</Text>
|
|
37
|
+
* )}
|
|
38
|
+
* </View>
|
|
39
|
+
* );
|
|
40
|
+
* }
|
|
41
|
+
* ```
|
|
42
|
+
*/
|
|
43
|
+
export function isMinVersionSupported(minVersions: {
|
|
44
|
+
android: `${number}.${number}.${number}` | 'always' | 'never';
|
|
45
|
+
ios: `${number}.${number}.${number}` | 'always' | 'never';
|
|
46
|
+
}): boolean {
|
|
47
|
+
const operationalEnvironment = AppsInTossModule.operationalEnvironment;
|
|
48
|
+
if (operationalEnvironment === 'sandbox') {
|
|
49
|
+
return true;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
const currentVersion = AppsInTossModule.tossAppVersion;
|
|
53
|
+
const isIOS = Platform.OS === 'ios';
|
|
54
|
+
const minVersion = isIOS ? minVersions.ios : minVersions.android;
|
|
55
|
+
|
|
56
|
+
if (minVersion === undefined) {
|
|
57
|
+
return false;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
if (minVersion === 'always') {
|
|
61
|
+
return true;
|
|
62
|
+
}
|
|
63
|
+
if (minVersion === 'never') {
|
|
64
|
+
return false;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
return compareVersions(currentVersion, minVersion) >= 0;
|
|
68
|
+
}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { AppsInTossModule } from './AppsInTossModule';
|
|
2
|
+
import { requestPermission } from './requestPermission';
|
|
3
|
+
import type { ImageResponse } from '../../types';
|
|
4
|
+
|
|
5
|
+
export interface OpenCameraOptions {
|
|
6
|
+
/**
|
|
7
|
+
* 이미지를 Base64 형식으로 반환할지 여부를 나타내는 불리언 값이에요.
|
|
8
|
+
*
|
|
9
|
+
* 기본값: `false`.
|
|
10
|
+
*/
|
|
11
|
+
base64?: boolean;
|
|
12
|
+
/**
|
|
13
|
+
* 이미지의 최대 너비를 나타내는 숫자 값이에요.
|
|
14
|
+
*
|
|
15
|
+
* 기본값: `1024`.
|
|
16
|
+
*/
|
|
17
|
+
maxWidth?: number;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* @public
|
|
22
|
+
* @category 카메라
|
|
23
|
+
* @name openCamera
|
|
24
|
+
* @description 카메라를 실행해서 촬영된 이미지를 반환하는 함수예요.
|
|
25
|
+
* @param {OpenCameraOptions} options - 카메라 실행 시 사용되는 옵션 객체예요.
|
|
26
|
+
* @param {boolean} [options.base64=false] - 이미지를 Base64 형식으로 반환할지 여부를 나타내는 불리언 값이에요. 기본값은 `false`예요. `true`로 설정하면 `dataUri` 대신 Base64 인코딩된 문자열을 반환해요.
|
|
27
|
+
* @param {number} [options.maxWidth=1024] - 이미지의 최대 너비를 나타내는 숫자 값이에요. 기본값은 `1024`예요.
|
|
28
|
+
* @returns {Promise<ImageResponse>}
|
|
29
|
+
* 촬영된 이미지 정보를 포함한 객체를 반환해요. 반환 객체의 구성은 다음과 같아요:
|
|
30
|
+
* - `id`: 이미지의 고유 식별자예요.
|
|
31
|
+
* - `dataUri`: 이미지 데이터를 표현하는 데이터 URI예요.
|
|
32
|
+
*
|
|
33
|
+
* @example
|
|
34
|
+
* ### 카메라 실행 후 촬영된 사진 가져오기
|
|
35
|
+
*
|
|
36
|
+
* ```tsx
|
|
37
|
+
* import React, { useState } from 'react';
|
|
38
|
+
* import { View, Text, Button, Image } from 'react-native';
|
|
39
|
+
* import { openCamera } from '@apps-in-toss/framework';
|
|
40
|
+
*
|
|
41
|
+
* const base64 = true;
|
|
42
|
+
*
|
|
43
|
+
* // 카메라를 실행하고 촬영된 이미지를 화면에 표시하는 컴포넌트
|
|
44
|
+
* function Camera() {
|
|
45
|
+
* const [image, setImage] = useState(null);
|
|
46
|
+
*
|
|
47
|
+
* const handlePress = async () => {
|
|
48
|
+
* try {
|
|
49
|
+
* const response = await openCamera({ base64 });
|
|
50
|
+
* setImage(response);
|
|
51
|
+
* } catch (error) {
|
|
52
|
+
* console.error('사진을 가져오는 데 실패했어요:', error);
|
|
53
|
+
* }
|
|
54
|
+
* };
|
|
55
|
+
*
|
|
56
|
+
* // base64 형식으로 반환된 이미지를 표시하려면 데이터 URL 스키마 Prefix를 붙여야해요.
|
|
57
|
+
* const imageUri = base64 ? 'data:image/jpeg;base64,' + image.dataUri : image.dataUri;
|
|
58
|
+
*
|
|
59
|
+
* return (
|
|
60
|
+
* <View>
|
|
61
|
+
* {image ? (
|
|
62
|
+
* <Image source={{ uri: imageUri }} style={{ width: 200, height: 200 }} />
|
|
63
|
+
* ) : (
|
|
64
|
+
* <Text>사진이 없어요</Text>
|
|
65
|
+
* )}
|
|
66
|
+
* <Button title="사진 촬영하기" onPress={handlePress} />
|
|
67
|
+
* </View>
|
|
68
|
+
* );
|
|
69
|
+
* }
|
|
70
|
+
* ```
|
|
71
|
+
*/
|
|
72
|
+
export async function openCamera(options?: OpenCameraOptions): Promise<ImageResponse> {
|
|
73
|
+
const permissionStatus = await requestPermission({ name: 'camera', access: 'access' });
|
|
74
|
+
if (permissionStatus === 'denied') {
|
|
75
|
+
throw new Error('카메라 권한이 거부되었어요.');
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
const photo = await AppsInTossModule.openCamera({ base64: false, maxWidth: 1024, ...options });
|
|
79
|
+
|
|
80
|
+
return photo;
|
|
81
|
+
}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { openURL } from '@granite-js/react-native';
|
|
2
|
+
import { isMinVersionSupported } from './isMinVersionSupported';
|
|
3
|
+
import { GAME_CENTER_MIN_VERSION } from '../constants';
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* @public
|
|
7
|
+
* @category 게임센터
|
|
8
|
+
* @name openGameCenterLeaderboard
|
|
9
|
+
* @description 게임센터 리더보드 웹뷰를 열어요.
|
|
10
|
+
* 앱 버전이 최소 지원 버전(`5.221.0`)보다 낮으면 아무 동작도 하지 않고 `undefined`를 반환해요.
|
|
11
|
+
* 게임센터를 사용하는 사용자는 반드시 최소 지원 버전 이상이어야 게임을 실행할 수 있어요.
|
|
12
|
+
* @returns 리더보드 웹뷰를 호출해요. 앱 버전이 낮으면 아무 동작도 하지 않고 `undefined`를 반환해요.
|
|
13
|
+
*
|
|
14
|
+
* @example
|
|
15
|
+
* import { Button } from 'react-native';
|
|
16
|
+
* import { openGameCenterLeaderboard } from '@apps-in-toss/framework';
|
|
17
|
+
*
|
|
18
|
+
* // '리더보드' 버튼을 누르면 게임센터 리더보드 웹뷰가 열려요.
|
|
19
|
+
* function LeaderboardButton() {
|
|
20
|
+
* const onPress = () => {
|
|
21
|
+
* openGameCenterLeaderboard();
|
|
22
|
+
* };
|
|
23
|
+
*
|
|
24
|
+
* return <Button title="리더보드 웹뷰 호출" onPress={onPress} />;
|
|
25
|
+
* }
|
|
26
|
+
*/
|
|
27
|
+
export async function openGameCenterLeaderboard(): Promise<void> {
|
|
28
|
+
if (!isMinVersionSupported(GAME_CENTER_MIN_VERSION)) {
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
const appName = global.__granite?.app?.name;
|
|
33
|
+
|
|
34
|
+
if (appName == null) {
|
|
35
|
+
throw new Error('Cannot get app name');
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
const url = new URL('servicetoss://game-center/leaderboard?_navbar=hide');
|
|
39
|
+
url.searchParams.set('appName', appName);
|
|
40
|
+
url.searchParams.set('referrer', `appsintoss.${appName}`);
|
|
41
|
+
return openURL(url.toString());
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
declare const global: { __granite: any };
|