Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

perf(Component UI): lazy load component logs #7346

Draft
wants to merge 26 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
1501d14
lazy load component logs
luvkapur May 1, 2023
eea4fd1
Merge branch 'master' into ui/lazy-load-logs
GiladShoham May 2, 2023
bcc9af0
Merge branch 'master' into ui/lazy-load-logs
GiladShoham May 2, 2023
c1aff40
simplify lazily fetching component logs
luvkapur May 2, 2023
adb8e7a
Merge remote-tracking branch 'origin/master' into ui/lazy-load-logs
luvkapur May 2, 2023
28d6c8f
lazy load version in version dropdown
luvkapur May 2, 2023
3ae0466
import code view
luvkapur May 2, 2023
7a0f346
lazy load snaps and tags separately in a single query + optimize comp…
luvkapur May 3, 2023
c6b8491
lazy load versions until dropdown is opened
luvkapur May 4, 2023
e61fac9
fix component compare to lazy load
luvkapur May 4, 2023
a1e492b
fix tests
luvkapur May 5, 2023
239adc2
fix loader - when lazy loading versions
luvkapur May 8, 2023
2ae24b0
query from head when on a version + fix loading state
luvkapur May 9, 2023
f7679a1
Merge remote-tracking branch 'origin/master' into ui/lazy-load-logs
luvkapur May 11, 2023
9e14c51
refator getLogs to allow fetching versions with advanced filters
luvkapur May 12, 2023
1fa4b10
clean up
luvkapur May 12, 2023
93c92d6
lint fix
luvkapur May 12, 2023
ae77ed8
fix updating offset while lazy loading
luvkapur May 12, 2023
72b64f2
allow lazy loading by type
luvkapur May 15, 2023
9027cd6
fix lazy loading backwards
luvkapur May 16, 2023
b95d6e4
refactor: extract to useComponentLogs
luvkapur May 16, 2023
3516bf5
clean up
luvkapur May 16, 2023
2c1ebd1
clean up
luvkapur May 16, 2023
09acbaf
clean up
luvkapur May 16, 2023
46ba4ba
fix handling getting logs for lane components without head filter
luvkapur May 17, 2023
5007b17
Merge branch 'master' into ui/lazy-load-logs
luvkapur May 17, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions .bitmap
Original file line number Diff line number Diff line change
Expand Up @@ -1329,6 +1329,12 @@
"mainFile": "index.ts",
"rootDir": "scopes/code/ui/code-tab-tree"
},
"ui/code-view": {
"scope": "teambit.code",
"version": "5e2b49420c500d4d88e1b05aa232061abc0fa0a7",
"mainFile": "index.ts",
"rootDir": "components/ui/code-view"
},
"ui/compare/lane-compare": {
"scope": "teambit.lanes",
"version": "0.0.77",
Expand Down
44 changes: 44 additions & 0 deletions components/ui/code-view/code-view.module.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
.codeView {
padding: 24px 40px;
width: 100%;
overflow: hidden;
box-sizing: border-box;

.codeSnippetWrapper {
width: 100%;
max-width: 100%;
.codeSnippet {
display: block;
overflow: auto;
height: 100vh;
luvkapur marked this conversation as resolved.
Show resolved Hide resolved
> code {
> code {
// this is to design the line numbers culumn
border-right: 1px solid #323232;
padding-right: 16px !important;
margin-right: 16px;
}
}
}
// TODO - fix this in code snippet component. it breaks when the component renders in different places
> span {
top: 13px !important;
}
}
}

.img {
width: 20px;
margin-right: 10px;
}

.fileName {
display: flex;
align-items: baseline;
}

.emptyCodeView {
margin: auto;
text-align: center;
font-size: 24px;
}
76 changes: 76 additions & 0 deletions components/ui/code-view/code-view.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
import { H1 } from '@teambit/documenter.ui.heading';
import classNames from 'classnames';
import React, { HTMLAttributes, useMemo } from 'react';
import { CodeSnippet } from '@teambit/documenter.ui.code-snippet';
import { useFileContent } from '@teambit/code.ui.queries.get-file-content';
import SyntaxHighlighter from 'react-syntax-highlighter/dist/esm/prism-light';
import markDownSyntax from 'react-syntax-highlighter/dist/esm/languages/prism/markdown';
import { staticStorageUrl } from '@teambit/base-ui.constants.storage';
import { ComponentID } from '@teambit/component';
import styles from './code-view.module.scss';

export type CodeViewProps = {
componentId: ComponentID;
currentFile?: string;
currentFileContent?: string;
icon?: string;
loading?: boolean;
codeSnippetClassName?: string;
} & HTMLAttributes<HTMLDivElement>;

SyntaxHighlighter.registerLanguage('md', markDownSyntax);

export function CodeView({
className,
componentId,
currentFile,
icon,
currentFileContent,
codeSnippetClassName,
loading: loadingFromProps,
}: CodeViewProps) {
const { fileContent: downloadedFileContent, loading: loadingFileContent } = useFileContent(
componentId,
currentFile,
!!currentFileContent
);
const loading = loadingFromProps || loadingFileContent;
const fileContent = currentFileContent || downloadedFileContent;
const title = useMemo(() => currentFile?.split('/').pop(), [currentFile]);
const lang = useMemo(() => {
const langFromFileEnding = currentFile?.split('.').pop();

// for some reason, SyntaxHighlighter doesnt support scss or sass highlighting, only css. I need to check how to fix this properly
if (langFromFileEnding === 'scss' || langFromFileEnding === 'sass') return 'css';
if (langFromFileEnding === 'mdx') return 'md';
return langFromFileEnding;
}, [fileContent]);

if (!fileContent && !loading && currentFile) return <EmptyCodeView />;

return (
<div className={classNames(styles.codeView, className)}>
<H1 size="sm" className={styles.fileName}>
{currentFile && <img className={styles.img} src={icon} />}
<span>{title}</span>
</H1>
<CodeSnippet
className={styles.codeSnippetWrapper}
frameClass={classNames(styles.codeSnippet, codeSnippetClassName)}
showLineNumbers
language={lang}
>
{fileContent || ''}
</CodeSnippet>
</div>
);
}

function EmptyCodeView() {
return (
<div className={styles.emptyCodeView}>
<img src={`${staticStorageUrl}/harmony/empty-code-view.svg`} />
<div>Nothing to show</div>
</div>
);
}
2 changes: 2 additions & 0 deletions components/ui/code-view/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export { CodeView } from './code-view';
export type { CodeViewProps } from './code-view';
4 changes: 3 additions & 1 deletion scopes/code/ui/code-tab-page/code-tab-page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -32,9 +32,10 @@ import styles from './code-tab-page.module.scss';
export type CodePageProps = {
fileIconSlot?: FileIconSlot;
host: string;
codeViewClassName?: string;
} & HTMLAttributes<HTMLDivElement>;

export function CodePage({ className, fileIconSlot, host }: CodePageProps) {
export function CodePage({ className, fileIconSlot, host, codeViewClassName }: CodePageProps) {
const urlParams = useCodeParams();
const component = useContext(ComponentContext);
const { mainFile, fileTree = [], dependencies, devFiles } = useCode(component.id);
Expand Down Expand Up @@ -67,6 +68,7 @@ export function CodePage({ className, fileIconSlot, host }: CodePageProps) {
<SplitPane layout={sidebarOpenness} size="85%" className={classNames(styles.codePage, className)}>
<Pane className={styles.left}>
<CodeView
codeSnippetClassName={codeViewClassName}
componentId={component.id}
currentFile={currentFile}
icon={icon}
Expand Down
4 changes: 3 additions & 1 deletion scopes/component/changelog/changelog.section.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,9 +4,11 @@ import { MenuWidgetIcon } from '@teambit/ui-foundation.ui.menu-widget-icon';
import { ChangeLogPage } from './ui/change-log-page';

export class ChangelogSection implements Section {
constructor(private host: string) {}

route = {
path: '~changelog',
element: <ChangeLogPage />,
element: <ChangeLogPage host={this.host} />,
};
navigationLink = {
href: '~changelog',
Expand Down
12 changes: 8 additions & 4 deletions scopes/component/changelog/changelog.ui.runtime.tsx
Original file line number Diff line number Diff line change
@@ -1,23 +1,27 @@
import { ComponentAspect, ComponentUI } from '@teambit/component';
import { UIRuntime } from '@teambit/ui';
import { Harmony } from '@teambit/harmony';
import React from 'react';

import { ChangelogAspect } from './changelog.aspect';
import { ChangelogSection } from './changelog.section';
import { ChangeLogPage } from './ui/change-log-page';

export class ChangeLogUI {
constructor(private host: string) {}
ChangeLog = () => {
return <ChangeLogPage />;
return <ChangeLogPage host={this.host} />;
};

static dependencies = [ComponentAspect];

static runtime = UIRuntime;

static async provider([component]: [ComponentUI]) {
const ui = new ChangeLogUI();
const section = new ChangelogSection();
static async provider([component]: [ComponentUI], _, __, harmony: Harmony) {
const { config } = harmony;
const host = String(config.get('teambit.harmony/bit'));
const ui = new ChangeLogUI(host);
const section = new ChangelogSection(host);

component.registerRoute(section.route);
component.registerWidget(section.navigationLink, section.order);
Expand Down
59 changes: 47 additions & 12 deletions scopes/component/changelog/ui/change-log-page.tsx
Original file line number Diff line number Diff line change
@@ -1,24 +1,58 @@
import { ComponentContext } from '@teambit/component';
import React, { HTMLAttributes } from 'react';
import { ComponentContext, useComponent } from '@teambit/component';
import { H1 } from '@teambit/documenter.ui.heading';
import { Separator } from '@teambit/design.ui.separator';
import { VersionBlock } from '@teambit/component.ui.version-block';
import classNames from 'classnames';
import { MDXLayout } from '@teambit/mdx.ui.mdx-layout';
import { ExportingComponents } from '@teambit/component.instructions.exporting-components';
import { AlertCard } from '@teambit/design.ui.alert-card';
import React, { HTMLAttributes, useContext } from 'react';

import styles from './change-log-page.module.scss';

type ChangeLogPageProps = {} & HTMLAttributes<HTMLDivElement>;
type ChangeLogPageProps = {
host: string;
} & HTMLAttributes<HTMLDivElement>;

export function ChangeLogPage({ className }: ChangeLogPageProps) {
const component = useContext(ComponentContext);
const { logs } = component;
export function ChangeLogPage({ className, host }: ChangeLogPageProps) {
const componentContext = React.useContext(ComponentContext);
const {
component,
loading,
loadMoreLogs,
hasMoreLogs: hasMore,
} = useComponent(host, componentContext?.id.toString(), {
logFilters: {
log: {
logLimit: 15,
},
},
});

if (!logs) return null;
const logs = component?.logs ?? [];

if (logs.length === 0) {
const observer = React.useRef<IntersectionObserver>();
const handleLoadMore = () => {
loadMoreLogs?.();
};

const lastLogRef = React.useCallback(
luvkapur marked this conversation as resolved.
Show resolved Hide resolved
(node) => {
if (loading) return;
if (observer.current) observer.current.disconnect();
observer.current = new IntersectionObserver((entries) => {
if (entries[0].isIntersecting && hasMore) {
handleLoadMore();
}
});
if (node) observer.current.observe(node);
},
[loading, hasMore]
);

if (loading) return null;

if (logs?.length === 0) {
return (
<div className={classNames(styles.changeLogPage, className)}>
<H1 className={styles.title}>History</H1>
Expand All @@ -42,16 +76,17 @@ export function ChangeLogPage({ className }: ChangeLogPageProps) {
<H1 className={styles.title}>History</H1>
<Separator isPresentational className={styles.separator} />
<div className={styles.logContainer}>
{logs.map((snap, index) => {
const isLatest = component.latest === snap.tag || component.latest === snap.hash;
const isCurrent = component.version === snap.tag || component.version === snap.hash;
{(logs || []).map((snap, index) => {
const isLatest = component?.latest === snap.tag || component?.latest === snap.hash;
const isCurrent = component?.version === snap.tag || component?.version === snap.hash;
return (
<VersionBlock
key={index}
componentId={component.id.fullName}
componentId={component?.id?.fullName || ''}
isLatest={isLatest}
snap={snap}
isCurrent={isCurrent}
ref={index === logs.length - 1 ? lastLogRef : null}
/>
);
})}
Expand Down
2 changes: 1 addition & 1 deletion scopes/component/component/ui/component.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ export function Component({
const componentId = _componentIdStr ? ComponentID.fromString(_componentIdStr) : undefined;
const resolvedComponentIdStr = path || idFromLocation;
const useComponentOptions = {
logFilters: useComponentFilters?.(),
logFilters: useComponentFilters?.() || { log: { logOffset: 0, logLimit: 10 } },
customUseComponent: useComponent,
};

Expand Down
1 change: 1 addition & 0 deletions scopes/component/component/ui/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,4 +6,5 @@ export { ComponentContext, ComponentProvider } from './context';
export { useComponent } from './use-component';
export { TopBarNav } from './top-bar-nav';
export { componentIdFields, componentOverviewFields, componentFields } from './use-component-query';
export type { ComponentQueryResult } from './use-component-query';
export { useIdFromLocation } from './use-component-from-location';
Loading