2014年2月11日火曜日

Google Visualization API を使ったボドゲ会カレンダの更新

今回の記事もボドゲとは関係なしです。

以前『Blogger API v3の使い方』の最後にちょっと書いた『ボドゲ会カレンダの簡易登録』を
実装してみました。

仕組みとしては、下記の3ステップで行います。


  • ボドゲ会情報をGoogle ドライブのスプレッドシートに登録
  • Google Visualization APIを使ってそのデータを習得
  • javascriptでテーブルの形式に編集して画面表示


では順番に解説していきます。



1.googleドライブにボドゲ会情報を登録します

たくさんありますね。
以降はページ側にjavascriptコードを書きます。


2.visualization APIを読み込むコード


<script type="text/javascript" src="http://www.google.com/jsapi"></script>
<script type="text/javascript">
    //<![CDATA[
    <!--

    //visualization APIの読込
    google.load("visualization", "1", {packages:['table']});
    //-->
    //]]>
</script>

こちらが今回の肝である『visualization API』を読み込むコードです。


3.スプレッドシートからデータを取得するコード


//データソース URL
//本番用
var dataSource = 'https://docs.google.com/spreadsheet/ccc?key=0Ajmd9EuHdR9ldF9zZ3pYZl9VTUxyQU5USDEwbDZmWmc&gid=0&pub=0';

//Query Language
//※今月以降の情報のみ取得する
var today = new Date();
var queryLanguage = 
    'SELECT B, C, D, E, F, G, H, I ' + 
    'WHERE ' +
        'YEAR(B) >= ' + today.getFullYear() + ' AND ' + 
        'MONTH(B) >= ' + today.getMonth() + ' ' +
    'ORDER BY B ';

/***********************
 * ページ読込完了後の処理
 ***********************/
google.setOnLoadCallback(function() {
    $("#result").html('<img src="' + base64img + '" />');
    //google spreadsheetの情報を取得しdispCalendar関数を実行する
    var query = new google.visualization.Query(dataSource);
    query.setQuery(queryLanguage);
    query.send(dispCalendar);
});

dataSourceにスプレッドシートのURLを指定し、queryLanguage にそのデータを取得するため命令をSQLに似たQuery Languageという言語で記述します。今回は『開催日が今月以降の登録データ』を取得します。

参考:Query Language Reference
https://developers.google.com/chart/interactive/docs/querylanguage?hl=ja

setOnLoadCallbackでデータ取得処理を実行し、取得したデータに対してdispCalendar関数を実行します。


4.テーブルに編集して画面表示するコード


/***********************
 * カレンダテーブル表示
 ***********************/
