4
3

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.

ブラウザバックで「ちょっと待って」的なアラートをだすhooks

Last updated at Posted at 2022-07-03

コチラでございます。

import { useEffect } from 'react';

/**
 * 一度入力した後にリロードすると、確認ダイアログを出す。
 */
export const useOnReloadAlert = (enable: boolean = true) => {
  useEffect(() => {
    if (!enable) return;

    window.history.pushState(null, '', window.location.pathname);
    const onReload = (event: BeforeUnloadEvent) => {
      event.preventDefault();
      event.returnValue = '';
    };
    window.addEventListener('beforeunload', onReload);
    return () => {
      window.removeEventListener('beforeunload', onReload);
    };
  }, [enable]);
};

enable: booleanオプションは、特定の条件化でブラウザバックアラートを出したくないので、追加しました。

具体的なケースは、SPA上で 入力 -> プレビュー -> 完了の画面遷移を行う場合です。
ページ遷移してしまえばいいのですが、わざわざrecoilやcontextで構築するほどでもないな〜という気持ちで、state管理で進めた結果以下の構成となり、

type PageType = 'create' | 'preview' | 'complete';

...

  const { values, ...handlers } = useXXXForm();

  switch (pageType) {
    case 'create':
      return <CreatePage values={values} {...handlers} />;
    case 'preview':
      return (
        <PreviewPage values={values} />
      );
    case 'complete':
      return <CompletePage handleSubmit={handlers.handleSubmit} />;
  }

で、この場合ですと、completeから離脱する際にもhooksが発火してしまう可能性があります。
(e.g. stripeの決済ページcheckoutなど)

そういうシーンで以下のようにオプションを渡す場合にお使いください!

const enableOnReloadAlertPages = new Set<UserPreviewPageType>(['create', 'preview']) 
useOnReloadAlert(enableOnReloadAlertPages.has(pageType));

不要な場合は以下で大丈夫です!

import { useEffect } from 'react';

/**
 * 一度入力した後にリロードすると、確認ダイアログを出す。
 */
export const useOnReloadAlert = () => {
  useEffect(() => {
    window.history.pushState(null, '', window.location.pathname);
    const onReload = (event: BeforeUnloadEvent) => {
      event.preventDefault();
      event.returnValue = '';
    };
    window.addEventListener('beforeunload', onReload);
    return () => {
      window.removeEventListener('beforeunload', onReload);
    };
  }, []);
};
4
3
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
4
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?