Skip to content

Commit

Permalink
feat(sources): widget grouping, sorting, etc
Browse files Browse the repository at this point in the history
  • Loading branch information
blackxored committed Dec 17, 2024
1 parent dcaec51 commit dc2a10c
Show file tree
Hide file tree
Showing 4 changed files with 170 additions and 11 deletions.
144 changes: 135 additions & 9 deletions app/components-react/windows/source-showcase/SourceGrid.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,14 @@ export default function SourceGrid(p: { activeTab: string }) {
'apps',
]);

const [widgetSections, setWidgetExpandedSections] = useState([
'essentialWidgets',
'interactive',
'goals',
'flair',
'charity',
]);

const { isLoggedIn, linkedPlatforms, primaryPlatform } = useVuex(() => ({
isLoggedIn: UserService.views.isLoggedIn,
linkedPlatforms: UserService.views.linkedPlatforms,
Expand Down Expand Up @@ -103,6 +111,18 @@ export default function SourceGrid(p: { activeTab: string }) {
}, []);

const essentialSourcesOrder = ['game_capture', 'dshow_input', 'ffmpeg_source'];
// Stream Label is last, we don't have a widget type for it
const essentialWidgetsOrder = [
WidgetType.AlertBox,
WidgetType.ChatBox,
WidgetType.EventList,
WidgetType.ViewerCount,
];

function customOrder<T, U>(orderArray: T[], getter: (a: U) => T) {
return (s1: U, s2: U): number =>
orderArray.indexOf(getter(s1)) - orderArray.indexOf(getter(s2));
}

const essentialSources = useMemo(() => {
const essentialDefaults = availableSources
Expand All @@ -114,10 +134,7 @@ export default function SourceGrid(p: { activeTab: string }) {
//byOS({ [OS.Windows]: 'screen_capture', [OS.Mac]: 'window_capture' }),
].includes(source.value),
)
.sort(
(s1, s2) =>
essentialSourcesOrder.indexOf(s1.value) - essentialSourcesOrder.indexOf(s2.value),
);
.sort(customOrder(essentialSourcesOrder, s => s.value));

const essentialWidgets = iterableWidgetTypes.filter(type =>
[WidgetType.AlertBox, WidgetType.ChatBox].includes(WidgetType[type]),
Expand Down Expand Up @@ -152,6 +169,12 @@ export default function SourceGrid(p: { activeTab: string }) {
<SourceTag key={source.value} type={source.value} essential excludeWrap={excludeWrap} />
);

// TODO: restrict type
// Hide widget descriptions on non-general tab
const toWidgetEl = (widget: string) => (
<SourceTag key={widget} type={widget} excludeWrap={excludeWrap} hideShortDescription />
);

const essentialSourcesList = useMemo(
() => (
<>
Expand All @@ -178,6 +201,7 @@ export default function SourceGrid(p: { activeTab: string }) {
);

const sourceDisplayData = useMemo(() => SourceDisplayData(), []);
const widgetDisplayData = useMemo(() => WidgetDisplayData(), []);

const byGroup = (group: 'capture' | 'av' | 'media') => (source: IObsListOption<TSourceType>) => {
const displayData = sourceDisplayData[source.value];
Expand All @@ -188,6 +212,15 @@ export default function SourceGrid(p: { activeTab: string }) {
return displayData.group === group;
};

const byWidgetGroup = (group: string) => (widget: string) => {
const displayData = widgetDisplayData[WidgetType[widget]];
if (!displayData) {
return true;
}

return displayData.group === group;
};

const captureSourcesList = useMemo(
() => availableSources.filter(byGroup('capture')).map(toSourceEl),
[availableSources, excludeWrap],
Expand Down Expand Up @@ -234,7 +267,12 @@ export default function SourceGrid(p: { activeTab: string }) {
) : (
<>
{iterableWidgetTypes.filter(filterEssential).map(widgetType => (
<SourceTag key={widgetType} type={widgetType} excludeWrap={excludeWrap} />
<SourceTag
key={widgetType}
type={widgetType}
excludeWrap={excludeWrap}
hideShortDescription
/>
))}
{p.activeTab !== 'all' && (
<SourceTag
Expand All @@ -251,6 +289,97 @@ export default function SourceGrid(p: { activeTab: string }) {
[isLoggedIn, iterableWidgetTypes, p.activeTab, excludeWrap],
);

const widgetGroupedList = useMemo(() => {
// TODO: restrict types
const widgetsInGroup = (group: string, sorter?: (s1: string, s2: string) => number) => {
const widgets = iterableWidgetTypes
.filter(byWidgetGroup(group))
// Sort lexographically by default, if sorter is not provided
.sort(sorter);

return widgets.map(toWidgetEl);
};

// Using essentials as a group for widgets since we wanna display more
const essentialWidgets = (
<>
{widgetsInGroup(
'essential',
customOrder(essentialWidgetsOrder, x => WidgetType[x]),
)}
<SourceTag
key="streamlabel"
name={$t('Stream Label')}
type="streamlabel"
excludeWrap={excludeWrap}
hideShortDescription
/>
</>
);

const interactiveWidgets = widgetsInGroup('interactive');
const goalWidgets = widgetsInGroup('goals');
const flairWidgets = <>{widgetsInGroup('flair')}</>;
const charityWidgets = widgetsInGroup('charity');

return (
<>
{!isLoggedIn ? (
<Empty
description={$t('You must be logged in to use Widgets')}
image={$i(`images/sleeping-kevin-${demoMode}.png`)}
>
<Button onClick={handleAuth}>{$t('Click here to log in')}</Button>
</Empty>
) : (
<Collapse
ghost
activeKey={widgetSections}
onChange={xs => setWidgetExpandedSections(xs as string[])}
>
<Panel
header={$t('Essentials')}
key="essentialWidgets"
collapsible="disabled"
showArrow={false}
>
<div className="collapse-section" data-testid="essential-widgets">
{essentialWidgets}
</div>
</Panel>
<Panel
header={$t('Interactive')}
key="interactive"
collapsible="disabled"
showArrow={false}
>
<div className="collapse-section" data-testid="interactive-widgets">
{interactiveWidgets}
</div>
</Panel>
<Panel header={$t('Goals')} key="goals" collapsible="disabled" showArrow={false}>
<div className="collapse-section" data-testid="goal-widgets">
{goalWidgets}
</div>
</Panel>
<Panel header={$t('Flair')} key="flair" collapsible="disabled" showArrow={false}>
<div className="collapse-section" data-testid="flair-widgets">
{flairWidgets}
</div>
</Panel>
{/* TODO: we don't have any charity widgets on Desktop
<Panel header={$t('Charity')} key="charity" collapsible="disabled" showArrow={false}>
<div className="collapse-section" data-testid="charity-widgets">
{charityWidgets}
</div>
</Panel>
*/}
</Collapse>
)}
</>
);
}, [isLoggedIn, iterableWidgetTypes, excludeWrap]);

const appsList = useMemo(
() => (
<>
Expand Down Expand Up @@ -308,10 +437,7 @@ export default function SourceGrid(p: { activeTab: string }) {
} else if (showContent('widgets')) {
return (
<>
<Col span={24}>
<PageHeader style={{ paddingLeft: 0 }} title={$t('Widgets')} />
</Col>
{widgetList}
<Col span={24}>{widgetGroupedList}</Col>
</>
);
} else if (showContent('apps')) {
Expand Down
3 changes: 2 additions & 1 deletion app/components-react/windows/source-showcase/SourceTag.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ export default function SourceTag(p: {
appSourceId?: string;
essential?: boolean;
excludeWrap?: boolean;
hideShortDescription?: boolean;
}) {
const { inspectSource, selectInspectedSource, store } = useSourceShowcaseSettings();

Expand Down Expand Up @@ -54,7 +55,7 @@ export default function SourceTag(p: {
</div>
<div className={styles.displayName}>{displayData?.name || p.name}</div>
</div>
{displayData?.shortDesc && (
{displayData?.shortDesc && !p.hideShortDescription && (
<div
style={{
opacity: '0.5',
Expand Down
6 changes: 5 additions & 1 deletion app/i18n/en-US/widgets.json
Original file line number Diff line number Diff line change
Expand Up @@ -188,5 +188,9 @@
"Superchat Goal": "Superchat Goal",
"YouTube Superchats": "YouTube Superchats",
"Custom Widget": "Custom Widget",
"Use HTML, CSS, and JavaScript to create a widget with custom functionality": "Use HTML, CSS, and JavaScript to create a widget with custom functionality"
"Use HTML, CSS, and JavaScript to create a widget with custom functionality": "Use HTML, CSS, and JavaScript to create a widget with custom functionality",
"Interactive": "Interactive",
"Goals": "Goals",
"Flair": "Flair",
"Charity": "Charity"
}
Loading

0 comments on commit dc2a10c

Please sign in to comment.