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

[Calendar] Remove promise based loading in favor of loading prop #1829

Merged
merged 5 commits into from
May 28, 2020
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
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
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -36,3 +36,5 @@ coverage
# editors
.vs
.DS_Store

.vercel
1 change: 1 addition & 0 deletions docs/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
.vercel
31 changes: 31 additions & 0 deletions docs/fakeApi/randomDate.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
import { getDaysInMonth, isValid } from 'date-fns';
import { NowRequest, NowResponse } from '@now/node';

function getRandomNumber(min: number, max: number) {
return Math.round(Math.random() * (max - min) + min);
}

export default function(req: NowRequest, res: NowResponse) {
const { month } = req.query;
if (!month || typeof month !== 'string') {
res.status(400);
return res.json({
reason: 'month query param is required',
});
}

const date = new Date(month);
if (!isValid(date)) {
res.status(422);
return res.json({
reason: 'cannot parsable month value',
});
}

setTimeout(() => {
const daysInMonth = getDaysInMonth(date);
const daysToHighlight = [1, 2, 3].map(_ => getRandomNumber(1, daysInMonth));

res.json({ daysToHighlight });
}, 500); // fake some long work
}
2 changes: 2 additions & 0 deletions docs/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
"@material-ui/core": "^4.9.14",
"@material-ui/icons": "^4.9.1",
"@material-ui/pickers": "^4.0.0-alpha.1",
"@now/node": "^1.6.1",
"@types/fuzzy-search": "^2.1.0",
"@types/isomorphic-fetch": "^0.0.35",
"@types/jss": "^10.0.0",
Expand Down Expand Up @@ -57,6 +58,7 @@
"next-images": "^1.4.0",
"next-transpile-modules": "^2.0.0",
"notistack": "^0.9.11",
"now": "^19.0.1",
"prismjs": "^1.20.0",
"raw-loader": "^1.0.0",
"react": "^16.13.0",
Expand Down
40 changes: 24 additions & 16 deletions docs/pages/demo/datepicker/ServerRequest.example.jsx
Original file line number Diff line number Diff line change
@@ -1,31 +1,39 @@
import React, { useState } from 'react';
import * as React from 'react';
import { Badge } from '@material-ui/core';
import { TextField } from '@material-ui/core';
import { DatePicker, Day } from '@material-ui/pickers';
import { makeJSDateObject } from '../../../utils/helpers';

function getRandomNumber(min, max) {
return Math.round(Math.random() * (max - min) + min);
}

function ServerRequest() {
const [selectedDays, setSelectedDays] = useState([1, 2, 15]);
const [selectedDate, handleDateChange] = useState(new Date());

const handleMonthChange = async () => {
// just select random days to simulate server side based data
return new Promise(resolve => {
setTimeout(() => {
setSelectedDays([1, 2, 3].map(() => getRandomNumber(1, 28)));
resolve();
}, 1000);
});
const requestAbortController = React.useRef(null);
const [selectedDays, setSelectedDays] = React.useState([1, 2, 15]);
const [selectedDate, handleDateChange] = React.useState(new Date());

const handleMonthChange = date => {
if (requestAbortController.current) {
requestAbortController.current.abort();
}

setSelectedDays(null)

const controller = new AbortController();
fetch(`/fakeApi/randomDate?month=${date.toString()}`, {
signal: controller.signal,
})
.then(res => res.json())
.then(({ daysToHighlight }) => {
setSelectedDays(daysToHighlight);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What if the component unmounts before the request resolves? I believe the set state call will warn/throw. The solution proposed in could be applied here too mui/mui-x#5 (comment)

Copy link
Member Author

@dmtrKovalenko dmtrKovalenko May 25, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We are saving abort controller. So needs to make an effect that will abort request on unmount. thanks

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's one way to solve this. I have no idea what's the best approach. I guess it's good enough, no need to look further 👍

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Interesting, but I think that this solution has one giant pitfall – it doesn't abort the request.

Probably if we want to support IE we must not use fetch and promises at all.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can requests be aborted? I mean, does it change something at the network layer?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actually request/response model cannot be aborted, but browsers can prevent some calculations on the responses if they are aborted (like processing low-level body and CORS)

Here network log when fast switching between months with request aborts:
image

Without:
image

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice :)

})
.catch(() => console.log("Ooopsy, something went wrong"))

requestAbortController.current = controller;
};

return (
<>
<DatePicker
value={selectedDate}
loading={selectedDays === null}
onChange={date => handleDateChange(date)}
onMonthChange={handleMonthChange}
renderInput={props => <TextField {...props} />}
Expand Down
154 changes: 96 additions & 58 deletions docs/prop-types.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 0 additions & 4 deletions lib/src/views/Calendar/Calendar.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -30,10 +30,6 @@ export interface ExportedCalendarProps
* @default currentWrapper !== 'static'
*/
allowKeyboardControl?: boolean;
/**
* Custom loading indicator.
*/
loadingIndicator?: JSX.Element;
}

export interface CalendarProps extends ExportedCalendarProps {
Expand Down
Loading