woocommerce/plugins/woocommerce-admin/client/two-column-tasks/sectioned-task-list.tsx

210 lines
5.2 KiB
TypeScript
Raw Normal View History

2022-03-21 17:04:15 +00:00
/**
* External dependencies
*/
import { useEffect, useRef, useState, useContext } from '@wordpress/element';
2022-03-21 17:04:15 +00:00
import { Panel, PanelBody, PanelRow } from '@wordpress/components';
import { useSelect, useDispatch } from '@wordpress/data';
2022-06-01 17:10:20 +00:00
import { ONBOARDING_STORE_NAME, getVisibleTasks } from '@woocommerce/data';
2022-03-21 17:04:15 +00:00
import { recordEvent } from '@woocommerce/tracks';
import { List } from '@woocommerce/experimental';
2022-03-21 17:04:15 +00:00
import classnames from 'classnames';
/**
* Internal dependencies
*/
import '../tasks/task-list.scss';
import './sectioned-task-list.scss';
import TaskListCompleted from './completed';
import { TaskListProps } from '~/tasks/task-list';
import { ProgressHeader } from '~/task-lists/progress-header';
import { SectionPanelTitle } from './section-panel-title';
import { TaskListItem } from './task-list-item';
import { TaskListCompletedHeader } from './completed-header';
import { LayoutContext } from '~/layout';
2022-03-21 17:04:15 +00:00
type PanelBodyProps = Omit< PanelBody.Props, 'title' | 'onToggle' > & {
title: string | React.ReactNode | undefined;
onToggle?: ( isOpen: boolean ) => void;
};
const PanelBodyWithUpdatedType =
PanelBody as React.ComponentType< PanelBodyProps >;
2022-03-21 17:04:15 +00:00
export const SectionedTaskList: React.FC< TaskListProps > = ( {
query,
id,
eventPrefix,
2022-03-21 17:04:15 +00:00
tasks,
keepCompletedTaskList,
isComplete,
sections,
displayProgressHeader,
cesHeader = true,
2022-03-21 17:04:15 +00:00
} ) => {
const { profileItems } = useSelect( ( select ) => {
2022-03-21 17:04:15 +00:00
const { getProfileItems } = select( ONBOARDING_STORE_NAME );
return {
profileItems: getProfileItems(),
};
} );
const { hideTaskList, keepCompletedTaskList: keepCompletedTasks } =
useDispatch( ONBOARDING_STORE_NAME );
2022-03-21 17:04:15 +00:00
const [ openPanel, setOpenPanel ] = useState< string | null >(
sections?.find( ( section ) => ! section.isComplete )?.id || null
);
const layoutContext = useContext( LayoutContext );
2022-03-21 17:04:15 +00:00
const prevQueryRef = useRef( query );
const visibleTasks = getVisibleTasks( tasks );
2022-03-21 17:04:15 +00:00
const recordTaskListView = () => {
if ( query.task ) {
return;
}
recordEvent( `${ eventPrefix }view`, {
2022-03-21 17:04:15 +00:00
number_tasks: visibleTasks.length,
store_connected: profileItems.wccom_connected,
context: layoutContext.toString(),
2022-03-21 17:04:15 +00:00
} );
};
useEffect( () => {
recordTaskListView();
}, [] );
useEffect( () => {
const { task: prevTask } = prevQueryRef.current;
const { task } = query;
if ( prevTask !== task ) {
window.document.documentElement.scrollTop = 0;
prevQueryRef.current = query;
}
}, [ query ] );
const hideTasks = () => {
hideTaskList( id );
};
const keepTasks = () => {
keepCompletedTasks( id );
2022-03-21 17:04:15 +00:00
};
let selectedHeaderCard = visibleTasks.find(
( listTask ) => listTask.isComplete === false
);
// If nothing is selected, default to the last task since everything is completed.
if ( ! selectedHeaderCard ) {
selectedHeaderCard = visibleTasks[ visibleTasks.length - 1 ];
}
const getSectionTasks = ( sectionTaskIds: string[] ) => {
return visibleTasks.filter( ( task ) =>
sectionTaskIds.includes( task.id )
);
};
if ( ! visibleTasks.length ) {
return <div className="woocommerce-task-dashboard__container"></div>;
}
if ( isComplete && keepCompletedTaskList !== 'yes' ) {
2022-03-21 17:04:15 +00:00
return (
<>
{ cesHeader ? (
<TaskListCompletedHeader
hideTasks={ hideTasks }
keepTasks={ keepTasks }
2022-05-17 17:25:06 +00:00
customerEffortScore={ true }
/>
) : (
<TaskListCompleted
hideTasks={ hideTasks }
keepTasks={ keepTasks }
twoColumns={ false }
/>
) }
2022-03-21 17:04:15 +00:00
</>
);
}
return (
<>
{ displayProgressHeader ? (
<ProgressHeader taskListId={ id } />
) : null }
2022-03-21 17:04:15 +00:00
<div
className={ classnames(
`woocommerce-task-dashboard__container woocommerce-sectioned-task-list two-column-experiment woocommerce-task-list__${ id }`
) }
>
<Panel>
{ ( sections || [] ).map( ( section ) => (
<PanelBodyWithUpdatedType
key={ section.id }
title={
<SectionPanelTitle
section={ section }
tasks={ tasks }
active={ openPanel === section.id }
/>
}
2022-03-21 17:04:15 +00:00
opened={ openPanel === section.id }
onToggle={ ( isOpen: boolean ) => {
if ( ! isOpen && openPanel === section.id ) {
recordEvent(
`${ eventPrefix }section_closed`,
{
id: section.id,
all: true,
}
);
2022-03-21 17:04:15 +00:00
setOpenPanel( null );
} else {
if ( openPanel ) {
recordEvent(
`${ eventPrefix }section_closed`,
{
id: openPanel,
all: false,
}
);
}
2022-03-21 17:04:15 +00:00
setOpenPanel( section.id );
}
if ( isOpen ) {
recordEvent(
`${ eventPrefix }section_opened`,
{
id: section.id,
}
);
}
2022-03-21 17:04:15 +00:00
} }
initialOpen={ false }
>
<PanelRow>
<List animation="custom">
{ getSectionTasks( section.tasks ).map(
( task ) => (
<TaskListItem
key={ task.id }
task={ task }
eventPrefix={ eventPrefix }
/>
)
2022-03-21 17:04:15 +00:00
) }
</List>
</PanelRow>
</PanelBodyWithUpdatedType>
) ) }
</Panel>
</div>
</>
);
};
export default SectionedTaskList;