0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

ReactNativeでstopwatch作ってみた

Posted at

#はじめに
前にswiftを使ってiosアプリを作ってみようとしたが、storyboardとかがややこしくて挫折。しかし、再びiosアプリ作ってみたいという欲求に駆られ、調べてみると、ReactNativeを見つけた。
これはクロスプラットフォームのアプリ開発用フレームワークでiosとAndroidどちらも動かせるものだった。
すげーと思いつつ、早速何かやってみようということで今回はstopwatchを作ってみた。
モチベを上げるために友達とどっちがきちんと実装できるかを競走しながら開発した。

#実際に完成したもの
ezgif.com-gif-maker.gif
iphoneに標準搭載されているstopwatchアプリを真似てスタイリングをした。
開発ツールとしてexpoを使用した。

#ソースコード
###ファイル構造

.
┣━━━ App.jsx
┗━━━ src
      ┗ component
          ┗ StopWatch.jsx

App.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であり、ここで主な機能の実装をしている。

StopWatch.jsx
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を使うのも初めてで、わからないところだらけだったが、色々調べたりしてなんとか機能を実装することができた。すでに友達とこのアプリの成果を見せあったが、どっちも機能自体は実装できていて、もう少し難しいものにもチャレンジして勝負に白黒つけたいという意欲が沸いた。

自分の知らないものを学んで、こんなものがあるのかと知ることができて嬉しいが、その一方で世の中の技術のすごさと自分のちっぽけさを感じた。

0
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?