|
|
@ -1,12 +1,9 @@
|
|
|
|
import _ from 'lodash';
|
|
|
|
|
|
|
|
import PropTypes from 'prop-types';
|
|
|
|
import PropTypes from 'prop-types';
|
|
|
|
import React, { Component } from 'react';
|
|
|
|
import React, { Component } from 'react';
|
|
|
|
import ReactDOM from 'react-dom';
|
|
|
|
|
|
|
|
import { Grid, WindowScroller } from 'react-virtualized';
|
|
|
|
import { Grid, WindowScroller } from 'react-virtualized';
|
|
|
|
import getIndexOfFirstCharacter from 'Utilities/Array/getIndexOfFirstCharacter';
|
|
|
|
import getIndexOfFirstCharacter from 'Utilities/Array/getIndexOfFirstCharacter';
|
|
|
|
import hasDifferentItemsOrOrder from 'Utilities/Object/hasDifferentItemsOrOrder';
|
|
|
|
import hasDifferentItemsOrOrder from 'Utilities/Object/hasDifferentItemsOrOrder';
|
|
|
|
import dimensions from 'Styles/Variables/dimensions';
|
|
|
|
import dimensions from 'Styles/Variables/dimensions';
|
|
|
|
import { sortDirections } from 'Helpers/Props';
|
|
|
|
|
|
|
|
import Measure from 'Components/Measure';
|
|
|
|
import Measure from 'Components/Measure';
|
|
|
|
import ArtistIndexItemConnector from 'Artist/Index/ArtistIndexItemConnector';
|
|
|
|
import ArtistIndexItemConnector from 'Artist/Index/ArtistIndexItemConnector';
|
|
|
|
import ArtistIndexOverview from './ArtistIndexOverview';
|
|
|
|
import ArtistIndexOverview from './ArtistIndexOverview';
|
|
|
@ -66,56 +63,44 @@ class ArtistIndexOverviews extends Component {
|
|
|
|
rowHeight: calculateRowHeight(238, null, props.isSmallScreen, {})
|
|
|
|
rowHeight: calculateRowHeight(238, null, props.isSmallScreen, {})
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
this._isInitialized = false;
|
|
|
|
|
|
|
|
this._grid = null;
|
|
|
|
this._grid = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
componentDidUpdate(prevProps, prevState) {
|
|
|
|
this._contentBodyNode = ReactDOM.findDOMNode(this.props.contentBody);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
componentDidUpdate(prevProps) {
|
|
|
|
|
|
|
|
const {
|
|
|
|
const {
|
|
|
|
items,
|
|
|
|
items,
|
|
|
|
filters,
|
|
|
|
|
|
|
|
sortKey,
|
|
|
|
sortKey,
|
|
|
|
sortDirection,
|
|
|
|
|
|
|
|
overviewOptions,
|
|
|
|
overviewOptions,
|
|
|
|
jumpToCharacter
|
|
|
|
jumpToCharacter
|
|
|
|
} = this.props;
|
|
|
|
} = this.props;
|
|
|
|
|
|
|
|
|
|
|
|
const itemsChanged = hasDifferentItemsOrOrder(prevProps.items, items);
|
|
|
|
const {
|
|
|
|
const overviewOptionsChanged = !_.isMatch(prevProps.overviewOptions, overviewOptions);
|
|
|
|
width,
|
|
|
|
|
|
|
|
rowHeight
|
|
|
|
|
|
|
|
} = this.state;
|
|
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
if (prevProps.sortKey !== sortKey ||
|
|
|
|
prevProps.sortKey !== sortKey ||
|
|
|
|
prevProps.overviewOptions !== overviewOptions) {
|
|
|
|
prevProps.overviewOptions !== overviewOptions ||
|
|
|
|
|
|
|
|
itemsChanged
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
this.calculateGrid();
|
|
|
|
this.calculateGrid();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
if (this._grid &&
|
|
|
|
prevProps.filters !== filters ||
|
|
|
|
(prevState.width !== width ||
|
|
|
|
prevProps.sortKey !== sortKey ||
|
|
|
|
prevState.rowHeight !== rowHeight ||
|
|
|
|
prevProps.sortDirection !== sortDirection ||
|
|
|
|
hasDifferentItemsOrOrder(prevProps.items, items))) {
|
|
|
|
itemsChanged ||
|
|
|
|
// recomputeGridSize also forces Grid to discard its cache of rendered cells
|
|
|
|
overviewOptionsChanged
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
this._grid.recomputeGridSize();
|
|
|
|
this._grid.recomputeGridSize();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (jumpToCharacter != null && jumpToCharacter !== prevProps.jumpToCharacter) {
|
|
|
|
if (jumpToCharacter != null && jumpToCharacter !== prevProps.jumpToCharacter) {
|
|
|
|
const index = getIndexOfFirstCharacter(items, jumpToCharacter);
|
|
|
|
const index = getIndexOfFirstCharacter(items, jumpToCharacter);
|
|
|
|
|
|
|
|
|
|
|
|
if (index != null) {
|
|
|
|
if (this._grid && index != null) {
|
|
|
|
const {
|
|
|
|
|
|
|
|
rowHeight
|
|
|
|
|
|
|
|
} = this.state;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const scrollTop = rowHeight * index;
|
|
|
|
this._grid.scrollToCell({
|
|
|
|
|
|
|
|
rowIndex: index,
|
|
|
|
this.props.onScroll({ scrollTop });
|
|
|
|
columnIndex: 0
|
|
|
|
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -123,21 +108,6 @@ class ArtistIndexOverviews extends Component {
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// Control
|
|
|
|
// Control
|
|
|
|
|
|
|
|
|
|
|
|
scrollToFirstCharacter(character) {
|
|
|
|
|
|
|
|
const items = this.props.items;
|
|
|
|
|
|
|
|
const {
|
|
|
|
|
|
|
|
rowHeight
|
|
|
|
|
|
|
|
} = this.state;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const index = getIndexOfFirstCharacter(items, character);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (index != null) {
|
|
|
|
|
|
|
|
const scrollTop = rowHeight * index;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
this.props.onScroll({ scrollTop });
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
setGridRef = (ref) => {
|
|
|
|
setGridRef = (ref) => {
|
|
|
|
this._grid = ref;
|
|
|
|
this._grid = ref;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -217,22 +187,14 @@ class ArtistIndexOverviews extends Component {
|
|
|
|
this.calculateGrid(width, this.props.isSmallScreen);
|
|
|
|
this.calculateGrid(width, this.props.isSmallScreen);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
onSectionRendered = () => {
|
|
|
|
|
|
|
|
if (!this._isInitialized && this._contentBodyNode) {
|
|
|
|
|
|
|
|
this.props.onRender();
|
|
|
|
|
|
|
|
this._isInitialized = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// Render
|
|
|
|
// Render
|
|
|
|
|
|
|
|
|
|
|
|
render() {
|
|
|
|
render() {
|
|
|
|
const {
|
|
|
|
const {
|
|
|
|
items,
|
|
|
|
items,
|
|
|
|
scrollTop,
|
|
|
|
|
|
|
|
isSmallScreen,
|
|
|
|
isSmallScreen,
|
|
|
|
onScroll
|
|
|
|
scroller
|
|
|
|
} = this.props;
|
|
|
|
} = this.props;
|
|
|
|
|
|
|
|
|
|
|
|
const {
|
|
|
|
const {
|
|
|
@ -241,29 +203,39 @@ class ArtistIndexOverviews extends Component {
|
|
|
|
} = this.state;
|
|
|
|
} = this.state;
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
return (
|
|
|
|
<Measure onMeasure={this.onMeasure}>
|
|
|
|
<Measure
|
|
|
|
|
|
|
|
whitelist={['width']}
|
|
|
|
|
|
|
|
onMeasure={this.onMeasure}
|
|
|
|
|
|
|
|
>
|
|
|
|
<WindowScroller
|
|
|
|
<WindowScroller
|
|
|
|
scrollElement={isSmallScreen ? undefined : this._contentBodyNode}
|
|
|
|
scrollElement={isSmallScreen ? undefined : scroller}
|
|
|
|
onScroll={onScroll}
|
|
|
|
|
|
|
|
>
|
|
|
|
>
|
|
|
|
{({ height, isScrolling }) => {
|
|
|
|
{({ height, registerChild, onChildScroll, scrollTop }) => {
|
|
|
|
|
|
|
|
if (!height) {
|
|
|
|
|
|
|
|
return <div />;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
return (
|
|
|
|
<Grid
|
|
|
|
<div ref={registerChild}>
|
|
|
|
ref={this.setGridRef}
|
|
|
|
<Grid
|
|
|
|
className={styles.grid}
|
|
|
|
ref={this.setGridRef}
|
|
|
|
autoHeight={true}
|
|
|
|
className={styles.grid}
|
|
|
|
height={height}
|
|
|
|
autoHeight={true}
|
|
|
|
columnCount={1}
|
|
|
|
height={height}
|
|
|
|
columnWidth={width}
|
|
|
|
columnCount={1}
|
|
|
|
rowCount={items.length}
|
|
|
|
columnWidth={width}
|
|
|
|
rowHeight={rowHeight}
|
|
|
|
rowCount={items.length}
|
|
|
|
width={width}
|
|
|
|
rowHeight={rowHeight}
|
|
|
|
scrollTop={scrollTop}
|
|
|
|
width={width}
|
|
|
|
overscanRowCount={2}
|
|
|
|
onScroll={onChildScroll}
|
|
|
|
cellRenderer={this.cellRenderer}
|
|
|
|
scrollTop={scrollTop}
|
|
|
|
onSectionRendered={this.onSectionRendered}
|
|
|
|
overscanRowCount={2}
|
|
|
|
isScrollingOptOut={true}
|
|
|
|
cellRenderer={this.cellRenderer}
|
|
|
|
/>
|
|
|
|
onSectionRendered={this.onSectionRendered}
|
|
|
|
|
|
|
|
scrollToAlignment={'start'}
|
|
|
|
|
|
|
|
isScrollingOptOut={true}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -275,20 +247,16 @@ class ArtistIndexOverviews extends Component {
|
|
|
|
|
|
|
|
|
|
|
|
ArtistIndexOverviews.propTypes = {
|
|
|
|
ArtistIndexOverviews.propTypes = {
|
|
|
|
items: PropTypes.arrayOf(PropTypes.object).isRequired,
|
|
|
|
items: PropTypes.arrayOf(PropTypes.object).isRequired,
|
|
|
|
filters: PropTypes.arrayOf(PropTypes.object).isRequired,
|
|
|
|
|
|
|
|
sortKey: PropTypes.string,
|
|
|
|
sortKey: PropTypes.string,
|
|
|
|
sortDirection: PropTypes.oneOf(sortDirections.all),
|
|
|
|
|
|
|
|
overviewOptions: PropTypes.object.isRequired,
|
|
|
|
overviewOptions: PropTypes.object.isRequired,
|
|
|
|
scrollTop: PropTypes.number.isRequired,
|
|
|
|
scrollTop: PropTypes.number.isRequired,
|
|
|
|
jumpToCharacter: PropTypes.string,
|
|
|
|
jumpToCharacter: PropTypes.string,
|
|
|
|
contentBody: PropTypes.object.isRequired,
|
|
|
|
scroller: PropTypes.instanceOf(Element).isRequired,
|
|
|
|
showRelativeDates: PropTypes.bool.isRequired,
|
|
|
|
showRelativeDates: PropTypes.bool.isRequired,
|
|
|
|
shortDateFormat: PropTypes.string.isRequired,
|
|
|
|
shortDateFormat: PropTypes.string.isRequired,
|
|
|
|
longDateFormat: PropTypes.string.isRequired,
|
|
|
|
longDateFormat: PropTypes.string.isRequired,
|
|
|
|
isSmallScreen: PropTypes.bool.isRequired,
|
|
|
|
isSmallScreen: PropTypes.bool.isRequired,
|
|
|
|
timeFormat: PropTypes.string.isRequired,
|
|
|
|
timeFormat: PropTypes.string.isRequired
|
|
|
|
onRender: PropTypes.func.isRequired,
|
|
|
|
|
|
|
|
onScroll: PropTypes.func.isRequired
|
|
|
|
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
export default ArtistIndexOverviews;
|
|
|
|
export default ArtistIndexOverviews;
|
|
|
|