@linktr.ee/messaging-react 1.33.3-rc-1777507218 → 1.34.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/Card-KHDSM8Zs.js +195 -0
- package/dist/Card-KHDSM8Zs.js.map +1 -0
- package/dist/Card-mHUN_YWq.js +167 -0
- package/dist/Card-mHUN_YWq.js.map +1 -0
- package/dist/assets/index.css +1 -1
- package/dist/{index-Ydi1pTAi.js → index-DWk0f1PF.js} +266 -263
- package/dist/index-DWk0f1PF.js.map +1 -0
- package/dist/index.d.ts +3 -1
- package/dist/index.js +1 -1
- package/package.json +1 -1
- package/src/components/CustomMessage/index.tsx +8 -5
- package/src/components/LockedAttachment/LockedAttachment.stories.tsx +41 -54
- package/src/components/LockedAttachment/components/Creator/Card.tsx +55 -23
- package/src/components/LockedAttachment/components/Creator/CardThumbnail.tsx +2 -2
- package/src/components/LockedAttachment/components/Visitor/Card.tsx +9 -16
- package/src/components/LockedAttachment/components/Visitor/CardActions.tsx +3 -12
- package/src/components/LockedAttachment/components/Visitor/CardThumbnail.tsx +2 -2
- package/src/components/MediaMessage/MediaMessage.stories.tsx +29 -61
- package/src/components/MediaMessage/index.tsx +1 -1
- package/src/styles.css +29 -3
- package/dist/Card-BfA8wq8O.js +0 -181
- package/dist/Card-BfA8wq8O.js.map +0 -1
- package/dist/Card-Bpud_enW.js +0 -177
- package/dist/Card-Bpud_enW.js.map +0 -1
- package/dist/index-Ydi1pTAi.js.map +0 -1
|
@@ -8,25 +8,17 @@ const meta: Meta<typeof MediaMessage> = {
|
|
|
8
8
|
title: 'Components/MediaMessage',
|
|
9
9
|
component: MediaMessage,
|
|
10
10
|
parameters: {
|
|
11
|
-
layout: '
|
|
11
|
+
layout: 'fullscreen',
|
|
12
12
|
},
|
|
13
13
|
}
|
|
14
14
|
export default meta
|
|
15
15
|
|
|
16
|
-
// ---------------------------------------------------------------------------
|
|
17
|
-
// Shared helpers
|
|
18
|
-
// ---------------------------------------------------------------------------
|
|
19
|
-
|
|
20
|
-
const SENDER = { id: 'user-other', name: 'Jane Creator', image: 'https://i.pravatar.cc/40?img=5' }
|
|
21
|
-
const ME = { id: 'user-me', name: 'Me' }
|
|
22
|
-
|
|
23
16
|
const base = (overrides: Partial<LocalMessage> = {}): LocalMessage => ({
|
|
24
17
|
id: 'msg-1',
|
|
25
18
|
text: '',
|
|
26
19
|
type: 'regular',
|
|
27
20
|
created_at: new Date(),
|
|
28
21
|
updated_at: new Date(),
|
|
29
|
-
user: SENDER,
|
|
30
22
|
attachments: [],
|
|
31
23
|
...overrides,
|
|
32
24
|
})
|
|
@@ -121,20 +113,17 @@ const LINK_VARIANTS = [
|
|
|
121
113
|
},
|
|
122
114
|
] as const
|
|
123
115
|
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
// ---------------------------------------------------------------------------
|
|
127
|
-
|
|
128
|
-
const GridTable: React.FC<{ children: React.ReactNode }> = ({ children }) => (
|
|
129
|
-
<div className="p-12 bg-[#F9F7F4]">
|
|
116
|
+
const Table: React.FC<{ children: React.ReactNode }> = ({ children }) => (
|
|
117
|
+
<div className="min-h-screen w-full p-12 bg-[#F9F7F4]">
|
|
130
118
|
<table className="border-separate border-spacing-4">{children}</table>
|
|
131
119
|
</div>
|
|
132
120
|
)
|
|
133
121
|
|
|
134
|
-
const
|
|
122
|
+
const TableHead: React.FC<{ variants: readonly { label: string }[] }> = ({ variants }) => (
|
|
135
123
|
<thead>
|
|
136
124
|
<tr>
|
|
137
|
-
|
|
125
|
+
<th className="text-left text-xs font-medium text-black/40 pb-2" />
|
|
126
|
+
{variants.map(({ label }) => (
|
|
138
127
|
<th key={label} className="text-left text-xs font-medium text-black/40 pb-2">
|
|
139
128
|
{label}
|
|
140
129
|
</th>
|
|
@@ -143,91 +132,70 @@ const GridHead: React.FC<{ labels: readonly string[] }> = ({ labels }) => (
|
|
|
143
132
|
</thead>
|
|
144
133
|
)
|
|
145
134
|
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
export const Received: StoryFn = () => (
|
|
151
|
-
<GridTable>
|
|
152
|
-
<GridHead labels={VARIANTS.map((v) => v.label)} />
|
|
135
|
+
export const Attachment: StoryFn = () => (
|
|
136
|
+
<Table>
|
|
137
|
+
<TableHead variants={VARIANTS} />
|
|
153
138
|
<tbody>
|
|
154
139
|
<tr>
|
|
140
|
+
<td className="text-xs text-right font-medium text-black/40 pr-4 align-top pt-2">
|
|
141
|
+
Received
|
|
142
|
+
</td>
|
|
155
143
|
{VARIANTS.map(({ label, attachment }) => (
|
|
156
144
|
<td key={label} className="align-top">
|
|
157
145
|
<MediaMessage
|
|
158
146
|
isMyMessage={false}
|
|
159
|
-
message={base({
|
|
147
|
+
message={base({ attachments: [attachment as LocalMessage['attachments'][number]] })}
|
|
160
148
|
/>
|
|
161
149
|
</td>
|
|
162
150
|
))}
|
|
163
151
|
</tr>
|
|
164
|
-
</tbody>
|
|
165
|
-
</GridTable>
|
|
166
|
-
)
|
|
167
|
-
Received.parameters = {
|
|
168
|
-
docs: {
|
|
169
|
-
description: {
|
|
170
|
-
story: 'Received messages — left-aligned with avatar, light gray card background.',
|
|
171
|
-
},
|
|
172
|
-
},
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
export const Sent: StoryFn = () => (
|
|
176
|
-
<GridTable>
|
|
177
|
-
<GridHead labels={VARIANTS.map((v) => v.label)} />
|
|
178
|
-
<tbody>
|
|
179
152
|
<tr>
|
|
153
|
+
<td className="text-xs text-right font-medium text-black/40 pr-4 align-top pt-2">
|
|
154
|
+
Sent
|
|
155
|
+
</td>
|
|
180
156
|
{VARIANTS.map(({ label, attachment }) => (
|
|
181
157
|
<td key={label} className="align-top">
|
|
182
158
|
<MediaMessage
|
|
183
159
|
isMyMessage={true}
|
|
184
|
-
message={base({
|
|
160
|
+
message={base({ attachments: [attachment as LocalMessage['attachments'][number]] })}
|
|
185
161
|
/>
|
|
186
162
|
</td>
|
|
187
163
|
))}
|
|
188
164
|
</tr>
|
|
189
165
|
</tbody>
|
|
190
|
-
</
|
|
166
|
+
</Table>
|
|
191
167
|
)
|
|
192
|
-
Sent.parameters = {
|
|
193
|
-
docs: {
|
|
194
|
-
description: {
|
|
195
|
-
story: 'Sent messages — right-aligned, no avatar, black card background.',
|
|
196
|
-
},
|
|
197
|
-
},
|
|
198
|
-
}
|
|
199
168
|
|
|
200
169
|
export const Links: StoryFn = () => (
|
|
201
|
-
<
|
|
202
|
-
<
|
|
170
|
+
<Table>
|
|
171
|
+
<TableHead variants={LINK_VARIANTS} />
|
|
203
172
|
<tbody>
|
|
204
173
|
<tr>
|
|
174
|
+
<td className="text-xs text-right font-medium text-black/40 pr-4 align-top pt-2">
|
|
175
|
+
Received
|
|
176
|
+
</td>
|
|
205
177
|
{LINK_VARIANTS.map(({ label, attachment }) => (
|
|
206
178
|
<td key={label} className="align-top">
|
|
207
179
|
<MediaMessage
|
|
208
180
|
isMyMessage={false}
|
|
209
|
-
message={base({
|
|
181
|
+
message={base({ attachments: [attachment as LocalMessage['attachments'][number]] })}
|
|
210
182
|
/>
|
|
211
183
|
</td>
|
|
212
184
|
))}
|
|
213
185
|
</tr>
|
|
214
186
|
<tr>
|
|
187
|
+
<td className="text-xs text-right font-medium text-black/40 pr-4 align-top pt-2">
|
|
188
|
+
Sent
|
|
189
|
+
</td>
|
|
215
190
|
{LINK_VARIANTS.map(({ label, attachment }) => (
|
|
216
191
|
<td key={label} className="align-top">
|
|
217
192
|
<MediaMessage
|
|
218
193
|
isMyMessage={true}
|
|
219
|
-
message={base({
|
|
194
|
+
message={base({ attachments: [attachment as LocalMessage['attachments'][number]] })}
|
|
220
195
|
/>
|
|
221
196
|
</td>
|
|
222
197
|
))}
|
|
223
198
|
</tr>
|
|
224
199
|
</tbody>
|
|
225
|
-
</
|
|
200
|
+
</Table>
|
|
226
201
|
)
|
|
227
|
-
Links.parameters = {
|
|
228
|
-
docs: {
|
|
229
|
-
description: {
|
|
230
|
-
story: 'Link preview cards — top row received, bottom row sent. Shows thumbnail, title, description, and URL with image fallback.',
|
|
231
|
-
},
|
|
232
|
-
},
|
|
233
|
-
}
|
|
@@ -113,7 +113,7 @@ const MediaMeta: React.FC<{
|
|
|
113
113
|
<div className="flex items-start gap-2 px-4 pb-3 pt-3">
|
|
114
114
|
<div className="min-w-0 flex-1">
|
|
115
115
|
{title && (
|
|
116
|
-
<p className={`
|
|
116
|
+
<p className={`truncate text-base font-medium ${primaryText(isMyMessage)}`}>
|
|
117
117
|
{title}
|
|
118
118
|
</p>
|
|
119
119
|
)}
|
package/src/styles.css
CHANGED
|
@@ -152,10 +152,37 @@
|
|
|
152
152
|
background: transparent;
|
|
153
153
|
}
|
|
154
154
|
|
|
155
|
-
|
|
156
|
-
|
|
155
|
+
.str-chat__li .str-chat__message--me {
|
|
156
|
+
position: relative;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
.str-chat__li .str-chat__message-inner {
|
|
160
|
+
grid-column-gap: 0;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
.str-chat__li .str-chat__message--me .str-chat__message-bubble-wrapper {
|
|
164
|
+
display: flex;
|
|
165
|
+
flex-direction: column;
|
|
166
|
+
align-items: flex-end;
|
|
167
|
+
gap: 2px;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
.str-chat__li .str-chat__message--other {
|
|
171
|
+
position: relative;
|
|
172
|
+
grid-template-columns: 40px 1fr;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
.str-chat__li .str-chat__message--other .str-chat__message-bubble-wrapper {
|
|
157
176
|
display: flex;
|
|
158
177
|
flex-direction: column;
|
|
178
|
+
align-items: flex-start;
|
|
179
|
+
gap: 2px;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
.str-chat__li .str-chat__message--other .str-chat__avatar {
|
|
183
|
+
position: absolute;
|
|
184
|
+
left: 0;
|
|
185
|
+
bottom: 0;
|
|
159
186
|
}
|
|
160
187
|
|
|
161
188
|
.str-chat__typing-indicator {
|
|
@@ -186,7 +213,6 @@
|
|
|
186
213
|
display: inline-flex;
|
|
187
214
|
align-items: center;
|
|
188
215
|
gap: 4px;
|
|
189
|
-
margin-top: 4px;
|
|
190
216
|
padding: 0;
|
|
191
217
|
font-size: 12px;
|
|
192
218
|
font-weight: 500;
|
package/dist/Card-BfA8wq8O.js
DELETED
|
@@ -1,181 +0,0 @@
|
|
|
1
|
-
import { jsxs as i, jsx as e } from "react/jsx-runtime";
|
|
2
|
-
import { CheckCircleIcon as N, XIcon as v, EyeIcon as k, EyeSlashIcon as z, LockOpenIcon as I, LockIcon as y } from "@phosphor-icons/react";
|
|
3
|
-
import d from "classnames";
|
|
4
|
-
import h, { useState as j, useCallback as _ } from "react";
|
|
5
|
-
import { r as b, g as C, M as S } from "./index-Ydi1pTAi.js";
|
|
6
|
-
const M = ({
|
|
7
|
-
title: a,
|
|
8
|
-
sourceUrl: l,
|
|
9
|
-
thumbnailUrl: t,
|
|
10
|
-
mimeType: s,
|
|
11
|
-
onToggle: r
|
|
12
|
-
}) => {
|
|
13
|
-
const n = r && l && t;
|
|
14
|
-
return /* @__PURE__ */ i(
|
|
15
|
-
"button",
|
|
16
|
-
{
|
|
17
|
-
type: "button",
|
|
18
|
-
disabled: !r,
|
|
19
|
-
className: d(
|
|
20
|
-
"relative block w-full overflow-hidden border-0 bg-black/5 p-0 text-left appearance-none",
|
|
21
|
-
{ "cursor-pointer": !!r, "cursor-default": !r }
|
|
22
|
-
),
|
|
23
|
-
onClick: r,
|
|
24
|
-
"aria-label": r ? "Toggle preview" : void 0,
|
|
25
|
-
children: [
|
|
26
|
-
n ? /* @__PURE__ */ e(
|
|
27
|
-
E,
|
|
28
|
-
{
|
|
29
|
-
sourceUrl: l,
|
|
30
|
-
thumbnailUrl: t,
|
|
31
|
-
mimeType: s
|
|
32
|
-
}
|
|
33
|
-
) : t ? /* @__PURE__ */ e("div", { className: "aspect-video overflow-hidden", children: /* @__PURE__ */ e(
|
|
34
|
-
"img",
|
|
35
|
-
{
|
|
36
|
-
src: t,
|
|
37
|
-
alt: a,
|
|
38
|
-
draggable: !1,
|
|
39
|
-
className: "absolute inset-0 h-full w-full object-cover"
|
|
40
|
-
}
|
|
41
|
-
) }) : /* @__PURE__ */ e("div", { className: "aspect-video flex items-center justify-center", children: b(s, {
|
|
42
|
-
className: "size-12 text-black/20",
|
|
43
|
-
weight: "regular"
|
|
44
|
-
}) }),
|
|
45
|
-
!n && /* @__PURE__ */ e("div", { className: "pointer-events-none absolute inset-0 bg-black/30" })
|
|
46
|
-
]
|
|
47
|
-
}
|
|
48
|
-
);
|
|
49
|
-
}, E = ({
|
|
50
|
-
sourceUrl: a,
|
|
51
|
-
thumbnailUrl: l,
|
|
52
|
-
mimeType: t
|
|
53
|
-
}) => {
|
|
54
|
-
const s = C(t);
|
|
55
|
-
return s === "video" || s === "audio" ? /* @__PURE__ */ e(
|
|
56
|
-
S,
|
|
57
|
-
{
|
|
58
|
-
mimeType: t,
|
|
59
|
-
source: a,
|
|
60
|
-
poster: l,
|
|
61
|
-
autoPlay: !0,
|
|
62
|
-
loop: !0,
|
|
63
|
-
controls: !0,
|
|
64
|
-
muted: !1
|
|
65
|
-
}
|
|
66
|
-
) : s === "image" ? /* @__PURE__ */ e("img", { src: a, alt: "", className: "block w-full", draggable: !1 }) : s === "document" ? /* @__PURE__ */ e(
|
|
67
|
-
"img",
|
|
68
|
-
{
|
|
69
|
-
src: l,
|
|
70
|
-
alt: "",
|
|
71
|
-
className: "block w-full",
|
|
72
|
-
draggable: !1
|
|
73
|
-
}
|
|
74
|
-
) : null;
|
|
75
|
-
}, X = ({
|
|
76
|
-
title: a,
|
|
77
|
-
mimeType: l = "application/octet-stream",
|
|
78
|
-
thumbnailUrl: t,
|
|
79
|
-
detail: s,
|
|
80
|
-
amountText: r,
|
|
81
|
-
placeholderTitle: n = "Attachment title",
|
|
82
|
-
placeholderAmountText: p,
|
|
83
|
-
paymentStatus: u,
|
|
84
|
-
onDismiss: x,
|
|
85
|
-
onPreviewClick: o
|
|
86
|
-
}) => {
|
|
87
|
-
const [c, m] = j(), g = c == null ? void 0 : c.sourceUrl, w = (c == null ? void 0 : c.thumbnailUrl) ?? t, f = _(() => {
|
|
88
|
-
c ? m(void 0) : o && m(o());
|
|
89
|
-
}, [c, o]);
|
|
90
|
-
return /* @__PURE__ */ i("div", { className: "relative w-[280px] select-none overflow-hidden rounded-[24px] bg-white shadow-[0_0_0_1px_rgba(0,0,0,0.04),0_4px_8px_rgba(0,0,0,0.06)]", children: [
|
|
91
|
-
/* @__PURE__ */ e(
|
|
92
|
-
F,
|
|
93
|
-
{
|
|
94
|
-
onDismiss: x,
|
|
95
|
-
onToggle: o ? f : void 0,
|
|
96
|
-
isExpanded: !!c,
|
|
97
|
-
paymentStatus: u
|
|
98
|
-
}
|
|
99
|
-
),
|
|
100
|
-
/* @__PURE__ */ e(
|
|
101
|
-
M,
|
|
102
|
-
{
|
|
103
|
-
title: a,
|
|
104
|
-
sourceUrl: g,
|
|
105
|
-
thumbnailUrl: w,
|
|
106
|
-
mimeType: l,
|
|
107
|
-
onToggle: o ? f : void 0
|
|
108
|
-
}
|
|
109
|
-
),
|
|
110
|
-
/* @__PURE__ */ i("div", { className: "px-4 pb-3 pt-3 bg-black", children: [
|
|
111
|
-
/* @__PURE__ */ e(
|
|
112
|
-
"p",
|
|
113
|
-
{
|
|
114
|
-
className: d("mb-1.5 truncate text-base font-medium", {
|
|
115
|
-
"text-white/30": !a,
|
|
116
|
-
"text-white": !!a
|
|
117
|
-
}),
|
|
118
|
-
children: a || n
|
|
119
|
-
}
|
|
120
|
-
),
|
|
121
|
-
/* @__PURE__ */ i("div", { className: "flex items-center gap-1", children: [
|
|
122
|
-
b(l, {
|
|
123
|
-
className: "size-5 shrink-0 text-white/55",
|
|
124
|
-
weight: "regular"
|
|
125
|
-
}),
|
|
126
|
-
s && /* @__PURE__ */ e("span", { className: "text-xs font-medium text-white/55", children: s }),
|
|
127
|
-
u === "paid" ? /* @__PURE__ */ i(h.Fragment, { children: [
|
|
128
|
-
/* @__PURE__ */ e("span", { className: "text-xs font-medium text-white/55", children: "•" }),
|
|
129
|
-
/* @__PURE__ */ e("span", { className: "text-xs font-medium text-[#4ade80]", children: "Sold" }),
|
|
130
|
-
/* @__PURE__ */ e(
|
|
131
|
-
N,
|
|
132
|
-
{
|
|
133
|
-
className: "size-4 text-[#4ade80]",
|
|
134
|
-
weight: "bold"
|
|
135
|
-
}
|
|
136
|
-
)
|
|
137
|
-
] }) : /* @__PURE__ */ i(h.Fragment, { children: [
|
|
138
|
-
/* @__PURE__ */ e("span", { className: "text-xs font-medium text-white/55", children: "•" }),
|
|
139
|
-
/* @__PURE__ */ e(
|
|
140
|
-
"span",
|
|
141
|
-
{
|
|
142
|
-
className: d("text-xs font-medium", {
|
|
143
|
-
"text-white/30": !r,
|
|
144
|
-
"text-white/55": !!r
|
|
145
|
-
}),
|
|
146
|
-
children: r || p
|
|
147
|
-
}
|
|
148
|
-
)
|
|
149
|
-
] })
|
|
150
|
-
] })
|
|
151
|
-
] })
|
|
152
|
-
] });
|
|
153
|
-
}, F = ({
|
|
154
|
-
onDismiss: a,
|
|
155
|
-
onToggle: l,
|
|
156
|
-
isExpanded: t,
|
|
157
|
-
paymentStatus: s
|
|
158
|
-
}) => a ? /* @__PURE__ */ e(
|
|
159
|
-
"button",
|
|
160
|
-
{
|
|
161
|
-
type: "button",
|
|
162
|
-
onClick: a,
|
|
163
|
-
className: "absolute top-3 z-50 flex size-8 items-center justify-center rounded-full bg-black/60 text-white right-3",
|
|
164
|
-
"aria-label": "Dismiss attachment",
|
|
165
|
-
children: /* @__PURE__ */ e(v, { className: "size-4", weight: "bold" })
|
|
166
|
-
}
|
|
167
|
-
) : l ? /* @__PURE__ */ e(
|
|
168
|
-
"button",
|
|
169
|
-
{
|
|
170
|
-
type: "button",
|
|
171
|
-
onClick: l,
|
|
172
|
-
className: "absolute top-3 z-50 flex size-8 items-center justify-center rounded-full bg-black/60 text-white left-3",
|
|
173
|
-
"aria-label": t ? "Hide preview" : "Show preview",
|
|
174
|
-
"aria-pressed": t,
|
|
175
|
-
children: /* @__PURE__ */ e(t ? k : z, { className: "size-4", weight: "fill" })
|
|
176
|
-
}
|
|
177
|
-
) : /* @__PURE__ */ e("div", { className: "absolute top-3 z-50 flex size-8 items-center justify-center rounded-full bg-black/60 text-white left-3", children: /* @__PURE__ */ e(s === "paid" ? I : y, { className: "size-4", weight: "fill" }) });
|
|
178
|
-
export {
|
|
179
|
-
X as default
|
|
180
|
-
};
|
|
181
|
-
//# sourceMappingURL=Card-BfA8wq8O.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"Card-BfA8wq8O.js","sources":["../src/components/LockedAttachment/components/Creator/CardThumbnail.tsx","../src/components/LockedAttachment/components/Creator/Card.tsx"],"sourcesContent":["import classNames from 'classnames'\nimport React from 'react'\n\nimport { renderTypeIcon } from '../../utils/icons'\nimport { getSourceType } from '../../utils/mimeType'\nimport MediaPlayer from '../MediaPlayer'\n\ninterface CardThumbnailProps {\n title?: string\n sourceUrl?: string\n thumbnailUrl?: string\n mimeType: string\n onToggle?: () => void\n}\n\nconst CardThumbnail: React.FC<CardThumbnailProps> = ({\n title,\n sourceUrl,\n thumbnailUrl,\n mimeType,\n onToggle,\n}) => {\n const isExpanded = onToggle && sourceUrl && thumbnailUrl\n\n return (\n <button\n type=\"button\"\n disabled={!onToggle}\n className={classNames(\n 'relative block w-full overflow-hidden border-0 bg-black/5 p-0 text-left appearance-none',\n { 'cursor-pointer': !!onToggle, 'cursor-default': !onToggle }\n )}\n onClick={onToggle}\n aria-label={onToggle ? 'Toggle preview' : undefined}\n >\n {isExpanded ? (\n <ThumbnailMedia\n sourceUrl={sourceUrl}\n thumbnailUrl={thumbnailUrl}\n mimeType={mimeType}\n />\n ) : thumbnailUrl ? (\n <div className=\"aspect-video overflow-hidden\">\n <img\n src={thumbnailUrl}\n alt={title}\n draggable={false}\n className=\"absolute inset-0 h-full w-full object-cover\"\n />\n </div>\n ) : (\n <div className=\"aspect-video flex items-center justify-center\">\n {renderTypeIcon(mimeType, {\n className: 'size-12 text-black/20',\n weight: 'regular',\n })}\n </div>\n )}\n\n {!isExpanded && (\n <div className=\"pointer-events-none absolute inset-0 bg-black/30\" />\n )}\n </button>\n )\n}\n\ninterface ThumbnailMediaProps {\n sourceUrl: string\n thumbnailUrl: string\n mimeType: string\n}\n\nconst ThumbnailMedia: React.FC<ThumbnailMediaProps> = ({\n sourceUrl,\n thumbnailUrl,\n mimeType,\n}) => {\n const sourceType = getSourceType(mimeType)\n\n if (sourceType === 'video' || sourceType === 'audio') {\n return (\n <MediaPlayer\n mimeType={mimeType}\n source={sourceUrl}\n poster={thumbnailUrl}\n autoPlay={true}\n loop={true}\n controls={true}\n muted={false}\n />\n )\n }\n\n if (sourceType === 'image') {\n return (\n <img src={sourceUrl} alt=\"\" className=\"block w-full\" draggable={false} />\n )\n }\n\n if (sourceType === 'document') {\n return (\n <img\n src={thumbnailUrl}\n alt=\"\"\n className=\"block w-full\"\n draggable={false}\n />\n )\n }\n\n return null\n}\n\nexport default CardThumbnail\n","import {\n CheckCircleIcon,\n EyeIcon,\n EyeSlashIcon,\n LockIcon,\n LockOpenIcon,\n XIcon,\n} from '@phosphor-icons/react'\nimport classNames from 'classnames'\nimport React, { useCallback, useState } from 'react'\n\nimport type {\n LockedAttachmentBaseProps,\n LockedAttachmentSource,\n PaymentStatus,\n} from '../../types'\nimport { renderTypeIcon } from '../../utils/icons'\n\nimport CardThumbnail from './CardThumbnail'\n\nexport interface CreatorCardProps extends LockedAttachmentBaseProps {\n placeholderTitle?: string\n placeholderAmountText?: string\n onDismiss?: () => void\n onPreviewClick?: () => LockedAttachmentSource\n}\n\nconst CreatorCard: React.FC<CreatorCardProps> = ({\n title,\n mimeType = 'application/octet-stream',\n thumbnailUrl,\n detail,\n amountText,\n placeholderTitle = 'Attachment title',\n placeholderAmountText,\n paymentStatus,\n onDismiss,\n onPreviewClick,\n}) => {\n const [source, setSource] = useState<LockedAttachmentSource | undefined>()\n\n const effectiveSourceUrl = source?.sourceUrl\n const effectiveThumbnailUrl = source?.thumbnailUrl ?? thumbnailUrl\n\n const handleToggle = useCallback(() => {\n if (source) {\n setSource(undefined)\n } else if (onPreviewClick) {\n setSource(onPreviewClick())\n }\n }, [source, onPreviewClick])\n\n return (\n <div className=\"relative w-[280px] select-none overflow-hidden rounded-[24px] bg-white shadow-[0_0_0_1px_rgba(0,0,0,0.04),0_4px_8px_rgba(0,0,0,0.06)]\">\n <CardHeader\n onDismiss={onDismiss}\n onToggle={onPreviewClick ? handleToggle : undefined}\n isExpanded={!!source}\n paymentStatus={paymentStatus}\n />\n\n <CardThumbnail\n title={title}\n sourceUrl={effectiveSourceUrl}\n thumbnailUrl={effectiveThumbnailUrl}\n mimeType={mimeType}\n onToggle={onPreviewClick ? handleToggle : undefined}\n />\n\n <div className=\"px-4 pb-3 pt-3 bg-black\">\n <p\n className={classNames('mb-1.5 truncate text-base font-medium', {\n 'text-white/30': !title,\n 'text-white': !!title,\n })}\n >\n {title || placeholderTitle}\n </p>\n\n <div className=\"flex items-center gap-1\">\n {renderTypeIcon(mimeType, {\n className: 'size-5 shrink-0 text-white/55',\n weight: 'regular',\n })}\n\n {detail && (\n <span className=\"text-xs font-medium text-white/55\">{detail}</span>\n )}\n\n {paymentStatus === 'paid' ? (\n <React.Fragment>\n <span className=\"text-xs font-medium text-white/55\">•</span>\n <span className=\"text-xs font-medium text-[#4ade80]\">Sold</span>\n <CheckCircleIcon\n className=\"size-4 text-[#4ade80]\"\n weight=\"bold\"\n />\n </React.Fragment>\n ) : (\n <React.Fragment>\n <span className=\"text-xs font-medium text-white/55\">•</span>\n <span\n className={classNames('text-xs font-medium', {\n 'text-white/30': !amountText,\n 'text-white/55': !!amountText,\n })}\n >\n {amountText || placeholderAmountText}\n </span>\n </React.Fragment>\n )}\n </div>\n </div>\n </div>\n )\n}\n\ninterface CardHeaderProps {\n onDismiss?: () => void\n onToggle?: () => void\n isExpanded?: boolean\n paymentStatus?: PaymentStatus\n}\n\nconst CardHeader: React.FC<CardHeaderProps> = ({\n onDismiss,\n onToggle,\n isExpanded,\n paymentStatus,\n}) => {\n if (onDismiss) {\n return (\n <button\n type=\"button\"\n onClick={onDismiss}\n className=\"absolute top-3 z-50 flex size-8 items-center justify-center rounded-full bg-black/60 text-white right-3\"\n aria-label=\"Dismiss attachment\"\n >\n <XIcon className=\"size-4\" weight=\"bold\" />\n </button>\n )\n }\n\n if (onToggle) {\n const Icon = isExpanded ? EyeIcon : EyeSlashIcon\n return (\n <button\n type=\"button\"\n onClick={onToggle}\n className=\"absolute top-3 z-50 flex size-8 items-center justify-center rounded-full bg-black/60 text-white left-3\"\n aria-label={isExpanded ? 'Hide preview' : 'Show preview'}\n aria-pressed={isExpanded}\n >\n <Icon className=\"size-4\" weight=\"fill\" />\n </button>\n )\n }\n\n const Icon = paymentStatus === 'paid' ? LockOpenIcon : LockIcon\n\n return (\n <div className=\"absolute top-3 z-50 flex size-8 items-center justify-center rounded-full bg-black/60 text-white left-3\">\n <Icon className=\"size-4\" weight=\"fill\" />\n </div>\n )\n}\n\nexport default CreatorCard\n"],"names":["CardThumbnail","title","sourceUrl","thumbnailUrl","mimeType","onToggle","isExpanded","jsxs","classNames","jsx","ThumbnailMedia","sourceType","getSourceType","MediaPlayer","CreatorCard","detail","amountText","placeholderTitle","placeholderAmountText","paymentStatus","onDismiss","onPreviewClick","source","setSource","useState","effectiveSourceUrl","effectiveThumbnailUrl","handleToggle","useCallback","CardHeader","renderTypeIcon","React","CheckCircleIcon","XIcon","EyeIcon","EyeSlashIcon","LockOpenIcon","LockIcon"],"mappings":"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"}
|
package/dist/Card-Bpud_enW.js
DELETED
|
@@ -1,177 +0,0 @@
|
|
|
1
|
-
import { jsx as e, jsxs as i } from "react/jsx-runtime";
|
|
2
|
-
import { LockSimpleIcon as k, DownloadSimpleIcon as R, LockOpenIcon as U, CheckCircleIcon as D } from "@phosphor-icons/react";
|
|
3
|
-
import w, { useState as y, useRef as x, useCallback as N, useEffect as M } from "react";
|
|
4
|
-
import { g as O, M as P, r as z } from "./index-Ydi1pTAi.js";
|
|
5
|
-
const A = (d) => {
|
|
6
|
-
const {
|
|
7
|
-
isUnlocking: l = !1,
|
|
8
|
-
sourceUrl: n,
|
|
9
|
-
redeemUrl: r,
|
|
10
|
-
onUnlockClicked: c,
|
|
11
|
-
onDownloadClicked: a
|
|
12
|
-
} = d, o = n === void 0;
|
|
13
|
-
return o && c != null ? /* @__PURE__ */ e(
|
|
14
|
-
"button",
|
|
15
|
-
{
|
|
16
|
-
type: "button",
|
|
17
|
-
onClick: c,
|
|
18
|
-
disabled: l,
|
|
19
|
-
className: "mt-3 inline-flex h-10 w-full items-center justify-center gap-2 rounded-full bg-white px-4 text-sm font-medium leading-none text-[#121110] hover:bg-white/90 disabled:opacity-70",
|
|
20
|
-
children: l ? /* @__PURE__ */ e(E, {}) : /* @__PURE__ */ i(w.Fragment, { children: [
|
|
21
|
-
/* @__PURE__ */ e(k, { className: "size-4", weight: "fill" }),
|
|
22
|
-
"Unlock"
|
|
23
|
-
] })
|
|
24
|
-
}
|
|
25
|
-
) : !o && a != null && n != null ? /* @__PURE__ */ i(
|
|
26
|
-
"a",
|
|
27
|
-
{
|
|
28
|
-
href: r ?? n,
|
|
29
|
-
target: "_blank",
|
|
30
|
-
rel: "noopener noreferrer",
|
|
31
|
-
onClick: a,
|
|
32
|
-
className: "mt-3 inline-flex h-10 w-full items-center justify-center gap-2 rounded-full bg-white px-4 text-sm font-medium leading-none !text-[#121110] hover:bg-white/90",
|
|
33
|
-
children: [
|
|
34
|
-
/* @__PURE__ */ e(R, { className: "size-4", weight: "bold" }),
|
|
35
|
-
"Download"
|
|
36
|
-
]
|
|
37
|
-
}
|
|
38
|
-
) : null;
|
|
39
|
-
}, E = () => /* @__PURE__ */ i("span", { className: "flex items-center gap-1", children: [
|
|
40
|
-
/* @__PURE__ */ e("span", { className: "size-1 rounded-full bg-white animate-bounce [animation-delay:-0.3s]" }),
|
|
41
|
-
/* @__PURE__ */ e("span", { className: "size-1 rounded-full bg-white animate-bounce [animation-delay:-0.15s]" }),
|
|
42
|
-
/* @__PURE__ */ e("span", { className: "size-1 rounded-full bg-white animate-bounce" })
|
|
43
|
-
] }), F = ({
|
|
44
|
-
title: d,
|
|
45
|
-
sourceUrl: l,
|
|
46
|
-
thumbnailUrl: n,
|
|
47
|
-
mimeType: r,
|
|
48
|
-
paymentStatus: c
|
|
49
|
-
}) => {
|
|
50
|
-
const [a, o] = y(!1), f = l === void 0, s = O(r);
|
|
51
|
-
return f ? /* @__PURE__ */ i("div", { className: "relative aspect-video overflow-hidden bg-black/5", children: [
|
|
52
|
-
n != null ? /* @__PURE__ */ e(
|
|
53
|
-
"img",
|
|
54
|
-
{
|
|
55
|
-
src: n,
|
|
56
|
-
alt: "",
|
|
57
|
-
className: "absolute inset-0 h-full w-full object-cover",
|
|
58
|
-
draggable: !1
|
|
59
|
-
}
|
|
60
|
-
) : /* @__PURE__ */ e("div", { className: "absolute inset-0 flex items-center justify-center", children: z(r, {
|
|
61
|
-
className: "size-12 text-black/20",
|
|
62
|
-
weight: "regular"
|
|
63
|
-
}) }),
|
|
64
|
-
f && /* @__PURE__ */ e("div", { className: "absolute inset-0 bg-black/30", children: /* @__PURE__ */ e("div", { className: "absolute left-3 top-3 flex size-8 items-center justify-center rounded-full bg-black/60 text-white", children: c === "paid" ? /* @__PURE__ */ e(U, {}) : /* @__PURE__ */ e(k, {}) }) })
|
|
65
|
-
] }) : s === "audio" || s === "video" ? /* @__PURE__ */ e(
|
|
66
|
-
P,
|
|
67
|
-
{
|
|
68
|
-
source: l,
|
|
69
|
-
poster: n,
|
|
70
|
-
mimeType: r
|
|
71
|
-
}
|
|
72
|
-
) : /* @__PURE__ */ e("div", { className: "relative overflow-hidden bg-black/5", children: /* @__PURE__ */ e(
|
|
73
|
-
"img",
|
|
74
|
-
{
|
|
75
|
-
src: s === "document" ? n : l,
|
|
76
|
-
alt: d,
|
|
77
|
-
className: `block w-full transition-opacity duration-300 ${a ? "opacity-100" : "opacity-0"}`,
|
|
78
|
-
draggable: !1,
|
|
79
|
-
onLoad: () => o(!0)
|
|
80
|
-
}
|
|
81
|
-
) });
|
|
82
|
-
}, q = ({
|
|
83
|
-
title: d,
|
|
84
|
-
amountText: l,
|
|
85
|
-
thumbnailUrl: n,
|
|
86
|
-
mimeType: r = "application/octet-stream",
|
|
87
|
-
detail: c,
|
|
88
|
-
onUnlockClick: a,
|
|
89
|
-
onFetchSource: o,
|
|
90
|
-
onDownloadClick: f,
|
|
91
|
-
paymentStatus: s,
|
|
92
|
-
isUnlocking: C = !1
|
|
93
|
-
}) => {
|
|
94
|
-
const [t, _] = y(), g = x(null), p = x(!1), m = x(o);
|
|
95
|
-
m.current = o;
|
|
96
|
-
const v = t == null ? void 0 : t.sourceUrl, j = (t == null ? void 0 : t.thumbnailUrl) ?? n, I = t == null ? void 0 : t.redeemUrl, h = N(async () => {
|
|
97
|
-
var u;
|
|
98
|
-
if (!p.current) {
|
|
99
|
-
p.current = !0;
|
|
100
|
-
try {
|
|
101
|
-
const b = await ((u = m.current) == null ? void 0 : u.call(m));
|
|
102
|
-
b && _(b);
|
|
103
|
-
} finally {
|
|
104
|
-
p.current = !1;
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
}, []), L = N(() => {
|
|
108
|
-
s === "paid" ? h() : a == null || a();
|
|
109
|
-
}, [s, h, a]);
|
|
110
|
-
return M(() => {
|
|
111
|
-
if (!g.current || s !== "paid" || t !== void 0) return;
|
|
112
|
-
const u = new IntersectionObserver(
|
|
113
|
-
([b]) => {
|
|
114
|
-
b.isIntersecting && (h(), u.disconnect());
|
|
115
|
-
},
|
|
116
|
-
{ threshold: 1 }
|
|
117
|
-
);
|
|
118
|
-
return u.observe(g.current), () => u.disconnect();
|
|
119
|
-
}, [s, t, h]), /* @__PURE__ */ i(
|
|
120
|
-
"div",
|
|
121
|
-
{
|
|
122
|
-
ref: g,
|
|
123
|
-
"data-testid": "locked-attachment",
|
|
124
|
-
className: "w-[280px] select-none overflow-hidden rounded-[24px] bg-white shadow-[0_0_0_1px_rgba(0,0,0,0.04),0_4px_8px_rgba(0,0,0,0.06)]",
|
|
125
|
-
children: [
|
|
126
|
-
/* @__PURE__ */ e(
|
|
127
|
-
F,
|
|
128
|
-
{
|
|
129
|
-
title: d,
|
|
130
|
-
sourceUrl: v,
|
|
131
|
-
thumbnailUrl: j,
|
|
132
|
-
mimeType: r,
|
|
133
|
-
paymentStatus: s
|
|
134
|
-
}
|
|
135
|
-
),
|
|
136
|
-
/* @__PURE__ */ i("div", { className: "px-4 pb-3 pt-3 bg-black", children: [
|
|
137
|
-
/* @__PURE__ */ e("p", { className: "mb-1.5 truncate text-base font-medium text-white", children: d }),
|
|
138
|
-
/* @__PURE__ */ i("div", { className: "flex items-center gap-1", children: [
|
|
139
|
-
z(r, {
|
|
140
|
-
className: "size-5 shrink-0 text-white/55",
|
|
141
|
-
weight: "regular"
|
|
142
|
-
}),
|
|
143
|
-
c && /* @__PURE__ */ e("span", { className: "text-xs font-medium text-white/55", children: c }),
|
|
144
|
-
s === "paid" ? /* @__PURE__ */ i(w.Fragment, { children: [
|
|
145
|
-
/* @__PURE__ */ e("span", { className: "text-xs font-medium text-white/55", children: "•" }),
|
|
146
|
-
/* @__PURE__ */ e("span", { className: "text-xs font-medium text-[#4ade80]", children: "Purchased" }),
|
|
147
|
-
/* @__PURE__ */ e(
|
|
148
|
-
D,
|
|
149
|
-
{
|
|
150
|
-
className: "size-4 text-[#4ade80]",
|
|
151
|
-
weight: "bold"
|
|
152
|
-
}
|
|
153
|
-
)
|
|
154
|
-
] }) : l != null ? /* @__PURE__ */ i(w.Fragment, { children: [
|
|
155
|
-
/* @__PURE__ */ e("span", { className: "text-xs font-medium text-white/55", children: "•" }),
|
|
156
|
-
/* @__PURE__ */ e("span", { className: "text-xs font-medium text-white/55", children: l })
|
|
157
|
-
] }) : null
|
|
158
|
-
] }),
|
|
159
|
-
/* @__PURE__ */ e(
|
|
160
|
-
A,
|
|
161
|
-
{
|
|
162
|
-
isUnlocking: C,
|
|
163
|
-
sourceUrl: v,
|
|
164
|
-
redeemUrl: I,
|
|
165
|
-
onUnlockClicked: L,
|
|
166
|
-
onDownloadClicked: f
|
|
167
|
-
}
|
|
168
|
-
)
|
|
169
|
-
] })
|
|
170
|
-
]
|
|
171
|
-
}
|
|
172
|
-
);
|
|
173
|
-
};
|
|
174
|
-
export {
|
|
175
|
-
q as default
|
|
176
|
-
};
|
|
177
|
-
//# sourceMappingURL=Card-Bpud_enW.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"Card-Bpud_enW.js","sources":["../src/components/LockedAttachment/components/Visitor/CardActions.tsx","../src/components/LockedAttachment/components/Visitor/CardThumbnail.tsx","../src/components/LockedAttachment/components/Visitor/Card.tsx"],"sourcesContent":["import { DownloadSimpleIcon, LockSimpleIcon } from '@phosphor-icons/react'\nimport React from 'react'\n\ninterface CardActionsProps {\n sourceUrl?: string\n redeemUrl?: string\n onUnlockClicked?: () => void\n onDownloadClicked?: () => void\n isUnlocking?: boolean\n}\n\nconst CardActions: React.FC<CardActionsProps> = (props) => {\n const {\n isUnlocking = false,\n sourceUrl,\n redeemUrl,\n onUnlockClicked,\n onDownloadClicked,\n } = props\n\n const isLocked = sourceUrl === undefined\n\n if (isLocked && onUnlockClicked != null) {\n return (\n <button\n type=\"button\"\n onClick={onUnlockClicked}\n disabled={isUnlocking}\n className=\"mt-3 inline-flex h-10 w-full items-center justify-center gap-2 rounded-full bg-white px-4 text-sm font-medium leading-none text-[#121110] hover:bg-white/90 disabled:opacity-70\"\n >\n {isUnlocking ? (\n <LoadingDots />\n ) : (\n <React.Fragment>\n <LockSimpleIcon className=\"size-4\" weight=\"fill\" />\n Unlock\n </React.Fragment>\n )}\n </button>\n )\n }\n\n if (!isLocked && onDownloadClicked != null && sourceUrl != null) {\n return (\n <a\n href={redeemUrl ?? sourceUrl}\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n onClick={onDownloadClicked}\n className=\"mt-3 inline-flex h-10 w-full items-center justify-center gap-2 rounded-full bg-white px-4 text-sm font-medium leading-none !text-[#121110] hover:bg-white/90\"\n >\n <DownloadSimpleIcon className=\"size-4\" weight=\"bold\" />\n Download\n </a>\n )\n }\n\n return null\n}\n\nconst LoadingDots: React.FC = () => {\n return (\n <span className=\"flex items-center gap-1\">\n <span className=\"size-1 rounded-full bg-white animate-bounce [animation-delay:-0.3s]\" />\n <span className=\"size-1 rounded-full bg-white animate-bounce [animation-delay:-0.15s]\" />\n <span className=\"size-1 rounded-full bg-white animate-bounce\" />\n </span>\n )\n}\n\nexport default CardActions\n","import { LockOpenIcon, LockSimpleIcon } from '@phosphor-icons/react'\nimport React, { useState } from 'react'\n\nimport { PaymentStatus } from '../../types'\nimport { renderTypeIcon } from '../../utils/icons'\nimport { getSourceType } from '../../utils/mimeType'\nimport MediaPlayer from '../MediaPlayer'\n\ninterface CardThumbnailProps {\n title?: string\n sourceUrl?: string\n thumbnailUrl?: string\n mimeType: string\n paymentStatus?: PaymentStatus\n}\n\nconst CardThumbnail: React.FC<CardThumbnailProps> = ({\n title,\n sourceUrl,\n thumbnailUrl,\n mimeType,\n paymentStatus,\n}) => {\n const [sourceReady, setSourceReady] = useState(false)\n\n const isLocked = sourceUrl === undefined\n const sourceType = getSourceType(mimeType)\n\n if (!isLocked) {\n if (sourceType === 'audio' || sourceType === 'video') {\n return (\n <MediaPlayer\n source={sourceUrl}\n poster={thumbnailUrl}\n mimeType={mimeType}\n />\n )\n }\n\n return (\n <div className=\"relative overflow-hidden bg-black/5\">\n <img\n src={sourceType === 'document' ? thumbnailUrl : sourceUrl}\n alt={title}\n className={`block w-full transition-opacity duration-300 ${sourceReady ? 'opacity-100' : 'opacity-0'}`}\n draggable={false}\n onLoad={() => setSourceReady(true)}\n />\n </div>\n )\n }\n\n return (\n <div className=\"relative aspect-video overflow-hidden bg-black/5\">\n {thumbnailUrl != null ? (\n <img\n src={thumbnailUrl}\n alt=\"\"\n className=\"absolute inset-0 h-full w-full object-cover\"\n draggable={false}\n />\n ) : (\n <div className=\"absolute inset-0 flex items-center justify-center\">\n {renderTypeIcon(mimeType, {\n className: 'size-12 text-black/20',\n weight: 'regular',\n })}\n </div>\n )}\n {isLocked && (\n <div className=\"absolute inset-0 bg-black/30\">\n <div className=\"absolute left-3 top-3 flex size-8 items-center justify-center rounded-full bg-black/60 text-white\">\n {paymentStatus === 'paid' ? <LockOpenIcon /> : <LockSimpleIcon />}\n </div>\n </div>\n )}\n </div>\n )\n}\n\nexport default CardThumbnail\n","import { CheckCircleIcon } from '@phosphor-icons/react'\nimport React, { useCallback, useEffect, useRef, useState } from 'react'\n\nimport type {\n LockedAttachmentBaseProps,\n LockedAttachmentSource,\n} from '../../types'\nimport { renderTypeIcon } from '../../utils/icons'\n\nimport CardActions from './CardActions'\nimport CardThumbnail from './CardThumbnail'\n\nexport interface VisitorCardProps extends LockedAttachmentBaseProps {\n /**\n * Called when the visitor clicks Unlock on an unpaid attachment.\n * Use this to open a checkout flow. Omit to hide the Unlock button.\n */\n onUnlockClick?: () => void\n /**\n * Called to fetch the attachment source — fired automatically when\n * paymentStatus transitions to 'paid', or immediately on click when\n * paymentStatus is already 'paid'. Return a LockedAttachmentSource to\n * unlock the card.\n */\n onFetchSource?: () => Promise<LockedAttachmentSource | void>\n /**\n * Called when the visitor clicks Download on an unlocked card.\n * Omit to hide the Download button.\n */\n onDownloadClick?: () => void\n /**\n * When true, shows loading dots on the Unlock button.\n * Driven by the LockedAttachmentContext (e.g. checkout in progress, payment processing).\n */\n isUnlocking?: boolean\n}\n\nconst VisitorCard: React.FC<VisitorCardProps> = ({\n title,\n amountText,\n thumbnailUrl,\n mimeType = 'application/octet-stream',\n detail,\n onUnlockClick,\n onFetchSource,\n onDownloadClick,\n paymentStatus,\n isUnlocking = false,\n}) => {\n const [source, setSource] = useState<LockedAttachmentSource | undefined>()\n\n const cardRef = useRef<HTMLDivElement>(null)\n const fetchingRef = useRef(false)\n\n const onFetchSourceRef = useRef(onFetchSource)\n onFetchSourceRef.current = onFetchSource\n\n const effectiveSourceUrl = source?.sourceUrl\n const effectiveThumbnail = source?.thumbnailUrl ?? thumbnailUrl\n const effectiveRedeemUrl = source?.redeemUrl\n\n const fetchSource = useCallback(async (): Promise<void> => {\n if (fetchingRef.current) return\n fetchingRef.current = true\n try {\n const result = await onFetchSourceRef.current?.()\n if (result) setSource(result)\n } finally {\n fetchingRef.current = false\n }\n }, [])\n\n const handleUnlockClick = useCallback(() => {\n if (paymentStatus === 'paid') {\n void fetchSource()\n } else {\n onUnlockClick?.()\n }\n }, [paymentStatus, fetchSource, onUnlockClick])\n\n // Fetch source when card is in viewport\n useEffect(() => {\n if (!cardRef.current) return\n if (paymentStatus !== 'paid' || source !== undefined) return\n\n const observer = new IntersectionObserver(\n ([entry]) => {\n if (entry.isIntersecting) {\n void fetchSource()\n observer.disconnect()\n }\n },\n { threshold: 1.0 }\n )\n\n observer.observe(cardRef.current)\n return () => observer.disconnect()\n }, [paymentStatus, source, fetchSource])\n\n return (\n <div\n ref={cardRef}\n data-testid=\"locked-attachment\"\n className=\"w-[280px] select-none overflow-hidden rounded-[24px] bg-white shadow-[0_0_0_1px_rgba(0,0,0,0.04),0_4px_8px_rgba(0,0,0,0.06)]\"\n >\n <CardThumbnail\n title={title}\n sourceUrl={effectiveSourceUrl}\n thumbnailUrl={effectiveThumbnail}\n mimeType={mimeType}\n paymentStatus={paymentStatus}\n />\n\n <div className=\"px-4 pb-3 pt-3 bg-black\">\n <p className=\"mb-1.5 truncate text-base font-medium text-white\">\n {title}\n </p>\n <div className=\"flex items-center gap-1\">\n {renderTypeIcon(mimeType, {\n className: 'size-5 shrink-0 text-white/55',\n weight: 'regular',\n })}\n\n {detail && (\n <span className=\"text-xs font-medium text-white/55\">{detail}</span>\n )}\n\n {paymentStatus === 'paid' ? (\n <React.Fragment>\n <span className=\"text-xs font-medium text-white/55\">•</span>\n <span className=\"text-xs font-medium text-[#4ade80]\">\n Purchased\n </span>\n <CheckCircleIcon\n className=\"size-4 text-[#4ade80]\"\n weight=\"bold\"\n />\n </React.Fragment>\n ) : amountText != null ? (\n <React.Fragment>\n <span className=\"text-xs font-medium text-white/55\">•</span>\n <span className=\"text-xs font-medium text-white/55\">\n {amountText}\n </span>\n </React.Fragment>\n ) : null}\n </div>\n\n <CardActions\n isUnlocking={isUnlocking}\n sourceUrl={effectiveSourceUrl}\n redeemUrl={effectiveRedeemUrl}\n onUnlockClicked={handleUnlockClick}\n onDownloadClicked={onDownloadClick}\n />\n </div>\n </div>\n )\n}\n\nexport default VisitorCard\n"],"names":["CardActions","props","isUnlocking","sourceUrl","redeemUrl","onUnlockClicked","onDownloadClicked","isLocked","jsx","LoadingDots","jsxs","React","LockSimpleIcon","DownloadSimpleIcon","CardThumbnail","title","thumbnailUrl","mimeType","paymentStatus","sourceReady","setSourceReady","useState","sourceType","getSourceType","LockOpenIcon","MediaPlayer","VisitorCard","amountText","detail","onUnlockClick","onFetchSource","onDownloadClick","source","setSource","cardRef","useRef","fetchingRef","onFetchSourceRef","effectiveSourceUrl","effectiveThumbnail","effectiveRedeemUrl","fetchSource","useCallback","result","_a","handleUnlockClick","useEffect","observer","entry","renderTypeIcon","CheckCircleIcon"],"mappings":"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"}
|