/** @format */ /** * External dependencies */ import { __, sprintf } from '@wordpress/i18n'; import { Component, Fragment } from '@wordpress/element'; import { format as formatDate } from '@wordpress/date'; import { map, orderBy } from 'lodash'; /** * Internal dependencies */ import { Card, Link, OrderStatus, TableCard, TablePlaceholder, ViewMoreList, } from '@woocommerce/components'; import { downloadCSVFile, generateCSVDataFromTable, generateCSVFileName } from 'lib/csv'; import { formatCurrency, getCurrencyFormatDecimal } from 'lib/currency'; import { getIntervalForQuery, getDateFormatsForInterval } from 'lib/date'; import { getAdminLink, onQueryChange } from 'lib/nav-utils'; import './style.scss'; export default class OrdersReportTable extends Component { constructor( props ) { super( props ); } onDownload( headers, rows, query ) { // @TODO The current implementation only downloads the contents displayed in the table. // Another solution is required when the data set is larger (see #311). return () => { downloadCSVFile( generateCSVFileName( 'orders', query ), generateCSVDataFromTable( headers, rows ) ); }; } getHeadersContent() { return [ { label: __( 'Date', 'wc-admin' ), key: 'date', required: true, defaultSort: true, isLeftAligned: true, isSortable: true, }, { label: __( 'Order #', 'wc-admin' ), key: 'id', required: true, isSortable: true, }, { label: __( 'Status', 'wc-admin' ), key: 'status', required: false, isSortable: false, }, { label: __( 'Customer', 'wc-admin' ), key: 'customer_id', required: false, isSortable: false, }, { label: __( 'Product(s)', 'wc-admin' ), key: 'products', required: false, isSortable: false, }, { label: __( 'Items Sold', 'wc-admin' ), key: 'items_sold', required: false, isSortable: false, isNumeric: true, }, { label: __( 'Coupon(s)', 'wc-admin' ), key: 'coupons', required: false, isSortable: false, }, { label: __( 'N. Revenue', 'wc-admin' ), key: 'net_revenue', required: true, isSortable: false, isNumeric: true, }, ]; } formatTableData( data ) { return map( data, row => { const { date_created, id, status, customer_id, line_items, coupon_lines, currency, total, total_tax, shipping_total, discount_total, } = row; return { date: date_created, id, status, customer_id, line_items, items_sold: line_items.reduce( ( acc, item ) => item.quantity + acc, 0 ), coupon_lines, currency, net_revenue: getCurrencyFormatDecimal( total - total_tax - shipping_total - discount_total ), }; } ); } getRowsContent( tableData ) { const { query } = this.props; const currentInterval = getIntervalForQuery( query ); const { tableFormat } = getDateFormatsForInterval( currentInterval ); return map( tableData, row => { const { date, id, status, customer_id, line_items, items_sold, coupon_lines, currency, net_revenue, } = row; const products = line_items .sort( ( itemA, itemB ) => itemB.quantity - itemA.quantity ) .map( item => ( { label: item.name, href: 'post.php?post=' + item.product_id + '&action=edit', quantity: item.quantity, } ) ); const coupons = coupon_lines.map( coupon => ( { label: coupon.code, // @TODO It should link to the coupons report href: 'edit.php?s=' + coupon.code + '&post_type=shop_coupon', } ) ); return [ { display: formatDate( tableFormat, date ), value: date, }, { display: { id }, value: id, }, { display: ( ), value: status, }, { // @TODO This should display customer type (new/returning) once it's // implemented in the API. display: customer_id, value: customer_id, }, { display: this.renderList( products.length ? [ products[ 0 ] ] : [], products.map( product => ( { label: sprintf( __( '%s× %s', 'wc-admin' ), product.quantity, product.label ), href: product.href, } ) ) ), value: products.map( product => product.label ).join( ' ' ), }, { display: items_sold, value: items_sold, }, { display: this.renderList( coupons.length ? [ coupons[ 0 ] ] : [], coupons ), value: coupons.map( item => item.code ).join( ' ' ), }, { display: formatCurrency( net_revenue, currency ), value: net_revenue, }, ]; } ); } renderLinks( items = [] ) { return items.map( ( item, i ) => ( { item.label } ) ); } renderList( visibleItems, popoverItems ) { return ( { this.renderLinks( visibleItems ) } { popoverItems.length > 1 && } ); } renderPlaceholderTable() { const headers = this.getHeadersContent(); return ( ); } renderTable() { const { orders, query, totalRows } = this.props; const rowsPerPage = parseInt( query.per_page ) || 25; const rows = this.getRowsContent( orderBy( this.formatTableData( orders ), query.orderby || 'date', query.order || 'asc' ) ); const headers = this.getHeadersContent(); const tableQuery = { ...query, orderby: query.orderby || 'date', order: query.order || 'asc', }; return ( ); } render() { const { isRequesting } = this.props; return isRequesting ? this.renderPlaceholderTable() : this.renderTable(); } }