function dispCalendar(response){
    var el = document.getElementById('result');

    if(response.isError()) {
        //エラーの場合はエラーメッセージを出力して処理を終了する
        el.innerHTML = response.getDetailedMessage();
        return;
    }

    //テーブルデータ
    //※indexは下記の通り
    //日付:0
    //曜日:1(未使用)
    //時間:2
    //名称:3
    //リンク:4
    //最寄駅:5
    //住所:6
    //施設名:7
    var data = response.getDataTable();

    //日付フォーマットパターン
    var formatter1 = new google.visualization.DateFormat({pattern: 'M/d(E)'});

    //日付と時間を結合するパターン
    var formatter2 = new google.visualization.PatternFormat(
        '{0}<br /><span style="font-size: x-small;">{1}</span>');
    //名称とリンクを結合するパターン
    var formatter3 = new google.visualization.PatternFormat(
        '<b>{0}</b><br /><span style="font-size: x-small;">' +
        '<a href="{1}" target="_blank">{1}</a></span>');
    //住所情報を結合するパターン
    var formatter4 = new google.visualization.PatternFormat(
        '最寄駅:{0}<br /><span style="font-size: x-small;">' +
        '<a href="https://maps.google.co.jp/maps?q={1}" target="_blank">' +
        '{1}<br />{2}</a></span>');

    //各パターンを適用する
    formatter1.format(data, 0);
    formatter2.format(data, [0, 2]);
    formatter3.format(data, [3, 4]);
    formatter4.format(data, [5, 6, 7]);
    
    var today = new Date();
    var tmpToday = new Date();
    var yesterday = new Date(tmpToday.setDate(today.getDate() - 1));

    //日付によってセルの色を変えるパターンを適用する
    //今日以前の日付は灰色・今日日付はピンク
    var formatterC = new google.visualization.ColorFormat();
    formatterC.addRange(0, yesterday, null, 'lightgrey');
    formatterC.addRange(yesterday, today, null, 'pink');
    formatterC.format(data, 0);
    
    //列名を設定する
    data.setColumnLabel(0, '日時');
    data.setColumnLabel(5, '会場');
    
    //曜日フォーマットの変換
    for(var i = 0; i < data.getNumberOfRows(); i++){
        var dispDate = data.getFormattedValue(i, 0)
            .replace('Sun','日')
            .replace('Mon','月')
            .replace('Tue','火')
            .replace('Wed','水')
            .replace('Thu','木')
            .replace('Fri','金')
            .replace('Sat','土')
            .replace('土','<span style="color: blue;">土</span>')
            .replace('日','<span style="color: red;">日</span>');
        data.setFormattedValue(i, 0, dispDate);
    }

    //表示する列を指定
    var view = new google.visualization.DataView(data);
    view.setColumns([0, 3, 5]);
    
    //テーブルを表示
    var visualization = new google.visualization.Table(el); 
    visualization.draw(view, {allowHtml: true});
}

まあこのへんは長いけど泥臭いコードであんまり面白みが無いですね。
簡単に処理を説明します。
◆スプレッドシートのデータの見た目を変更
 APIのDateFormatとPatternFormat関数を利用します。
 例:2014/02/01 → 2/1(土) に変換、
   2/1(土)と10:00~21:00を1つのセルにまとめる、などです。

◆セルの色を変更
 APIのColorFormat関数を利用します。
 開催日が『過去日付』の場合は灰色で、『今日』の場合はピンクに変更します。

◆列名を変更する
 indexが0と5の列名を変更します。(indexは0から始まるので1つ目の列が0になります。)

◆曜日フォーマットを変更する
 『土』・『日』の文字色を青と赤に変更します。

◆テーブルを表示する
 indexが0,3,5の列だけ表示します。

参考:Google Visualization API Reference
https://developers.google.com/chart/interactive/docs/reference?hl=ja


5.コードまとめ


