1
0
mirror of https://github.com/franjsco/tomadoro synced 2025-04-03 02:21:00 +02:00
This commit is contained in:
Francesco Esposito 2019-03-15 15:24:47 +01:00
parent 88f4919a22
commit 4d76a04872
6 changed files with 141 additions and 50 deletions

5
package-lock.json generated
View File

@ -13014,6 +13014,11 @@
"react-lifecycles-compat": "^3.0.4" "react-lifecycles-compat": "^3.0.4"
} }
}, },
"react-web-notification": {
"version": "0.5.0",
"resolved": "https://registry.npmjs.org/react-web-notification/-/react-web-notification-0.5.0.tgz",
"integrity": "sha512-xdUHyOMBP+RlhAwuvYd0WlNvT11tW7V4/vXnp3ApbmChyT6FE/4KYEcmeXw+RtP1cJH8B1KbT7RiYNHFnY/bcQ=="
},
"reactstrap": { "reactstrap": {
"version": "7.1.0", "version": "7.1.0",
"resolved": "https://registry.npmjs.org/reactstrap/-/reactstrap-7.1.0.tgz", "resolved": "https://registry.npmjs.org/reactstrap/-/reactstrap-7.1.0.tgz",

View File

@ -2,11 +2,16 @@
"name": "tomadoro", "name": "tomadoro",
"version": "0.1.0", "version": "0.1.0",
"private": true, "private": true,
"engines": {
"npm": "6.4.1",
"node": "10.15.0"
},
"dependencies": { "dependencies": {
"bootstrap": "^4.3.1", "bootstrap": "^4.3.1",
"react": "^16.8.4", "react": "^16.8.4",
"react-dom": "^16.8.4", "react-dom": "^16.8.4",
"react-scripts": "2.1.8", "react-scripts": "2.1.8",
"react-web-notification": "^0.5.0",
"reactstrap": "^7.1.0" "reactstrap": "^7.1.0"
}, },
"scripts": { "scripts": {

View File

@ -1,5 +1,5 @@
body { body {
background-color: #d9534f; background-color: #db4f4a;
} }
.title { .title {
@ -13,26 +13,21 @@ body {
} }
.App-header { .App-header {
background-color: #d9534f; background-color: #c34944;
display: flex; display: flex;
min-height: 20vh; min-height: 10vh;
flex-direction: column; flex-direction: column;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
font-size: calc(50px + 2vmin); font-size: calc(40px + 2vmin);
color: white; color: white;
text-shadow: 0 0 2px #cf4242, 0 0 4px #2B2D1F; text-shadow: 0 0 2px #cf4242, 0 0 4px #2B2D1F;
margin-bottom: 25px;
box-shadow: 0 0 2px #cf4242, 0 0 8px #2B2D1F;
} }
.App-link { .App-link {
color: #61dafb; color: #61dafb;
} }
@keyframes App-logo-spin {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
}

View File

@ -3,35 +3,33 @@
font-weight: 600; font-weight: 600;
text-shadow: 0 0 2px #cf4242, 0 0 4px #2B2D1F; text-shadow: 0 0 2px #cf4242, 0 0 4px #2B2D1F;
color: white; color: white;
opacity: 1;
} }
.buttons { .buttons {
font-size: 22px; font-size: 20px;
font-weight: bold; font-weight: normal;
height: 70px; height: 60px;
border-radius: 30px; border-radius: 25px;
box-shadow: 0 0 1px #03030349, 0 0 10px rgb(155, 156, 150); box-shadow: 0 0 5px #161616, 0 0 2px #2B2D1F;
opacity: 0.9; opacity: 0.9;
} }
.top-buffer { .top-margin {
margin-top: 15px; margin-top: 18px;
} }
.buttons-box { .buttons-box {
background-color: rgba(255, 255, 255, 0.9); /* transparent */ background-color: rgba(255, 255, 255, 1.0);
padding: 20px; padding: 20px;
margin-top: 5px; margin-top: 5px;
border-radius: 30px; border-radius: 25px;
opacity: 1; box-shadow: 0 0 1px #cf4242, 0 0 10px #2B2D1F;
box-shadow: 0 0 2px #cf4242, 0 0 5px rgb(243, 243, 243);
} }
.App-logo { .App-logo {
height: 25vmin; height: 25vmin;
pointer-events: none; pointer-events: none;
filter: drop-shadow(0 0 1.00rem rgb(211, 204, 206)); filter: drop-shadow(0 0 0.30rem rgb(250, 223, 174));
align-items: center; align-items: center;
justify-content: center; justify-content: center;
text-align: center; text-align: center;
@ -50,14 +48,4 @@
to { to {
transform: rotate(360deg); transform: rotate(360deg);
} }
}
.App-logo-scale {
animation: crescendo 0.5s alternate infinite ease-in;
}
@keyframes crescendo {
0% {transform: scale(.8);}
100% {transform: scale(1.2);}
} }

View File

@ -2,29 +2,103 @@ import React, { Component } from 'react';
import { Button, Container, Row, Col } from 'reactstrap'; import { Button, Container, Row, Col } from 'reactstrap';
import './Timer.css'; import './Timer.css';
import logo from '../logo.svg'; import logo from '../logo.svg';
import Notification from 'react-web-notification';
import sound from '../sound.mp3';
class Timer extends Component { class Timer extends Component {
constructor(props) { constructor(props) {
super(props); super(props);
this.defaultSeconds = 1500; this.defaultSeconds = 4;
this.defaultLogoSpin = 'App-logo-rotation'; this.defaultLogoSpin = 'App-logo-rotation';
this.state = { this.state = {
seconds: this.defaultSeconds, seconds: this.defaultSeconds,
started: false, started: false,
logoSpin: '' logoSpin: '',
ignore: true,
title: ''
}; };
// binding
this.startTimer = this.startTimer.bind(this); this.startTimer = this.startTimer.bind(this);
this.stopTimer = this.stopTimer.bind(this); this.stopTimer = this.stopTimer.bind(this);
this.resetTimer = this.resetTimer.bind(this); this.resetTimer = this.resetTimer.bind(this);
this.handlePermissionGranted = this.handlePermissionGranted.bind(this);
this.handlePermissionDenied = this.handlePermissionDenied.bind(this);
this.handleNotSupported = this.handleNotSupported.bind(this);
this.sendNotification = this.sendNotification.bind(this);
this.handleNotificationOnShow = this.handleNotificationOnShow.bind(this);
this.playSound = this.playSound.bind(this);
} }
fmtMSS(s){return(s-(s%=60))/60+(9<s?':':':0')+s} formatMinute(s){
return ( s - ( s%=60 )) / 60 + ( 9 < s ?':':':0') +s;
}
// handleNotification
handlePermissionGranted() {
this.setState({
ignore: false
});
}
handlePermissionDenied() {
this.setState({
ignore: true
});
}
handleNotSupported() {
this.setState({
ignore: true
});
}
handleNotificationOnError(e, tag){
console.log(e, 'Notification error tag:' + tag);
}
playSound(){
document.getElementById('sound').play();
}
handleNotificationOnShow() {
this.playSound();
}
sendNotification() {
if (this.state.ignore) {
return;
}
const title = 'tomadoro';
const body = 'Time is Up! 🍅';
const tag = Date.now();
const options = {
tag: tag,
body: body,
lang: 'en',
sound: {sound}
};
this.setState({
title: title,
options: options
});
}
tick() { tick() {
this.setState(state => ({ this.setState(state => ({
seconds: state.seconds -1 seconds: state.seconds -1
})); }));
if (this.state.seconds <= 0) {
this.sendNotification();
this.stopTimer();
this.resetTimer();
}
} }
startTimer() { startTimer() {
@ -54,22 +128,27 @@ class Timer extends Component {
<Container> <Container>
<Row> <Row>
<Col> <Col>
<img className={`App-logo ${this.state.logoSpin}`} src={logo}/> <img
className={`App-logo ${this.state.logoSpin}`}
src={logo}
alt="Tomato"
>
</img>
</Col> </Col>
</Row> </Row>
<Row> <Row>
<Col> <Col>
<p <p
class="timer" className="timer"
> >
{this.fmtMSS(this.state.seconds)} {this.formatMinute(this.state.seconds)}
</p> </p>
</Col> </Col>
</Row> </Row>
<div className="buttons-box"> <div className="buttons-box">
<Row> <Row>
<Col> <Col>
<Button <Button
className="buttons" className="buttons"
block block
size="lg" size="lg"
@ -77,13 +156,15 @@ class Timer extends Component {
onClick={this.startTimer} onClick={this.startTimer}
disabled={this.state.started} disabled={this.state.started}
> >
Start START
</Button> </Button>
</Col> </Col>
</Row> </Row>
<Row className="top-buffer"> <Row
className="top-margin"
>
<Col> <Col>
<Button <Button
className="buttons" className="buttons"
block block
color="danger" color="danger"
@ -91,26 +172,43 @@ class Timer extends Component {
onClick={this.stopTimer} onClick={this.stopTimer}
disabled={!this.state.started} disabled={!this.state.started}
> >
Stop STOP
</Button> </Button>
</Col> </Col>
<Col> <Col>
<Button <Button
className="buttons" className="buttons"
block block
color="secondary" color="secondary"
size="lg" size="lg"
onClick={this.resetTimer} onClick={this.resetTimer}
disabled={this.state.started} disabled={this.state.started || this.state.seconds === this.defaultSeconds}
> >
Reset RESET
</Button> </Button>
</Col> </Col>
</Row> </Row>
</div> </div>
</Container> </Container>
<Notification
ignore={this.state.ignore}
onPermissionGranted = {this.handlePermissionGranted}
onPermissionDenied = {this.handlePermissionDenied}
notSupported = {this.handleNotSupported}
onError = {this.onError}
timeout = {5000}
title = {this.state.title}
options = {this.state.options}
onShow = {this.handleNotificationOnShow}
>
</Notification>
<audio id='sound' preload='auto'>
<source src={sound} type='audio/mpeg' />
<embed hidden src={sound} />
</audio>
</div> </div>
); );
} }

BIN
src/sound.mp3 Normal file

Binary file not shown.