LoginSignup
6
1

kintoneで他のユーザの予定を確認するJSカスタマイズを作る

Last updated at Posted at 2020-12-10

この記事はkintone Advent Calendar 2020の10日目の記事です。

こんにちは、@tasshiです。
最近は12/3に発売したフィットボクシング2の体験版をやっています。1
前作で微妙だったところが細かく改善されていて良い感じですね。

概要

kintone上で他のユーザの予定を確認できるJSカスタマイズを作成します。
コメントのユーザにカーソルを当てるとツールチップでGaroonの予定が確認できるようにします。

kintoneからGaroonの予定が確認できる.gif

ソースコードはこちら
https://gist.github.com/tasshi-me/43ca377788ee46a7f7a5a572cf707bfb

Garoon REST APIについて

2018年に公開されたGaroon REST APIではスケジュールの取得/登録/更新/削除や、ワークフローの申請データ取得などが可能です。2

同一サブドメインのkintoneからのGETリクエストはセッション認証を利用して簡単に実行することができます。
Garoon REST APIの共通仕様 – cybozu developer network

詳しい利用方法などは公式HPを参照してください。
Garoon API – cybozu developer network

※実際の開発では有志の方が提供されているGaroon REST APIクライアントライブラリを使用すると便利です。3
本記事のサンプルコードでは説明の簡略化のためFetch APIを使用しています。

Fetch の使用 - Web API | MDN

手順

ざっくりと次のような順番で進めていきます。

  1. ツールチップを表示する
  2. ユーザのログイン名を取得する
  3. ログイン名からGaroonユーザIDを取得する
  4. GaroonユーザIDからGaroonの予定を取得する
  5. ツールチップに予定を表示する
  6. 動作確認
  7. おまけ:キャッシュ

※この記事のサンプルコードではエラーハンドリングやデータの検査などは行っていません。
実際にJSカスタマイズを作成する際はkintoneのセキュアコーディング ガイドラインなどを参考にして適切な処理を行ってください。4

ツールチップを表示する

まずはkintoneのスレッド画面でユーザ(アイコン・表示名)にカーソルを当てた際に、ツールチップを表示します。

ユーザ名とアイコン.png

このユーザのアイコンと名前の要素はそれぞれ以下のセレクタで取ることができます。

要素 セレクタ
ユーザ名 .ocean-ui-comments-commentbase-user
ユーザアイコン .ocean-ui-comments-commentbase-entity > a

※これらのDOM構造やid/class属性は予告なく変更されることがあります5

まずツールチップの要素を作成します。

// ツールチップの要素を作成
const scheduleTooltipEl = document.createElement("div");
scheduleTooltipEl.classList.add("schedule-viewer-root");
scheduleTooltipEl.textContent = "ここにスケジュールが表示されます";
scheduleTooltipEl.style.display = "none";
scheduleTooltipEl.style.position = "absolute";
scheduleTooltipEl.style.pointerEvents = "none";
scheduleTooltipEl.style.backgroundColor = "white";
scheduleTooltipEl.style.padding = "0 10px";
scheduleTooltipEl.style.border = "1px solid black";
document.body.appendChild(scheduleTooltipEl);

次に、対象となる要素に対してmouseenter, mouseleaveイベントのイベントリスナーを設定します。
Element: mouseenter イベント - Web API | MDN

今回はユーザ名の要素を対象としましょう。

// 対象の要素にカーソルが当たっている間、ツールチップを表示する
const userLinkEl = document.querySelector(
  ".ocean-ui-comments-commentbase-user"
);

// カーソルが当たった時の処理
userLinkEl.addEventListener("mouseenter", async (event) => {
  const targetEl = event.target;

  //1秒後にまだカーソルが当たっていた場合、予定一覧の要素を表示する
  setTimeout(async () => {
    if (targetEl.querySelector(":hover") != null) {
      // カーソルの位置に要素を表示
      scheduleTooltipEl.style.left = event.pageX + "px";
      scheduleTooltipEl.style.top = event.pageY + "px";
      scheduleTooltipEl.style.display = "block";
    }
  }, 1000);
});

