284 lines
6.3 KiB
JavaScript
284 lines
6.3 KiB
JavaScript
/** @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: <a href={ getAdminLink( 'post.php?post=' + id + '&action=edit' ) }>{ id }</a>,
|
||
value: id,
|
||
},
|
||
{
|
||
display: (
|
||
<OrderStatus className="woocommerce-orders-table__status" order={ { status } } />
|
||
),
|
||
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 ) => (
|
||
<Link href={ item.href } key={ i } type={ 'wp-admin' }>
|
||
{ item.label }
|
||
</Link>
|
||
) );
|
||
}
|
||
|
||
renderList( visibleItems, popoverItems ) {
|
||
return (
|
||
<Fragment>
|
||
{ this.renderLinks( visibleItems ) }
|
||
{ popoverItems.length > 1 && <ViewMoreList items={ this.renderLinks( popoverItems ) } /> }
|
||
</Fragment>
|
||
);
|
||
}
|
||
|
||
renderPlaceholderTable( tableQuery ) {
|
||
const headers = this.getHeadersContent();
|
||
|
||
return (
|
||
<Card
|
||
title={ __( 'Orders', 'wc-admin' ) }
|
||
className="woocommerce-analytics__table-placeholder"
|
||
>
|
||
<TablePlaceholder
|
||
caption={ __( 'Orders', 'wc-admin' ) }
|
||
headers={ headers }
|
||
query={ tableQuery }
|
||
/>
|
||
</Card>
|
||
);
|
||
}
|
||
|
||
renderTable( tableQuery ) {
|
||
const { orders, query, totalRows } = this.props;
|
||
|
||
const rowsPerPage = parseInt( tableQuery.per_page ) || 25;
|
||
const rows = this.getRowsContent(
|
||
orderBy( this.formatTableData( orders ), tableQuery.orderby, tableQuery.order )
|
||
);
|
||
|
||
const headers = this.getHeadersContent();
|
||
|
||
return (
|
||
<TableCard
|
||
title={ __( 'Orders', 'wc-admin' ) }
|
||
rows={ rows }
|
||
totalRows={ totalRows }
|
||
rowsPerPage={ rowsPerPage }
|
||
headers={ headers }
|
||
onClickDownload={ this.onDownload( headers, rows, query ) }
|
||
onQueryChange={ onQueryChange }
|
||
query={ tableQuery }
|
||
summary={ null }
|
||
/>
|
||
);
|
||
}
|
||
|
||
render() {
|
||
const { isRequesting, query } = this.props;
|
||
|
||
const tableQuery = {
|
||
...query,
|
||
orderby: query.orderby || 'date',
|
||
order: query.order || 'asc',
|
||
};
|
||
|
||
return isRequesting
|
||
? this.renderPlaceholderTable( tableQuery )
|
||
: this.renderTable( tableQuery );
|
||
}
|
||
}
|