Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Heatmap #8

Merged
merged 63 commits into from
Oct 10, 2020
Merged
Show file tree
Hide file tree
Changes from 60 commits
Commits
Show all changes
63 commits
Select commit Hold shift + click to select a range
444acc7
header task: first commit
bahobab Sep 25, 2020
d7e6cf6
header task: fixed missing default 'javascript'
bahobab Sep 25, 2020
34c1fd3
header task: fixed default link: '/search/javascript'
bahobab Sep 25, 2020
c6faa67
header after first review
bahobab Sep 27, 2020
f46dc2d
header fixing test
bahobab Sep 27, 2020
ae0229e
implemented unit test
bahobab Sep 27, 2020
84feae2
header: before rebase after app-skeleton-review
bahobab Sep 27, 2020
e74b390
resolved conflicts. Fixed unit tests
bahobab Sep 28, 2020
60db4f7
footer: initial commit
bahobab Sep 28, 2020
a462336
implemented Footer.js and Footer.style.js
bahobab Sep 28, 2020
b7c4094
footer: completed unit tests
bahobab Sep 28, 2020
0b12d62
hero-section: initial commit with with fixes from previous PR
bahobab Sep 28, 2020
0817a00
Merge branch 'master' into hero-section
bahobab Sep 28, 2020
e99e5dc
hero-section: completed unit tests
bahobab Sep 29, 2020
2c55849
Merge branch 'hero-section' of https://github.com/ooloo-io/reddit-tim…
bahobab Sep 29, 2020
88238ac
hero-section: fixed linting issues in unit tests
bahobab Sep 29, 2020
0e72262
hero-section: fixed Apps to have all unit tests pass
bahobab Sep 29, 2020
035041f
hero-section: added missing test descriptions in footer.js
bahobab Sep 29, 2020
880d020
hero-section: refactor components and file structure after review
bahobab Sep 29, 2020
2c629e2
info-section: initial commit
bahobab Sep 29, 2020
97ed18d
Merge branch 'master' into info-section
bahobab Sep 30, 2020
3c568ed
info-section: implemented About and How it works sections
bahobab Sep 30, 2020
c4f0d4d
Merge branch 'info-section' of https://github.com/ooloo-io/reddit-tim…
bahobab Sep 30, 2020
bb4486d
info-section: fix link to https://ooloo.io/employers
bahobab Sep 30, 2020
13bdf67
info-section: implemented react-router-hash-link
bahobab Sep 30, 2020
b4a1e4e
info-section: fixed ooloo.io link
bahobab Sep 30, 2020
ee27a75
info-section: implemented unit tests
bahobab Sep 30, 2020
bf59a70
info-section: fixed linting errors
bahobab Sep 30, 2020
d71c19d
info-section: fixed linting errors
bahobab Sep 30, 2020
3ec116c
info-section: fix unit test using this: https://github.com/testing-li…
bahobab Sep 30, 2020
242aefa
info-section: commit after review
bahobab Oct 1, 2020
2780c04
subreddit-form: initial commit. Setup folder structure
bahobab Oct 1, 2020
5b11f42
Merge branch 'master' into subreddit-form
bahobab Oct 1, 2020
f2c2e2d
subreddit-form: implement tests
bahobab Oct 1, 2020
4bb12dc
Merge branch 'subreddit-form' of https://github.com/ooloo-io/reddit-t…
bahobab Oct 1, 2020
c112f47
subreddit-form: fix lint error in Header.js
bahobab Oct 1, 2020
04cf9d2
subreddit-form: completed unit and e2e tests
bahobab Oct 2, 2020
0de4627
subreddit-form: fixes and cleanup after review
bahobab Oct 2, 2020
56554d7
load-the-data: initial commit: added LoadTheData.js
bahobab Oct 2, 2020
56d31c6
load-the-data: fetching data implemented. But unit test and e2e test …
bahobab Oct 3, 2020
7326c4f
Merge branch 'master' into load-the-data
bahobab Oct 3, 2020
1af6086
load-the-data: troubleshooting tests
bahobab Oct 3, 2020
5a60619
Merge branch 'load-the-data' of https://github.com/ooloo-io/reddit-ti…
bahobab Oct 3, 2020
7f50064
load-the-data: continue troubleshooting tests
bahobab Oct 3, 2020
9179df7
load-the-data: fix linting error
bahobab Oct 3, 2020
02223cf
load-the-data: fix the code to fetch top 500 posts
bahobab Oct 3, 2020
1bd7a8e
load-the-data: troubleshooting unit test - 'invalid json response bod…
bahobab Oct 4, 2020
4361e96
load-the-data: integration test passing
bahobab Oct 4, 2020
e061c4a
load-the-data: refactor code after review. still 1 test to fix in Sea…
bahobab Oct 4, 2020
baaccb4
laod-the-data: all tests pass
bahobab Oct 5, 2020
49ccf49
heatmap: first commit: setup heatmap table
bahobab Oct 6, 2020
dd0d221
Merge branch 'master' into heatmap
bahobab Oct 6, 2020
623001e
heatmap: implemented heatmap architecture
bahobab Oct 6, 2020
b552fcb
Merge branch 'heatmap' of https://github.com/ooloo-io/reddit-timer-ba…
bahobab Oct 6, 2020
cf83cc2
heatmap: implemented user timezone
bahobab Oct 6, 2020
2ccfbe7
heatmap: implemented hover, highlit and timezone
bahobab Oct 7, 2020
1b15f5f
heatmap: completed implementation - start testing
bahobab Oct 7, 2020
8040245
heatmap: refactor component structure - set test skeleton
bahobab Oct 7, 2020
6f72e72
heatmap: complete integration tests
bahobab Oct 8, 2020
33c282b
heatmap: complete. fix user timezone in tests
bahobab Oct 8, 2020
62c79e4
heatmap: refactor code after review
bahobab Oct 9, 2020
ae2d7de
heatmap: fix all test errors. ready to merge pr
bahobab Oct 10, 2020
278ff8e
heatmap: fix last test errors. ready to merge pr
bahobab Oct 10, 2020
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
"eslint-plugin-styled-components-a11y": "^0.0.16",
"history": "^5.0.0",
"jest-fetch-mock": "^3.0.3",
"jest-styled-components": "^7.0.3",
"msw": "^0.21.2",
"prop-types": "^15.7.2",
"react": "^16.13.0",
Expand Down
115 changes: 115 additions & 0 deletions src/__tests__/Heatmap.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
import React from 'react';
import { MemoryRouter, Route } from 'react-router-dom';
import {
render, screen, waitForElementToBeRemoved, within,
} from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import '@testing-library/jest-dom/extend-expect';
import 'jest-styled-components';