<html>
    <head>
        <meta http-equiv="content-type" content="text/html;charset=utf-8">
        <script src="http://ajax.googleapis.com/ajax/libs/jquery/1.10.2/jquery.min.js"></script>
        <script type="text/javascript" src="http://www.google.com/jsapi"></script>
        <script type="text/javascript">
            //<![CDATA[
            <!--

            //visualization APIの読込
            google.load("visualization", "1", {packages:['table']});
            
            /**********************
             * 定数・変数の定義
             **********************/
            //データソース URL
            //本番用
            var dataSource = 'https://docs.google.com/spreadsheet/ccc?key=0Ajmd9EuHdR9ldF9zZ3pYZl9VTUxyQU5USDEwbDZmWmc&gid=0&pub=0';

            //Query Language
            //※今月以降の情報のみ取得する
            var today = new Date();
            var queryLanguage = 
                'SELECT B, C, D, E, F, G, H, I ' + 
                'WHERE ' +
                    'YEAR(B) >= ' + today.getFullYear() + ' AND ' + 
                    'MONTH(B) >= ' + today.getMonth() + ' ' +
                'ORDER BY B ';
            
            //ローディングGIF
            var base64img = '';
            
            /***********************
             * ページ読込完了後の処理
             ***********************/
            google.setOnLoadCallback(function() {
                $("#result").html('<img src="' + base64img + '" />');
                //google spreadsheetの情報を取得しdispCalendar関数を実行する
                var query = new google.visualization.Query(dataSource);
                query.setQuery(queryLanguage);
                query.send(dispCalendar);
            });
            
            /***********************
             * カレンダテーブル表示
             ***********************/
            function dispCalendar(response){
                var el = document.getElementById('result');

                if(response.isError()) {
                    //エラーの場合はエラーメッセージを出力して処理を終了する
                    el.innerHTML = response.getDetailedMessage();
                    return;
                }

                //テーブルデータ
                //※indexは下記の通り
                //日付:0
                //曜日:1(未使用)
                //時間:2
                //名称:3
                //リンク:4
                //最寄駅:5
                //住所:6
                //施設名:7
                var data = response.getDataTable();

                //日付フォーマットパターン
                var formatter1 = new google.visualization.DateFormat({pattern: 'M/d(E)'});

                //日付と時間を結合するパターン
                var formatter2 = new google.visualization.PatternFormat(
                    '{0}<br /><span style="font-size: x-small;">{1}</span>');
                //名称とリンクを結合するパターン
                var formatter3 = new google.visualization.PatternFormat(
                    '<b>{0}</b><br /><span style="font-size: x-small;">' +
                    '<a href="{1}" target="_blank">{1}</a></span>');
                //住所情報を結合するパターン
                var formatter4 = new google.visualization.PatternFormat(
                    '最寄駅:{0}<br /><span style="font-size: x-small;">' +
                    '<a href="https://maps.google.co.jp/maps?q={1}" target="_blank">' +
                    '{1}<br />{2}</a></span>');

                //各パターンを適用する
                formatter1.format(data, 0);
                formatter2.format(data, [0, 2]);
                formatter3.format(data, [3, 4]);
                formatter4.format(data, [5, 6, 7]);
                
                var today = new Date();
                var tmpToday = new Date();
                var yesterday = new Date(tmpToday.setDate(today.getDate() - 1));

                //日付によってセルの色を変えるパターンを適用する
                //今日以前の日付は灰色・今日日付はピンク
                var formatterC = new google.visualization.ColorFormat();
                formatterC.addRange(0, yesterday, null, 'lightgrey');
                formatterC.addRange(yesterday, today, null, 'pink');
                formatterC.format(data, 0);
                
                //列名を設定する
                data.setColumnLabel(0, '日時');
                data.setColumnLabel(5, '会場');
                
                //曜日フォーマットの変換
                for(var i = 0; i < data.getNumberOfRows(); i++){
                    var dispDate = data.getFormattedValue(i, 0)
                        .replace('Sun','日')
                        .replace('Mon','月')
                        .replace('Tue','火')
                        .replace('Wed','水')
                        .replace('Thu','木')
                        .replace('Fri','金')
                        .replace('Sat','土')
                        .replace('土','<span style="color: blue;">土</span>')
                        .replace('日','<span style="color: red;">日</span>');
                    data.setFormattedValue(i, 0, dispDate);
                }

                //表示する列を指定
                var view = new google.visualization.DataView(data);
                view.setColumns([0, 3, 5]);
                
                //テーブルを表示
                var visualization = new google.visualization.Table(el); 
                visualization.draw(view, {allowHtml: true});
            }

            //-->
            //]]>
        </script>
    </head>
    <body>
        <div id="result" ></div>
    </body>
</html>


★まとめ

今回の対応により、スプレッドシートにボドゲ会情報を登録するだけでブログのカレンダが自動更新されます。今までは、htmlを直接編集していたのでかなりの負荷軽減になりました。

あとはtwitter BOTへの登録が未だ手動なのがちょっと面倒です。理想はスプレッドシートに登録した情報を自動でつぶやいてくれればいいんですが。

ちょくちょく時間を見つけて改善していきたいと思います。

0 件のコメント:

コメントを投稿