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

ZOZOAdvent Calendar 2023

Day 23

みなさん、ReactでGTMをどう設定していますか?悪戦苦闘した試行錯誤の末に。

Posted at

TL;DR

ReactでGTMを利用するなら、それぞれのトリガーについて以下の設定をする必要があります。

  1. 要素の表示トリガー:ページ遷移をGTMにアピールする
  2. クリックのトリガー:リンク以外のものをトリガーするなら変数にカスタムJavaScriptを設定する
  3. スクロールのトリガー:ページ遷移をGTMに通知し、拡張イベント機能計測と重複しないように気を付ける

背景

Google Analytics 4 (GA4)で要素の表示やクリック、スクロールを計測しています。計測対象のWebサイトは、Gatsby.jsで開発しており、GA4への接続はGoogle Tag Manager (GTM)を利用しています。GTMの標準的なトリガーは、SPAであるReactでは期待したように動かないことが多々あります。それらの原因を探り、様々な試行錯誤の末に辿り着いた方法をまとめます。

今回の実装ではトリガーの設定はGTMで、トリガーの対象となる要素の指定はコードで管理するという方法をとっています。具体的には以下の記事と同様の実装です。このような実装としたのは、GTM周りの管理を開発者が担当しているためで、非開発者が管理する必要がある場合には追加の設定や実装を検討する必要があると思います。

実装

1. 要素の表示トリガー:ページ遷移をGTMにアピールする

GTMでは、SPAを利用していると、期待通りに発火しないことがあります。そしてその解決方法が一般的に確立されていないということまで含めて、あるあるだと思います。

当初はGTMで設定されているトリガーを利用して試行錯誤していました(後述します)が、どんなに頑張っても解決に至らなかったため、以下の記事と同様にIntersection Observerを利用しました。

上記の記事を参考に、react-intersection-observerを利用した上で、同じ設定を使い回しやすいよう、カスタムフックにしてみました。

useElementVisibility.ts
import { useInView } from 'react-intersection-observer'

export const useElementVisibility = (viewedComponent: string) => {
  return useInView({
    threshold: 0.5,
    triggerOnce: true, // GTMの、要素の表示トリガーと同様にページごとに1回発火させたい
    onChange(inView) {
      if (inView) {
        window.dataLayer = window.dataLayer || []
        window.dataLayer.push({
          event: 'viewComponent',
          viewedComponent,
        })
      }
    },
  })
}

計測対象のコンポーネントを呼び出す場所で、以下のように設定します。

import { useElementVisibility } from '../../models/ElementVisibility'
...
  const [targetRef] = useElementVisibility('計測対象として管理したい変数を入れる')

// `key`の属性は、子コンポーネントの一部のみ切り替わるようなページ遷移の場合に必要となります。続けて詳しい説明をします。
...

            <div key={pageIndex}>
                <div ref={targetRef}>
                  {/* 計測対象のコンポーネント */}
                </div>
            </div>

...

ここで1つ気をつけるべきポイントがあります。それはページの一部のみを切り替えるような遷移の場合です。例えばニュースサイトの場合、複数のページから成る記事の、記事内における遷移です。(タイトルや目次などの構造の大部分は切り替わらず内側のコンテンツ部分が切り替わるイメージ、クエリパラメータで?page=2みたいなイメージです。)

この場合、切り替わらなかったコンポーネントは既にトリガーが発火しており、ページ遷移(クエリパラメータが変わっても)しても再度発火しません。これを解決するには、要素の表示を計測するコンポーネントより親の、ページ切り替えを担ってるコンポーネントに、ページごとに異なる値をkeyに渡す必要があります。こうすることでページ遷移と共に配下のReactコンポーネントが別のものとして描画され、Intersection Observerがページ遷移後でも発火するようになります。

当初利用していた方法

当初はライブラリを入れなくてもいい以下の方法で運用していたのですが、上記の問題のようにページの一部のコンポーネントのみを切り替えるページ遷移が必要となりました。ライブラリを増やさないようにと試行錯誤したのですが、Safariでどうしても解決できず上記の結論に至りました。

コンポーネントの一部を切り替える遷移が無いならこちらの方法も使えます

概ねこちらの記事と同様に、計測対象にdata属性で設定します。

しかしながら、同じ構造を持つページ間、つまり同じReactのテンプレートコンポーネントを利用しているなどの場合に、なぜかページ遷移が検知されません。

解決策は、トリガー対象となるdata属性の値をマウント後に更新することです。今回要素の表示トリガーの対象として設定したのがdata-gtm-viewで、この属性がついていれば発火します。gtmViewという変数に計測対象として記録したい文字列を渡し、マウント後にuseEffect内で設定するようにします。こうすることでページ遷移する度にdata-gtm-viewの値が更新していると認識され、発火するようになります。

...

  const [gtmView, setGtmView] = useState<string>()

  useEffect(() => {
    setGtmView('toppage-latest-container')
  }, [])

...

        <div data-gtm-view={gtmView}>

...

2. クリックのトリガー:リンク以外のものをトリガーするなら変数にカスタムJavaScriptを設定する

クリックトリガーはGTM標準のトリガーを使います。その場合全ての要素リンクのみのどちらかを選択できます。リンクのみを選ぶメリットとしては、確実にページ遷移が行われるaタグのみを指定できることです。

全ての要素を利用すると、ページ遷移しないボタンなどの任意の要素のクリックでトリガーすることができます。しかしトリガーされた要素の値を取得して保存したい場合に問題が発生します。例えば<button>の要素にdata属性で値を設定していたとします。もし<button>の子要素で、クリックのトリガーが発生しても、通常では親要素である<button>の値にアクセスできません。

解決策はまさにこちらにまとめてくださっているので、ご紹介します。

計測対象のコンポーネント
...

              <div
                data-gtm-click="計測対象として管理したい変数を入れる"
                data-gtm-id={articleId}
                data-gtm-order={index + 1}
              >

...
GTMの変数に設定するカスタムJavaScript
function() {
  var attribute = 'data-gtm-click'
  return {{Click Element}}.closest('[' + attribute + ']').getAttribute(attribute)
}

3. スクロールのトリガー:ページ遷移をGTMに通知し、拡張イベント機能計測と重複しないように気を付ける

スクロールもGTM標準のトリガーを利用しますが、計測する上で気を付けるべきポイントは以下です。

a. SPAでページ遷移していることを認識させるために、履歴の変更をトリガーに、gtm.loadを実行する
b. 重複しないように拡張イベント計測機能のスクロールをオフにすることを忘れない

aのコードは以下になります。ページ遷移時、今回は履歴の変更トリガー時に以下を実行するようにしました。カスタムHTMLとしてタグに登録する。

<script>
  window.dataLayer = window.dataLayer || [];
  setTimeout(function() {
      window.dataLayer.push({
        'event': 'gtm.load'
      });
  }, 500); // ページ全体が読み込まれるまで待つ
</script>

感想

大変でした。みなさんどうやって計測しているだろうって思ってます。
もっといいツールないかなー。無いなら作るしかないのかなー。

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