import {
cellBackgroundColorMap, mapWeekday, postsTimeSlots, getUserTimeZone,
} from '../config';

import App from '../app';

const weekdays = Object.values(mapWeekday);
const cellBackgroundColor = Object.values(cellBackgroundColorMap);

const setup = (initialPath = '/') => {
let history;

render(
<MemoryRouter initialEntries={[initialPath]}>
<App />
<Route
path="*"
render={(props) => {
history = props.history;
return null;
}}
/>
</MemoryRouter>,
);
return { history };
};

describe('heatmap', () => {
it.each(weekdays)(
'weekdays are shown from Sunday to Saturday', async (weekday) => {
setup('/search/less-than-500-posts');
// find weekday container element tr.weekday
// assert that it contains Sunday, Monday, Tuesday, Wednesday
// Thursday, Friday, Saturday
const spinner = screen.getByText('loading-spinner.svg');

await waitForElementToBeRemoved(spinner);

screen.getByRole('table');
// weekday
screen.getByRole('cell', { name: weekday });
},
);

it.each(postsTimeSlots)(
'hours are shown from 12:00am to 10:00pm', async (postsTimeSlot) => {
setup('/search/reactjs');
// find hours container element th.timeline
// assert that it contains 12:00am, 2:00am..
const spinner = screen.getByText('loading-spinner.svg');

await waitForElementToBeRemoved(spinner);

screen.getByRole('table');

const timeSlots = screen.getAllByRole('columnheader');
// screen.debug(timeSlot);
within(timeSlots[1]).getByText(postsTimeSlot);
},
);

it('each combination of weekday and hour is represented by a square box', async () => {
setup('/search/less-than-500-posts');

const spinner = screen.getByText('loading-spinner.svg');
await waitForElementToBeRemoved(spinner);
screen.getByRole('table');

const cells = await document.querySelector('td.cell');
// select cells td.cell
const el = within(cells).getByText(cells.textContent);
const numPosts = Number(el.textContent);

expect(el).toHaveStyle({ border: 'unset', backgroundColor: '#a0ce93' });
// assert that background-color correspond to numeric innerText value
expect(el).toHaveStyle({ backgroundColor: `${cellBackgroundColor[numPosts]}` });

// assert that each has width === height
expect(el).toHaveStyleRule('width', '40px');
expect(el).toHaveStyleRule('height', '40px');

// assert that innerText.value is a number
// expect(el).toHaveValue(numPosts);

// assert that element is highlighted when hovered
userEvent.hover(el);
expect(el).toHaveStyle({ border: '1px solid red' });
userEvent.unhover(el);
expect(el).toHaveStyle({ border: 'unset' });

// assert that element is highlighted when clicked
userEvent.click(el);
expect(el).toHaveStyle({ border: '1px solid red' });
});

it("the user' timezone is shown below the heatmap", async () => {
setup('/search/less-than-500-posts');
// select div.timezone
// assert that the element contains "America/Chicago - Central Saving Time"
const spinner = screen.getByText('loading-spinner.svg');

await waitForElementToBeRemoved(spinner);
screen.getByRole('table');
screen.getByText(getUserTimeZone());
});
});
47 changes: 47 additions & 0 deletions src/config.js
Original file line number Diff line number Diff line change
@@ -1,2 +1,49 @@
// eslint-disable-next-line import/prefer-default-export
export const defaultSubReddit = 'javascript';