// カーソルが離れた時の処理
userLinkEl.addEventListener("mouseleave", () => {
  // ツールチップを非表示にする
  scheduleTooltipEl.style.display = "none";
});

これでユーザ名にカーソルを当てるとツールチップが表示されるようになりました。
ここにREST APIで取得した予定を表示します。

ツールチップが表示される.gif

ユーザのログイン名を取得する

ユーザ名の要素から、そのユーザのログイン名を取得します。

ユーザ名の要素にはピープルへのリンクが設定されているので、そこから正規表現でログイン名を取得します。
先ほどのmouseenterイベントリスナのコールバック内で、処理を行います。

// 対象の要素からユーザログイン名を取得する
// ピープルへのリンクのフラグメント: "#/people/user/<ログイン名>"
const peopleUrlHash = targetEl.hash;

// ログイン名を正規表現で抽出
const regex = /^#\/people\/user\/(.+)$/g;
const targetUserCode = [...peopleUrlHash.matchAll(regex)][0][1];

ログイン名からGaroonユーザIDを取得する

対象ユーザのログイン名が分かりましたが、Garoonの予定を取得するにはGaroonユーザIDが必要です。

GaroonユーザIDは「ユーザーの一覧取得(クエリで条件を指定)」APIで取得できます。

ユーザーの取得(GET) – cybozu developer network

今回はログイン名が分かっているので、nameパラメータにログイン名を指定して実行します。

// Garoonユーザを取得する
const response = await fetch("/g/api/v1/base/users?name=" + code, {
  method: "GET",
  headers: {
    "Content-Type": "application/json",
    "X-Requested-With": "XMLHttpRequest",
  },
});
const body = await response.json();

nameパラメータの検索対象は以下の4つなので、レスポンスには目的のユーザ以外も含まれる場合があります。

  • 表示名
  • ログイン名
  • 個人設定のローカライズで設定された名前(クラウド版のみ)
  • 英語表記(パッケージ版のみ)

そのため、レスポンスから users[].codeが指定したログイン名と一致するユーザを取り出します。

// 得られたユーザのうち、users[].codeが指定したログイン名と一致するユーザを取り出す
const targetUser = body.users.filter((user) => user.code === targetUserCode)[0];
const targetUserId = targetUser.id; // GaroonユーザID

GaroonユーザIDからGaroonの予定を取得する

GaroonユーザIDが分かったのでいよいよ予定を取得します。

予定の取得には「予定の一覧取得(クエリで条件を指定)」APIを使用します。

予定の取得(GET) – cybozu developer network

リクエストに指定するパラメータは以下の通りです。

|パラメータの種類|指定する値|
|---|---|---|
|取得するプロパティ|ID, 開始日時, 終了日時, 予定メニュー, タイトル|
|ソート条件|開始日時 昇順|
|検索範囲の日時|当日の00:00から11:59まで|
|絞り込み条件|参加者に<対象ユーザ>が含まれる|

// 対象ユーザの当日の予定を取得する
// 現在日時
const currentDate = new Date();
// 検索対象の開始日時: 当日00:00:00
const rangeStart = new Date(
  currentDate.getFullYear(),
  currentDate.getMonth(),
  currentDate.getDate()
);
// 検索対象の終了日時: 翌日00:00:00
const rangeEnd = new Date(
  currentDate.getFullYear(),
  currentDate.getMonth(),
  currentDate.getDate() + 1
);
// リクエストパラメータ
const params = {
  fields: ["id", "start", "end", "eventType", "eventMenu", "subject"],
  orderBy: "start asc",
  rangeStart: rangeStart.toISOString(),
  rangeEnd: rangeEnd.toISOString(),
  target: targetUserId,
  targetType: "user",
};

const response = await fetch(
  "/g/api/v1/schedule/events?" + new URLSearchParams(params),
  {
    method: "GET",
    headers: {
      "Content-Type": "application/json",
      "X-Requested-With": "XMLHttpRequest",
    },
  }
);
const body = await response.json();
const todayEvents =  body.events;

