2018-04-13 20:25:52 +00:00
|
|
|
/**
|
|
|
|
* @prettier
|
|
|
|
*/
|
2018-04-13 00:45:14 +00:00
|
|
|
import moment from 'moment';
|
2018-04-13 20:25:52 +00:00
|
|
|
import { compact, groupBy, sortBy } from 'lodash';
|
2018-04-13 00:45:14 +00:00
|
|
|
|
2018-04-25 20:42:08 +00:00
|
|
|
import { Message } from './types/Message';
|
2018-04-13 21:54:53 +00:00
|
|
|
// import { missingCaseError } from '../../../util/missingCaseError';
|
2018-04-13 00:45:14 +00:00
|
|
|
|
2018-04-13 20:27:06 +00:00
|
|
|
type StaticSectionType = 'today' | 'yesterday' | 'thisWeek' | 'thisMonth';
|
|
|
|
type YearMonthSectionType = 'yearMonth';
|
|
|
|
|
|
|
|
interface GenericSection<T> {
|
|
|
|
type: T;
|
|
|
|
messages: Array<Message>;
|
|
|
|
}
|
|
|
|
type StaticSection = GenericSection<StaticSectionType>;
|
|
|
|
type YearMonthSection = GenericSection<YearMonthSectionType> & {
|
|
|
|
year: number;
|
|
|
|
month: number;
|
|
|
|
};
|
|
|
|
export type Section = StaticSection | YearMonthSection;
|
2018-04-13 20:25:52 +00:00
|
|
|
export const groupMessagesByDate = (
|
|
|
|
timestamp: number,
|
|
|
|
messages: Array<Message>
|
2018-04-13 20:27:06 +00:00
|
|
|
): Array<Section> => {
|
2018-04-13 00:45:14 +00:00
|
|
|
const referenceDateTime = moment.utc(timestamp);
|
|
|
|
|
2018-04-13 20:27:06 +00:00
|
|
|
const sortedMessages = sortBy(messages, message => -message.received_at);
|
|
|
|
const messagesWithSection = sortedMessages.map(
|
2018-04-24 16:09:09 +00:00
|
|
|
withSection(referenceDateTime)
|
2018-04-13 20:27:06 +00:00
|
|
|
);
|
|
|
|
const groupedMessages = groupBy(messagesWithSection, 'type');
|
|
|
|
const yearMonthMessages = Object.values(
|
|
|
|
groupBy(groupedMessages.yearMonth, 'order')
|
|
|
|
).reverse();
|
|
|
|
return compact([
|
|
|
|
toSection(groupedMessages.today),
|
|
|
|
toSection(groupedMessages.yesterday),
|
|
|
|
toSection(groupedMessages.thisWeek),
|
|
|
|
toSection(groupedMessages.thisMonth),
|
2018-04-24 16:09:17 +00:00
|
|
|
...yearMonthMessages.map(toSection),
|
2018-04-13 20:27:06 +00:00
|
|
|
]);
|
|
|
|
};
|
2018-04-13 00:45:14 +00:00
|
|
|
|
2018-04-13 20:27:06 +00:00
|
|
|
const toSection = (
|
|
|
|
messagesWithSection: Array<MessageWithSection> | undefined
|
|
|
|
): Section | null => {
|
|
|
|
if (!messagesWithSection || messagesWithSection.length === 0) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const firstMessageWithSection: MessageWithSection = messagesWithSection[0];
|
|
|
|
if (!firstMessageWithSection) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const messages = messagesWithSection.map(
|
|
|
|
messageWithSection => messageWithSection.message
|
|
|
|
);
|
|
|
|
switch (firstMessageWithSection.type) {
|
|
|
|
case 'today':
|
|
|
|
case 'yesterday':
|
|
|
|
case 'thisWeek':
|
|
|
|
case 'thisMonth':
|
2018-04-13 00:45:14 +00:00
|
|
|
return {
|
2018-04-13 20:27:06 +00:00
|
|
|
type: firstMessageWithSection.type,
|
2018-04-14 02:09:56 +00:00
|
|
|
messages,
|
2018-04-13 00:45:14 +00:00
|
|
|
};
|
2018-04-13 20:27:06 +00:00
|
|
|
case 'yearMonth':
|
2018-04-13 00:45:14 +00:00
|
|
|
return {
|
2018-04-13 20:27:06 +00:00
|
|
|
type: firstMessageWithSection.type,
|
|
|
|
year: firstMessageWithSection.year,
|
|
|
|
month: firstMessageWithSection.month,
|
|
|
|
messages,
|
2018-04-13 00:45:14 +00:00
|
|
|
};
|
2018-04-13 20:27:06 +00:00
|
|
|
default:
|
|
|
|
// NOTE: Investigate why we get the following error:
|
|
|
|
// error TS2345: Argument of type 'any' is not assignable to parameter
|
|
|
|
// of type 'never'.
|
|
|
|
// return missingCaseError(firstMessageWithSection.type);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-04-14 02:09:56 +00:00
|
|
|
interface GenericMessageWithSection<T> {
|
2018-04-13 20:27:06 +00:00
|
|
|
order: number;
|
|
|
|
type: T;
|
|
|
|
message: Message;
|
2018-04-14 02:09:56 +00:00
|
|
|
}
|
2018-04-13 20:27:06 +00:00
|
|
|
type MessageWithStaticSection = GenericMessageWithSection<StaticSectionType>;
|
|
|
|
type MessageWithYearMonthSection = GenericMessageWithSection<
|
|
|
|
YearMonthSectionType
|
|
|
|
> & {
|
|
|
|
year: number;
|
|
|
|
month: number;
|
|
|
|
};
|
|
|
|
type MessageWithSection =
|
|
|
|
| MessageWithStaticSection
|
|
|
|
| MessageWithYearMonthSection;
|
2018-04-13 00:45:14 +00:00
|
|
|
|
2018-04-24 16:09:09 +00:00
|
|
|
const withSection = (referenceDateTime: moment.Moment) => (
|
|
|
|
message: Message
|
|
|
|
): MessageWithSection => {
|
|
|
|
const today = moment(referenceDateTime).startOf('day');
|
|
|
|
const yesterday = moment(referenceDateTime)
|
|
|
|
.subtract(1, 'day')
|
|
|
|
.startOf('day');
|
|
|
|
const thisWeek = moment(referenceDateTime).startOf('isoWeek');
|
|
|
|
const thisMonth = moment(referenceDateTime).startOf('month');
|
|
|
|
|
2018-04-13 20:27:06 +00:00
|
|
|
const messageReceivedDate = moment.utc(message.received_at);
|
|
|
|
if (messageReceivedDate.isAfter(today)) {
|
|
|
|
return {
|
|
|
|
order: 0,
|
|
|
|
type: 'today',
|
|
|
|
message,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
if (messageReceivedDate.isAfter(yesterday)) {
|
|
|
|
return {
|
|
|
|
order: 1,
|
|
|
|
type: 'yesterday',
|
|
|
|
message,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
if (messageReceivedDate.isAfter(thisWeek)) {
|
|
|
|
return {
|
|
|
|
order: 2,
|
|
|
|
type: 'thisWeek',
|
|
|
|
message,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
if (messageReceivedDate.isAfter(thisMonth)) {
|
2018-04-13 00:45:14 +00:00
|
|
|
return {
|
2018-04-13 20:27:06 +00:00
|
|
|
order: 3,
|
|
|
|
type: 'thisMonth',
|
2018-04-13 00:45:14 +00:00
|
|
|
message,
|
|
|
|
};
|
2018-04-13 20:27:06 +00:00
|
|
|
}
|
2018-04-13 00:45:14 +00:00
|
|
|
|
2018-04-13 20:27:06 +00:00
|
|
|
const month: number = messageReceivedDate.month();
|
|
|
|
const year: number = messageReceivedDate.year();
|
|
|
|
return {
|
|
|
|
order: year * 100 + month,
|
|
|
|
type: 'yearMonth',
|
|
|
|
month,
|
|
|
|
year,
|
|
|
|
message,
|
|
|
|
};
|
2018-04-13 00:45:14 +00:00
|
|
|
};
|