@dcl/asset-packs 2.14.3 → 2.15.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (63) hide show
  1. package/bin/index.js +2169 -616
  2. package/catalog.json +4 -4
  3. package/dist/actions.js +8 -8
  4. package/dist/admin-toolkit-ui/Button.d.ts +1 -1
  5. package/dist/admin-toolkit-ui/Button.js +2 -2
  6. package/dist/admin-toolkit-ui/Error.d.ts +2 -2
  7. package/dist/admin-toolkit-ui/Error.js +4 -4
  8. package/dist/admin-toolkit-ui/Header.js +1 -1
  9. package/dist/admin-toolkit-ui/Modal.d.ts +19 -0
  10. package/dist/admin-toolkit-ui/Modal.js +37 -0
  11. package/dist/admin-toolkit-ui/ModerationControl/UsersList.d.ts +3 -3
  12. package/dist/admin-toolkit-ui/ModerationControl/UsersList.js +14 -8
  13. package/dist/admin-toolkit-ui/ModerationControl/index.d.ts +4 -5
  14. package/dist/admin-toolkit-ui/ModerationControl/index.js +19 -11
  15. package/dist/admin-toolkit-ui/ModerationControl/styles/AddUserInputStyles.js +4 -4
  16. package/dist/admin-toolkit-ui/ModerationControl/styles/UsersListStyles.d.ts +2 -0
  17. package/dist/admin-toolkit-ui/ModerationControl/styles/UsersListStyles.js +13 -5
  18. package/dist/admin-toolkit-ui/RewardsControl.d.ts +2 -2
  19. package/dist/admin-toolkit-ui/RewardsControl.js +9 -5
  20. package/dist/admin-toolkit-ui/SmartItemsControl.d.ts +2 -2
  21. package/dist/admin-toolkit-ui/SmartItemsControl.js +6 -4
  22. package/dist/admin-toolkit-ui/TextAnnouncements.d.ts +2 -2
  23. package/dist/admin-toolkit-ui/TextAnnouncements.js +9 -5
  24. package/dist/admin-toolkit-ui/TextAnnouncementsControl.d.ts +3 -3
  25. package/dist/admin-toolkit-ui/TextAnnouncementsControl.js +15 -7
  26. package/dist/admin-toolkit-ui/VideoControl/DclCast/CompactDclCast.d.ts +11 -0
  27. package/dist/admin-toolkit-ui/VideoControl/DclCast/CompactDclCast.js +90 -0
  28. package/dist/admin-toolkit-ui/VideoControl/DclCast/DclCastInfo.d.ts +6 -4
  29. package/dist/admin-toolkit-ui/VideoControl/DclCast/DclCastInfo.js +71 -57
  30. package/dist/admin-toolkit-ui/VideoControl/DclCast/PresentationPanel.d.ts +10 -0
  31. package/dist/admin-toolkit-ui/VideoControl/DclCast/PresentationPanel.js +67 -0
  32. package/dist/admin-toolkit-ui/VideoControl/DclCast/SharePresentationModal.d.ts +6 -0
  33. package/dist/admin-toolkit-ui/VideoControl/DclCast/SharePresentationModal.js +74 -0
  34. package/dist/admin-toolkit-ui/VideoControl/DclCast/SpeakerShowcase.d.ts +11 -0
  35. package/dist/admin-toolkit-ui/VideoControl/DclCast/SpeakerShowcase.js +110 -0
  36. package/dist/admin-toolkit-ui/VideoControl/DclCast/index.d.ts +17 -3
  37. package/dist/admin-toolkit-ui/VideoControl/DclCast/index.js +208 -35
  38. package/dist/admin-toolkit-ui/VideoControl/DclCast/styles.d.ts +73 -1
  39. package/dist/admin-toolkit-ui/VideoControl/DclCast/styles.js +193 -3
  40. package/dist/admin-toolkit-ui/VideoControl/LiveStream/ShowStreamKey.js +17 -13
  41. package/dist/admin-toolkit-ui/VideoControl/LiveStream/index.d.ts +1 -1
  42. package/dist/admin-toolkit-ui/VideoControl/LiveStream/index.js +8 -8
  43. package/dist/admin-toolkit-ui/VideoControl/VideoUrl.d.ts +2 -2
  44. package/dist/admin-toolkit-ui/VideoControl/VideoUrl.js +9 -9
  45. package/dist/admin-toolkit-ui/VideoControl/VolumeControl.d.ts +3 -2
  46. package/dist/admin-toolkit-ui/VideoControl/VolumeControl.js +8 -10
  47. package/dist/admin-toolkit-ui/VideoControl/api.d.ts +49 -0
  48. package/dist/admin-toolkit-ui/VideoControl/api.js +185 -1
  49. package/dist/admin-toolkit-ui/VideoControl/index.d.ts +16 -15
  50. package/dist/admin-toolkit-ui/VideoControl/index.js +57 -21
  51. package/dist/admin-toolkit-ui/VideoControl/utils.d.ts +2 -2
  52. package/dist/admin-toolkit-ui/VideoControl/utils.js +3 -3
  53. package/dist/admin-toolkit-ui/constants.d.ts +2 -3
  54. package/dist/admin-toolkit-ui/constants.js +4 -4
  55. package/dist/admin-toolkit-ui/fetch-utils.js +2 -2
  56. package/dist/admin-toolkit-ui/index.d.ts +4 -4
  57. package/dist/admin-toolkit-ui/index.js +40 -18
  58. package/dist/admin-toolkit-ui/types.d.ts +19 -0
  59. package/dist/admin-toolkit-ui/types.js +1 -1
  60. package/dist/admin.js +2 -2
  61. package/dist/bin/index.js +2169 -616
  62. package/dist/ui.js +4 -4
  63. 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 { getDclCastInfo, resetStreamKey } from '../api';