得られた予定一覧を、現在の予定、今後の予定、終日の予定に分けます。

// 現在の予定、今後の予定、終日の予定に分ける
// 現在日時
const currentDate = new Date();

const currentEvents = new Array();
const upcomingEvents = new Array();
const allDayEvents = new Array();

for (const event of events) {
  const startDate = new Date(event.start.dateTime);
  const endDate = new Date(event.end.dateTime);
  // 終日の予定
  if (
    startDate.getHours() === 0 &&
    startDate.getMinutes() === 0 &&
    startDate.getSeconds() === 0 &&
    endDate.getHours() === 23 &&
    endDate.getMinutes() === 59 &&
    endDate.getSeconds() === 59
  ) {
    allDayEvents.push(event);
    continue;
  }
  // 現在の予定
  if (startDate <= currentDate && endDate >= currentDate) {
    currentEvents.push(event);
    continue;
  }
  // 今後の予定
  if (startDate > currentDate) {
    upcomingEvents.push(event);
    continue;
  }
}

ツールチップに予定を表示する

いよいよkintone上に予定を表示します。

手順1のイベントリスナのコールバック内でツールチップにスケジュールの内容を表示します。

// ツールチップに予定を表示する

// 予定一覧の要素を作成する関数
const createEventsEl = (events, description) => {
  const rootEl = document.createElement("div");
  const headerEl = document.createElement("p");
  headerEl.textContent = description;
  rootEl.appendChild(headerEl);

  const bodyEl = document.createElement("ol");
  for (const event of events) {
    const startDate = new Date(event.start.dateTime);
    const endDate = new Date(event.end.dateTime);

    const startDateStr =
      startDate.getHours().toString().padStart(2, "0") +
      ":" +
      startDate.getMinutes().toString().padStart(2, "0");
    const endDateStr =
      endDate.getHours().toString().padStart(2, "0") +
      ":" +
      endDate.getMinutes().toString().padStart(2, "0");

    const el = document.createElement("li");
    el.textContent = `${startDateStr}-${endDateStr} [${event.eventMenu}] ${event.subject}`;
    bodyEl.appendChild(el);
  }
  rootEl.appendChild(bodyEl);

  return rootEl;
};

// ツールチップをクリア
scheduleTooltipEl.textContent = "";
// タイトル(ユーザ名)
const userNameEl = document.createElement("p");
userNameEl.textContent = `${targetUser.name} さんの予定`;
scheduleTooltipEl.appendChild(userNameEl);
// 現在の予定
scheduleTooltipEl.appendChild(createEventsEl(currentEvents, "現在の予定"));
// 今後の予定
scheduleTooltipEl.appendChild(createEventsEl(upcomingEvents, "今後の予定"));
// 終日の予定
scheduleTooltipEl.appendChild(createEventsEl(allDayEvents, "終日の予定"));

動作確認

それでは実際の動作を確認してみましょう。
kintoneシステム管理 > JavaScript / CSSでカスタマイズ からJSカスタマイズを適用します。

JSカスタマイズを適用.png

ツールチップで予定を確認できるようになりました。

動作確認.gif

最終的なコードは以下のようになります。
実行タイミング調整のため全体をloadイベントリスナのコールバック内で実行しています。

