@dcl/asset-packs 2.14.2 → 2.15.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/bin/index.js +2169 -616
- package/catalog.json +127 -1
- package/dist/actions.js +8 -8
- package/dist/admin-toolkit-ui/Button.d.ts +1 -1
- package/dist/admin-toolkit-ui/Button.js +2 -2
- package/dist/admin-toolkit-ui/Error.d.ts +2 -2
- package/dist/admin-toolkit-ui/Error.js +4 -4
- package/dist/admin-toolkit-ui/Header.js +1 -1
- package/dist/admin-toolkit-ui/Modal.d.ts +19 -0
- package/dist/admin-toolkit-ui/Modal.js +37 -0
- package/dist/admin-toolkit-ui/ModerationControl/UsersList.d.ts +3 -3
- package/dist/admin-toolkit-ui/ModerationControl/UsersList.js +14 -8
- package/dist/admin-toolkit-ui/ModerationControl/index.d.ts +4 -5
- package/dist/admin-toolkit-ui/ModerationControl/index.js +19 -11
- package/dist/admin-toolkit-ui/ModerationControl/styles/AddUserInputStyles.js +4 -4
- package/dist/admin-toolkit-ui/ModerationControl/styles/UsersListStyles.d.ts +2 -0
- package/dist/admin-toolkit-ui/ModerationControl/styles/UsersListStyles.js +13 -5
- package/dist/admin-toolkit-ui/RewardsControl.d.ts +2 -2
- package/dist/admin-toolkit-ui/RewardsControl.js +9 -5
- package/dist/admin-toolkit-ui/SmartItemsControl.d.ts +2 -2
- package/dist/admin-toolkit-ui/SmartItemsControl.js +6 -4
- package/dist/admin-toolkit-ui/TextAnnouncements.d.ts +2 -2
- package/dist/admin-toolkit-ui/TextAnnouncements.js +9 -5
- package/dist/admin-toolkit-ui/TextAnnouncementsControl.d.ts +3 -3
- package/dist/admin-toolkit-ui/TextAnnouncementsControl.js +15 -7
- package/dist/admin-toolkit-ui/VideoControl/DclCast/CompactDclCast.d.ts +11 -0
- package/dist/admin-toolkit-ui/VideoControl/DclCast/CompactDclCast.js +90 -0
- package/dist/admin-toolkit-ui/VideoControl/DclCast/DclCastInfo.d.ts +6 -4
- package/dist/admin-toolkit-ui/VideoControl/DclCast/DclCastInfo.js +71 -57
- package/dist/admin-toolkit-ui/VideoControl/DclCast/PresentationPanel.d.ts +10 -0
- package/dist/admin-toolkit-ui/VideoControl/DclCast/PresentationPanel.js +67 -0
- package/dist/admin-toolkit-ui/VideoControl/DclCast/SharePresentationModal.d.ts +6 -0
- package/dist/admin-toolkit-ui/VideoControl/DclCast/SharePresentationModal.js +74 -0
- package/dist/admin-toolkit-ui/VideoControl/DclCast/SpeakerShowcase.d.ts +11 -0
- package/dist/admin-toolkit-ui/VideoControl/DclCast/SpeakerShowcase.js +110 -0
- package/dist/admin-toolkit-ui/VideoControl/DclCast/index.d.ts +17 -3
- package/dist/admin-toolkit-ui/VideoControl/DclCast/index.js +208 -35
- package/dist/admin-toolkit-ui/VideoControl/DclCast/styles.d.ts +73 -1
- package/dist/admin-toolkit-ui/VideoControl/DclCast/styles.js +193 -3
- package/dist/admin-toolkit-ui/VideoControl/LiveStream/ShowStreamKey.js +17 -13
- package/dist/admin-toolkit-ui/VideoControl/LiveStream/index.d.ts +1 -1
- package/dist/admin-toolkit-ui/VideoControl/LiveStream/index.js +8 -8
- package/dist/admin-toolkit-ui/VideoControl/VideoUrl.d.ts +2 -2
- package/dist/admin-toolkit-ui/VideoControl/VideoUrl.js +9 -9
- package/dist/admin-toolkit-ui/VideoControl/VolumeControl.d.ts +3 -2
- package/dist/admin-toolkit-ui/VideoControl/VolumeControl.js +8 -10
- package/dist/admin-toolkit-ui/VideoControl/api.d.ts +49 -0
- package/dist/admin-toolkit-ui/VideoControl/api.js +185 -1
- package/dist/admin-toolkit-ui/VideoControl/index.d.ts +16 -15
- package/dist/admin-toolkit-ui/VideoControl/index.js +57 -21
- package/dist/admin-toolkit-ui/VideoControl/utils.d.ts +2 -2
- package/dist/admin-toolkit-ui/VideoControl/utils.js +3 -3
- package/dist/admin-toolkit-ui/constants.d.ts +2 -3
- package/dist/admin-toolkit-ui/constants.js +4 -4
- package/dist/admin-toolkit-ui/fetch-utils.js +2 -2
- package/dist/admin-toolkit-ui/index.d.ts +4 -4
- package/dist/admin-toolkit-ui/index.js +40 -18
- package/dist/admin-toolkit-ui/types.d.ts +19 -0
- package/dist/admin-toolkit-ui/types.js +1 -1
- package/dist/admin.js +2 -2
- package/dist/bin/index.js +2169 -616
- package/dist/ui.js +4 -4
- package/package.json +5 -5
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import ReactEcs, { Dropdown, UiEntity, Label } from '@dcl/react-ecs';
|
|
2
|
+
import { getContentUrl } from '../../constants';
|
|
3
|
+
import { Button } from '../../Button';
|
|
4
|
+
import { Modal } from '../../Modal';
|
|
5
|
+
import { getSourceLabel, isPresentationBot } from '../api';
|
|
6
|
+
import { getSpeakerShowcaseStyles, getShowcaseColors, getShowcaseBackgrounds, getShowcaseIconBackgrounds, getPaginationColor, SHOWCASE_DROPDOWN_COLORS, SHOWCASE_PAGE_INDICATOR_COLOR, } from './styles';
|
|
7
|
+
const SPEAKERS_PER_PAGE = 10;
|
|
8
|
+
const ICONS = {
|
|
9
|
+
get BACK() {
|
|
10
|
+
return `${getContentUrl()}/admin_toolkit/assets/icons/chevron-back.png`;
|
|
11
|
+
},
|
|
12
|
+
get NEXT() {
|
|
13
|
+
return `${getContentUrl()}/admin_toolkit/assets/icons/chevron-forward.png`;
|
|
14
|
+
},
|
|
15
|
+
get STAR() {
|
|
16
|
+
return `${getContentUrl()}/admin_toolkit/assets/icons/star.png`;
|
|
17
|
+
},
|
|
18
|
+
};
|
|
19
|
+
function getDropdownOptions(participant) {
|
|
20
|
+
return participant.tracks.map(track => `${getSourceLabel(track.sourceType)}`);
|
|
21
|
+
}
|
|
22
|
+
function getActiveIndex(participant, activeTrackSid) {
|
|
23
|
+
if (!activeTrackSid)
|
|
24
|
+
return -1;
|
|
25
|
+
return participant.tracks.findIndex(t => t.sid === activeTrackSid);
|
|
26
|
+
}
|
|
27
|
+
function ParticipantRow({ participant, activeTrackSid, isHovered, onSelectTrack, onHoverEnter, onHoverLeave, }) {
|
|
28
|
+
const styles = getSpeakerShowcaseStyles();
|
|
29
|
+
const backgrounds = getShowcaseBackgrounds();
|
|
30
|
+
const colors = getShowcaseColors();
|
|
31
|
+
const iconBgs = getShowcaseIconBackgrounds();
|
|
32
|
+
const isBot = isPresentationBot(participant.name);
|
|
33
|
+
const displayName = isBot ? 'Presentation' : participant.name;
|
|
34
|
+
const activeIndex = getActiveIndex(participant, activeTrackSid);
|
|
35
|
+
const isActive = activeIndex !== -1;
|
|
36
|
+
const options = getDropdownOptions(participant);
|
|
37
|
+
const displayLabel = isActive
|
|
38
|
+
? getSourceLabel(participant.tracks[activeIndex].sourceType)
|
|
39
|
+
: 'Showcase';
|
|
40
|
+
const dropdownTextColor = isHovered
|
|
41
|
+
? SHOWCASE_DROPDOWN_COLORS.hover
|
|
42
|
+
: isActive
|
|
43
|
+
? SHOWCASE_DROPDOWN_COLORS.active
|
|
44
|
+
: SHOWCASE_DROPDOWN_COLORS.idle;
|
|
45
|
+
const dropdownBgColor = isHovered
|
|
46
|
+
? SHOWCASE_DROPDOWN_COLORS.hoverBg
|
|
47
|
+
: SHOWCASE_DROPDOWN_COLORS.transparentBg;
|
|
48
|
+
return (ReactEcs.createElement(UiEntity, { uiTransform: styles.userItem },
|
|
49
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.userRow },
|
|
50
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.userInfo },
|
|
51
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.personIconContainer },
|
|
52
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.personIcon, uiBackground: backgrounds.personIcon })),
|
|
53
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.userDetails },
|
|
54
|
+
ReactEcs.createElement(Label, { value: `<b>${displayName}</b>`, fontSize: 14, color: colors.white }))),
|
|
55
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.rowCenter }, isBot ? (ReactEcs.createElement(Button, { id: `showcase-activate-${participant.identity}`, value: isActive ? '<b>Active</b>' : '<b>Activate</b>', variant: isActive ? 'primary' : 'secondary', disabled: isActive, fontSize: 14, color: colors.white, uiTransform: styles.dropdownTransform, onMouseDown: () => {
|
|
56
|
+
const track = participant.tracks[0];
|
|
57
|
+
if (track) {
|
|
58
|
+
onSelectTrack(track);
|
|
59
|
+
}
|
|
60
|
+
} })) : (ReactEcs.createElement(UiEntity, { uiTransform: styles.rowCenter },
|
|
61
|
+
ReactEcs.createElement(UiEntity, { uiTransform: { ...styles.starIcon, display: isActive ? 'flex' : 'none' }, uiBackground: iconBgs.star }),
|
|
62
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.dropdownWrapper, onMouseEnter: onHoverEnter, onMouseLeave: onHoverLeave },
|
|
63
|
+
ReactEcs.createElement(Dropdown, { key: `showcase-dropdown-${participant.identity}-${activeTrackSid ?? 'none'}`, acceptEmpty: true, emptyLabel: displayLabel, options: options, selectedIndex: -1, onChange: (optionIndex) => {
|
|
64
|
+
const track = participant.tracks[optionIndex];
|
|
65
|
+
if (track) {
|
|
66
|
+
onSelectTrack(track);
|
|
67
|
+
}
|
|
68
|
+
}, textAlign: "middle-left", fontSize: 14, uiTransform: styles.dropdownTransform, uiBackground: { color: dropdownBgColor }, color: dropdownTextColor })))))),
|
|
69
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.divider, uiBackground: backgrounds.divider })));
|
|
70
|
+
}
|
|
71
|
+
export function SpeakerShowcase({ participants, activeTrackSid, onSelectTrack, onSetDefault, onClose, }) {
|
|
72
|
+
const [page, setPage] = ReactEcs.useState(1);
|
|
73
|
+
const [hoveredDropdown, setHoveredDropdown] = ReactEcs.useState(undefined);
|
|
74
|
+
const isDefaultActive = !activeTrackSid;
|
|
75
|
+
const styles = getSpeakerShowcaseStyles();
|
|
76
|
+
const colors = getShowcaseColors();
|
|
77
|
+
const iconBgs = getShowcaseIconBackgrounds();
|
|
78
|
+
const totalPages = Math.ceil(participants.length / SPEAKERS_PER_PAGE);
|
|
79
|
+
const startIndex = (page - 1) * SPEAKERS_PER_PAGE;
|
|
80
|
+
const endIndex = Math.min(startIndex + SPEAKERS_PER_PAGE, participants.length);
|
|
81
|
+
const currentPageParticipants = participants.slice(startIndex, endIndex);
|
|
82
|
+
const paginationFooter = participants.length > SPEAKERS_PER_PAGE && (ReactEcs.createElement(UiEntity, { uiTransform: styles.pagination },
|
|
83
|
+
ReactEcs.createElement(Button, { id: "showcase-prev", value: "Prev", variant: "secondary", disabled: page <= 1, fontSize: 18, icon: ICONS.BACK, iconTransform: styles.prevIcon, iconBackground: { color: getPaginationColor(page <= 1) }, color: getPaginationColor(page <= 1), labelTransform: styles.prevLabel, uiTransform: styles.paginationButton, onMouseDown: () => setPage(page - 1) }),
|
|
84
|
+
ReactEcs.createElement(Label, { value: `Page ${page}/${totalPages}`, fontSize: 14, color: SHOWCASE_PAGE_INDICATOR_COLOR }),
|
|
85
|
+
ReactEcs.createElement(Button, { id: "showcase-next", value: "<b>Next</b>", variant: "secondary", fontSize: 18, iconRight: ICONS.NEXT, iconRightTransform: styles.nextIcon, labelTransform: styles.nextLabel, iconRightBackground: {
|
|
86
|
+
color: getPaginationColor(page >= totalPages),
|
|
87
|
+
}, color: getPaginationColor(page >= totalPages), disabled: page >= totalPages, uiTransform: styles.paginationButton, onMouseDown: () => setPage(page + 1) })));
|
|
88
|
+
return (ReactEcs.createElement(Modal, { id: "showcase", title: "SPEAKER SHOWCASE", titleFontSize: 20, headerIcon: iconBgs.showcase, headerIconSize: 24, headerMarginBottom: 16, counterText: `(${participants.length} Speakers)`, counterFontSize: 14, width: 650, height: 650, padding: 16, onClose: onClose, footer: paginationFooter || undefined },
|
|
89
|
+
ReactEcs.createElement(UiEntity, { uiTransform: { flexDirection: 'column', width: '100%' } },
|
|
90
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.toggleRow, uiBackground: { color: colors.softBlack } },
|
|
91
|
+
ReactEcs.createElement(UiEntity, { uiTransform: { flexDirection: 'column' } },
|
|
92
|
+
ReactEcs.createElement(Label, { value: "Automatic Showcase", fontSize: 14, color: colors.white }),
|
|
93
|
+
ReactEcs.createElement(Label, { value: "Speakers will be automatically featured when they speak", fontSize: 10, color: colors.gray })),
|
|
94
|
+
ReactEcs.createElement(Button, { id: "showcase-default-speaker", value: isDefaultActive ? 'Active' : 'Turn On', variant: "secondary", disabled: isDefaultActive, fontSize: 14, color: colors.white, icon: ICONS.STAR, iconTransform: {
|
|
95
|
+
...styles.starIcon,
|
|
96
|
+
display: isDefaultActive ? 'flex' : 'none',
|
|
97
|
+
}, iconBackground: { color: colors.white }, uiTransform: {
|
|
98
|
+
...styles.toggleButton,
|
|
99
|
+
...(isDefaultActive ? { borderColor: colors.transparent } : {}),
|
|
100
|
+
}, onMouseDown: () => {
|
|
101
|
+
if (!isDefaultActive) {
|
|
102
|
+
onSetDefault();
|
|
103
|
+
}
|
|
104
|
+
} })),
|
|
105
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.listContainer },
|
|
106
|
+
participants.length === 0 && (ReactEcs.createElement(UiEntity, { uiTransform: styles.messageContainer },
|
|
107
|
+
ReactEcs.createElement(Label, { value: "No current active participants in the Cast", fontSize: 16, color: colors.gray }))),
|
|
108
|
+
currentPageParticipants.map(participant => (ReactEcs.createElement(ParticipantRow, { key: participant.identity, participant: participant, activeTrackSid: activeTrackSid, isHovered: hoveredDropdown === participant.identity, onSelectTrack: onSelectTrack, onHoverEnter: () => setHoveredDropdown(participant.identity), onHoverLeave: () => setHoveredDropdown(undefined) })))))));
|
|
109
|
+
}
|
|
110
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SpeakerShowcase.js","sourceRoot":"","sources":["../../../../src/admin-toolkit-ui/VideoControl/DclCast/SpeakerShowcase.tsx"],"names":[],"mappings":"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","sourcesContent":["import ReactEcs, { Dropdown, UiEntity, Label } from '@dcl/react-ecs';\nimport { getContentUrl } from '../../constants';\nimport { Button } from '../../Button';\nimport { Modal } from '../../Modal';\nimport { getSourceLabel, isPresentationBot, type FlattenedTrack, type Participant } from '../api';\nimport {\n  getSpeakerShowcaseStyles,\n  getShowcaseColors,\n  getShowcaseBackgrounds,\n  getShowcaseIconBackgrounds,\n  getPaginationColor,\n  SHOWCASE_DROPDOWN_COLORS,\n  SHOWCASE_PAGE_INDICATOR_COLOR,\n} from './styles';\n\nconst SPEAKERS_PER_PAGE = 10;\n\nconst ICONS = {\n  get BACK() {\n    return `${getContentUrl()}/admin_toolkit/assets/icons/chevron-back.png`;\n  },\n  get NEXT() {\n    return `${getContentUrl()}/admin_toolkit/assets/icons/chevron-forward.png`;\n  },\n  get STAR() {\n    return `${getContentUrl()}/admin_toolkit/assets/icons/star.png`;\n  },\n};\n\ntype SpeakerShowcaseProps = {\n  participants: Participant[];\n  activeTrackSid: string | undefined;\n  onSelectTrack: (track: FlattenedTrack) => void;\n  onSetDefault: () => void;\n  onClose: () => void;\n};\n\nfunction getDropdownOptions(participant: Participant): string[] {\n  return participant.tracks.map(track => `${getSourceLabel(track.sourceType)}`);\n}\n\nfunction getActiveIndex(participant: Participant, activeTrackSid: string | undefined): number {\n  if (!activeTrackSid) return -1;\n  return participant.tracks.findIndex(t => t.sid === activeTrackSid);\n}\n\nfunction ParticipantRow({\n  participant,\n  activeTrackSid,\n  isHovered,\n  onSelectTrack,\n  onHoverEnter,\n  onHoverLeave,\n}: {\n  key?: string;\n  participant: Participant;\n  activeTrackSid: string | undefined;\n  isHovered: boolean;\n  onSelectTrack: (track: FlattenedTrack) => void;\n  onHoverEnter: () => void;\n  onHoverLeave: () => void;\n}) {\n  const styles = getSpeakerShowcaseStyles();\n  const backgrounds = getShowcaseBackgrounds();\n  const colors = getShowcaseColors();\n  const iconBgs = getShowcaseIconBackgrounds();\n\n  const isBot = isPresentationBot(participant.name);\n  const displayName = isBot ? 'Presentation' : participant.name;\n  const activeIndex = getActiveIndex(participant, activeTrackSid);\n  const isActive = activeIndex !== -1;\n  const options = getDropdownOptions(participant);\n  const displayLabel = isActive\n    ? getSourceLabel(participant.tracks[activeIndex].sourceType)\n    : 'Showcase';\n\n  const dropdownTextColor = isHovered\n    ? SHOWCASE_DROPDOWN_COLORS.hover\n    : isActive\n      ? SHOWCASE_DROPDOWN_COLORS.active\n      : SHOWCASE_DROPDOWN_COLORS.idle;\n\n  const dropdownBgColor = isHovered\n    ? SHOWCASE_DROPDOWN_COLORS.hoverBg\n    : SHOWCASE_DROPDOWN_COLORS.transparentBg;\n\n  return (\n    <UiEntity uiTransform={styles.userItem}>\n      <UiEntity uiTransform={styles.userRow}>\n        <UiEntity uiTransform={styles.userInfo}>\n          <UiEntity uiTransform={styles.personIconContainer}>\n            <UiEntity\n              uiTransform={styles.personIcon}\n              uiBackground={backgrounds.personIcon}\n            />\n          </UiEntity>\n          <UiEntity uiTransform={styles.userDetails}>\n            <Label\n              value={`<b>${displayName}</b>`}\n              fontSize={14}\n              color={colors.white}\n            />\n          </UiEntity>\n        </UiEntity>\n        <UiEntity uiTransform={styles.rowCenter}>\n          {isBot ? (\n            <Button\n              id={`showcase-activate-${participant.identity}`}\n              value={isActive ? '<b>Active</b>' : '<b>Activate</b>'}\n              variant={isActive ? 'primary' : 'secondary'}\n              disabled={isActive}\n              fontSize={14}\n              color={colors.white}\n              uiTransform={styles.dropdownTransform}\n              onMouseDown={() => {\n                const track = participant.tracks[0];\n                if (track) {\n                  onSelectTrack(track);\n                }\n              }}\n            />\n          ) : (\n            <UiEntity uiTransform={styles.rowCenter}>\n              <UiEntity\n                uiTransform={{ ...styles.starIcon, display: isActive ? 'flex' : 'none' }}\n                uiBackground={iconBgs.star}\n              />\n              <UiEntity\n                uiTransform={styles.dropdownWrapper}\n                onMouseEnter={onHoverEnter}\n                onMouseLeave={onHoverLeave}\n              >\n                <Dropdown\n                  key={`showcase-dropdown-${participant.identity}-${activeTrackSid ?? 'none'}`}\n                  acceptEmpty\n                  emptyLabel={displayLabel}\n                  options={options}\n                  selectedIndex={-1}\n                  onChange={(optionIndex: number) => {\n                    const track = participant.tracks[optionIndex];\n                    if (track) {\n                      onSelectTrack(track);\n                    }\n                  }}\n                  textAlign=\"middle-left\"\n                  fontSize={14}\n                  uiTransform={styles.dropdownTransform}\n                  uiBackground={{ color: dropdownBgColor }}\n                  color={dropdownTextColor}\n                />\n              </UiEntity>\n            </UiEntity>\n          )}\n        </UiEntity>\n      </UiEntity>\n      <UiEntity\n        uiTransform={styles.divider}\n        uiBackground={backgrounds.divider}\n      />\n    </UiEntity>\n  );\n}\n\nexport function SpeakerShowcase({\n  participants,\n  activeTrackSid,\n  onSelectTrack,\n  onSetDefault,\n  onClose,\n}: SpeakerShowcaseProps) {\n  const [page, setPage] = ReactEcs.useState(1);\n  const [hoveredDropdown, setHoveredDropdown] = ReactEcs.useState<string | undefined>(undefined);\n  const isDefaultActive = !activeTrackSid;\n\n  const styles = getSpeakerShowcaseStyles();\n  const colors = getShowcaseColors();\n  const iconBgs = getShowcaseIconBackgrounds();\n\n  const totalPages = Math.ceil(participants.length / SPEAKERS_PER_PAGE);\n  const startIndex = (page - 1) * SPEAKERS_PER_PAGE;\n  const endIndex = Math.min(startIndex + SPEAKERS_PER_PAGE, participants.length);\n  const currentPageParticipants = participants.slice(startIndex, endIndex);\n\n  const paginationFooter = participants.length > SPEAKERS_PER_PAGE && (\n    <UiEntity uiTransform={styles.pagination}>\n      <Button\n        id=\"showcase-prev\"\n        value=\"Prev\"\n        variant=\"secondary\"\n        disabled={page <= 1}\n        fontSize={18}\n        icon={ICONS.BACK}\n        iconTransform={styles.prevIcon}\n        iconBackground={{ color: getPaginationColor(page <= 1) }}\n        color={getPaginationColor(page <= 1)}\n        labelTransform={styles.prevLabel}\n        uiTransform={styles.paginationButton}\n        onMouseDown={() => setPage(page - 1)}\n      />\n      <Label\n        value={`Page ${page}/${totalPages}`}\n        fontSize={14}\n        color={SHOWCASE_PAGE_INDICATOR_COLOR}\n      />\n      <Button\n        id=\"showcase-next\"\n        value=\"<b>Next</b>\"\n        variant=\"secondary\"\n        fontSize={18}\n        iconRight={ICONS.NEXT}\n        iconRightTransform={styles.nextIcon}\n        labelTransform={styles.nextLabel}\n        iconRightBackground={{\n          color: getPaginationColor(page >= totalPages),\n        }}\n        color={getPaginationColor(page >= totalPages)}\n        disabled={page >= totalPages}\n        uiTransform={styles.paginationButton}\n        onMouseDown={() => setPage(page + 1)}\n      />\n    </UiEntity>\n  );\n\n  return (\n    <Modal\n      id=\"showcase\"\n      title=\"SPEAKER SHOWCASE\"\n      titleFontSize={20}\n      headerIcon={iconBgs.showcase}\n      headerIconSize={24}\n      headerMarginBottom={16}\n      counterText={`(${participants.length} Speakers)`}\n      counterFontSize={14}\n      width={650}\n      height={650}\n      padding={16}\n      onClose={onClose}\n      footer={paginationFooter || undefined}\n    >\n      <UiEntity uiTransform={{ flexDirection: 'column', width: '100%' }}>\n        <UiEntity\n          uiTransform={styles.toggleRow}\n          uiBackground={{ color: colors.softBlack }}\n        >\n          <UiEntity uiTransform={{ flexDirection: 'column' }}>\n            <Label\n              value=\"Automatic Showcase\"\n              fontSize={14}\n              color={colors.white}\n            />\n            <Label\n              value=\"Speakers will be automatically featured when they speak\"\n              fontSize={10}\n              color={colors.gray}\n            />\n          </UiEntity>\n          <Button\n            id=\"showcase-default-speaker\"\n            value={isDefaultActive ? 'Active' : 'Turn On'}\n            variant=\"secondary\"\n            disabled={isDefaultActive}\n            fontSize={14}\n            color={colors.white}\n            icon={ICONS.STAR}\n            iconTransform={{\n              ...styles.starIcon,\n              display: isDefaultActive ? 'flex' : 'none',\n            }}\n            iconBackground={{ color: colors.white }}\n            uiTransform={{\n              ...styles.toggleButton,\n              ...(isDefaultActive ? { borderColor: colors.transparent } : {}),\n            }}\n            onMouseDown={() => {\n              if (!isDefaultActive) {\n                onSetDefault();\n              }\n            }}\n          />\n        </UiEntity>\n\n        <UiEntity uiTransform={styles.listContainer}>\n          {participants.length === 0 && (\n            <UiEntity uiTransform={styles.messageContainer}>\n              <Label\n                value=\"No current active participants in the Cast\"\n                fontSize={16}\n                color={colors.gray}\n              />\n            </UiEntity>\n          )}\n          {currentPageParticipants.map(participant => (\n            <ParticipantRow\n              key={participant.identity}\n              participant={participant}\n              activeTrackSid={activeTrackSid}\n              isHovered={hoveredDropdown === participant.identity}\n              onSelectTrack={onSelectTrack}\n              onHoverEnter={() => setHoveredDropdown(participant.identity)}\n              onHoverLeave={() => setHoveredDropdown(undefined)}\n            />\n          ))}\n        </UiEntity>\n      </UiEntity>\n    </Modal>\n  );\n}\n"]}
|
|
@@ -1,11 +1,25 @@
|
|
|
1
1
|
import ReactEcs from '@dcl/react-ecs';
|
|
2
|
-
import { DeepReadonlyObject, Entity, IEngine, PBVideoPlayer } from '@dcl/ecs';
|
|
3
|
-
import { State } from '../../types';
|
|
2
|
+
import type { DeepReadonlyObject, Entity, IEngine, PBVideoPlayer } from '@dcl/ecs';
|
|
3
|
+
import type { State } from '../../types';
|
|
4
|
+
import { type FlattenedTrack, type Participant } from '../api';
|
|
5
|
+
export declare const showcaseState: {
|
|
6
|
+
show: boolean;
|
|
7
|
+
participants: Participant[];
|
|
8
|
+
activeTrackSid: string | undefined;
|
|
9
|
+
onSelectTrack: ((track: FlattenedTrack) => void) | undefined;
|
|
10
|
+
onSetDefault: (() => void) | undefined;
|
|
11
|
+
onClose: (() => void) | undefined;
|
|
12
|
+
};
|
|
13
|
+
export declare const sharePresentationState: {
|
|
14
|
+
show: boolean;
|
|
15
|
+
onClose: (() => void) | undefined;
|
|
16
|
+
};
|
|
4
17
|
export declare function handleGetDclCastInfo(state: State): Promise<import("../api").DclCastResponse | null | undefined>;
|
|
5
|
-
declare const DclCast: ({ engine, state, entity, video, }: {
|
|
18
|
+
declare const DclCast: ({ engine, state, entity, video, playerAddress, }: {
|
|
6
19
|
engine: IEngine;
|
|
7
20
|
state: State;
|
|
8
21
|
entity: Entity;
|
|
9
22
|
video: DeepReadonlyObject<PBVideoPlayer> | undefined;
|
|
23
|
+
playerAddress: string | undefined;
|
|
10
24
|
}) => ReactEcs.JSX.Element;
|
|
11
25
|
export default DclCast;
|
|
@@ -1,16 +1,41 @@
|
|
|
1
1
|
import ReactEcs, { UiEntity } from '@dcl/react-ecs';
|
|
2
2
|
import { Color4 } from '@dcl/sdk/math';
|
|
3
|
-
import {
|
|
4
|
-
import {
|
|
5
|
-
import {
|
|
6
|
-
import DclCastInfo from './DclCastInfo';
|
|
7
|
-
import { LoadingDots } from '../../Loading';
|
|
3
|
+
import { getComponents } from '../../../definitions';
|
|
4
|
+
import { getContentUrl } from '../../constants';
|
|
5
|
+
import { setInterval, clearInterval } from '../../utils';
|
|
8
6
|
import { Button } from '../../Button';
|
|
7
|
+
import { LoadingDots } from '../../Loading';
|
|
8
|
+
import { nextTickFunctions } from '../..';
|
|
9
|
+
import { LIVEKIT_STREAM_SRC } from '../LiveStream';
|
|
10
|
+
import { getDclCastInfo, getActiveStreams, groupTracksByParticipant, resetStreamKey, hasPresentationTrack, subscribeToPresentationTopic, consumePresentationMessages, ensurePresenterRole, } from '../api';
|
|
11
|
+
import { createVideoPlayerControls, isDclCast } from '../utils';
|
|
12
|
+
import DclCastInfo from './DclCastInfo';
|
|
13
|
+
import CompactDclCast from './CompactDclCast';
|
|
9
14
|
import { getDclCastStyles, getDclCastColors } from './styles';
|
|
10
|
-
import { getComponents } from '../../../definitions';
|
|
11
15
|
const ICONS = {
|
|
12
|
-
DCL_CAST_ICON
|
|
16
|
+
get DCL_CAST_ICON() {
|
|
17
|
+
return `${getContentUrl()}/admin_toolkit/assets/icons/video-control-dcl-cast.png`;
|
|
18
|
+
},
|
|
19
|
+
get CHEVRON_UP() {
|
|
20
|
+
return `${getContentUrl()}/admin_toolkit/assets/icons/chevron-up.png`;
|
|
21
|
+
},
|
|
22
|
+
};
|
|
23
|
+
export const showcaseState = {
|
|
24
|
+
show: false,
|
|
25
|
+
participants: [],
|
|
26
|
+
activeTrackSid: undefined,
|
|
27
|
+
onSelectTrack: undefined,
|
|
28
|
+
onSetDefault: undefined,
|
|
29
|
+
onClose: undefined,
|
|
30
|
+
};
|
|
31
|
+
export const sharePresentationState = {
|
|
32
|
+
show: false,
|
|
33
|
+
onClose: undefined,
|
|
13
34
|
};
|
|
35
|
+
let presentationSubscribed = false;
|
|
36
|
+
let consuming = false;
|
|
37
|
+
let presentationSystem = null;
|
|
38
|
+
let participantPollingSystem = null;
|
|
14
39
|
export async function handleGetDclCastInfo(state) {
|
|
15
40
|
const [error, data] = await getDclCastInfo();
|
|
16
41
|
if (error) {
|
|
@@ -24,22 +49,119 @@ export async function handleGetDclCastInfo(state) {
|
|
|
24
49
|
}
|
|
25
50
|
}
|
|
26
51
|
}
|
|
27
|
-
|
|
52
|
+
function startPresentationSystem(engine, state) {
|
|
53
|
+
if (presentationSystem)
|
|
54
|
+
return;
|
|
55
|
+
presentationSubscribed = true;
|
|
56
|
+
subscribeToPresentationTopic();
|
|
57
|
+
const system = () => {
|
|
58
|
+
if (consuming)
|
|
59
|
+
return;
|
|
60
|
+
consuming = true;
|
|
61
|
+
consumePresentationMessages()
|
|
62
|
+
.then(latestState => {
|
|
63
|
+
if (latestState === 'stopped') {
|
|
64
|
+
state.videoControl.presentationState = undefined;
|
|
65
|
+
}
|
|
66
|
+
else if (latestState) {
|
|
67
|
+
state.videoControl.presentationState = latestState;
|
|
68
|
+
}
|
|
69
|
+
})
|
|
70
|
+
.catch(() => {
|
|
71
|
+
console.log('[DclCast] Failed to consume presentation messages');
|
|
72
|
+
})
|
|
73
|
+
.finally(() => {
|
|
74
|
+
consuming = false;
|
|
75
|
+
});
|
|
76
|
+
};
|
|
77
|
+
engine.addSystem(system);
|
|
78
|
+
presentationSystem = system;
|
|
79
|
+
}
|
|
80
|
+
function stopPresentationSystem(engine, state) {
|
|
81
|
+
if (presentationSystem) {
|
|
82
|
+
engine.removeSystem(presentationSystem);
|
|
83
|
+
presentationSystem = null;
|
|
84
|
+
}
|
|
85
|
+
state.videoControl.presentationState = undefined;
|
|
86
|
+
presentationSubscribed = false;
|
|
87
|
+
}
|
|
88
|
+
function startParticipantPolling(engine, state) {
|
|
89
|
+
if (participantPollingSystem)
|
|
90
|
+
return;
|
|
91
|
+
const poll = async () => {
|
|
92
|
+
const tracks = await getActiveStreams();
|
|
93
|
+
if (!tracks)
|
|
94
|
+
return;
|
|
95
|
+
showcaseState.participants = groupTracksByParticipant(tracks);
|
|
96
|
+
const hasPresentation = hasPresentationTrack(tracks);
|
|
97
|
+
if (hasPresentation && !presentationSubscribed) {
|
|
98
|
+
startPresentationSystem(engine, state);
|
|
99
|
+
}
|
|
100
|
+
else if (!hasPresentation && presentationSubscribed) {
|
|
101
|
+
stopPresentationSystem(engine, state);
|
|
102
|
+
}
|
|
103
|
+
};
|
|
104
|
+
poll();
|
|
105
|
+
participantPollingSystem = setInterval(engine, poll, 5000);
|
|
106
|
+
}
|
|
107
|
+
function stopParticipantPolling(engine, state) {
|
|
108
|
+
if (participantPollingSystem) {
|
|
109
|
+
clearInterval(engine, participantPollingSystem);
|
|
110
|
+
participantPollingSystem = null;
|
|
111
|
+
}
|
|
112
|
+
stopPresentationSystem(engine, state);
|
|
113
|
+
}
|
|
114
|
+
const DclCast = ({ engine, state, entity, video, playerAddress, }) => {
|
|
28
115
|
const { VideoControlState } = getComponents(engine);
|
|
116
|
+
const controls = createVideoPlayerControls(entity, engine);
|
|
29
117
|
const styles = getDclCastStyles();
|
|
30
118
|
const colors = getDclCastColors();
|
|
31
119
|
const [isLoading, setIsLoading] = ReactEcs.useState(false);
|
|
32
120
|
const [error, setError] = ReactEcs.useState(false);
|
|
121
|
+
const onShowShowcaseModal = async () => {
|
|
122
|
+
const latestTracks = await getActiveStreams();
|
|
123
|
+
if (!latestTracks)
|
|
124
|
+
return;
|
|
125
|
+
console.log('[DclCast] Active tracks:', JSON.stringify(latestTracks, null, 2));
|
|
126
|
+
const closeModal = () => {
|
|
127
|
+
showcaseState.show = false;
|
|
128
|
+
};
|
|
129
|
+
showcaseState.participants = groupTracksByParticipant(latestTracks);
|
|
130
|
+
showcaseState.onSelectTrack = (track) => {
|
|
131
|
+
controls.setSource(track.sid);
|
|
132
|
+
showcaseState.activeTrackSid = track.sid;
|
|
133
|
+
state.videoControl.selectedStream = 'dcl-cast';
|
|
134
|
+
};
|
|
135
|
+
showcaseState.onSetDefault = () => {
|
|
136
|
+
controls.setSource(LIVEKIT_STREAM_SRC);
|
|
137
|
+
showcaseState.activeTrackSid = undefined;
|
|
138
|
+
state.videoControl.selectedStream = 'dcl-cast';
|
|
139
|
+
};
|
|
140
|
+
showcaseState.onClose = closeModal;
|
|
141
|
+
nextTickFunctions.push(() => {
|
|
142
|
+
showcaseState.show = true;
|
|
143
|
+
});
|
|
144
|
+
};
|
|
33
145
|
const fetchDclCastInfo = async () => {
|
|
34
|
-
console.log('fetching DCL cast info...');
|
|
35
146
|
setIsLoading(true);
|
|
36
147
|
setError(false);
|
|
37
148
|
const result = await handleGetDclCastInfo(state);
|
|
38
149
|
if (!result) {
|
|
39
150
|
setError(true);
|
|
40
151
|
}
|
|
152
|
+
else if (video?.src?.startsWith('livekit-video://') && !state.videoControl.selectedStream) {
|
|
153
|
+
state.videoControl.selectedStream = 'dcl-cast';
|
|
154
|
+
}
|
|
41
155
|
setIsLoading(false);
|
|
42
156
|
};
|
|
157
|
+
const onSharePresentation = () => {
|
|
158
|
+
sharePresentationState.onClose = () => {
|
|
159
|
+
sharePresentationState.show = false;
|
|
160
|
+
};
|
|
161
|
+
nextTickFunctions.push(() => {
|
|
162
|
+
sharePresentationState.show = true;
|
|
163
|
+
});
|
|
164
|
+
};
|
|
43
165
|
const handleResetRoomId = async () => {
|
|
44
166
|
setIsLoading(true);
|
|
45
167
|
const [error, data] = await resetStreamKey();
|
|
@@ -55,33 +177,84 @@ const DclCast = ({ engine, state, entity, video, }) => {
|
|
|
55
177
|
};
|
|
56
178
|
ReactEcs.useEffect(() => {
|
|
57
179
|
fetchDclCastInfo();
|
|
180
|
+
if (playerAddress) {
|
|
181
|
+
ensurePresenterRole(playerAddress);
|
|
182
|
+
}
|
|
58
183
|
}, []);
|
|
184
|
+
const videoSrc = video?.src;
|
|
185
|
+
ReactEcs.useEffect(() => {
|
|
186
|
+
if (videoSrc?.startsWith('livekit-video://') && !state.videoControl.selectedStream) {
|
|
187
|
+
state.videoControl.selectedStream = 'dcl-cast';
|
|
188
|
+
}
|
|
189
|
+
}, [videoSrc]);
|
|
190
|
+
const isCastActive = !!(video?.src && isDclCast(video.src));
|
|
191
|
+
ReactEcs.useEffect(() => {
|
|
192
|
+
if (isCastActive) {
|
|
193
|
+
startParticipantPolling(engine, state);
|
|
194
|
+
}
|
|
195
|
+
else {
|
|
196
|
+
stopParticipantPolling(engine, state);
|
|
197
|
+
}
|
|
198
|
+
}, [isCastActive]);
|
|
199
|
+
ReactEcs.useEffect(() => {
|
|
200
|
+
if (state.videoControl.presentationState) {
|
|
201
|
+
state.videoControl.isMinimized = true;
|
|
202
|
+
}
|
|
203
|
+
}, [!!state.videoControl.presentationState]);
|
|
204
|
+
const isMinimized = state.videoControl.isMinimized;
|
|
59
205
|
return (ReactEcs.createElement(UiEntity, { uiTransform: styles.fullContainer },
|
|
60
|
-
ReactEcs.createElement(
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
206
|
+
ReactEcs.createElement(UiEntity, { uiTransform: { display: isMinimized ? 'flex' : 'none', width: '100%' } },
|
|
207
|
+
ReactEcs.createElement(CompactDclCast, { engine: engine, state: state, entity: entity, video: video, onShowShowcaseModal: onShowShowcaseModal })),
|
|
208
|
+
ReactEcs.createElement(UiEntity, { uiTransform: {
|
|
209
|
+
display: isMinimized ? 'none' : 'flex',
|
|
210
|
+
flexDirection: 'column',
|
|
211
|
+
width: '100%',
|
|
212
|
+
} },
|
|
213
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.rowCenterSpaceBetween },
|
|
214
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.rowCenter },
|
|
215
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.headerIcon, uiBackground: {
|
|
216
|
+
textureMode: 'stretch',
|
|
217
|
+
texture: { src: ICONS.DCL_CAST_ICON },
|
|
218
|
+
} }),
|
|
219
|
+
ReactEcs.createElement(UiEntity, { uiText: {
|
|
220
|
+
value: '<b>DCL Cast</b>',
|
|
221
|
+
fontSize: 24,
|
|
222
|
+
color: Color4.White(),
|
|
223
|
+
textAlign: 'middle-left',
|
|
224
|
+
}, uiTransform: { margin: { left: 10 } } })),
|
|
225
|
+
ReactEcs.createElement(UiEntity, { onMouseDown: () => {
|
|
226
|
+
state.videoControl.isMinimized = true;
|
|
227
|
+
}, uiTransform: styles.chevronButton, uiBackground: {
|
|
228
|
+
textureMode: 'stretch',
|
|
229
|
+
color: Color4.White(),
|
|
230
|
+
texture: {
|
|
231
|
+
src: ICONS.CHEVRON_UP,
|
|
232
|
+
},
|
|
233
|
+
} })),
|
|
234
|
+
ReactEcs.createElement(UiEntity, { uiTransform: styles.fullWidthWithBottomMargin },
|
|
235
|
+
ReactEcs.createElement(UiEntity, { uiText: {
|
|
236
|
+
value: 'Use a browser-based DCL Cast room to easily stream camera and screen feed to a screen in your scene.',
|
|
237
|
+
fontSize: 16,
|
|
238
|
+
color: Color4.fromHexString('#A09BA8'),
|
|
239
|
+
textAlign: 'top-left',
|
|
240
|
+
textWrap: 'wrap',
|
|
241
|
+
}, uiTransform: styles.marginBottomSmall })),
|
|
242
|
+
isLoading && (ReactEcs.createElement(LoadingDots, { uiTransform: styles.loadingContainer, engine: engine })),
|
|
243
|
+
error && (ReactEcs.createElement(UiEntity, { uiTransform: styles.columnCentered },
|
|
244
|
+
ReactEcs.createElement(UiEntity, { uiText: {
|
|
245
|
+
value: '<b>Failed to fetch DCL Cast info</b>',
|
|
246
|
+
fontSize: 16,
|
|
247
|
+
color: Color4.White(),
|
|
248
|
+
}, uiTransform: styles.marginBottomSmall }),
|
|
249
|
+
ReactEcs.createElement(UiEntity, { uiText: {
|
|
250
|
+
value: 'Please retry.',
|
|
251
|
+
fontSize: 16,
|
|
252
|
+
color: Color4.Gray(),
|
|
253
|
+
} }),
|
|
254
|
+
ReactEcs.createElement(Button, { id: "dcl_cast_retry", value: "<b>Retry</b>", variant: "secondary", fontSize: 16, color: colors.white, onMouseDown: () => {
|
|
255
|
+
handleGetDclCastInfo(state);
|
|
256
|
+
}, uiTransform: styles.retryButton }))),
|
|
257
|
+
!isLoading && !error && (ReactEcs.createElement(DclCastInfo, { state: state, entity: entity, engine: engine, video: video, onResetRoomId: handleResetRoomId, onShowShowcaseModal: onShowShowcaseModal, onSharePresentation: onSharePresentation })))));
|
|
85
258
|
};
|
|
86
259
|
export default DclCast;
|
|
87
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/admin-toolkit-ui/VideoControl/DclCast/index.tsx"],"names":[],"mappings":"AAAA,OAAO,QAAQ,EAAE,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAEpD,OAAO,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAEvC,OAAO,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,QAAQ,CAAC;AACxD,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAG9C,OAAO,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AACtC,OAAO,WAAW,MAAM,eAAe,CAAC;AACxC,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAC5C,OAAO,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AACtC,OAAO,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,MAAM,UAAU,CAAC;AAC9D,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAErD,MAAM,KAAK,GAAG;IACZ,aAAa,EAAE,GAAG,WAAW,wDAAwD;CACtF,CAAC;AAEF,MAAM,CAAC,KAAK,UAAU,oBAAoB,CAAC,KAAY;IACrD,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,GAAG,MAAM,cAAc,EAAE,CAAC;IAC7C,IAAI,KAAK,EAAE,CAAC;QACV,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACrB,OAAO,IAAI,CAAC;IACd,CAAC;SAAM,CAAC;QACN,IAAI,IAAI,EAAE,CAAC;YACT,KAAK,CAAC,YAAY,CAAC,OAAO,GAAG,IAAI,CAAC;YAClC,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;AACH,CAAC;AAED,MAAM,OAAO,GAAG,CAAC,EACf,MAAM,EACN,KAAK,EACL,MAAM,EACN,KAAK,GAMN,EAAE,EAAE;IACH,MAAM,EAAE,iBAAiB,EAAE,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC;IACpD,MAAM,MAAM,GAAG,gBAAgB,EAAE,CAAC;IAClC,MAAM,MAAM,GAAG,gBAAgB,EAAE,CAAC;IAClC,MAAM,CAAC,SAAS,EAAE,YAAY,CAAC,GAAG,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC3D,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IAEnD,MAAM,gBAAgB,GAAG,KAAK,IAAI,EAAE;QAClC,OAAO,CAAC,GAAG,CAAC,2BAA2B,CAAC,CAAC;QACzC,YAAY,CAAC,IAAI,CAAC,CAAC;QACnB,QAAQ,CAAC,KAAK,CAAC,CAAC;QAEhB,MAAM,MAAM,GAAG,MAAM,oBAAoB,CAAC,KAAK,CAAC,CAAC;QAEjD,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,QAAQ,CAAC,IAAI,CAAC,CAAC;QACjB,CAAC;QAED,YAAY,CAAC,KAAK,CAAC,CAAC;IACtB,CAAC,CAAC;IAEF,MAAM,iBAAiB,GAAG,KAAK,IAAI,EAAE;QACnC,YAAY,CAAC,IAAI,CAAC,CAAC;QACnB,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,GAAG,MAAM,cAAc,EAAE,CAAC;QAC7C,IAAI,KAAK,EAAE,CAAC;YACV,YAAY,CAAC,KAAK,CAAC,CAAC;YACpB,QAAQ,CAAC,IAAI,CAAC,CAAC;QACjB,CAAC;aAAM,CAAC;YACN,MAAM,YAAY,GAAG,iBAAiB,CAAC,UAAU,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;YAC9E,YAAY,CAAC,MAAM,GAAG,IAAI,EAAE,MAAM,CAAC;YACnC,gBAAgB,EAAE,CAAC;QACrB,CAAC;IACH,CAAC,CAAC;IAEF,QAAQ,CAAC,SAAS,CAAC,GAAG,EAAE;QACtB,gBAAgB,EAAE,CAAC;IACrB,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,OAAO,CACL,uBAAC,QAAQ,IAAC,WAAW,EAAE,MAAM,CAAC,aAAa;QACzC,uBAAC,MAAM,IACL,OAAO,EAAE,KAAK,CAAC,aAAa,EAC5B,KAAK,EAAC,UAAU,GAChB;QACF,uBAAC,QAAQ,IAAC,WAAW,EAAE,MAAM,CAAC,yBAAyB;YACrD,uBAAC,QAAQ,IACP,MAAM,EAAE;oBACN,KAAK,EACH,sGAAsG;oBACxG,QAAQ,EAAE,EAAE;oBACZ,KAAK,EAAE,MAAM,CAAC,aAAa,CAAC,SAAS,CAAC;oBAEtC,SAAS,EAAE,UAAU;oBACrB,QAAQ,EAAE,MAAM;iBACjB,EACD,WAAW,EAAE,MAAM,CAAC,iBAAiB,GACrC,CACO;QACV,SAAS,IAAI,CACZ,uBAAC,WAAW,IACV,WAAW,EAAE,EAAE,SAAS,EAAE,GAAG,EAAE,EAC/B,MAAM,EAAE,MAAM,GACd,CACH;QACA,KAAK,IAAI,CACR,uBAAC,QAAQ,IAAC,WAAW,EAAE,MAAM,CAAC,cAAc;YAC1C,uBAAC,QAAQ,IACP,MAAM,EAAE;oBACN,KAAK,EAAE,sCAAsC;oBAC7C,QAAQ,EAAE,EAAE;oBACZ,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE;iBACtB,EACD,WAAW,EAAE,EAAE,MAAM,EAAE,EAAE,MAAM,EAAE,CAAC,EAAE,EAAE,GACtC;YACF,uBAAC,QAAQ,IACP,MAAM,EAAE;oBACN,KAAK,EAAE,eAAe;oBACtB,QAAQ,EAAE,EAAE;oBACZ,KAAK,EAAE,MAAM,CAAC,IAAI,EAAE;iBACrB,GACD;YACF,uBAAC,MAAM,IACL,EAAE,EAAC,gBAAgB,EACnB,KAAK,EAAC,cAAc,EACpB,OAAO,EAAC,WAAW,EACnB,QAAQ,EAAE,EAAE,EACZ,KAAK,EAAE,MAAM,CAAC,KAAK,EACnB,WAAW,EAAE,GAAG,EAAE;oBAChB,oBAAoB,CAAC,KAAK,CAAC,CAAC;gBAC9B,CAAC,EACD,WAAW,EAAE,MAAM,CAAC,WAAW,GAC/B,CACO,CACZ;QAEA,CAAC,SAAS,IAAI,CAAC,KAAK,IAAI,CACvB,uBAAC,WAAW,IACV,KAAK,EAAE,KAAK,EACZ,MAAM,EAAE,MAAM,EACd,MAAM,EAAE,MAAM,EACd,KAAK,EAAE,KAAK,EACZ,aAAa,EAAE,iBAAiB,GAChC,CACH,CACQ,CACZ,CAAC;AACJ,CAAC,CAAC;AAEF,eAAe,OAAO,CAAC","sourcesContent":["import ReactEcs, { UiEntity } from '@dcl/react-ecs';\nimport { DeepReadonlyObject, Entity, IEngine, PBVideoPlayer } from '@dcl/ecs';\nimport { Color4 } from '@dcl/sdk/math';\n\nimport { getDclCastInfo, resetStreamKey } from '../api';\nimport { CONTENT_URL } from '../../constants';\nimport { State } from '../../types';\n\nimport { Header } from '../../Header';\nimport DclCastInfo from './DclCastInfo';\nimport { LoadingDots } from '../../Loading';\nimport { Button } from '../../Button';\nimport { getDclCastStyles, getDclCastColors } from './styles';\nimport { getComponents } from '../../../definitions';\n\nconst ICONS = {\n  DCL_CAST_ICON: `${CONTENT_URL}/admin_toolkit/assets/icons/video-control-dcl-cast.png`,\n};\n\nexport async function handleGetDclCastInfo(state: State) {\n  const [error, data] = await getDclCastInfo();\n  if (error) {\n    console.error(error);\n    return null;\n  } else {\n    if (data) {\n      state.videoControl.dclCast = data;\n      return data;\n    }\n  }\n}\n\nconst DclCast = ({\n  engine,\n  state,\n  entity,\n  video,\n}: {\n  engine: IEngine;\n  state: State;\n  entity: Entity;\n  video: DeepReadonlyObject<PBVideoPlayer> | undefined;\n}) => {\n  const { VideoControlState } = getComponents(engine);\n  const styles = getDclCastStyles();\n  const colors = getDclCastColors();\n  const [isLoading, setIsLoading] = ReactEcs.useState(false);\n  const [error, setError] = ReactEcs.useState(false);\n\n  const fetchDclCastInfo = async () => {\n    console.log('fetching DCL cast info...');\n    setIsLoading(true);\n    setError(false);\n\n    const result = await handleGetDclCastInfo(state);\n\n    if (!result) {\n      setError(true);\n    }\n\n    setIsLoading(false);\n  };\n\n  const handleResetRoomId = async () => {\n    setIsLoading(true);\n    const [error, data] = await resetStreamKey();\n    if (error) {\n      setIsLoading(false);\n      setError(true);\n    } else {\n      const videoControl = VideoControlState.getMutable(state.adminToolkitUiEntity);\n      videoControl.endsAt = data?.endsAt;\n      fetchDclCastInfo();\n    }\n  };\n\n  ReactEcs.useEffect(() => {\n    fetchDclCastInfo();\n  }, []);\n\n  return (\n    <UiEntity uiTransform={styles.fullContainer}>\n      <Header\n        iconSrc={ICONS.DCL_CAST_ICON}\n        title=\"DCL Cast\"\n      />\n      <UiEntity uiTransform={styles.fullWidthWithBottomMargin}>\n        <UiEntity\n          uiText={{\n            value:\n              'Use a browser-based DCL Cast room to easily stream camera and screen feed to a screen in your scene.',\n            fontSize: 16,\n            color: Color4.fromHexString('#A09BA8'),\n\n            textAlign: 'top-left',\n            textWrap: 'wrap',\n          }}\n          uiTransform={styles.marginBottomSmall}\n        />\n      </UiEntity>\n      {isLoading && (\n        <LoadingDots\n          uiTransform={{ minHeight: 400 }}\n          engine={engine}\n        />\n      )}\n      {error && (\n        <UiEntity uiTransform={styles.columnCentered}>\n          <UiEntity\n            uiText={{\n              value: '<b>Failed to fetch DCL Cast info</b>',\n              fontSize: 16,\n              color: Color4.White(),\n            }}\n            uiTransform={{ margin: { bottom: 8 } }}\n          />\n          <UiEntity\n            uiText={{\n              value: 'Please retry.',\n              fontSize: 16,\n              color: Color4.Gray(),\n            }}\n          />\n          <Button\n            id=\"dcl_cast_retry\"\n            value=\"<b>Retry</b>\"\n            variant=\"secondary\"\n            fontSize={16}\n            color={colors.white}\n            onMouseDown={() => {\n              handleGetDclCastInfo(state);\n            }}\n            uiTransform={styles.retryButton}\n          />\n        </UiEntity>\n      )}\n\n      {!isLoading && !error && (\n        <DclCastInfo\n          state={state}\n          entity={entity}\n          engine={engine}\n          video={video}\n          onResetRoomId={handleResetRoomId}\n        />\n      )}\n    </UiEntity>\n  );\n};\n\nexport default DclCast;\n"]}
|
|
260
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/admin-toolkit-ui/VideoControl/DclCast/index.tsx"],"names":[],"mappings":"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","sourcesContent":["import ReactEcs, { UiEntity } from '@dcl/react-ecs';\nimport type { DeepReadonlyObject, Entity, IEngine, PBVideoPlayer } from '@dcl/ecs';\nimport { Color4 } from '@dcl/sdk/math';\nimport { getComponents } from '../../../definitions';\nimport { getContentUrl } from '../../constants';\nimport type { State } from '../../types';\nimport { setInterval, clearInterval } from '../../utils';\nimport { Button } from '../../Button';\nimport { LoadingDots } from '../../Loading';\nimport { nextTickFunctions } from '../..';\nimport { LIVEKIT_STREAM_SRC } from '../LiveStream';\nimport {\n  getDclCastInfo,\n  getActiveStreams,\n  groupTracksByParticipant,\n  resetStreamKey,\n  hasPresentationTrack,\n  subscribeToPresentationTopic,\n  consumePresentationMessages,\n  ensurePresenterRole,\n  type FlattenedTrack,\n  type Participant,\n} from '../api';\nimport { createVideoPlayerControls, isDclCast } from '../utils';\nimport DclCastInfo from './DclCastInfo';\nimport CompactDclCast from './CompactDclCast';\nimport { getDclCastStyles, getDclCastColors } from './styles';\n\nconst ICONS = {\n  get DCL_CAST_ICON() {\n    return `${getContentUrl()}/admin_toolkit/assets/icons/video-control-dcl-cast.png`;\n  },\n  get CHEVRON_UP() {\n    return `${getContentUrl()}/admin_toolkit/assets/icons/chevron-up.png`;\n  },\n};\n\nexport const showcaseState: {\n  show: boolean;\n  participants: Participant[];\n  activeTrackSid: string | undefined;\n  onSelectTrack: ((track: FlattenedTrack) => void) | undefined;\n  onSetDefault: (() => void) | undefined;\n  onClose: (() => void) | undefined;\n} = {\n  show: false,\n  participants: [],\n  activeTrackSid: undefined,\n  onSelectTrack: undefined,\n  onSetDefault: undefined,\n  onClose: undefined,\n};\n\nexport const sharePresentationState: {\n  show: boolean;\n  onClose: (() => void) | undefined;\n} = {\n  show: false,\n  onClose: undefined,\n};\n\nlet presentationSubscribed = false;\nlet consuming = false;\nlet presentationSystem: (() => void) | null = null;\nlet participantPollingSystem: ((dt: number) => void) | null = null;\n\nexport async function handleGetDclCastInfo(state: State) {\n  const [error, data] = await getDclCastInfo();\n  if (error) {\n    console.error(error);\n    return null;\n  } else {\n    if (data) {\n      state.videoControl.dclCast = data;\n      return data;\n    }\n  }\n}\n\nfunction startPresentationSystem(engine: IEngine, state: State): void {\n  if (presentationSystem) return; // Already running\n  presentationSubscribed = true;\n  subscribeToPresentationTopic();\n\n  const system = () => {\n    if (consuming) return; // Prevent concurrent requests\n    consuming = true;\n    consumePresentationMessages()\n      .then(latestState => {\n        if (latestState === 'stopped') {\n          state.videoControl.presentationState = undefined;\n        } else if (latestState) {\n          state.videoControl.presentationState = latestState;\n        }\n      })\n      .catch(() => {\n        // Silently ignore — will retry next frame\n        console.log('[DclCast] Failed to consume presentation messages');\n      })\n      .finally(() => {\n        consuming = false;\n      });\n  };\n\n  engine.addSystem(system);\n  presentationSystem = system;\n}\n\nfunction stopPresentationSystem(engine: IEngine, state: State): void {\n  if (presentationSystem) {\n    engine.removeSystem(presentationSystem);\n    presentationSystem = null;\n  }\n  state.videoControl.presentationState = undefined;\n  presentationSubscribed = false;\n}\n\nfunction startParticipantPolling(engine: IEngine, state: State): void {\n  if (participantPollingSystem) return; // Already polling\n\n  const poll = async () => {\n    const tracks = await getActiveStreams();\n    if (!tracks) return;\n\n    // Debug: log tracks on each poll to review presentation bot metadata\n    // console.log('[DclCast] Poll tracks:', JSON.stringify(tracks, null, 2));\n\n    // Keep showcase modal data fresh\n    showcaseState.participants = groupTracksByParticipant(tracks);\n\n    // Check for presentation track\n    const hasPresentation = hasPresentationTrack(tracks);\n\n    // console.log(`[DclCast] Presentation track ${hasPresentation ? 'found' : 'not found'} in poll`);\n\n    if (hasPresentation && !presentationSubscribed) {\n      startPresentationSystem(engine, state);\n    } else if (!hasPresentation && presentationSubscribed) {\n      stopPresentationSystem(engine, state);\n    }\n  };\n\n  // Poll immediately, then every 5 seconds\n  poll();\n  participantPollingSystem = setInterval(engine, poll, 5000);\n}\n\nfunction stopParticipantPolling(engine: IEngine, state: State): void {\n  if (participantPollingSystem) {\n    clearInterval(engine, participantPollingSystem);\n    participantPollingSystem = null;\n  }\n  stopPresentationSystem(engine, state);\n}\n\nconst DclCast = ({\n  engine,\n  state,\n  entity,\n  video,\n  playerAddress,\n}: {\n  engine: IEngine;\n  state: State;\n  entity: Entity;\n  video: DeepReadonlyObject<PBVideoPlayer> | undefined;\n  playerAddress: string | undefined;\n}) => {\n  const { VideoControlState } = getComponents(engine);\n  const controls = createVideoPlayerControls(entity, engine);\n  const styles = getDclCastStyles();\n  const colors = getDclCastColors();\n  const [isLoading, setIsLoading] = ReactEcs.useState(false);\n  const [error, setError] = ReactEcs.useState(false);\n\n  const onShowShowcaseModal = async () => {\n    const latestTracks = await getActiveStreams();\n    if (!latestTracks) return;\n\n    // Debug: log all track data to review presentation bot metadata\n    console.log('[DclCast] Active tracks:', JSON.stringify(latestTracks, null, 2));\n\n    const closeModal = () => {\n      showcaseState.show = false;\n    };\n\n    showcaseState.participants = groupTracksByParticipant(latestTracks);\n\n    showcaseState.onSelectTrack = (track: FlattenedTrack) => {\n      controls.setSource(track.sid);\n      showcaseState.activeTrackSid = track.sid;\n      state.videoControl.selectedStream = 'dcl-cast';\n    };\n\n    showcaseState.onSetDefault = () => {\n      controls.setSource(LIVEKIT_STREAM_SRC);\n      showcaseState.activeTrackSid = undefined;\n      state.videoControl.selectedStream = 'dcl-cast';\n    };\n\n    showcaseState.onClose = closeModal;\n\n    nextTickFunctions.push(() => {\n      showcaseState.show = true;\n    });\n  };\n\n  const fetchDclCastInfo = async () => {\n    setIsLoading(true);\n    setError(false);\n\n    const result = await handleGetDclCastInfo(state);\n\n    if (!result) {\n      setError(true);\n    } else if (video?.src?.startsWith('livekit-video://') && !state.videoControl.selectedStream) {\n      state.videoControl.selectedStream = 'dcl-cast';\n    }\n\n    setIsLoading(false);\n  };\n\n  const onSharePresentation = () => {\n    sharePresentationState.onClose = () => {\n      sharePresentationState.show = false;\n    };\n    nextTickFunctions.push(() => {\n      sharePresentationState.show = true;\n    });\n  };\n\n  const handleResetRoomId = async () => {\n    setIsLoading(true);\n    const [error, data] = await resetStreamKey();\n    if (error) {\n      setIsLoading(false);\n      setError(true);\n    } else {\n      const videoControl = VideoControlState.getMutable(state.adminToolkitUiEntity);\n      videoControl.endsAt = data?.endsAt;\n      fetchDclCastInfo();\n    }\n  };\n\n  ReactEcs.useEffect(() => {\n    fetchDclCastInfo();\n    if (playerAddress) {\n      ensurePresenterRole(playerAddress);\n    }\n  }, []);\n\n  // Sync selectedStream when video.src arrives after mount (late-joiner fix)\n  const videoSrc = video?.src;\n  ReactEcs.useEffect(() => {\n    if (videoSrc?.startsWith('livekit-video://') && !state.videoControl.selectedStream) {\n      state.videoControl.selectedStream = 'dcl-cast';\n    }\n  }, [videoSrc]);\n\n  const isCastActive = !!(video?.src && isDclCast(video.src));\n\n  ReactEcs.useEffect(() => {\n    if (isCastActive) {\n      startParticipantPolling(engine, state);\n    } else {\n      stopParticipantPolling(engine, state);\n    }\n  }, [isCastActive]);\n\n  // Auto-minimize when a presentation starts so the compact view\n  // (with presentation controls) is immediately visible\n  ReactEcs.useEffect(() => {\n    if (state.videoControl.presentationState) {\n      state.videoControl.isMinimized = true;\n    }\n  }, [!!state.videoControl.presentationState]);\n\n  const isMinimized = state.videoControl.isMinimized;\n\n  return (\n    <UiEntity uiTransform={styles.fullContainer}>\n      {/* Compact bar — always rendered, toggled via display */}\n      <UiEntity uiTransform={{ display: isMinimized ? 'flex' : 'none', width: '100%' }}>\n        <CompactDclCast\n          engine={engine}\n          state={state}\n          entity={entity}\n          video={video}\n          onShowShowcaseModal={onShowShowcaseModal}\n        />\n      </UiEntity>\n\n      {/* Full panel — always rendered, toggled via display */}\n      <UiEntity\n        uiTransform={{\n          display: isMinimized ? 'none' : 'flex',\n          flexDirection: 'column',\n          width: '100%',\n        }}\n      >\n        <UiEntity uiTransform={styles.rowCenterSpaceBetween}>\n          <UiEntity uiTransform={styles.rowCenter}>\n            <UiEntity\n              uiTransform={styles.headerIcon}\n              uiBackground={{\n                textureMode: 'stretch',\n                texture: { src: ICONS.DCL_CAST_ICON },\n              }}\n            />\n            <UiEntity\n              uiText={{\n                value: '<b>DCL Cast</b>',\n                fontSize: 24,\n                color: Color4.White(),\n                textAlign: 'middle-left',\n              }}\n              uiTransform={{ margin: { left: 10 } }}\n            />\n          </UiEntity>\n          <UiEntity\n            onMouseDown={() => {\n              state.videoControl.isMinimized = true;\n            }}\n            uiTransform={styles.chevronButton}\n            uiBackground={{\n              textureMode: 'stretch',\n              color: Color4.White(),\n              texture: {\n                src: ICONS.CHEVRON_UP,\n              },\n            }}\n          />\n        </UiEntity>\n        <UiEntity uiTransform={styles.fullWidthWithBottomMargin}>\n          <UiEntity\n            uiText={{\n              value:\n                'Use a browser-based DCL Cast room to easily stream camera and screen feed to a screen in your scene.',\n              fontSize: 16,\n              color: Color4.fromHexString('#A09BA8'),\n\n              textAlign: 'top-left',\n              textWrap: 'wrap',\n            }}\n            uiTransform={styles.marginBottomSmall}\n          />\n        </UiEntity>\n        {isLoading && (\n          <LoadingDots\n            uiTransform={styles.loadingContainer}\n            engine={engine}\n          />\n        )}\n        {error && (\n          <UiEntity uiTransform={styles.columnCentered}>\n            <UiEntity\n              uiText={{\n                value: '<b>Failed to fetch DCL Cast info</b>',\n                fontSize: 16,\n                color: Color4.White(),\n              }}\n              uiTransform={styles.marginBottomSmall}\n            />\n            <UiEntity\n              uiText={{\n                value: 'Please retry.',\n                fontSize: 16,\n                color: Color4.Gray(),\n              }}\n            />\n            <Button\n              id=\"dcl_cast_retry\"\n              value=\"<b>Retry</b>\"\n              variant=\"secondary\"\n              fontSize={16}\n              color={colors.white}\n              onMouseDown={() => {\n                handleGetDclCastInfo(state);\n              }}\n              uiTransform={styles.retryButton}\n            />\n          </UiEntity>\n        )}\n\n        {!isLoading && !error && (\n          <DclCastInfo\n            state={state}\n            entity={entity}\n            engine={engine}\n            video={video}\n            onResetRoomId={handleResetRoomId}\n            onShowShowcaseModal={onShowShowcaseModal}\n            onSharePresentation={onSharePresentation}\n          />\n        )}\n      </UiEntity>\n    </UiEntity>\n  );\n};\n\nexport default DclCast;\n"]}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { Color4 } from '@dcl/sdk/math';
|
|
2
|
-
import { UiTransformProps } from '@dcl/react-ecs';
|
|
2
|
+
import type { UiTransformProps } from '@dcl/react-ecs';
|
|
3
|
+
import { getPaginationColor } from '../../ModerationControl/styles/UsersListStyles';
|
|
3
4
|
export declare const getDclCastStyles: () => Record<string, UiTransformProps>;
|
|
4
5
|
export declare const getDclCastColors: () => {
|
|
5
6
|
white: Color4.MutableColor4;
|
|
@@ -18,3 +19,74 @@ export declare const getDclCastBackgrounds: () => {
|
|
|
18
19
|
textureMode: "stretch";
|
|
19
20
|
};
|
|
20
21
|
};
|
|
22
|
+
export declare const getCompactBarStyles: () => Record<string, UiTransformProps>;
|
|
23
|
+
export declare const SHOWCASE_DROPDOWN_COLORS: {
|
|
24
|
+
idle: Color4.MutableColor4;
|
|
25
|
+
hover: Color4.MutableColor4;
|
|
26
|
+
active: Color4.MutableColor4;
|
|
27
|
+
hoverBg: Color4.MutableColor4;
|
|
28
|
+
transparentBg: Color4.MutableColor4;
|
|
29
|
+
};
|
|
30
|
+
export declare const SHOWCASE_PAGE_INDICATOR_COLOR: Color4.MutableColor4;
|
|
31
|
+
export declare const getSpeakerShowcaseStyles: () => Record<string, UiTransformProps>;
|
|
32
|
+
export declare const getShowcaseColors: () => {
|
|
33
|
+
white: Color4.MutableColor4;
|
|
34
|
+
gray: Color4.MutableColor4;
|
|
35
|
+
addressGray: Color4.MutableColor4;
|
|
36
|
+
removeRed: Color4.MutableColor4;
|
|
37
|
+
disabledGray: Color4.MutableColor4;
|
|
38
|
+
black: Color4.MutableColor4;
|
|
39
|
+
softBlack: Color4.MutableColor4;
|
|
40
|
+
transparent: Color4.MutableColor4;
|
|
41
|
+
};
|
|
42
|
+
export declare function getShowcaseIconBackgrounds(): {
|
|
43
|
+
showcase: {
|
|
44
|
+
textureMode: "stretch";
|
|
45
|
+
texture: {
|
|
46
|
+
src: string;
|
|
47
|
+
};
|
|
48
|
+
};
|
|
49
|
+
star: {
|
|
50
|
+
textureMode: "stretch";
|
|
51
|
+
texture: {
|
|
52
|
+
src: string;
|
|
53
|
+
};
|
|
54
|
+
};
|
|
55
|
+
};
|
|
56
|
+
export declare function getShowcaseBackgrounds(): {
|
|
57
|
+
container: {
|
|
58
|
+
color: Color4.MutableColor4;
|
|
59
|
+
};
|
|
60
|
+
divider: {
|
|
61
|
+
color: Color4.MutableColor4;
|
|
62
|
+
};
|
|
63
|
+
roleBadge: {
|
|
64
|
+
color: Color4.MutableColor4;
|
|
65
|
+
};
|
|
66
|
+
headerIcon: {
|
|
67
|
+
textureMode: "stretch";
|
|
68
|
+
texture: {
|
|
69
|
+
src: string;
|
|
70
|
+
};
|
|
71
|
+
};
|
|
72
|
+
personIcon: {
|
|
73
|
+
textureMode: "stretch";
|
|
74
|
+
texture: {
|
|
75
|
+
src: string;
|
|
76
|
+
};
|
|
77
|
+
};
|
|
78
|
+
banIcon: {
|
|
79
|
+
textureMode: "stretch";
|
|
80
|
+
texture: {
|
|
81
|
+
src: string;
|
|
82
|
+
};
|
|
83
|
+
};
|
|
84
|
+
verifiedIcon: {
|
|
85
|
+
textureMode: "stretch";
|
|
86
|
+
texture: {
|
|
87
|
+
src: string;
|
|
88
|
+
};
|
|
89
|
+
color: Color4.MutableColor4;
|
|
90
|
+
};
|
|
91
|
+
};
|
|
92
|
+
export { getPaginationColor };
|