@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.
- package/dist/index.d.ts +4 -0
- package/dist/index.js +9 -0
- package/dist/index.js.map +1 -0
- package/dist/lane-info/index.d.ts +2 -0
- package/dist/lane-info/index.js +6 -0
- package/dist/lane-info/index.js.map +1 -0
- package/dist/lane-info/lane-info.d.ts +6 -0
- package/dist/lane-info/lane-info.js +20 -0
- package/dist/lane-info/lane-info.js.map +1 -0
- package/dist/lane-info/lane-info.module.scss +27 -0
- package/dist/preview-1753277474239.js +7 -0
- package/dist/version-dropdown-placeholder.d.ts +16 -0
- package/dist/version-dropdown-placeholder.js +92 -0
- package/dist/version-dropdown-placeholder.js.map +1 -0
- package/dist/version-dropdown-placeholder.module.scss +50 -0
- package/dist/version-dropdown.composition.d.ts +3 -0
- package/dist/version-dropdown.composition.js +28 -0
- package/dist/version-dropdown.composition.js.map +1 -0
- package/dist/version-dropdown.d.ts +53 -0
- package/dist/version-dropdown.docs.d.ts +35 -0
- package/dist/version-dropdown.docs.js +67 -0
- package/dist/version-dropdown.docs.js.map +1 -0
- package/dist/version-dropdown.js +153 -0
- package/dist/version-dropdown.js.map +1 -0
- package/dist/version-dropdown.module.scss +128 -0
- package/dist/version-dropdown.spec.d.ts +1 -0
- package/dist/version-dropdown.spec.js +33 -0
- package/dist/version-dropdown.spec.js.map +1 -0
- package/dist/version-info/index.d.ts +2 -0
- package/dist/version-info/index.js +6 -0
- package/dist/version-info/index.js.map +1 -0
- package/dist/version-info/version-info.d.ts +10 -0
- package/dist/version-info/version-info.js +79 -0
- package/dist/version-info/version-info.js.map +1 -0
- package/dist/version-info/version-info.module.scss +54 -0
- package/index.ts +4 -0
- package/lane-info/index.ts +2 -0
- package/lane-info/lane-info.module.scss +27 -0
- package/lane-info/lane-info.tsx +22 -0
- package/package.json +71 -0
- package/types/asset.d.ts +29 -0
- package/types/style.d.ts +42 -0
- package/version-dropdown-placeholder.module.scss +50 -0
- package/version-dropdown-placeholder.tsx +108 -0
- package/version-dropdown.composition.tsx +36 -0
- package/version-dropdown.docs.tsx +68 -0
- package/version-dropdown.module.scss +128 -0
- package/version-dropdown.spec.tsx +29 -0
- package/version-dropdown.tsx +330 -0
- package/version-info/index.ts +2 -0
- package/version-info/version-info.module.scss +54 -0
- 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
|
+
}
|