@blocklet/launcher-workflow 2.3.80 → 2.3.82

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/es/checkout.js DELETED
@@ -1,601 +0,0 @@
1
- /* eslint-disable react-hooks/exhaustive-deps */
2
- import Dialog from '@arcblock/ux/lib/Dialog';
3
- import { NFT_TYPE_DEDICATE, NFT_TYPE_SERVERLESS } from '@blocklet/launcher-util/es/constant';
4
- import { getBlockletDisplayName } from '@blocklet/launcher-util/es/util';
5
- import Button from '@blocklet/launcher-ux/lib/button';
6
- import useMobile from '@blocklet/launcher-ux/lib/use-mobile';
7
- import Payment from '@blocklet/payment/lib/payment';
8
- import styled from '@emotion/styled';
9
- import Grid from '@mui/material/Grid';
10
- import Skeleton from '@mui/material/Skeleton';
11
- import dsbridge from 'dsbridge';
12
- import get from 'lodash.get';
13
- import PropTypes from 'prop-types';
14
- import React, { useEffect } from 'react';
15
- import { useNavigate, useSearchParams } from 'react-router-dom';
16
- import useAsync from 'react-use/lib/useAsync';
17
- import useSetState from 'react-use/lib/useSetState';
18
- import joinURL from 'url-join';
19
- import useBrowser from '@arcblock/react-hooks/lib/useBrowser';
20
- import Confirm from '@arcblock/ux/lib/Dialog/confirm';
21
- import ResultMessage from '@blocklet/launcher-layout/lib/launch-result-message';
22
- import formatError from '@blocklet/launcher-util/es/format-error';
23
- var SpaceIcon = function SpaceIcon(props) {
24
- return /*#__PURE__*/_jsxs("svg", {
25
- ...props,
26
- children: [/*#__PURE__*/_jsx("rect", {
27
- width: "40",
28
- height: "40",
29
- rx: "8",
30
- fill: "url(#a)"
31
- }), /*#__PURE__*/_jsxs("defs", {
32
- children: [/*#__PURE__*/_jsx("pattern", {
33
- id: "a",
34
- patternContentUnits: "objectBoundingBox",
35
- width: "1",
36
- height: "1",
37
- children: /*#__PURE__*/_jsx("use", {
38
- xlinkHref: "#b",
39
- transform: "scale(.00833)"
40
- })
41
- }), /*#__PURE__*/_jsx("image", {
42
- id: "b",
43
- width: "120",
44
- height: "120",
45
- xlinkHref: ""
46
- })]
47
- })]
48
- });
49
- };
50
- SpaceIcon.defaultProps = {
51
- width: "40",
52
- height: "40",
53
- fill: "none",
54
- xmlns: "http://www.w3.org/2000/svg",
55
- xmlnsXlink: "http://www.w3.org/1999/xlink"
56
- };
57
- import Agreement from './components/agreement';
58
- import Checkbox from './components/checkbox';
59
- import Layout from './components/layout';
60
- import Body from './components/layout/body';
61
- import Footer from './components/layout/footer';
62
- import Header from './components/layout/header';
63
- import { useLocaleContext } from './contexts/locale';
64
- import useRequest from './contexts/request';
65
- import { useSessionContext } from './contexts/session';
66
- import { useWorkflowContext } from './contexts/workflow';
67
- import { BLOCKLET_STORE_URL, getPrice, launchSession } from './util';
68
- import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
69
- function CheckoutPage({
70
- blocklet
71
- }) {
72
- const {
73
- t,
74
- locale
75
- } = useLocaleContext();
76
- const {
77
- session,
78
- events
79
- } = useSessionContext();
80
- const [params] = useSearchParams();
81
- const [state, setState] = useSetState({
82
- error: '',
83
- showAgreement: false,
84
- showPayment: false,
85
- showAgreeHint: false,
86
- showPaidDialog: {
87
- open: false,
88
- nftId: ''
89
- },
90
- agreed: true,
91
- paying: false,
92
- planId: params.get('planId'),
93
- sessionId: params.get('sessionId'),
94
- paymentMethod: params.get('paymentMethod'),
95
- autoLogin: true
96
- });
97
- const browser = useBrowser();
98
- const navigate = useNavigate();
99
- const {
100
- create,
101
- api: launchSessionAPI
102
- } = useRequest();
103
- const {
104
- mobileSize,
105
- isMobile
106
- } = useMobile();
107
- const {
108
- routerPrefix,
109
- didPayPrefix,
110
- embed,
111
- isPurchaseOnly
112
- } = useWorkflowContext();
113
- const api = create({
114
- baseURL: didPayPrefix
115
- });
116
- useEffect(() => {
117
- if (isPurchaseOnly) {
118
- return;
119
- }
120
- if (session.user) {
121
- launchSession.connect(launchSessionAPI, routerPrefix, state.sessionId, session.user.did);
122
- }
123
- }, [session.user]);
124
- const enableBlockletAgreement = !!blocklet;
125
- const context = useAsync(async () => {
126
- const [{
127
- data: {
128
- data: paymentMethods
129
- }
130
- }, {
131
- data: {
132
- data: plan
133
- }
134
- }] = await Promise.all([api.get('/payment-methods'), api.get(`/plans/${state.planId}`)]);
135
- if (!plan || paymentMethods.length === 0) {
136
- throw new Error(t('purchase.noProducts'));
137
- }
138
- if (paymentMethods.length > 0 && !state.paymentMethod) {
139
- setState({
140
- paymentMethod: paymentMethods[0]._id
141
- });
142
- }
143
- return {
144
- paymentMethods,
145
- plan
146
- };
147
- });
148
-
149
- // 用户退出的时候自动重新创建启动会话
150
- useEffect(() => {
151
- if (isPurchaseOnly) {
152
- return;
153
- }
154
- events.on('logout', () => {
155
- const metaUrl = params.get('blocklet_meta_url');
156
- launchSession.create(launchSessionAPI, routerPrefix, metaUrl, (err, launch) => {
157
- setState({
158
- sessionId: launch._id
159
- });
160
- params.set('sessionId', launch._id);
161
- const type = get(context.value?.plan, 'extra.type') || 'serverless';
162
- launchSession.select(launchSessionAPI, routerPrefix, launch._id, type, state.planId);
163
- });
164
- });
165
- }, []);
166
- const switchPayment = method => {
167
- setState({
168
- paymentMethod: method
169
- });
170
- };
171
- const handleCancelPay = () => {
172
- setState({
173
- showPayment: false,
174
- paying: false
175
- });
176
- };
177
- const ensureLogin = () => {
178
- return new Promise(resolve => {
179
- if (!session.user) {
180
- session.login(() => {
181
- resolve();
182
- });
183
- } else {
184
- resolve();
185
- }
186
- });
187
- };
188
- const handlePay = () => {
189
- if (enableBlockletAgreement && !state.agreed) {
190
- setState({
191
- showAgreeHint: true
192
- });
193
- setTimeout(() => {
194
- setState({
195
- showAgreeHint: false
196
- });
197
- }, 2000);
198
- return;
199
- }
200
- if (browser.wallet) {
201
- ensureLogin().then(() => {
202
- setState({
203
- showPayment: true,
204
- paying: true,
205
- autoLogin: false
206
- });
207
- });
208
- } else {
209
- setState({
210
- showPayment: true,
211
- paying: true,
212
- autoLogin: true
213
- });
214
- }
215
- };
216
- const handleConnected = result => {
217
- if (isPurchaseOnly) {
218
- session.refresh();
219
- return;
220
- }
221
- launchSession.connect(launchSessionAPI, routerPrefix, state.sessionId, result.did, () => {
222
- session.refresh();
223
- });
224
- };
225
- const handlePaid = ({
226
- nftId,
227
- nftState
228
- }) => {
229
- if ((nftState.tags || []).includes(NFT_TYPE_SERVERLESS)) {
230
- if (!embed) {
231
- setState({
232
- showPaidDialog: {
233
- open: true,
234
- nftId
235
- }
236
- });
237
- handleCancelPay();
238
- return;
239
- }
240
- params.set('launchType', 'serverless');
241
- }
242
- if (isPurchaseOnly && !(nftState.tags || []).includes(NFT_TYPE_DEDICATE)) {
243
- handleCancelPay();
244
- dsbridge.call('arcClosePage');
245
- return;
246
- }
247
- handleCancelPay();
248
- navigate({
249
- pathname: joinURL(routerPrefix, `/launch/${nftId}`),
250
- search: params.toString()
251
- });
252
- };
253
- const handlePayFailed = () => {
254
- // TODO: Toast 错误
255
- handleCancelPay();
256
- };
257
- const calculateTotalPayment = prices => {
258
- if (!state.paymentMethod) {
259
- return '';
260
- }
261
- const providerName = paymentMethods?.find(x => x._id === state.paymentMethod)?.provider.name;
262
- return getPrice(prices, providerName);
263
- };
264
- const toggleAgreement = () => setState({
265
- agreed: !state.agreed
266
- });
267
- const {
268
- paymentMethods,
269
- plan
270
- } = context.value || {};
271
- const paymentMethod = paymentMethods?.find(x => x._id === state.paymentMethod);
272
- const providerName = paymentMethod?.provider.name;
273
- return /*#__PURE__*/_jsxs(Container, {
274
- mobileSize: mobileSize,
275
- enableBlockletAgreement: enableBlockletAgreement,
276
- children: [/*#__PURE__*/_jsx(Header, {
277
- title: t('checkout.pageTitle')
278
- }), context.error && /*#__PURE__*/_jsx(Body, {
279
- children: /*#__PURE__*/_jsx(ResultMessage, {
280
- variant: "error",
281
- title: t('common.error'),
282
- subTitle: formatError(context.error)
283
- })
284
- }), !context.error && /*#__PURE__*/_jsxs(_Fragment, {
285
- children: [/*#__PURE__*/_jsxs(Body, {
286
- children: [/*#__PURE__*/_jsx("div", {
287
- className: "header",
288
- children: t('common.details')
289
- }), context.loading ? /*#__PURE__*/_jsx(Skeleton, {
290
- className: "skeleton",
291
- sx: {
292
- width: '100%',
293
- height: '100%'
294
- }
295
- }) : /*#__PURE__*/_jsxs("div", {
296
- className: "space details",
297
- children: [/*#__PURE__*/_jsxs("div", {
298
- className: "basic",
299
- children: [/*#__PURE__*/_jsx(SpaceIcon, {}), /*#__PURE__*/_jsxs("div", {
300
- className: "basic-info",
301
- children: [/*#__PURE__*/_jsx("div", {
302
- className: "title",
303
- children: plan.name[locale]
304
- }), /*#__PURE__*/_jsx("div", {
305
- className: "sub-title",
306
- children: t('common.space')
307
- })]
308
- })]
309
- }), /*#__PURE__*/_jsx("div", {
310
- className: "price",
311
- children: getPrice(plan.prices, providerName)
312
- })]
313
- }), /*#__PURE__*/_jsx("div", {
314
- className: "divider"
315
- }), context.loading ? /*#__PURE__*/_jsx(Skeleton, {
316
- className: "skeleton",
317
- sx: {
318
- width: '100%',
319
- height: '100%'
320
- }
321
- }) : /*#__PURE__*/_jsxs("div", {
322
- className: "total-order space",
323
- children: [/*#__PURE__*/_jsx("div", {
324
- children: t('checkout.orderTotal')
325
- }), /*#__PURE__*/_jsx("div", {
326
- children: getPrice(plan?.prices || [], providerName)
327
- })]
328
- }), /*#__PURE__*/_jsxs("div", {
329
- className: "total-payment space",
330
- children: [/*#__PURE__*/_jsx("div", {
331
- children: t('checkout.paymentTotal')
332
- }), /*#__PURE__*/_jsx("div", {
333
- children: calculateTotalPayment(plan?.prices || [])
334
- })]
335
- }), /*#__PURE__*/_jsx("div", {
336
- className: "payment-method space",
337
- children: context.loading ? /*#__PURE__*/_jsx(Skeleton, {
338
- className: "skeleton",
339
- sx: {
340
- width: '100%',
341
- height: '100%'
342
- }
343
- }) : /*#__PURE__*/_jsxs("div", {
344
- className: "body",
345
- children: [/*#__PURE__*/_jsx("div", {
346
- className: "payment-method__title",
347
- children: t('checkout.paymentMethod.title')
348
- }), /*#__PURE__*/_jsx("div", {
349
- className: "payment-method__list",
350
- children: /*#__PURE__*/_jsx(Grid, {
351
- container: true,
352
- spacing: 3,
353
- justifyContent: "flex-start",
354
- children: paymentMethods.map(method => /*#__PURE__*/_jsx(Grid, {
355
- item: true,
356
- xs: 12,
357
- sm: 12,
358
- md: 3,
359
- lg: 2,
360
- xl: 2,
361
- children: /*#__PURE__*/_jsx(Checkbox, {
362
- label: method.currency.displayName[locale] || method.currency.displayName.en,
363
- checked: state.paymentMethod === method._id,
364
- onClick: () => switchPayment(method._id)
365
- })
366
- }, method._id))
367
- })
368
- })]
369
- })
370
- })]
371
- }), !context.loading && /*#__PURE__*/_jsxs(Footer, {
372
- className: "footer",
373
- children: [enableBlockletAgreement && /*#__PURE__*/_jsx("div", {
374
- className: "agreement",
375
- children: /*#__PURE__*/_jsx(Checkbox, {
376
- checked: state.agreed,
377
- showHint: state.showAgreeHint,
378
- onClick: () => toggleAgreement(),
379
- label: /*#__PURE__*/_jsxs("div", {
380
- style: {
381
- display: 'flex',
382
- alignItems: 'center'
383
- },
384
- children: [t('common.agreeTo'), /*#__PURE__*/_jsx("span", {
385
- className: "agreement-label",
386
- onClick: e => {
387
- e.stopPropagation();
388
- setState({
389
- showAgreement: true
390
- });
391
- },
392
- children: t('checkout.agreement', {
393
- name: getBlockletDisplayName(blocklet)
394
- })
395
- })]
396
- })
397
- })
398
- }), /*#__PURE__*/_jsxs("div", {
399
- className: "actions",
400
- children: [isMobile && /*#__PURE__*/_jsx(Button, {
401
- className: "button-previous",
402
- variant: "outlined",
403
- onClick: () => navigate(-1),
404
- children: t('common.previous')
405
- }), /*#__PURE__*/_jsx(Button, {
406
- loading: state.paying,
407
- className: "button-pay",
408
- disabled: state.paying || state.showPayment || !state.planId || !state.paymentMethod,
409
- variant: "contained",
410
- onClick: handlePay,
411
- children: t('common.pay')
412
- })]
413
- })]
414
- })]
415
- }), state.showPayment && /*#__PURE__*/_jsx(Payment, {
416
- baseURL: didPayPrefix,
417
- method: paymentMethod,
418
- userDid: session?.user?.did,
419
- productId: state.planId,
420
- useSocket: false,
421
- onConnect: handleConnected,
422
- onComplete: handlePaid,
423
- onFailed: handlePayFailed,
424
- onCancel: handleCancelPay,
425
- metadata: {
426
- sessionId: state.sessionId,
427
- currentUserDid: session?.user?.did,
428
- forceSwitch: !session?.user,
429
- // HACK: 在 openPopup 阶段需要手动添加 sourceAppPid 字段
430
- sourceAppPid: session?.user?.sourceAppPid
431
- },
432
- autoLogin: state.autoLogin
433
- }), /*#__PURE__*/_jsx(Confirm, {
434
- open: state.showPaidDialog.open,
435
- disableEscapeKeyDown: true,
436
- title: t('purchase.serverlessDialog.title'),
437
- confirmButton: {
438
- text: t('purchase.serverlessDialog.redirectToStoreButton')
439
- },
440
- PaperProps: {
441
- style: {
442
- maxWidth: 1200,
443
- minWidth: isMobile ? undefined : 600
444
- }
445
- },
446
- onCancel: () => setState({
447
- showPaidDialog: {
448
- open: false,
449
- nftId: ''
450
- }
451
- }),
452
- showCancelButton: false,
453
- onConfirm: () => {
454
- const url = new URL(BLOCKLET_STORE_URL);
455
- url.searchParams.set('launcher_nft_id', state.showPaidDialog.nftId);
456
- window.location = url.href;
457
- },
458
- children: /*#__PURE__*/_jsx("div", {
459
- children: t('purchase.serverlessDialog.purchaseServerlessSuccess')
460
- })
461
- }), /*#__PURE__*/_jsx(Dialog, {
462
- open: state.showAgreement,
463
- title: t('common.agreement'),
464
- onClose: () => setState({
465
- showAgreement: false
466
- }),
467
- children: /*#__PURE__*/_jsx(Agreement, {})
468
- })]
469
- });
470
- }
471
- CheckoutPage.propTypes = {
472
- blocklet: PropTypes.object
473
- };
474
- CheckoutPage.defaultProps = {
475
- blocklet: null
476
- };
477
- const Container = styled(Layout)`
478
- .header {
479
- font-weight: bolder;
480
- font-size: 1.2em;
481
- }
482
-
483
- .divider {
484
- margin-top: 20px;
485
- border: 1.2px solid #f6f6f6;
486
- }
487
-
488
- .skeleton {
489
- height: 64px;
490
- }
491
-
492
- .space {
493
- margin-top: 18px;
494
- }
495
-
496
- .total-order {
497
- display: flex;
498
- justify-content: space-between;
499
- align-items: center;
500
- font-weight: bolder;
501
- }
502
-
503
- .total-payment {
504
- display: flex;
505
- justify-content: space-between;
506
- align-items: center;
507
- color: ${props => props.theme.palette.primary.main};
508
- font-size: 22px;
509
- font-weight: bolder;
510
- }
511
-
512
- .details {
513
- display: flex;
514
- justify-content: space-between;
515
- align-items: center;
516
- height: 40px;
517
-
518
- .basic {
519
- display: flex;
520
- height: 40px;
521
-
522
- .basic-info {
523
- display: flex;
524
- flex-direction: column;
525
- justify-content: space-between;
526
- margin-left: 8px;
527
-
528
- .title {
529
- font-weight: bolder;
530
- }
531
-
532
- .sub-title {
533
- color: grey;
534
- font-size: 0.8em;
535
- }
536
- }
537
- }
538
-
539
- .period {
540
- display: flex;
541
- align-items: center;
542
-
543
- .info-icon {
544
- color: grey;
545
- margin-left: 8px;
546
- }
547
- }
548
- }
549
-
550
- .payment-method {
551
- background-color: #fbfbfb;
552
- padding: 18px;
553
-
554
- .payment-method__title {
555
- color: #9397a1;
556
- }
557
-
558
- .payment-method__list {
559
- margin-top: 8px;
560
- }
561
- }
562
-
563
- .footer {
564
- display: flex;
565
- align-items: center;
566
- justify-content: ${props => props.enableBlockletAgreement ? 'space-between' : 'flex-end'};
567
-
568
- .agreement {
569
- .agreement-label {
570
- margin-left: 4px;
571
- color: ${props => props.theme.palette.primary.main};
572
- text-decoration: underline;
573
- }
574
- }
575
-
576
- @media (min-width: ${props => props.mobileSize}) {
577
- .actions {
578
- .button-pay {
579
- width: 200px;
580
- }
581
- }
582
- }
583
-
584
- @media (max-width: ${props => props.mobileSize}) {
585
- flex-direction: column;
586
- align-items: center;
587
-
588
- .actions {
589
- display: flex;
590
- justify-content: space-between;
591
- width: 100%;
592
- margin-top: 20px;
593
-
594
- & > button {
595
- width: 48%;
596
- }
597
- }
598
- }
599
- }
600
- `;
601
- export default CheckoutPage;