2
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.

kintone カレンダーPlusに祝日を表示

Posted at

カレンダーPlus Advent Calendar 2020の12/19 担当分です。
カレンダーPlus に Google Calendar の祝日を表示してみます。

はじめに

こんにちは、kintone では rex0220 プラグインを開発・販売しております。
カレンダーPlus に連携するプラグインとして、計算式プラグインがあります。
今回は、カレンダーPlus のJavaScript API を利用して、Google Calendar の祝日を表示する JavaScript カスタマイズをやってみました。
処理ごとに解説を入れましたので、他に応用する場合もご参考になると思います。

またカレンダーPlusアドオンを募集との知らせを受け、さっそく休日Plusプラグインを開発しました。名前からしてカレンダーPlusに乗っかっています。
JavaScript カスタマイズなんてかったるいことやってられないという方は、そちらをご覧ください。

もっと簡単に祝日を表示したいという方は、カレンダーPlus 機能追加のリクエストを出してプレッシャーをかけましょう!!

概要

Google Calendar API で、日本の祝日を取得して、カレンダーPlusの月表示に表示します。

  • カレンダーPlusで使用しているタグ名やクラス名を元にカスタマイズしますので、仕様変更で動作しなくなるリスクがあります。
  • Google Calendar の日本の祝日で取得できるのは、2〜3年分です。

PC の祝日表示。いい感じですね。
92C4DD99-66DA-4E33-8ECB-A44DDDAE6C69_1_105_c.jpeg

モバイル表示だと、祝日名が枠からはみ出ちゃってます。ここはご愛敬ということで!
3EB515E7-AF15-48FC-ADBE-8BF879BFB6B5_1_105_c.jpeg

Google Calendar API の準備

JavaScriptでGoogle Calendar APIを実装する方法を現役エンジニアが解説【初心者向け】 に、「API Key」 と 「カレンダーID」を取得する方法が載っています。
手順通りにやれば簡単です。

ただ、「API Key」を作るページが探しにくかったので記載しておきます。
Browser Quickstart

取得した「API Key」 、 「カレンダーID」とGoogleAPI 呼び出し用のPATHを変数に入れておきます。

.js
    const API_KEY = 'xxxxxxxxxxxxxxxxxxxxxxxxxx'; // 取得した API KEY を指定
    const CALENDAR_ID = 'ja.japanese#holiday@group.v.calendar.google.com';
    const GAPI_PATH = 'https://www.googleapis.com/calendar/v3/calendars/' + encodeURIComponent(CALENDAR_ID) + '/events';

カレンダーPlus JavaScript API の利用

Calendar Plus JavaScript API リファレンス に説明が載っていますが、サンプルコードが少なく、少しとっつきにくいかもしれません。

カレンダーPlus カレンダー表示イベント('cp.calendar.show')

今回は、カレンダーPlusの月表示のカスタマイズを行いますので、カレンダー表示イベント('cp.calendar.show')を利用します。

リファレンスの説明

カレンダー画面の描画後イベント
カレンダー画面の描画終了後に発生するイベントです。
・表示対象の日付範囲切り替え時や月別・週別・日別表示の切り替え時にもイベントが発生します。
※2019/3/6 現在、ドラッグ&ドロップによるイベントレコード更新時には当イベントが 2 回発生します。

カレンダー表示イベントは、カレンダー画面の描画終了後だけではなく、ドラッグ&ドロップによるイベントレコード更新時にも発生するので、短時間に何回実行されてもいいように処理を検討しましょう。

カレンダーPlus カレンダー表示イベントハンドラーを登録

カレンダーPlus が表示されるのは、kintone の一覧表示カスタム画面ですので、そのタイミングでカレンダー表示イベントハンドラーを登録します。

.js
    // kintone 一覧表示
    kintone.events.on(['app.record.index.show','mobile.app.record.index.show'], function (event) {
        if (event.viewType !== 'custom') return event;
        // カレンダー表示イベント
        calendarplus.events.on('cp.calendar.show', function (ev) {
            if (ev.view.type !== 'month') return;
            // 祝日表示
            gapi.load('client', start);
        });
        return event;
    });
  • kintone の一覧表示イベント:'app.record.index.show','mobile.app.record.index.show'
    PC とモバイル両方のカレンダーPlus表示に対応します。

  • イベント情報の変数名
    kintone とカレンダーPlusの両方にイベント情報がありますので、下記のようにして区別します。

    • kintone: event
    • カレンダーPlus: ev
  • event.viewType: kintone の一覧画面の種類で、'custom' のみを処理します。
    通常の一覧画面や標準のカレンダー表示では、処理しないように return event; します。
    今回の場合、一覧画面でカレンダーPlus のイベントハンドラーを登録しても、呼び出されないので問題ないのですが、できるだけ他の画面では余計なことはしないようにしましょう。

  • ev.view.type: カレンダーPlus で表示される表示形式を取得できます。
    これは、カレンダーPlus で、使用している fullcalendar のViewオブジェクトを参照しています。
    fullcalendar docs view object
    カレンダーPlus の月形式の表示では、'month' となります。
    docs を見ると、他にも title やら calendar オブジェクトなども利用できますので、多くの情報を取得できます。

  • gapi.load('client', start);
    Google カレンダーを取得するための GoogleAPI を使い、start 関数で処理を記述します。

