@teambit/component.ui.version-dropdown 0.0.0-0fcec20801a4b40c3b5cc91a8c8ae02e1ff20ddb

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 (52) hide show
  1. package/dist/index.d.ts +4 -0
  2. package/dist/index.js +9 -0
  3. package/dist/index.js.map +1 -0
  4. package/dist/lane-info/index.d.ts +2 -0
  5. package/dist/lane-info/index.js +6 -0
  6. package/dist/lane-info/index.js.map +1 -0
  7. package/dist/lane-info/lane-info.d.ts +6 -0
  8. package/dist/lane-info/lane-info.js +20 -0
  9. package/dist/lane-info/lane-info.js.map +1 -0
  10. package/dist/lane-info/lane-info.module.scss +27 -0
  11. package/dist/preview-1753277474239.js +7 -0
  12. package/dist/version-dropdown-placeholder.d.ts +16 -0
  13. package/dist/version-dropdown-placeholder.js +92 -0
  14. package/dist/version-dropdown-placeholder.js.map +1 -0
  15. package/dist/version-dropdown-placeholder.module.scss +50 -0
  16. package/dist/version-dropdown.composition.d.ts +3 -0
  17. package/dist/version-dropdown.composition.js +28 -0
  18. package/dist/version-dropdown.composition.js.map +1 -0
  19. package/dist/version-dropdown.d.ts +53 -0
  20. package/dist/version-dropdown.docs.d.ts +35 -0
  21. package/dist/version-dropdown.docs.js +67 -0
  22. package/dist/version-dropdown.docs.js.map +1 -0
  23. package/dist/version-dropdown.js +153 -0
  24. package/dist/version-dropdown.js.map +1 -0
  25. package/dist/version-dropdown.module.scss +128 -0
  26. package/dist/version-dropdown.spec.d.ts +1 -0
  27. package/dist/version-dropdown.spec.js +33 -0
  28. package/dist/version-dropdown.spec.js.map +1 -0
  29. package/dist/version-info/index.d.ts +2 -0
  30. package/dist/version-info/index.js +6 -0
  31. package/dist/version-info/index.js.map +1 -0
  32. package/dist/version-info/version-info.d.ts +10 -0
  33. package/dist/version-info/version-info.js +79 -0
  34. package/dist/version-info/version-info.js.map +1 -0
  35. package/dist/version-info/version-info.module.scss +54 -0
  36. package/index.ts +4 -0
  37. package/lane-info/index.ts +2 -0
  38. package/lane-info/lane-info.module.scss +27 -0
  39. package/lane-info/lane-info.tsx +22 -0
  40. package/package.json +71 -0
  41. package/types/asset.d.ts +29 -0
  42. package/types/style.d.ts +42 -0
  43. package/version-dropdown-placeholder.module.scss +50 -0
  44. package/version-dropdown-placeholder.tsx +108 -0
  45. package/version-dropdown.composition.tsx +36 -0
  46. package/version-dropdown.docs.tsx +68 -0
  47. package/version-dropdown.module.scss +128 -0
  48. package/version-dropdown.spec.tsx +29 -0
  49. package/version-dropdown.tsx +330 -0
  50. package/version-info/index.ts +2 -0
  51. package/version-info/version-info.module.scss +54 -0
  52. package/version-info/version-info.tsx +87 -0