export const cellHighlight = '1px solid red';

export const mapWeekday = {
0: 'Sunday',
1: 'Monday',
2: 'Tuesday',
3: 'Wednesday',
4: 'Thursday',
5: 'Friday',
6: 'Saturday',
};

export const cellBackgroundColorMap = {
0: '#e0e592',
1: '#aed396',
2: '#a9d194',
3: '#a0ce93',
4: '#99cd94',
5: '#8cc894',
6: '#5eb391',
7: '#5db492',
8: '#5cb391',
9: '#5aad8c',
10: '#3984a3',
};

export const postsTimeSlots = [
'12:00am',
'2:00am',
'4:00am',
'6:00am',
'8:00am',
'10:00am',
'12:00pm',
'2:00pm',
'4:00pm',
'6:00pm',
'8:00pm',
'10:00pm',
];

export function getUserTimeZone() {
const globalTimeZone = Intl.DateTimeFormat().resolvedOptions().timeZone;
const localTimeZone = (new Date()).toTimeString().split('(')[1].replace(')', '');
return `${globalTimeZone} - ${localTimeZone}`;
}
59 changes: 59 additions & 0 deletions src/page-search/CellWrapper.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
import React, { useState } from 'react';
import { array } from 'prop-types';

import { cellBackgroundColorMap, cellHighlight } from '../config';
import { Cell } from './PostsTable.style';

function CellWrapper({ postsInHour }) {
const numPosts = postsInHour.length;
function getCellBGColor() {
return cellBackgroundColorMap[numPosts] || cellBackgroundColorMap[10];
}
const cellDefaultStyle = {
border: 'unset',
backgroundColor: `${getCellBGColor()}`,
};

const [cellStyle, setCellStyle] = useState(cellDefaultStyle);
const [isSelected, setIsSelected] = useState(false);

function handleCellClick() {
setCellStyle(
{ ...cellStyle, border: '1px solid red' },
);
setIsSelected(true);
}

function handleOnMouseEnter(event) {
if (event.target.style.border !== cellHighlight) {
// eslint-disable-next-line no-param-reassign
event.target.style.border = cellHighlight;
}
}

function handleOnMouseLeave(event) {
if (!isSelected) {
// eslint-disable-next-line no-param-reassign
event.target.style.border = 'none';
}
}

return (
<Cell
style={cellStyle}
onClick={handleCellClick}
onMouseEnter={handleOnMouseEnter}
onMouseLeave={handleOnMouseLeave}
className="cell"
>
{numPosts}
</Cell>
);
}