index.js
(async () => {
  window.addEventListener("load", async () => {
    // ツールチップの要素を作成
    const scheduleTooltipEl = document.createElement("div"); // 要素を作成
    scheduleTooltipEl.classList.add("schedule-viewer-root");
    scheduleTooltipEl.textContent = "ここにスケジュールが表示されます";
    scheduleTooltipEl.style.display = "none";
    scheduleTooltipEl.style.position = "absolute";
    scheduleTooltipEl.style.pointerEvents = "none";
    scheduleTooltipEl.style.backgroundColor = "white";
    scheduleTooltipEl.style.padding = "0 10px";
    scheduleTooltipEl.style.border = "1px solid black";
    document.body.appendChild(scheduleTooltipEl);

    // 対象となる要素
    const userLinkSelectors = [
      ".ocean-ui-comments-commentbase-user",
      ".ocean-ui-comments-commentbase-entity > a",
      ".ocean-space-thread-update",
      ".ocean-ui-plugin-mention-user.ocean-ui-plugin-linkbubble-no",
    ];
    const userLinkEls = document.querySelectorAll(userLinkSelectors.join());

    // 対象の要素にカーソルが当たっている間、ツールチップを表示する
    for (const userLinkEl of userLinkEls) {
      // カーソルが離れた時の処理
      userLinkEl.addEventListener("mouseleave", () => {
        // ツールチップを非表示にする
        scheduleTooltipEl.style.display = "none";
      });

      // カーソルが当たった時の処理
      userLinkEl.addEventListener("mouseenter", async (event) => {
        const targetEl = event.target;

        //1秒後にまだカーソルが当たっていた場合、予定一覧の要素を表示する
        setTimeout(async () => {
          if (targetEl.querySelector(":hover") != null) {
            // カーソルの位置に要素を表示
            scheduleTooltipEl.style.left = event.pageX + "px";
            scheduleTooltipEl.style.top = event.pageY + "px";
            scheduleTooltipEl.textContent = "";
            scheduleTooltipEl.style.display = "block";

            // 対象の要素からログイン名を取得する
            const targetUserCode = getUserCodeFromAnchorElement(targetEl);

            // Garoonユーザを取得する
            const targetUser = await fetchGaroonUserByCode(targetUserCode);
            const targetUserId = targetUser.id; // GaroonユーザID

            // 対象ユーザの当日の予定を取得する
            const todayEvents = await fetchTodayEventsByUserId(targetUserId);

            // 現在の予定、今後の予定、終日の予定に分ける
            const [
              currentEvents,
              upcomingEvents,
              allDayEvents,
            ] = categorizeEvents(todayEvents);

            // ツールチップに予定を表示する
            // タイトル(ユーザ名)
            const userNameEl = document.createElement("p");
            userNameEl.textContent = `${targetUser.name} さんの予定`;
            scheduleTooltipEl.appendChild(userNameEl);
            // 現在の予定
            scheduleTooltipEl.appendChild(
              createEventsEl(currentEvents, "現在の予定")
            );
            // 今後の予定
            scheduleTooltipEl.appendChild(
              createEventsEl(upcomingEvents, "今後の予定")
            );
            // 終日の予定
            scheduleTooltipEl.appendChild(
              createEventsEl(allDayEvents, "終日の予定")
            );
          }
        }, 1000);
      });
    }
  });

  // 対象の要素からログイン名を取得する
  const getUserCodeFromAnchorElement = (el) => {
    // ピープルへのリンクのフラグメント: "#/people/user/<ログイン名>"
    const peopleUrlHash = el.hash;

    // ログイン名を正規表現で抽出
    const regex = /^#\/people\/user\/(.+)$/g;
    const targetUserCode = [...peopleUrlHash.matchAll(regex)][0][1];
    return targetUserCode;
  };

  // Garoonユーザを取得する
  const fetchGaroonUserByCode = async (code) => {
    const response = await fetch("/g/api/v1/base/users?name=" + code, {
      method: "GET",
      headers: {
        "Content-Type": "application/json",
        "X-Requested-With": "XMLHttpRequest",
      },
    });
    const body = await response.json();

    // 得られたユーザのうち、users[].codeが指定したログイン名と一致するユーザを取り出す
    const targetUser = body.users.filter((user) => user.code === code)[0];
    return targetUser;
  };

  // 対象ユーザの当日の予定を取得する
  const fetchTodayEventsByUserId = async (garoonUserId) => {
    // 現在日時
    const currentDate = new Date();
    // 検索対象の開始日時: 当日00:00:00
    const rangeStart = new Date(
      currentDate.getFullYear(),
      currentDate.getMonth(),
      currentDate.getDate()
    );
    // 検索対象の終了日時: 翌日00:00:00
    const rangeEnd = new Date(
      currentDate.getFullYear(),
      currentDate.getMonth(),
      currentDate.getDate() + 1
    );
    // リクエストパラメータ
    const params = {
      fields: ["id", "start", "end", "eventType", "eventMenu", "subject"],
      orderBy: "start asc",
      rangeStart: rangeStart.toISOString(),
      rangeEnd: rangeEnd.toISOString(),
      target: garoonUserId,
      targetType: "user",
    };

    const response = await fetch(
      "/g/api/v1/schedule/events?" + new URLSearchParams(params),
      {
        method: "GET",
        headers: {
          "Content-Type": "application/json",
          "X-Requested-With": "XMLHttpRequest",
        },
      }
    );
    const body = await response.json();
    return body.events;
  };

  // 現在の予定、今後の予定、終日の予定に分ける
  const categorizeEvents = (events) => {
    // 現在日時
    const currentDate = new Date();

    const currentEvents = new Array();
    const upcomingEvents = new Array();
    const allDayEvents = new Array();

    for (const event of events) {
      const startDate = new Date(event.start.dateTime);
      const endDate = new Date(event.end.dateTime);
      // 終日の予定
      if (
        startDate.getHours() === 0 &&
        startDate.getMinutes() === 0 &&
        startDate.getSeconds() === 0 &&
        endDate.getHours() === 23 &&
        endDate.getMinutes() === 59 &&
        endDate.getSeconds() === 59
      ) {
        allDayEvents.push(event);
        continue;
      }
      // 現在の予定
      if (startDate <= currentDate && endDate >= currentDate) {
        currentEvents.push(event);
        continue;
      }
      // 今後の予定
      if (startDate > currentDate) {
        upcomingEvents.push(event);
        continue;
      }
    }

    return [currentEvents, upcomingEvents, allDayEvents];
  };

  // 予定一覧の要素を作成する関数
  const createEventsEl = (events, description) => {
    const rootEl = document.createElement("div");
    const headerEl = document.createElement("p");
    headerEl.textContent = description;
    rootEl.appendChild(headerEl);

    const bodyEl = document.createElement("ol");
    for (const event of events) {
      const startDate = new Date(event.start.dateTime);
      const endDate = new Date(event.end.dateTime);

      const startDateStr =
        startDate.getHours().toString().padStart(2, "0") +
        ":" +
        startDate.getMinutes().toString().padStart(2, "0");
      const endDateStr =
        endDate.getHours().toString().padStart(2, "0") +
        ":" +
        endDate.getMinutes().toString().padStart(2, "0");

      const el = document.createElement("li");
      el.textContent = `${startDateStr}-${endDateStr} [${event.eventMenu}] ${event.subject}`;
      bodyEl.appendChild(el);
    }
    rootEl.appendChild(bodyEl);

    return rootEl;
  };
})();

