JetThemeでトップページ・関連記事・ガジェット等の画像なし記事に専用のサムネイルを表示する【Blogger】

JetThmeでは、画像のない投稿記事に関して、トップページの投稿一覧や記事検索画面、サイドバーの新着記事・人気の投稿ガジェット、記事下の関連記事セクションなどでサムネイル画像が表示されないため、レイアウトに若干崩れが生じる場合があります。
画像の有無に関係なくレイアウトの統一感を保つため、画像がない投稿には専用の代替サムネイル(No Image画像)を自動で表示するカスタマイズ方法を紹介します。
テンプレートファイルの編集を行うため、事前にバックアップすることを推奨します。
以降の掲載コードおよびテンプレート内の行数は素のJetTheme v2.9のものです。既にカスタマイズ済みの場合、行数は目安としてください。また、筆者はHTMLやJavaScriptについては素人・勉強中です。筆者環境で動作確認はしていますが、コードに間違いや非効率的な部分などあるかもしれません。
目次
トップページ・人気の投稿ガジェットのNo Image対応方法
トップページ系(記事一覧・アーカイブ・記事検索結果など)
<b:includable id='JetArchive' var='post'>
で検索し、その少し下1940行目付近にある
<b:if cond='data:post.featuredImage'>
<div class='item-thumbnail'>
<a class='jt-bg-light d-block ratio ratio-16x9' expr:href='data:post.url.canonical' rel='bookmark'>
<img expr:alt='data:post.title' expr:class='data:post.featuredImage.isYoutube ? "object-cover lazyload" : "lazyload"' expr:data-src='data:post.featuredImage.isYoutube ? data:post.thumbnailUrl : data:post.featuredImage' loading='lazy' src=''/>
</a>
</div>
</b:if>
の部分を以下のコードで置き換えます。
<div class='item-thumbnail'>
<a class='jt-bg-light d-block ratio ratio-16x9' expr:href='data:post.url.canonical' rel='bookmark'>
<b:if cond='data:post.featuredImage'>
<img expr:alt='data:post.title' expr:class='data:post.featuredImage.isYoutube ? "object-cover lazyload" : "lazyload"' expr:data-src='data:post.featuredImage.isYoutube ? data:post.thumbnailUrl : data:post.featuredImage' loading='lazy' src=''/>
<b:else />
<img expr:alt='data:post.title' class='lazyload' expr:data-src='【NoImage用画像のURL】' loading='lazy' src=''/>
</b:if>
</a>
</div>
【No Image用画像のURL】部分については、ご自身で用意したBlogger内画像URLに書き換えてください。JetTheme内のJavaScriptによりBlogger画像URLのパラメータに準拠して書き換えられサイズ調整されるため、必ずBloggerにアップロードした画像のURLを使ってください。
人気の投稿ガジェット
同様に、<b:includable id='JetPopularPosts'>
で検索し、その少し下1794行目付近の
<b:if cond='data:post.featuredImage'>
<div class='item-thumbnail me-3' style='width:85px'>
<a class='rounded jt-bg-light overflow-hidden d-block ratio ratio-1x1' expr:href='data:post.url.canonical'>
<img expr:alt='data:post.title' expr:class='data:post.featuredImage.isYoutube ? "object-cover lazyload" : "lazyload"' expr:data-src='data:post.featuredImage.isYoutube ? data:post.featuredImage.youtubeMaxResDefaultUrl : data:post.featuredImage' loading='lazy' src=''/>
</a>
</div>
</b:if>
を以下のコードで置き換えます。
<div class='item-thumbnail me-3' style='width:85px'>
<a class='rounded jt-bg-light overflow-hidden d-block ratio ratio-1x1' expr:href='data:post.url.canonical'>
<b:if cond='data:post.featuredImage'>
<img expr:alt='data:post.title' expr:class='data:post.featuredImage.isYoutube ? "object-cover lazyload" : "lazyload"' expr:data-src='data:post.featuredImage.isYoutube ? data:post.featuredImage.youtubeMaxResDefaultUrl : data:post.featuredImage' loading='lazy' src=''/>
<else />
<img expr:alt='data:post.title' class='lazyload' data-src='【NoImage用画像のURL】' loading='lazy' src=''/>
</b:if>
</a>
</div>
こちらの【NoImage用画像のURL】も同様にBlogger内画像を指定してください。
変更点の解説
トップページおよび人気の投稿のどちらもBloggerの制御タグを使って同じような分岐をさせています。
元のコードでは<b:if cond='data:post.featuredImage'>
によって投稿内画像の有無を判定し、存在する場合にのみdiv
タグやa
タグを含むサムネイル要素を出力する設定となっています。
一方、改修したコードでは<b:else />
を追加し、投稿内に画像がない場合には【NoImage用画像のURL】を含む方のimg
タグを出力するよう追記しています。また、投稿内画像のあるなしに関わらず<div class='item-thumbnail ~>
等の画像関連のタグを出力するように<b:if ~>~</b:if>
で囲む箇所を変更しています。
<b:if cond='data:post.featuredImage'>
data:post.featuredImageが真(投稿内に画像がある)の場合に出力されるimgタグ
<b:else />
それ以外(投稿内に画像がない)の場合に出力されるimgタグ
</b:if>
また、「data:post.featuredImage
が真(投稿内に画像がある)の場合に出力されるimg
タグ」について、expr
属性と三項演算子を用いて投稿内の最初の画像がYouTubeのサムネイルかBlogger内画像か判定し、それに基づいて画像URLや遅延読み込み用のクラスを動的に変更しています。この条件部分と値を変更することで<b:else />
を使う代わりに以下のように書くこともできます(この場合は前後の<b:if cond='data:post.featuredImage'>
および</b:if>
も不要になるので削除してください)。
<img expr:alt='data:post.title' expr:class='data:post.featuredImage.isYoutube ? "object-cover lazyload" : "lazyload"' expr:data-src='data:post.featuredImage.isYoutube ? data:post.thumbnailUrl : (data:post.featuredImage ? data:post.featuredImage : "【NoImage用画像のURL】")' loading='lazy' src=''/>
関連記事・最新記事のNo Image対応方法
ブログ記事下部の関連記事セクションおよびサイドバーの最新記事ガジェットに関してはJavaScriptを修正して専用サムネイルを表示させます。
関連記事
function related_temp(e)
で検索し、2686行目付近の以下のコードを
function related_temp(e) {
return (e.title ? "<div class='widget-title position-relative mb-3 text-uppercase fw-light'><span>" + e.title + "</span></div>" : "") + "<div class='row row-cols-sm-2'>" + e.posts.map(function (data, i) {
return "<article class='mb-4'><div class='h-100 overflow-hidden rounded position-relative border jt-border-light bg-archive shadow-sm'>" + (data.img ? "<div class='item-thumbnail'><a class='jt-bg-light d-block ratio ratio-21x9' href='" + data.url + "'><img alt='" + data.title + "' class='object-cover lazy-" + data.grup_id + " lazyload' data-src='" + data.img + "' loading='lazy' src=''/></a></div>" : "") + "<div class='item-content p-4'><h2 class='item-title fs-6 mb-2' itemprop='headline'><a class='text-reset' href='" + data.url + "'>" + data.title + "</a></h2><div class='item-meta text-secondary d-flex flex-wrap fw-light'>" + (data.author != "Unknown" ? "<small class='me-2'><svg aria-hidden='true' class='me-1 jt-icon'><use xlink:href='#i-user'/></svg>" + data.author + "</small>" : "") + "<small class='me-2'><svg aria-hidden='true' class='me-1 jt-icon'><use xlink:href='#i-clock'/></svg>" + data.date + "</small></div></div></div></article>";
}).join("") + "</div>";
}
まるまる以下のコードで置き換えます。
// 関連記事NoImage対応
function related_temp(e) {
return (e.title ? "<div class='widget-title position-relative mb-3 text-uppercase fw-light'><span>" + e.title + "</span></div>" : "") + "<div class='row row-cols-2 row-cols-md-3'>" + e.posts.map(function (data, i) {
const imgSrc = data.img ? data.img : "【NoImage用画像のURL】";
return "<article class='mb-3 px-2'><div class='h-100 overflow-hidden rounded position-relative border jt-border-light bg-archive shadow-sm'>" + "<div class='item-thumbnail'><a class='jt-bg-light d-block ratio ratio-21x9' href='" + data.url + "'><img alt='" + data.title + "' class='object-cover lazy-" + data.grup_id + " lazyload' data-src='" + imgSrc + "' loading='lazy' src=''/></a></div>" + "<div class='item-content p-2'><h2 class='item-title fs-7 mb-0' itemprop='headline'><a class='text-reset' href='" + data.url + "'>" + data.title + "</a></h2><div class='item-meta text-secondary d-flex flex-wrap fw-light'>" + "<small class='me-2'><svg aria-hidden='true' class='me-1 jt-icon'><use xlink:href='#i-clock'/></svg>" + data.date + "</small></div></div></div></article>";
}).join("") + "</div>";
}
【NoImage用画像のURL】は他と同様にBloggerにアップロードした画像URLを指定してください。
最新記事ガジェット
function sidebar_temp(e)
で検索し、2717行目付近の以下のコードを
function sidebar_temp(e) {
return (e.title ? "<div class='widget-title position-relative fs-6 mb-3'><span>" + e.title + "</span></div>" : "") + "<div class='mb-4'>" + e.posts.map(function (data, i) {
return "<div class='item-post d-flex mb-3'>" + (data.img ? "<div class='item-thumbnail me-3' style='width:85px'><a class='rounded jt-bg-light overflow-hidden d-block ratio ratio-1x1' href='" + data.url + "'><img alt='" + data.title + "' class='object-cover lazy-" + data.grup_id + " lazyload' data-src='" + data.img + "' loading='lazy' src=''/></a></div>" : "") + "<div class='item-content col'><h3 class='item-title fs-7 mb-2' itemprop='headline'><a class='text-reset' href='" + data.url + "'>" + data.title + "</a></h3><div class='item-meta text-secondary d-flex flex-wrap fs-8'><small class='me-2'><svg aria-hidden='true' class='me-1 jt-icon'><use xlink:href='#i-clock'/></svg>" + data.date + "</small></div></div></div>";
}).join("") + "</div>";
}
まるまる以下のコードで置き換えます。
// 最新記事NoImage対応
function sidebar_temp(e) {
return (e.title ? "<div class='widget-title position-relative fs-6 mb-3'><span>" + e.title + "</span></div>" : "") + "<div class='mb-4'>" + e.posts.map(function (data, i) {
const imgSrc = data.img ? data.img : "【NoImage用画像のURL】";
return "<div class='item-post d-flex mb-3'>" + "<div class='item-thumbnail me-3' style='width:85px'><a class='rounded jt-bg-light overflow-hidden d-block ratio ratio-1x1' href='" + data.url + "'><img alt='" + data.title + "' class='object-cover lazy-" + data.grup_id + " lazyload' data-src='" + imgSrc + "' loading='lazy' src=''/></a></div>" + "<div class='item-content col'><h3 class='item-title fs-7 mb-2' itemprop='headline'><a class='text-reset' href='" + data.url + "'>" + data.title + "</a></h3><div class='item-meta text-secondary d-flex flex-wrap fs-8'><small class='me-2'><svg aria-hidden='true' class='me-1 jt-icon'><use xlink:href='#i-clock'/></svg>" + data.date + "</small></div></div></div>";
}).join("") + "</div>";
}
これも同様に【NoImage用画像のURL】は他と同様にBloggerにアップロードした画像URLを指定してください。
変更箇所について
関連記事・最新記事ともに同様の変更を加えています。
もとのコードでは、data.img
に値があればimg
タグやそれに関連するdiv
タグ・a
タグを生成し、data.img
が空ならそれらを生成しない、という動作でした((data.img ? "<div ~ </div>" : "")
の部分)。
改修したコードでは、専用画像を利用するための変数imgSrc
を用意し、data.img
に値があればその画像を、画像がない場合は【NoImage用画像のURL】をimgSrc
に格納するよう変更しました(const imgSrc = data.img ? data.img : "【NoImage用画像のURL】";
の部分)。サムネイルにはimgSrc
の画像を表示し、記事内に画像がない場合でも常に表示するので、関連する条件分岐を削除しimg
タグなどが常に生成されるように変更しました。