祝日処理

GoogleAPI の手順に従っているので少しわかりにくいのですが、API_KEY を GAPI_PATH に渡して、祝日情報を取得しているだけです。
「// カレンダーPlus 祝日処理」以降が、実際に祝日を表示する処理になります。

.js
    // 祝日処理
    function start() {
        // 祝日取得
        gapi.client.init({
            'apiKey': API_KEY,
        }).then(function () {
            return gapi.client.request({
                'path': GAPI_PATH
            })
        }).then(function (response) {
            // カレンダーPlus 祝日処理
            const items = response.result.items;
            setHolidays(items);
        }, function (reason) {
            console.log('google calendar api Error: ' + reason.result.error.message);
        });
    };
  • const items = response.result.items;

    Google カレンダーから取得した祝日情報

カレンダーPlus 祝日処理

items に、祝日情報が配列で入っていますので、祝日数分ループして処理します。
Chrome の開発者ツールで、items の中身を見てみましょう。
同じく DOM 構造もChrome の開発者ツールで確認できます。

  • items の中身
    2020-12-18_13h36_15.png

    • start.date 祝日の日付
    • summary 祝日の名称
  • カレンダーPlus の DOM 構造

    • 月表示の日付枠は、テーブルタグ"td"内に、リンクタグ"a" という構造です。
    • 日付を識別できるものとしては、td の属性にdata-date があり、日付が入っています。

      これをキーにして、element を取得できますね。

      data-date="2021-01-05"

2020-12-18_13h45_40.png

  • クラス名に曜日の識別

    • fc-sun, fc-mon... そのまま曜日です。土日の識別になります。
      2020-12-18_13h55_04.png
  • 祝日を該当箇所に挿入
    data-date 属性で、対象日付箇所を取得して、日付名称を挿入します。
    赤表示は、css で指定します。

