9
4

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 1 year has passed since last update.

オープンストリームAdvent Calendar 2018

Day 16

FitbitAPIを使って心拍数をデスクトップに表示してみた。

Last updated at Posted at 2018-12-15

Fitbitとは、「歩数」「移動距離」「消費カロリー」「運動強度」「睡眠状態」「心拍数」を測定してくれる、有名どころのウェアブル端末ですがガシェット好きにはたまらない、自分でデータを取得したり操作したりを可能にするAPIを公開してくれている。
https://dev.fitbit.com/

今回は、FitbitAPIからデータを取得して、デスクトップに表示するWindowsアプリを作成してみた。
アプリ.png

ステップ1 - アプリケーションを登録する

最初に行うことは、Fitbit開発者サイトに移動し、アプリケーションを登録して、アクセスキーを取得します。
「REGISTER AN APP」タブを選択し、詳細を入力します。
基本的には適当な入力でかまいませんが、今回はリアルタイムに自分の個人データ全部にアクセスするためにOAuth 2.0 Application Typeの項目だけは、Personalを選択してください。

FireShot Capture 6 - Applications - https___dev.fitbit.com_apps_new.png

Application Name - 「FBDisplay」
Description - 「個人的な趣味のためのFitbitデータの使用」(適当でOK)
Application Website - (https://localhost/)
Organization - (localhost)
Organization Website - (https://localhost/)
Terms Of Service Url - (https://localhost/)
Privacy Policy Url - (https://localhost/)
OAuth 2.0 Application Type - Personal
Callback URL - (https://localhost/)
Default Access Type - Read-Only

登録が完了すると「MANAGE MY APPS」タブに必要になるClient ID、Client Secret項目が表示されます。
FireShot Capture 7 - Edit Application - https___dev.fitbit.com_apps_details_22DF97.png

ステップ2 - 認証コード、心拍数を取得

Fitbit APIは OAuth 2.0のAuthorization Code Grant Flowで認証認可を行うことで各データにアクセスする事ができる。

今回作成したアプリでは、アプリケーション登録時のClient ID、Client Secret入力後にGetAuthCodeを押下するとブラウザが起動して
アプリケーションのアクセス許可を求められます。
FireShot Capture 10 - アプリの認証_ - https___www.fitbit.com_oauth2_authorize.png

許可すると、画面遷移してlocalhostへアクセスしてエラーとなります。
この時のURL(code=XXXXXXXXXXXXX#=)のXXXXXXXXXXXXX部分をアプリのAuth Codeに入力します。

以下に画面部分のコードを示すが、FitBitのデータ取得部分に関しては、Fitbit.NETというライブラリが使えるという事で使用した。
しかし、Nugetでインストールできる現在のバージョンだと心拍数が取得できない。なので開発者Gitで最新版を自前でコンパイルして利用しなければならない。

MainForm.cs
using Fitbit.Api.Portable;
using Fitbit.Api.Portable.OAuth2;
using Fitbit.Models;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Diagnostics;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace FBDisplay
{
    public partial class MainForm : Form
    {
        FitbitAppCredentials credentials = new FitbitAppCredentials();
        FitbitClient fitbitClient = null;
        OAuth2AccessToken token = null;
        string result = "";

        public MainForm()
        {
            InitializeComponent();

            loadProperties();
        }

        private void saveProperties()
        {
            Properties.Settings.Default.ClientId = this.textBoxClientId.Text;
            Properties.Settings.Default.ClientSecret = this.textBoxClientSecret.Text;
            Properties.Settings.Default.CallbackURL = this.textBoxCallbackURL.Text;
            
            Properties.Settings.Default.Save();
        }

        private void loadProperties()
        {
            this.textBoxClientId.Text = Properties.Settings.Default.ClientId;
            this.textBoxClientSecret.Text = Properties.Settings.Default.ClientSecret;
            this.textBoxCallbackURL.Text = Properties.Settings.Default.CallbackURL;
        }

        //GetAuthCodeボタン押下
        private void buttonAuthCode_Click(object sender, EventArgs e)
        {
            saveProperties();

            credentials.ClientId = this.textBoxClientId.Text;
            credentials.ClientSecret = this.textBoxClientSecret.Text;

            // authenticate
            OAuth2Helper helper = new OAuth2Helper(credentials, this.textBoxCallbackURL.Text);
            string authUrl = helper.GenerateAuthUrl(new[] { "heartrate" });

            //ブラウザ起動
            Process.Start(authUrl);
        }

        //GetTokenボタン押下
        private void buttonToken_Click(object sender, EventArgs e)
        {
            saveProperties();

            Task.Run(async () =>
            {
                if (!String.IsNullOrEmpty(this.textBoxAuthCode.Text))
                {
                    if (this.fitbitClient == null)
                    {
                        //アクセストークンを取得する
                        OAuth2Helper helper = new OAuth2Helper(credentials, this.textBoxCallbackURL.Text);
                        this.token = await helper.ExchangeAuthCodeForAccessTokenAsync(this.textBoxAuthCode.Text);
                        this.fitbitClient = new FitbitClient(credentials, token);
                    }
                }


            }).Wait();


        }

        private void MainForm_FormClosed(object sender, FormClosedEventArgs e)
        {
            Application.Exit();
        }

        //心拍数取得(外部から呼び出す)
        public string GetHeartRate()
        {
            Task.Run(async () =>
                {
                    if (!String.IsNullOrEmpty(this.textBoxAuthCode.Text))
                    {
                        if (this.fitbitClient != null)
                        {
                            DateTime dt = DateTime.Now;
                            HeartActivitiesIntraday heart = await fitbitClient.GetHeartRateIntraday(dt, HeartRateResolution.oneSecond);

                            this.result = string.Format("[{0}], ", heart.Dataset.Last().Value);
                        }
                    }
                    Thread.Sleep(500);


            }).Wait();

            return result;
        }
    }
}

ステップ3 - 心拍数表示

心拍数の表示部分に関しては、DxLibDLLを利用して表示した。(MainLoopAsyncがフレーム毎に呼び出される)
なぜDxLibDLLかというと、当初の予定ではMMDモデルを心拍数に応じてモーションを早くする等したかったからだが今回は大人の事情で数値のみとした。

DisplayForm.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

using DxLibDLL;
using System.Threading;

namespace FBDisplay
{
    public partial class DisplayForm : Form
    {
        public DisplayForm()
        {
            InitializeComponent();

            ClientSize = new Size(Screen.PrimaryScreen.Bounds.Width, Screen.PrimaryScreen.Bounds.Height);
            
            DX.SetOutApplicationLogValidFlag(DX.FALSE);
            DX.SetUserWindow(Handle);


            DX.DxLib_Init();
            DX.SetDrawScreen(DX.DX_SCREEN_BACK);

            FormBorderStyle = FormBorderStyle.None;
            TransparencyKey = Color.FromArgb(1, 1, 1);
            //TopMost = true;
        }

        public void MainLoopAsync(String heart)
        {
            DX.ClearDrawScreen();
            DX.DrawBox(0, 0, Screen.PrimaryScreen.Bounds.Width, Screen.PrimaryScreen.Bounds.Height, DX.GetColor(1, 1, 1), DX.TRUE);


            int ScreenWidth = Screen.PrimaryScreen.WorkingArea.Width;
            int Screenheigth = Screen.PrimaryScreen.WorkingArea.Height;

            int AppLeftXPos = ScreenWidth - 150;
            int AppLeftYPos = Screenheigth - 100;
    
            DX.SetFontSize(64);
            if (!String.IsNullOrEmpty(heart))
            {
                DX.DrawString(AppLeftXPos, AppLeftYPos, heart, DX.GetColor(0, 0, 0));
            } else
            {
                DX.DrawString(AppLeftXPos, AppLeftYPos, "-", DX.GetColor(0, 0, 0));
            }

            //ESCキーを押したら終了
            if (DX.CheckHitKey(DX.KEY_INPUT_ESCAPE) != 0)
            {
                Close();
            }

            DX.ScreenFlip();
        }

        private void DisplayForm_FormClosed(object sender, FormClosedEventArgs e)
        {
            DX.DxLib_End();
        }
    }
}

これで、プレゼン中に自分の心拍数を表示できる。
(取得間隔を短くしすぎるとリクエストが多すぎるとロックされるし、完全にリアルタイムデータが取得できるわけではないので注意が必要です)

9
4
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
9
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?