4
- import { CONTENT_URL } from '../../constants';
5
- import { Header } from '../../Header';
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: `${CONTENT_URL}/admin_toolkit/assets/icons/video-control-dcl-cast.png`,
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
- const DclCast = ({ engine, state, entity, video, }) => {
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(Header, { iconSrc: ICONS.DCL_CAST_ICON, title: "DCL Cast" }),
61
- ReactEcs.createElement(UiEntity, { uiTransform: styles.fullWidthWithBottomMargin },
62
- ReactEcs.createElement(UiEntity, { uiText: {
63
- value: 'Use a browser-based DCL Cast room to easily stream camera and screen feed to a screen in your scene.',
64
- fontSize: 16,
65
- color: Color4.fromHexString('#A09BA8'),
66
- textAlign: 'top-left',
67
- textWrap: 'wrap',
68
- }, uiTransform: styles.marginBottomSmall })),
69
- isLoading && (ReactEcs.createElement(LoadingDots, { uiTransform: { minHeight: 400 }, engine: engine })),
70
- error && (ReactEcs.createElement(UiEntity, { uiTransform: styles.columnCentered },
71
- ReactEcs.createElement(UiEntity, { uiText: {
72
- value: '<b>Failed to fetch DCL Cast info</b>',
73
- fontSize: 16,
74
- color: Color4.White(),
75
- }, uiTransform: { margin: { bottom: 8 } } }),
76
- ReactEcs.createElement(UiEntity, { uiText: {
77
- value: 'Please retry.',
78
- fontSize: 16,
79
- color: Color4.Gray(),
80
- } }),
81
- ReactEcs.createElement(Button, { id: "dcl_cast_retry", value: "<b>Retry</b>", variant: "secondary", fontSize: 16, color: colors.white, onMouseDown: () => {
82
- handleGetDclCastInfo(state);
83
- }, uiTransform: styles.retryButton }))),
84
- !isLoading && !error && (ReactEcs.createElement(DclCastInfo, { state: state, entity: entity, engine: engine, video: video, onResetRoomId: handleResetRoomId }))));
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 };