.js
    // カレンダーPlus 祝日表示
    function setHolidays(items) {
        // 祝日数分ループ
        items.forEach(function(item) {
            const dt = item.start.date;
            const sel = 'td[data-date="' + dt + '"] > a';
            var ele1 = document.querySelector(sel);
            if (ele1) {
                if (!ele1.parentElement.querySelector('span.xp-cplus-holiday')) {
                    ele1.classList.add('xp-cplus-holiday');
                    var html = '<span class="xp-cplus-holiday">' + item.summary + '</span>';
                    ele1.insertAdjacentHTML('beforebegin',html);
                }
            }
        });
    }
  • const sel = 'td[data-date="' + dt + '"] > a';

    dt に日付 "2020-01-01" 等が入ってきますので、文字列は、

    'td[data-date="2020-01-01"] > a' のようになります。
  • var ele1 = document.querySelector(sel);

    'td[data-date="2020-01-01"] > a' のリンクタグを取得します。

    属性 data-date="2020-01-01" を持つ td タグ直下の a リンクタグが対象です。
  • if (!ele1.parentElement.querySelector('span.xp-cplus-holiday')) {

    対象枠に「span.xp-cplus-holiday」が無かったら(未処理だったら)祝日を追加します。

    カレンダーPlus のカレンダー表示イベントは、何度も呼び出されるため、祝日追加は、1回のみにします。
  • ele1.classList.add('xp-cplus-holiday');

    CSS のクラス名をリンクタグに追加します。CSS で、文字色等を指定できます。
  • le1.insertAdjacentHTML('beforebegin',html);

    xp-cplus-holiday のクラス名を付けて、祝日表示の span タグを組み立てて、日付リンクタグの前に挿入します。

    クラス名があると、ブラウザーの開発ツールでいろいろ調整するのが簡単にできて便利です。

JavaScript 全体

以上をまとめたものが下記になります。

cplus-holiday.js
////////////////////////////////////////////////////////////////////////////
// kintone JavaScript: カレンダープラスプラグイン google calendar 祝日表示(month)
//  2020.12.18 by rex0220
////////////////////////////////////////////////////////////////////////////
(function () {
    'use strict';
    const API_KEY = 'xxxxxxxxxxxxxxxxxxxxxxxxxx'; // 取得した API KEY を指定
    const CALENDAR_ID = 'ja.japanese#holiday@group.v.calendar.google.com';
    const GAPI_PATH = 'https://www.googleapis.com/calendar/v3/calendars/' + encodeURIComponent(CALENDAR_ID) + '/events';

    // kintone 一覧表示
    kintone.events.on(['app.record.index.show','mobile.app.record.index.show'], function (event) {
        if (event.viewType !== 'custom') return event;
        // カレンダー表示イベント
        calendarplus.events.on('cp.calendar.show', function (ev) {
            if (ev.view.type !== 'month') return;
            // 祝日処理
            gapi.load('client', start);
        });
        return event;
    });

    // 祝日処理
    function start() {
        // 祝日取得
        gapi.client.init({
            'apiKey': API_KEY,
        }).then(function () {
            return gapi.client.request({
                'path': GAPI_PATH
            })
        }).then(function (response) {
            // カレンダーPlus 祝日表示
            const items = response.result.items;
            setHolidays(items);
        }, function (reason) {
            console.log('google calendar api Error: ' + reason.result.error.message);
        });
    };

    // カレンダーPlus 祝日表示
    function setHolidays(items) {
        // 祝日数分ループ
        items.forEach(function(item) {
            const dt = item.start.date;
            const sel = 'td[data-date="' + dt + '"] > a';
            var ele1 = document.querySelector(sel);
            if (ele1) {
                if (!ele1.parentElement.querySelector('span.xp-cplus-holiday')) {
                    ele1.classList.add('xp-cplus-holiday');
                    var html = '<span class="xp-cplus-holiday">' + item.summary + '</span>';
                    ele1.insertAdjacentHTML('beforebegin',html);
                }
            }
        });
    }
})();

CSS で土日祝日を赤表示

CSS で、土日祝日の表示を赤に指定します。
fc-sun の部分を変えれば、好きな曜日の色を変えられます。
PC/スマートフォン兼用の CSS です。

cplus-holiday.css
/* calendar plus holiday color */
.fc-view-container .fc-sun > a,
.fc-view-container .fc-sat > a,
.fc-view-container .fc-sun > span,
.fc-view-container .fc-sat > span {
  color: red;
}
.fc-view-container .xp-cplus-holiday {
  color: red;
}
  • .fc-view-container .fc-sun > a,

    CSS は、アプリ全体に適用されるため、カレンダーPlus で利用している FullCarendar で使用している fc-view-container クラスの中だけに、限定するように指定しています。

    適用範囲を限定することで、他の画面で変な現象がでる問題の防止になります。

    ただし、タグ名・クラス名などは仕様変更で変わってしまうリスクもあります。
  • .fc-view-container .xp-cplus-holiday {

    xp-cplus-holiday は、祝日の名称を追加したときに付与したクラス名です。

    色以外に、文字の大きさなどもここで変えられます。

JavaScript/CSS の指定

URL指定で Google API のライブラリ、カスタマイズ用JavaScript,CSSファイルを指定します。
今回の JavaScript, CSS は、PC、モバイル兼用ですので同じものをアップロードします。

736AD10C-8901-4DF2-8CD4-71B9596F5BA1_1_105_c.jpeg

改善点

以上で、祝日をカレンダーに表示することができました。
しかしちょっと気になるのが、カレンダーが表示されてから祝日が表示されるまでに少しタイムラグがあることです。
カレンダー表示イベント毎に祝日データを取得しているせいですね。
「>」ボタンなどで月の切り替えを繰り返すとタイムラグが目立ちます。

祝日データは、何回取得しても内容は変わりませんので、1回取得したらそれを使いまわしたほうがよさそうです。
ぜひタイムラグの解消に挑戦してみてください。

注意事項

  • カレンダーPlusで使用しているタグ名やクラス名を元にカスタマイズしますので、仕様変更で動作しなくなるリスクがあります。

  • カレンダーPlus へ祝日・休日を表示したいという要望が多いとのことなので、将来のバージョンアップでカレンダーPlus に機能が追加される可能性があります。急がない方は、期待して待っていましょう。

  • Google Calendar API には、回数制限や割り当て制限があります。

500人の kintone ユーザーが一斉にカレンダーを見る職場だと制限値を超えちゃいますね。
(どんな職場だ!朝9時にカレンダーチェックするルールとか?)
問題が起きたときは、想定外だったとシラを切りましょう。

あとがき

多くの kintone プラグインがありますがカレンダーPlus は、API を公開している数少ないプラグインの一つです。
標準機能に足りないものは、APIを利用すると自前で何とか出来ちゃうのがいいですね。
ただし、保守を考えると後々困る場合が出てきます。
とりあえず欲しい機能は、カレンダーPlusへリクエストを上げておきましょう。
斎藤さん、リクエスト対応よろしくお願いします!

参考

2
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
2
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?