前回記事ページを作成しました。
NotionAPIで取得したslugの値で記事リンクを作成する | getStaticPathとDynamic routesが便利だ_Blog learn04
今こんな感じ。
記事ページに表示させている「Post: notion-api_make-blog」はNotionDBのslug列で記入している文字。
const RenderPost = (post) => {
const router = useRouter();
const { slug } = router.query;
return (
<div className={styles.container}>
<div className={styles.mainContent}>
<p>Post: {slug}</p> 👉 これ
</div>
</div>
);
};
export default RenderPost;
記事一覧ページ(blog/index.tsx)で使うような
componentでペトペト貼って使えるようにするぞ......!!!!!
っていうのが今回のテーマです\(^o^)/
const RenderPosts = ({ posts = [] }) => {
return (
<div className={styles.container}>
<div className={styles.mainContent}>
<p>***main-content***</p>
<h2>BlogList page</h2>
<br />
{""}
<section>
<div>
{posts.map((post) => {
return (
<div key={post.Slug}>
<PostDate post={post} /> 👉 これ使いたいなー
<PostTitle post={post} /> 👉 これもあったらいいよなー
<PostExcerpt post={post} /> 👉 これは本文にはいらないかなーーー
</div>
);
})}
</div>
/// 以下省略......
いつものようにcomponentを使いたい場所に貼ってみました。
import { PostDate } from "../../components/blog-parts";
//省略
const RenderPost = (post) => {
const router = useRouter();
const { slug } = router.query;
return (
<div className={styles.container}>
<div className={styles.mainContent}>
<p>Post: {slug}</p>
<PostDate post={post} /> 👉 ペトっ
</div>
</div>
);
};
export default RenderPost;
日付がNanNanNanです.....。
なんでしょう......(´゚д゚`)
以下、useEffectが原因だと思い込んで説明してしまっているので、
useEffectの中のifがあってredirectが設定されていないから条件を満たしていない
ということを頭の隅に置きながら読み進めていってください....(^_^;)
結果を先に知りたい方は、「原因を振り返ると....」へ飛んでくださいm(_ _)m
見慣れないコードを発見。
useEffect
は何をやっているのか?
レンダー後に何かの処理をしないといけない、ということを React に伝えます。
React はあなたが渡した関数を覚えており(これを「副作用(関数)」と呼ぶこととします)、DOM の更新の後にそれを呼び出します。
useEffect
は毎回のレンダー後に呼ばれるのか?
「マウント」と「更新」という観点で考えるのではなく、「レンダーの後」に副作用は起こる、というように考える方が簡単かもしれません。
React は、副作用が実行される時点では DOM が正しく更新され終わっていることを保証します。
ちょっと公式Docは私には難解だったので、
こちらの記事がしっくりと理解できました....😅
【React Hooks】useEffectの基本的な動きを理解して使いこなそう
useEffect
は、関数コンポーネント内で副作用(side-effect)を実行するためのHookです。
副作用とは、関数コンポーネントの出力(レンダリング)に関係ない処理のことです。
つまり、useEffect
を用いることでレンダリングと副作用を切り離すことが可能になります。
useEffect(callback[, dependencies]);
callback関数について調べつつ構文を解釈すると.....
っていうイメージ。
Reactがやっと出てきました。
React Hookってよく耳にしますが、その一味なようです。
実行させるタイミングを指定できるのがuseEffectの特徴。
import { useEffect } from "react";
//省略
const RenderPost = ({ post, redirect }) => {
const router = useRouter();
const { slug } = router.query;
/// ▼ ここから....
useEffect(() => {
if (redirect && !post) {
router.replace(redirect);
}
}, [router, redirect, post]);
//// ▲ ここまでの部分
return (
<div className={styles.container}>
<div className={styles.mainContent}>
<p>Post: {slug}</p>
<PostDate post={post} />
</div>
</div>
);
};
export default RenderPost;
日付が表示されたーーー\(^o^)/
結局useEffectはpostが無かった時に処理するためのものだ.....
**** 追加しました 2022/05/18 ******
後日もう一度作り直してみて、NaNNanNaNの原因を調べるマスーーーー。(´・ω・`)
単純な記述ミスかもしれない.....。
NaNNaNNaNはいったい何なんだ~~~~~~~~
*** 追加しました 2022/05/21 ***
Blog learn04をForkして作り直してみると....
NaN-aN-aNが再び出現!!!!
....全然分からない.....m(_ _)m
みつけたーーーーーー(´゚д゚`)
useEffectを追加する時にredirectを引数として加える際
{ }でくくったことでNaNが直った....ってことが分かりました。
やはりアルパカ先生のご指摘どおり、
useEffectの不足によるものではなく、別の理由(単に文法ミス)でした(;・∀・)
ひょいなことからuseEffectをゴリゴリ調べることとなりましたが、
これはこれで勉強になりました\(^o^)/www
import { useEffect } from "react";
//省略
const RenderPost = ({ post }) => {
const router = useRouter();
const { slug } = router.query;
return (
<div className={styles.container}>
<div className={styles.mainContent}>
<p>Post: {slug}</p>
<PostDate post={post} />
</div>
</div>
);
};
export default RenderPost;
import { PostDate,PostTitle } from "../../components/blog-parts";
//省略
const RenderPost = ({ post }) => {
//省略
return (
<div className={styles.container}>
<div className={styles.mainContent}>
<p>Post: {slug}</p>
<PostDate post={post} />
<PostTitle post={post} /> 👉 追加
</div>
</div>
);
};
export default RenderPost;
タイトルが表示された\(^o^)/
本家のコードを確認すると、
PostTitleのcomponentにこんな記述が(´゚д゚`)
三項演算子で、
デフォルトでenableLink = true
になっているので、
不要な時にfalse
にしてあげれば良さそう★
componentにenableLinkの三項演算子を追加してあげて.....
export const PostTitle = ({ post, enableLink = true }) => {
const postTitle = post.Title ? post.Title : "";
return (
<h3>
{enableLink ? (
<Link href="/blog/[slug]" as={getBlogLink(post.Slug)} passHref>
<a>{postTitle}</a>
</Link>
) : (
postTitle
)}
</h3>
);
};
記事ページのタイトルはenableLinkを使わないように設定する。
//省略
const RenderPost = ({ post }) => {
//省略
return (
<div className={styles.container}>
<div className={styles.mainContent}>
<p>Post: {slug}</p>
<PostDate post={post} />
<PostTitle post={post} enableLink={false}/> 👉 追加
</div>
</div>
);
};
export default RenderPost;
リンクの埋め込みのないタイトルになった\(^o^)/
記事ページ([slug].tsx)にNotionDBの情報を2つ(Date・Title)を表示させるには.....
{ }
でくくることで解決今回使用したcode sandboxはこちら
useEffectがあることで、
記事ページごとに表示させる情報の切り替えができている
っていうことが分かりました\(^o^)/
次回はNotionの本文テキストを表示させちゃいます🔥
enableLinkのような使い方って、
カスタマイズする時に役立ちそう.....。
componentにあった三項演算子も勉強になるなーー🤨
私の場合で言えば、
記事一覧ページでは2カラムサイズの表示にしたいけど、
記事ページでは1つの記事だけしかないから幅をいっぱい使って表示させたい。
今の段階では2パターンでわざわざcomponentを用意して、
ガッチャガチャになってるけど、
今日学べた使い方を取り入れればだいぶ清潔になるのかな〜
なんてことを思いました(´;ω;`)
まだまだやることたくさんアリマス.....な(´゚д゚`)
Twitterでは更新のお知らせを随時行っています
興味ある方はLet'sフォロー★▼ この記事に興味があったら同じタグから関連記事をのぞいてみてね
RSSリーダーにatomのリンクを登録すると通知が行くよ🐌
https://herohoro.com/atom
やってみてね(*´ω`*)(*´ω`*)
フォロー大歓迎\(^o^)/