/** * External dependencies */ import { __ } from '@wordpress/i18n'; import { useEffect, useRef, useState, createElement, useContext, } from '@wordpress/element'; import { Button, Card } from '@wordpress/components'; import { useSelect, useDispatch } from '@wordpress/data'; import { EllipsisMenu } from '@woocommerce/components'; import { navigateTo, getNewPath } from '@woocommerce/navigation'; import { ONBOARDING_STORE_NAME, TaskType, useUserPreferences, getVisibleTasks, TaskListType, WCDataSelector, } from '@woocommerce/data'; import { recordEvent } from '@woocommerce/tracks'; import { List, useSlot } from '@woocommerce/experimental'; import classnames from 'classnames'; /** * Internal dependencies */ import '../tasks/task-list.scss'; import taskHeaders from './task-headers'; import DismissModal from './dismiss-modal'; import TaskListCompleted from './completed'; import { ProgressHeader } from '~/task-lists/progress-header'; import { TaskListItemTwoColumn } from './task-list-item-two-column'; import { TaskListCompletedHeader } from './completed-header'; import { LayoutContext } from '~/layout'; import { TaskListCompletionSlot, EXPERIMENTAL_WC_TASK_LIST_COMPLETION_SLOT_NAME, } from './task-list-completion-slot'; export type TaskListProps = TaskListType & { eventName?: string; twoColumns?: boolean; query: { task?: string; }; cesHeader?: boolean; }; export const TaskList: React.FC< TaskListProps > = ( { query, id, eventName, eventPrefix, tasks, twoColumns, keepCompletedTaskList, isComplete, displayProgressHeader, cesHeader = true, } ) => { const listEventPrefix = eventName ? eventName + '_' : eventPrefix; const { profileItems } = useSelect( ( select: WCDataSelector ) => { const { getProfileItems } = select( ONBOARDING_STORE_NAME ); return { profileItems: getProfileItems(), }; } ); const { hideTaskList, visitedTask, keepCompletedTaskList: keepCompletedTasks, } = useDispatch( ONBOARDING_STORE_NAME ); const userPreferences = useUserPreferences(); const [ headerData, setHeaderData ] = useState< { task?: TaskType; goToTask?: () => void; trackClick?: () => void; } >( {} ); const [ activeTaskId, setActiveTaskId ] = useState( '' ); const [ showDismissModal, setShowDismissModal ] = useState( false ); const layoutContext = useContext( LayoutContext ); const prevQueryRef = useRef( query ); const visibleTasks = getVisibleTasks( tasks ); const recordTaskListView = () => { if ( query.task ) { return; } recordEvent( `${ listEventPrefix }view`, { number_tasks: visibleTasks.length, store_connected: profileItems.wccom_connected, context: layoutContext.toString(), } ); }; useEffect( () => { recordTaskListView(); }, [] ); const taskListCompletionSlot = useSlot( EXPERIMENTAL_WC_TASK_LIST_COMPLETION_SLOT_NAME ); useEffect( () => { const { task: prevTask } = prevQueryRef.current; const { task } = query; if ( prevTask !== task ) { window.document.documentElement.scrollTop = 0; prevQueryRef.current = query; } }, [ query ] ); const incompleteTasks = tasks.filter( ( task ) => ! task.isComplete && ! task.isDismissed ); const hideTasks = () => { hideTaskList( id ); }; const keepTasks = () => { keepCompletedTasks( id ); }; const renderMenu = () => { return (
void; } ) => (
) } />
); }; 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 getTaskStartedCount = ( taskId: string ) => { const trackedStartedTasks = userPreferences.task_list_tracked_started_tasks; if ( ! trackedStartedTasks || ! trackedStartedTasks[ taskId ] ) { return 0; } return trackedStartedTasks[ taskId ]; }; // @todo This would be better as a task endpoint that handles updating the count. const updateTrackStartedCount = ( taskId: string ) => { const newCount = getTaskStartedCount( taskId ) + 1; const trackedStartedTasks = userPreferences.task_list_tracked_started_tasks || {}; visitedTask( taskId ); userPreferences.updateUserPreferences( { task_list_tracked_started_tasks: { ...( trackedStartedTasks || {} ), [ taskId ]: newCount, }, } ); }; const trackClick = ( task: TaskType ) => { recordEvent( `${ listEventPrefix }click`, { task_name: task.id, context: layoutContext.toString(), } ); }; const goToTask = ( task: TaskType ) => { trackClick( task ); if ( ! task.isComplete ) { updateTrackStartedCount( task.id ); } if ( task.actionUrl ) { navigateTo( { url: task.actionUrl, } ); return; } navigateTo( { url: getNewPath( { task: task.id }, '/', {} ) } ); }; const showTaskHeader = ( task: TaskType ) => { if ( taskHeaders[ task.id ] ) { setHeaderData( { task, goToTask: () => goToTask( task ), trackClick: () => trackClick( task ), } ); setActiveTaskId( task.id ); } }; useEffect( () => { if ( selectedHeaderCard ) { showTaskHeader( selectedHeaderCard ); } }, [ selectedHeaderCard ] ); if ( ! visibleTasks.length ) { return
; } const hasTaskListCompletionSlotFills = Boolean( taskListCompletionSlot?.fills?.length ); if ( isComplete && keepCompletedTaskList !== 'yes' ) { if ( hasTaskListCompletionSlotFills ) { return ( ); } return ( <> { cesHeader ? ( ) : ( ) } ); } return ( <> { showDismissModal && ( ) } { displayProgressHeader ? ( ) : null }
{ headerData?.task && createElement( taskHeaders[ headerData.task.id ], headerData ) }
{ ! displayProgressHeader && renderMenu() }
{ visibleTasks.map( ( task, index ) => { return ( goToTask( task ) } trackClick={ () => trackClick( task ) } /> ); } ) }
); }; export default TaskList;