@@ -0,0 +1,108 @@
1
+ import React, { HTMLAttributes } from 'react';
2
+ import { Ellipsis } from '@teambit/design.ui.styles.ellipsis';
3
+ import classNames from 'classnames';
4
+ import * as semver from 'semver';
5
+ import { Icon } from '@teambit/evangelist.elements.icon';
6
+ import { TimeAgo } from '@teambit/design.ui.time-ago';
7
+ import { UserAvatar } from '@teambit/design.ui.avatar';
8
+ import { WordSkeleton } from '@teambit/base-ui.loaders.skeleton';
9
+ import { DropdownComponentVersion } from './version-dropdown';
10
+
11
+ import styles from './version-dropdown-placeholder.module.scss';
12
+
13
+ export type VersionProps = {
14
+ currentVersion?: string;
15
+ isTag?: (version?: string) => boolean;
16
+ disabled?: boolean;
17
+ hasMoreVersions?: boolean;
18
+ showFullVersion?: boolean;
19
+ loading?: boolean;
20
+ useCurrentVersionLog?: (props: { skip?: boolean; version?: string }) => DropdownComponentVersion | undefined;
21
+ } & HTMLAttributes<HTMLDivElement>;
22
+
23
+ export function SimpleVersion({
24
+ currentVersion,
25
+ className,
26
+ disabled,
27
+ hasMoreVersions,
28
+ isTag = (version) => semver.valid(version) !== null,
29
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
30
+ useCurrentVersionLog,
31
+ showFullVersion,
32
+ loading,
33
+ ...rest
34
+ }: VersionProps) {
35
+ if (loading) return <WordSkeleton className={styles.loader} length={9} />;
36
+ const formattedVersion = showFullVersion || isTag(currentVersion) ? currentVersion : currentVersion?.slice(0, 6);
37
+
38
+ return (
39
+ <div {...rest} className={classNames(styles.simple, className, disabled && styles.disabled)}>
40
+ <Ellipsis className={classNames(styles.versionName)} onClick={rest.onClick}>
41
+ {formattedVersion}
42
+ </Ellipsis>
43
+ {hasMoreVersions && <Icon of="fat-arrow-down" onClick={rest.onClick} />}
44
+ </div>
45
+ );
46
+ }
47
+
48
+ export function DetailedVersion({
49
+ currentVersion,
50
+ className,
51
+ disabled,
52
+ hasMoreVersions,
53
+ isTag = (version) => semver.valid(version) !== null,
54
+ loading,
55
+ useCurrentVersionLog,
56
+ showFullVersion,
57
+ ...rest
58
+ }: VersionProps) {
59
+ const currentVersionLog = useCurrentVersionLog?.({ skip: loading, version: currentVersion });
60
+ const { displayName, message, username, email, date: _date, profileImage } = currentVersionLog || {};
61
+ const author = React.useMemo(() => {
62
+ return {
63
+ displayName: displayName ?? '',
64
+ email,
65
+ name: username ?? '',
66
+ profileImage,
67
+ };
68
+ }, [displayName, email, username, profileImage]);
69
+ const formattedVersion = showFullVersion || isTag(currentVersion) ? currentVersion : currentVersion?.slice(0, 6);
70
+
71
+ const date = _date ? new Date(+_date) : undefined;
72
+ const timestamp = React.useMemo(() => (date ? new Date(+date).toString() : new Date().toString()), [date]);
73
+ if (loading) return <WordSkeleton className={styles.loader} length={9} />;
74
+
75
+ return (
76
+ <div {...rest} className={classNames(styles.detailed, className, disabled && styles.disabled)}>
77
+ <UserAvatar
78
+ size={24}
79
+ account={author ?? {}}
80
+ className={styles.versionUserAvatar}
81
+ showTooltip={true}
82
+ onClick={rest.onClick}
83
+ />
84
+ <Ellipsis className={classNames(styles.versionName)} onClick={rest.onClick}>
85
+ {formattedVersion}
86
+ </Ellipsis>
87
+ {commitMessage(message, rest.onClick)}
88
+ <Ellipsis className={styles.versionTimestamp} onClick={rest.onClick}>
89
+ <TimeAgo date={timestamp} onClick={rest.onClick} />
90
+ </Ellipsis>
91
+ {hasMoreVersions && <Icon of="fat-arrow-down" onClick={rest.onClick} />}
92
+ </div>
93
+ );
94
+ }
95
+
96
+ function commitMessage(message?: string, onClick?: React.MouseEventHandler<HTMLDivElement> | undefined) {
97
+ if (!message || message === '')
98
+ return (
99
+ <Ellipsis className={styles.emptyMessage} onClick={onClick}>
100
+ No commit message
101
+ </Ellipsis>
102
+ );
103
+ return (
104
+ <Ellipsis className={styles.commitMessage} onClick={onClick}>
105
+ {message}
106
+ </Ellipsis>
107
+ );
108
+ }
@@ -0,0 +1,36 @@
1
+ import React from 'react';
2
+ import { MemoryRouter } from 'react-router-dom';
3
+ import { ThemeCompositions } from '@teambit/documenter.theme.theme-compositions';
4
+ import { VersionDropdown } from './version-dropdown';
5
+
6
+ const style = { display: 'flex', justifyContent: 'center', alignContent: 'center' };
7
+
8
+ export const VersionDropdownWithOneVersion = () => {
9
+ return (
10
+ <ThemeCompositions style={style}>
11
+ <VersionDropdown
12
+ useComponentVersions={() => ({
13
+ tags: [{ version: '0.1' }],
14
+ })}
15
+ currentVersion="0.1"
16
+ />
17
+ </ThemeCompositions>
18
+ );
19
+ };
20
+
21
+ export const VersionDropdownWithMultipleVersions = () => {
22
+ const versions = ['0.3', '0.2', '0.1'].map((version) => ({ version }));
23
+
24
+ return (
25
+ <ThemeCompositions style={style}>
26
+ <MemoryRouter>
27
+ <VersionDropdown
28
+ useComponentVersions={() => ({
29
+ tags: [{ version: '0.1' }],
30
+ })}
31
+ currentVersion={versions[0].version}
32
+ />
33
+ </MemoryRouter>
34
+ </ThemeCompositions>
35
+ );
36
+ };
@@ -0,0 +1,68 @@
1
+ import React from 'react';
2
+ import { MemoryRouter } from 'react-router-dom';
3
+ import { Section } from '@teambit/documenter.ui.section';
4
+ import { ThemeCompositions } from '@teambit/documenter.theme.theme-compositions';
5
+ import { Separator } from '@teambit/documenter.ui.separator';
6
+ import { VersionDropdown } from './version-dropdown';
7
+
8
+ export default function Overview() {
9
+ return (
10
+ <ThemeCompositions>
11
+ <>
12
+ <Section>
13
+ The version-dropdown displays the latest version of the viewed component. <br />
14
+ If previous versions are available, the component will display a list of them, when clicked. <br />
15
+ This allows the user to navigate to previous versions, and explore them.
16
+ </Section>
17
+ <Separator />
18
+ </>
19
+ </ThemeCompositions>
20
+ );
21
+ }
22
+
23
+ Overview.abstract = 'The version-dropdown lists the latest and previous versions of the viewed component.';
24
+
25
+ Overview.labels = ['react', 'typescript', 'version', 'dropdown'];
26
+
27
+ const style = { display: 'flex', justifyContent: 'center', alignContent: 'center' };
28
+
29
+ Overview.examples = [
30
+ {
31
+ scope: {
32
+ VersionDropdown,
33
+ style,
34
+ },
35
+ title: 'Version Dropdown',
36
+ description: 'Using the Version Dropdown component with one verion',
37
+ code: `
38
+ () => {
39
+ return (
40
+ <div style={{...style, minHeight: 150 }}>
41
+ <VersionDropdown versions={['0.1']} currentVersion="0.1" />
42
+ </div>
43
+ );
44
+ }
45
+ `,
46
+ },
47
+ {
48
+ scope: {
49
+ VersionDropdown,
50
+ style,
51
+ MemoryRouter,
52
+ },
53
+ title: 'Version Dropdown with multiple versions',
54
+ description: 'Using the Version Dropdown component with more than one version',
55
+ code: `
56
+ () => {
57
+ const versions = ['0.3', '0.2', '0.1'];
58
+ return (
59
+ <div style={{...style, minHeight: 400, alignItems: 'end', justifyContent: 'flex-end', margin: 10 }}>
60
+ <MemoryRouter>
61
+ <VersionDropdown versions={versions} currentVersion={versions[0]} />
62
+ </MemoryRouter>
63
+ </div>
64
+ );
65
+ }
66
+ `,
67
+ },
68
+ ];
@@ -0,0 +1,128 @@
1
+ @import '@teambit/ui-foundation.ui.constants.z-indexes/z-indexes.module.scss';
2
+
3
+ .versionDropdown {
4
+ height: 100%;
5
+
6
+ > div {
7
+ height: 100%;
8
+ display: flex;
9
+ align-items: center;
10
+ }
11
+ .menu {
12
+ padding: 0;
13
+ right: 0;
14
+ // top: 43px;
15
+ font-size: var(--bit-p-xs);
16
+ border-radius: 6px;
17
+ z-index: $modal-z-index;
18
+ }
19
+ }
20
+
21
+ .title {
22
+ padding: 16px;
23
+ padding-bottom: 12px;
24
+ border-bottom: 1px solid var(--bit-border-color-lightest, #ededed);
25
+ }
26
+
27
+ .titleContainer {
28
+ margin-bottom: 2px;
29
+ }
30
+
31
+ .versionContainerRoot {
32
+ max-height: 240px;
33
+ overflow-y: scroll;
34
+ padding-bottom: 8px;
35
+ position: relative;
36
+ }
37
+
38
+ .versionRow {
39
+ display: flex;
40
+ justify-content: space-between;
41
+ align-items: center;
42
+ height: 40px;
43
+ padding: 0 16px;
44
+
45
+ &:hover {
46
+ background-color: var(--surface-hover-color, #eceaff) !important;
47
+ }
48
+
49
+ &.localVersion {
50
+ border-bottom: 1px solid var(--bit-border-color-lightest, #ededed);
51
+ }
52
+ .versionTimestamp {
53
+ margin-right: 2px;
54
+ }
55
+ .versionUserAvatar {
56
+ padding: 0px 8px;
57
+ }
58
+ .laneIcon {
59
+ padding: 0px 4px;
60
+ }
61
+ .version {
62
+ width: 60%;
63
+ display: flex;
64
+ flex-direction: row;
65
+ align-content: space-between;
66
+ align-items: center;
67
+ }
68
+ .versionName {
69
+ padding: 0px 8px;
70
+ min-width: fit-content;
71
+ }
72
+ }
73
+
74
+ .withVersions {
75
+ cursor: pointer;
76
+ > div {
77
+ margin-right: 5px;
78
+ }
79
+ > span {
80
+ display: unset;
81
+ }
82
+ &:hover {
83
+ background-color: var(--bit-bg-heavy);
84
+ }
85
+ [data-open='true'] & {
86
+ transition:
87
+ color 300ms,
88
+ background-color 300ms ease-in-out;
89
+ background-color: var(--bit-bg-color, #ffffff);
90
+ color: var(--bit-accent-color, #6c5ce7);
91
+ &:hover {
92
+ background-color: var(--bit-bg-color, #ffffff);
93
+ }
94
+ }
95
+ }
96
+
97
+ .tabs {
98
+ display: flex;
99
+ padding: 0 24px;
100
+ line-height: 14px;
101
+ border-bottom: 1px solid var(--bit-border-color-lightest, #ededed);
102
+ overflow-x: auto;
103
+ margin-top: 8px;
104
+
105
+ .tab {
106
+ font-weight: bold;
107
+ padding-top: 4px;
108
+ font-size: 12px;
109
+ }
110
+ }
111
+
112
+ .loading {
113
+ color: var(--bit-bg-dent, #f6f6f6);
114
+ }
115
+
116
+ .loader {
117
+ color: var(--bit-bg-dent, #f6f6f6);
118
+ > div {
119
+ padding: 8px 0px;
120
+ }
121
+ }
122
+
123
+ .versionMenuContainer {
124
+ display: initial;
125
+ &.hide {
126
+ display: none;
127
+ }
128
+ }
@@ -0,0 +1,29 @@
1
+ import React from 'react';
2
+ import { render, screen } from '@testing-library/react';
3
+ import { expect } from 'chai';
4
+ import { VersionDropdownWithOneVersion, VersionDropdownWithMultipleVersions } from './version-dropdown.composition';
5
+
6
+ describe('version dropdown tests', () => {
7
+ /**
8
+ * https://github.com/jsdom/jsdom/issues/1695
9
+ * scrollIntoView is not implemented in jsdom
10
+ * */
11
+ beforeEach(() => {
12
+ Element.prototype.scrollIntoView = jest.fn();
13
+ });
14
+ it('should render one version', () => {
15
+ const { getByText } = render(<VersionDropdownWithOneVersion />);
16
+ const textVersion = getByText(/^0.1$/);
17
+ expect(textVersion).to.exist;
18
+ });
19
+ it('should not return multiple versions when mounted (lazy loading)', () => {
20
+ render(<VersionDropdownWithMultipleVersions />);
21
+ const textVersionOne = screen.queryByText(/^0.1$/);
22
+ const textVersionTwo = screen.queryByText(/^0.2$/);
23
+ const textVersionThree = screen.getAllByText(/^0.3$/);
24
+
25
+ expect(textVersionOne).to.be.null;
26
+ expect(textVersionTwo).to.be.null;
27
+ expect(textVersionThree).to.have.lengthOf.at.least(1);
28
+ });
29
+ });
@@ -0,0 +1,330 @@
1
+ import React, { useState } from 'react';
2
+ import { MenuLinkItem } from '@teambit/design.ui.surfaces.menu.link-item';
3
+ import { Dropdown } from '@teambit/evangelist.surfaces.dropdown';
4
+ import { Tab } from '@teambit/ui-foundation.ui.use-box.tab';
5
+ import { LegacyComponentLog } from '@teambit/legacy-component-log';
6
+ import { UserAvatar } from '@teambit/design.ui.avatar';
7
+ import { LineSkeleton } from '@teambit/base-ui.loaders.skeleton';
8
+ import { LaneModel } from '@teambit/lanes.ui.models.lanes-model';
9
+ import classNames from 'classnames';
10
+ import styles from './version-dropdown.module.scss';
11
+ import { VersionInfo } from './version-info';
12
+ import { LaneInfo } from './lane-info';
13
+ import { SimpleVersion, VersionProps } from './version-dropdown-placeholder';
14
+
15
+ export const LOCAL_VERSION = 'workspace';
16
+
17
+ export type DropdownComponentVersion = Partial<LegacyComponentLog> & { version: string };
18
+
19
+ export type UseComponentDropdownVersionsResult = {
20
+ tags?: DropdownComponentVersion[];
21
+ snaps?: DropdownComponentVersion[];
22
+ loading?: boolean;
23
+ };
24
+ export type UseComponentDropdownVersionsProps = {
25
+ skip?: boolean;
26
+ };
27
+ export type UseComponentDropdownVersions = (
28
+ props?: UseComponentDropdownVersionsProps
29
+ ) => UseComponentDropdownVersionsResult;
30
+ export type GetActiveTabIndex = (
31
+ currentVersion?: string,
32
+ tabs?: Array<VersionMenuTab>,
33
+ tags?: DropdownComponentVersion[],
34
+ snaps?: DropdownComponentVersion[],
35
+ currentLane?: LaneModel
36
+ ) => number;
37
+ export type VersionDropdownProps = {
38
+ localVersion?: boolean;
39
+ latestVersion?: string;
40
+ currentVersion: string;
41
+ useCurrentVersionLog?: (props?: { skip?: boolean; version?: string }) => DropdownComponentVersion | undefined;
42
+ hasMoreVersions?: boolean;
43
+ loading?: boolean;
44
+ useComponentVersions?: UseComponentDropdownVersions;
45
+ currentLane?: LaneModel;
46
+ lanes?: LaneModel[];
47
+ getActiveTabIndex?: GetActiveTabIndex;
48
+ overrideVersionHref?: (version: string) => string;
49
+ placeholderClassName?: string;
50
+ dropdownClassName?: string;
51
+ menuClassName?: string;
52
+ showVersionDetails?: boolean;
53
+ disabled?: boolean;
54
+ PlaceholderComponent?: React.ComponentType<VersionProps>;
55
+ } & React.HTMLAttributes<HTMLDivElement>;
56
+
57
+ export const VersionDropdown = React.memo(_VersionDropdown);
58
+ const VersionMenu = React.memo(_VersionMenu);
59
+ function _VersionDropdown({
60
+ currentVersion,
61
+ latestVersion,
62
+ localVersion,
63
+ useCurrentVersionLog,
64
+ hasMoreVersions,
65
+ loading,
66
+ overrideVersionHref,
67
+ className,
68
+ placeholderClassName,
69
+ getActiveTabIndex,
70
+ dropdownClassName,
71
+ menuClassName,
72
+ showVersionDetails = true,
73
+ disabled,
74
+ PlaceholderComponent: _PlaceholderComponent,
75
+ currentLane,
76
+ useComponentVersions,
77
+ lanes,
78
+ ...rest
79
+ }: VersionDropdownProps) {
80
+ const [key, setKey] = useState(0);
81
+ const singleVersion = !hasMoreVersions;
82
+ const [open, setOpen] = useState(false);
83
+
84
+ React.useEffect(() => {
85
+ if (loading && open) {
86
+ setOpen(false);
87
+ }
88
+ }, [loading]);
89
+
90
+ const handlePlaceholderClicked = (e: React.MouseEvent<HTMLDivElement>) => {
91
+ if (loading) return;
92
+ if (e.target === e.currentTarget) {
93
+ setOpen((o) => !o);
94
+ }
95
+ };
96
+
97
+ const defaultPlaceholder = (
98
+ <SimpleVersion
99
+ useCurrentVersionLog={useCurrentVersionLog}
100
+ disabled={disabled}
101
+ className={placeholderClassName}
102
+ currentVersion={currentVersion}
103
+ onClick={handlePlaceholderClicked}
104
+ hasMoreVersions={hasMoreVersions}
105
+ loading={loading}
106
+ showFullVersion={currentVersion === 'workspace'}
107
+ />
108
+ );
109
+
110
+ const PlaceholderComponent = _PlaceholderComponent ? (
111
+ <_PlaceholderComponent
112
+ useCurrentVersionLog={useCurrentVersionLog}
113
+ disabled={disabled}
114
+ className={placeholderClassName}
115
+ currentVersion={currentVersion}
116
+ onClick={handlePlaceholderClicked}
117
+ hasMoreVersions={hasMoreVersions}
118
+ loading={loading}
119
+ showFullVersion={currentVersion === 'workspace'}
120
+ />
121
+ ) : (
122
+ defaultPlaceholder
123
+ );
124
+
125
+ if (disabled || (singleVersion && !loading)) {
126
+ return <div className={classNames(styles.noVersions, className)}>{PlaceholderComponent}</div>;
127
+ }
128
+
129
+ return (
130
+ <div {...rest} className={classNames(styles.versionDropdown, className)}>
131
+ <Dropdown
132
+ className={classNames(styles.dropdown, dropdownClassName)}
133
+ dropClass={classNames(styles.menu, menuClassName)}
134
+ open={open}
135
+ onClick={handlePlaceholderClicked}
136
+ onClickOutside={() => setOpen(false)}
137
+ onChange={(_e, _open) => _open && setKey((x) => x + 1)} // to reset menu to initial state when toggling
138
+ PlaceholderComponent={({ children, ...other }) => (
139
+ <div {...other} className={placeholderClassName} onClick={handlePlaceholderClicked}>
140
+ {children}
141
+ </div>
142
+ )}
143
+ placeholder={PlaceholderComponent}
144
+ >
145
+ <VersionMenu
146
+ className={menuClassName}
147
+ key={key}
148
+ currentVersion={currentVersion}
149
+ latestVersion={latestVersion}
150
+ localVersion={localVersion}
151
+ overrideVersionHref={overrideVersionHref}
152
+ showVersionDetails={showVersionDetails}
153
+ currentLane={currentLane}
154
+ getActiveTabIndex={getActiveTabIndex}
155
+ lanes={lanes}
156
+ useVersions={useComponentVersions}
157
+ onVersionClicked={() => setOpen(false)}
158
+ open={open}
159
+ />
160
+ </Dropdown>
161
+ </div>
162
+ );
163
+ }
164
+
165
+ type VersionMenuProps = {
166
+ localVersion?: boolean;
167
+ currentVersion?: string;
168
+ latestVersion?: string;
169
+ useVersions?: UseComponentDropdownVersions;
170
+ currentLane?: LaneModel;
171
+ lanes?: LaneModel[];
172
+ overrideVersionHref?: (version: string) => string;
173
+ showVersionDetails?: boolean;
174
+ loading?: boolean;
175
+ getActiveTabIndex?: GetActiveTabIndex;
176
+ open?: boolean;
177
+ onVersionClicked?: () => void;
178
+ } & React.HTMLAttributes<HTMLDivElement>;
179
+
180
+ export type VersionMenuTab =
181
+ | {
182
+ name: 'SNAP';
183
+ payload: DropdownComponentVersion[];
184
+ }
185
+ | {
186
+ name: 'LANE';
187
+ payload: LaneModel[];
188
+ }
189
+ | {
190
+ name: 'TAG';
191
+ payload: DropdownComponentVersion[];
192
+ };
193
+
194
+ const defaultActiveTabIndex: GetActiveTabIndex = (currentVersion, tabs = [], tags, snaps) => {
195
+ if ((snaps || []).some((snap) => snap.version === currentVersion))
196
+ return tabs.findIndex((tab) => tab.name === 'SNAP');
197
+ return 0;
198
+ };
199
+
200
+ const VERSION_TAB_NAMES = ['TAG', 'SNAP', 'LANE'] as const;
201
+ function _VersionMenu({
202
+ currentVersion,
203
+ localVersion,
204
+ latestVersion,
205
+ overrideVersionHref,
206
+ showVersionDetails,
207
+ useVersions,
208
+ currentLane,
209
+ lanes,
210
+ getActiveTabIndex = defaultActiveTabIndex,
211
+ loading: loadingFromProps,
212
+ open,
213
+ onVersionClicked,
214
+ ...rest
215
+ }: VersionMenuProps) {
216
+ const { snaps, tags, loading: loadingVersions } = useVersions?.() || {};
217
+ const loading = loadingFromProps || loadingVersions;
218
+
219
+ const tabs = React.useMemo(
220
+ () =>
221
+ VERSION_TAB_NAMES.map((name) => {
222
+ switch (name) {
223
+ case 'SNAP':
224
+ return { name, payload: snaps || [] };
225
+ case 'LANE':
226
+ return { name, payload: lanes || [] };
227
+ default:
228
+ return { name, payload: tags || [] };
229
+ }
230
+ }).filter((tab) => tab.payload.length > 0),
231
+ [snaps?.length, tags?.length, lanes?.length, loading]
232
+ );
233
+
234
+ const [activeTabIndex, setActiveTab] = React.useState<number | undefined>(
235
+ getActiveTabIndex(currentVersion, tabs, tags, snaps, currentLane)
236
+ );
237
+
238
+ const activeTab = React.useMemo(
239
+ () => (activeTabIndex !== undefined ? tabs[activeTabIndex] : undefined),
240
+ [activeTabIndex, tabs]
241
+ );
242
+
243
+ React.useEffect(() => {
244
+ if (!currentLane) return;
245
+ if (tabs.length === 0) return;
246
+ const _activeTabIndex = getActiveTabIndex(currentVersion, tabs, tags, snaps, currentLane);
247
+ if (_activeTabIndex !== activeTabIndex) setActiveTab(_activeTabIndex);
248
+ }, [currentLane, tabs.length, tags?.length, snaps?.length, currentVersion, loading]);
249
+
250
+ const multipleTabs = tabs.length > 1;
251
+ const message = multipleTabs
252
+ ? 'Switch to view tags, snaps, or lanes'
253
+ : `Switch between ${tabs[0]?.name.toLocaleLowerCase()}s`;
254
+
255
+ const showTab = activeTabIndex !== undefined && tabs[activeTabIndex]?.payload.length > 0;
256
+
257
+ const _rowRenderer = React.useCallback(
258
+ function VersionRowRenderer({ index }) {
259
+ const { name, payload = [] } = activeTab || {};
260
+ const item = payload[index];
261
+ if (!item) return null;
262
+ if (name === 'LANE') {
263
+ const lane = item as LaneModel;
264
+ return <LaneInfo key={lane.id.toString()} currentLane={currentLane} {...lane}></LaneInfo>;
265
+ }
266
+ const version = item as DropdownComponentVersion;
267
+ return (
268
+ <VersionInfo
269
+ key={version.version}
270
+ currentVersion={currentVersion}
271
+ latestVersion={latestVersion}
272
+ overrideVersionHref={overrideVersionHref}
273
+ showDetails={showVersionDetails}
274
+ onVersionClicked={onVersionClicked}
275
+ {...version}
276
+ ></VersionInfo>
277
+ );
278
+ },
279
+ [activeTab, currentVersion, latestVersion, showVersionDetails, currentLane?.id.toString(), showTab]
280
+ );
281
+
282
+ const rowRenderer = React.useMemo(
283
+ () => (showTab && activeTab ? _rowRenderer : () => null),
284
+ [showTab, activeTab, _rowRenderer]
285
+ );
286
+
287
+ const ActiveTab = React.useMemo(() => {
288
+ return activeTab?.payload.map((payload, index) => {
289
+ return rowRenderer({ index });
290
+ });
291
+ }, [activeTab]);
292
+
293
+ return (
294
+ <div {...rest} className={classNames(styles.versionMenuContainer, !open && styles.hide)}>
295
+ <div className={styles.top}>
296
+ {loading && <LineSkeleton count={6} className={styles.loader} />}
297
+ {!loading && <div className={classNames(styles.titleContainer, styles.title)}>{message}</div>}
298
+ {!loading && localVersion && (
299
+ <MenuLinkItem
300
+ href={'?'}
301
+ active={currentVersion === LOCAL_VERSION}
302
+ className={classNames(styles.versionRow, styles.localVersion)}
303
+ onClick={onVersionClicked}
304
+ >
305
+ <div className={styles.version}>
306
+ <UserAvatar size={24} account={{}} className={styles.versionUserAvatar} />
307
+ <span className={styles.versionName}>{LOCAL_VERSION}</span>
308
+ </div>
309
+ </MenuLinkItem>
310
+ )}
311
+ </div>
312
+ <div className={classNames(multipleTabs && styles.tabs)}>
313
+ {multipleTabs &&
314
+ tabs.map(({ name }, index) => {
315
+ return (
316
+ <Tab
317
+ className={styles.tab}
318
+ key={name}
319
+ isActive={activeTabIndex === index}
320
+ onClick={() => setActiveTab(index)}
321
+ >
322
+ {name}
323
+ </Tab>
324
+ );
325
+ })}
326
+ </div>
327
+ <div className={styles.versionContainerRoot}>{ActiveTab}</div>
328
+ </div>
329
+ );
330
+ }
@@ -0,0 +1,2 @@
1
+ export { VersionInfo } from './version-info';
2
+ export type { VersionInfoProps } from './version-info';