#はじめに
前にswiftを使ってiosアプリを作ってみようとしたが、storyboardとかがややこしくて挫折。しかし、再びiosアプリ作ってみたいという欲求に駆られ、調べてみると、ReactNativeを見つけた。
これはクロスプラットフォームのアプリ開発用フレームワークでiosとAndroidどちらも動かせるものだった。
すげーと思いつつ、早速何かやってみようということで今回はstopwatchを作ってみた。
モチベを上げるために友達とどっちがきちんと実装できるかを競走しながら開発した。
#実際に完成したもの
iphoneに標準搭載されているstopwatchアプリを真似てスタイリングをした。
開発ツールとしてexpoを使用した。
#ソースコード
###ファイル構造
.
┣━━━ App.jsx
┗━━━ src
┗ component
┗ StopWatch.jsx
App.jsx
メインとなるファイル
import React from 'react';
import {
StyleSheet, View,
} from 'react-native';
import { StatusBar } from 'expo-status-bar';
import StopWatch from './src/component/StopWatch';
export default function App() {
return (
<View style={styles.container}>
<StopWatch />
<StatusBar style="light" />
</View>
);
}
const styles = StyleSheet.create({
container: {
flex: 1,
},
});
今回背景色が黒なので、StatusBarはlightとした。
StopWatch.jsx
componentであり、ここで主な機能の実装をしている。
import React, { Component } from 'react';
import {
View, Text, StyleSheet, TouchableHighlight, ScrollView,
} from 'react-native';
import formatTime from 'minutes-seconds-milliseconds';
class StopWatch extends Component {
constructor(props) {
super(props);
this.state = {
// 経過時間
timeElapsed: 0,
// タイマーが動いているかどうか
running: false,
startTime: 0,
stopTime: 0,
laps: [],
};
// それぞれ作成した関数でthisを正しく認識するためbindしている
this.handleStartPress = this.handleStartPress.bind(this);
this.handleLapPress = this.handleLapPress.bind(this);
this.handleStopPress = this.handleStopPress.bind(this);
this.handleResetPress = this.handleResetPress.bind(this);
}
handleStartPress() {
this.setState({ startTime: new Date() - this.state.stopTime });
this.interval = setInterval(() => {
this.setState({
timeElapsed: new Date() - this.state.startTime,
running: true,
});
}, 30);
}
handleStopPress() {
// タイマーを止める
clearInterval(this.interval);
this.setState({
running: false,
stopTime: this.state.timeElapsed,
});
}
handleLapPress() {
const currentLapTime = this.state.timeElapsed;
// 配列の最後を取得する
let end = 0;
if (currentLapTime !== 0) {
if (this.state.laps.length !== 0) {
end = this.state.laps[this.state.laps.length - 1];
}
// それぞれのlapの差をとっている
const difference = currentLapTime - end;
this.setState({
// concatで配列の結合
laps: this.state.laps.concat([difference]),
});
}
}
handleResetPress() {
this.setState({
timeElapsed: 0,
startTime: 0,
stopTime: 0,
laps: [],
running: false,
});
}
renderStartStopButton() {
const style = this.state.running ? styles.stopButton : styles.startButton;
return (
<TouchableHighlight
style={[styles.button, style]}
underlayColor="gray"
onPress={this.state.running ? this.handleStopPress : this.handleStartPress}
>
<Text style={this.state.running ? styles.stopLabel : styles.startLabel}>
{this.state.running ? 'Stop' : 'Start'}
</Text>
</TouchableHighlight>
);
}
renderLapResetButton() {
const style = this.state.timeElapsed !== 0 ? styles.lightGrayButton : styles.darkGrayButton;
return (
<TouchableHighlight
style={[styles.button, style]}
underlayColor="gray"
onPress={!this.state.running && this.state.timeElapsed !== 0 ? this.handleResetPress : this.handleLapPress}
>
<Text style={this.state.timeElapsed !== 0 ? styles.lightGrayLabel : styles.darkGrayLabel}>
{!this.state.running && this.state.timeElapsed !== 0 ? 'Reset' : 'Lap'}
</Text>
</TouchableHighlight>
);
}
renderLaps() {
return (
<View>
{this.state.laps.map((time, index) => (
<View style={styles.lap} key={index}>
<Text style={styles.lapText}>Lap {index + 1}</Text>
<Text style={styles.lapText}>{formatTime(time)}</Text>
</View>
))}
</View>
);
}
render() {
return (
<View style={styles.container}>
<View style={styles.timerHeader}>
<View>
<Text style={styles.timer}>
{formatTime(this.state.timeElapsed)}
</Text>
</View>
</View>
<View style={styles.buttonItem}>
{this.renderLapResetButton()}
{this.renderStartStopButton()}
</View>
<ScrollView style={styles.footer}>
{this.renderLaps()}
</ScrollView>
</View>
);
}
}
const styles = StyleSheet.create({
container: {
display: 'flex',
flex: 1,
backgroundColor: '#000000',
width: '100%',
justifyContent: 'flex-start',
},
timerHeader: {
backgroundColor: '#000000',
height: 375,
justifyContent: 'flex-end',
alignItems: 'center',
},
timer: {
fontSize: 60,
color: '#ffffff',
// eslint-disable-next-line no-dupe-keys
fontSize: 85,
lineHeight: 100,
fontWeight: '200',
marginBottom: 80,
},
buttonItem: {
flexDirection: 'row',
justifyContent: 'space-between',
alignItems: 'center',
height: 110,
backgroundColor: 'black',
},
button: {
width: 80,
height: 80,
borderRadius: 40,
justifyContent: 'center',
alignItems: 'center',
},
startButton: {
backgroundColor: 'rgba(1,79,9,0.35)',
marginRight: 20,
},
startLabel: {
color: '#3AC219',
fontSize: 16,
},
stopButton: {
backgroundColor: 'rgba(108,3,19,0.35)',
marginRight: 20,
},
stopLabel: {
color: '#C70E2A',
fontSize: 16,
},
lightGrayButton: {
backgroundColor: '#212121',
marginLeft: 20,
},
darkGrayButton: {
backgroundColor: 'rgba(40,42,55,0.5)',
marginLeft: 20,
},
lightGrayLabel: {
color: '#E0E0E0',
fontSize: 16,
},
darkGrayLabel: {
color: '#83859F',
fontSize: 16,
},
lap: {
justifyContent: 'space-between',
flexDirection: 'row',
borderBottomColor: 'rgba(155,155,155,0.25)',
borderBottomWidth: 1,
height: 45,
alignItems: 'center',
},
lapText: {
fontSize: 22,
color: '#E0E0E0',
},
footer: {
flex: 1,
borderTopColor: 'rgba(155,155,155,0.25)',
borderTopWidth: 1,
marginLeft: 15,
marginRight: 15,
},
});
export default StopWatch;
#満足していないところ
・動画をみていただくとわかる通り、タイマーが少し左右にブレてしまっている。これがブレないように実装することができれば、もっと見やすいstopwatchになったと思う。
・lapのスタイルをもっとiphoneのstopwatchに寄せたかった。iphoneのstopwatchでは、lapの値が一番大きいものと一番小さいものはわかりやすく、色が振られる。また、lapされていくたびに新しいlapが次々に上に積み上げられていく。
これをもう少し実現することができればもっと似たと思う笑 時間ある時にもう少し拘りたい。
感想
今回reactを使うのも初めてで、わからないところだらけだったが、色々調べたりしてなんとか機能を実装することができた。すでに友達とこのアプリの成果を見せあったが、どっちも機能自体は実装できていて、もう少し難しいものにもチャレンジして勝負に白黒つけたいという意欲が沸いた。
自分の知らないものを学んで、こんなものがあるのかと知ることができて嬉しいが、その一方で世の中の技術のすごさと自分のちっぽけさを感じた。