CellWrapper.propTypes = {
// eslint-disable-next-line react/forbid-prop-types
postsInHour: array.isRequired,
};

export default CellWrapper;
Empty file.
6 changes: 5 additions & 1 deletion src/page-search/Heatmap.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import { useParams } from 'react-router-dom';

import useFetchPosts from './useFetchPosts';
import { ErrorContainer, LoadingContainer, LoadingSpinner } from './Heatmap.style';
import PostsTable from './PostsTable';

function Heatmap() {
const { subreddit } = useParams(); // route param set in App
Expand All @@ -25,7 +26,10 @@ function Heatmap() {
}

return (
<div>{posts.length}</div>
<>
<div>{posts.length}</div>
<PostsTable posts={posts} />
</>
);
}

Expand Down
97 changes: 97 additions & 0 deletions src/page-search/PostsTable.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
import React from 'react';
import { array } from 'prop-types';

import CellWrapper from './CellWrapper';

import { mapWeekday, getUserTimeZone, postsTimeSlots } from '../config';
import {
WeekRow, Weekday, TimeFrame, HeatmapTable, TimeSlice, TimeSliceWrapper,
} from './PostsTable.style';

function PostsTable({ posts }) {
// let myHeatmap;

const postsPerHourPerDay = new Array(7)
.fill([])
.map(() => new Array(24)
.fill([])
.map(() => []));

function getDayTimeFromTimeCreated(postCreatedAt) {
// https://duckduckgo.com/?q=javascript+date+always+point+to+epoch+time&t=ffab&atb=v196-1&ia=web
const d = new Date(0);
const day = new Date(d.setUTCSeconds(postCreatedAt));
return {
weekday: day.getDay(),
timeOfDay: day.getHours(),
};
}

function buildPostsPerHourPerDayTable(subredditPosts) {
subredditPosts.forEach((post) => {
const {
id, author, created, title, url,
} = post.data;
const { weekday, timeOfDay } = getDayTimeFromTimeCreated(created);
postsPerHourPerDay[Number(weekday)][Number(timeOfDay)].push({
id, author, title, url, weekday, timeOfDay,
});
});
}

function generateHeatmap() {
buildPostsPerHourPerDayTable(posts);
// try {
return postsPerHourPerDay.map((weekDay, weekDayIndex) => {
// console.log('weekday', weekDay);
const weekday = mapWeekday[weekDayIndex];
return (
<WeekRow key={weekday}>
<Weekday className="weekday">{weekday}</Weekday>
{weekDay.map((postsInHour, hourIndex) => (
// eslint-disable-next-line react/no-array-index-key
<CellWrapper key={hourIndex} postsInHour={postsInHour} />
))}
</WeekRow>
);
});
// } catch (error) {
// console.log('XXXXXXXXXXX', error);
// }
}

const timeLine = (
<TimeSliceWrapper>
{
postsTimeSlots.map((timeSlot) => (
<TimeSlice key={timeSlot}>{timeSlot}</TimeSlice>
))
}
</TimeSliceWrapper>
);

return (
<div>
<HeatmapTable>
<TimeFrame>
<tr>
{/* eslint-disable-next-line jsx-a11y/control-has-associated-label */}
<th />
<th className="timeline" colSpan="24">{timeLine}</th>
</tr>
</TimeFrame>
<tbody>
{generateHeatmap()}
</tbody>
</HeatmapTable>
<div className="timezone" style={{ textAlign: 'center' }}>{getUserTimeZone()}</div>
</div>
);
}

PostsTable.propTypes = {
// eslint-disable-next-line react/forbid-prop-types
posts: array.isRequired,
};

export default PostsTable;
Loading