おまけ:キャッシュ

無事ツールチップにイベント情報を表示することができましたが、今の実装だとアイコンにカーソルを当てる度にAPIリクエストが発行されてしまいます。
このままだと無駄が多いので取得したユーザ情報、予定を再利用するようにします。

// キャッシュ
const users = new Map();

let user = users.get("<ログイン名>");
if (user == undefined){
    user = await ユーザ情報をAPIで取得();
    users.set(user.code, user);
}

まとめ

いかがでしたでしょうか?
kintone REST API, Garoon REST APIで相互にデータを参照/変更できるのは便利ですね。

今回はJSカスタマイズを作成しましたが、kintoneプラグインではそれぞれの環境に合わせて設定を変更できるため、より再利用性の高いカスタマイズを作成できます。
公式HPの開発手順にしたがって簡単に作成できるので、興味のある方は作成してみてはいかがでしょうか?
kintone プラグイン開発手順 – cybozu developer network

ここまで読んでいただきありがとうございました。

  1. Fit Boxing 2(フィットボクシング2)| Nintendo Switch

  2. 2018/08/12の定期メンテナンスにおけるkintone API、Garoon API、User API更新情報(2018/07/13) – cybozu developer network

  3. @miyajan/garoon-rest - npm

  4. セキュアコーディング ガイドライン – cybozu developer network

  5. kintone JavaScript コーディングガイドライン – cybozu developer network

6
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
6
1