コンテンツの読み込みに時間がかかっています

Entry_Summaryループ内クラス設定を運営者でも触れるようにする


現在提供中の標準テーマの一覧ページでは、横に並ぶコンテンツの数(2列、3列、4列)は、テンプレートの CSS クラスの設定によって決まる仕様になっています。 この記事では、タイトルの通り、運営側でもこの設定を変更できるようにする方法について考えてみます。


問題点の検証

実際にどのような問題が起こるのかを UTSUWA のテーマ「事業紹介」を例に説明していきます。

納品直後の状態

当初、打ち合わせ時には「事業紹介」については増えないと聞いており、コンテンツとしては 2つなので、col-6 と Class を設定し、横に2枚並ぶようにテンプレートに書かれていました。



事業が追加される事に

これがサイトを運営している途中で、新しい事業が増えて 3つになる事になりました。3つ目のコンテンツは下に表示され隣は空白となります。



3つ並べたいと相談される

同じページ内の「ブログ」コンテンツは横に4枚並べて表示していますし、横に3枚並べた方がいいと思う人も多い事でしょう。



CMS で運用している中で、よくある話ではないかと思います。 詳細ページについては、ある程度レイアウト調整もできるのですが、沢山の情報を管理可能な CMS では一覧ページについて自由度が低くなる傾向があります。

対策を考えてみる

対策としては、その制御を行っているのはテンプレート上に書かれている 1つの Class という事になるので、それを管理画面側から変更できるようにする事になります。

実は準備済みでした

実は Ver. 2.5.1(2015年8月) から、モジュールの設定に ループ内クラス という設定が追加されています。これを活用する事で、管理画面上から変更できるようになります。 テンプレートに直接書かれている Class を {entry:loop.class} という変数に置き換えるだけです。



運用担当者に分かるのか?

次の問題は、この記述内容が運用担当者に理解できるのかというところになります。

  1. acms-col-6 acms-col-sm-4 acms-col-md-3

標準テーマの一覧ページでは、コンテンツの横幅を指定するために acms-col-* というクラスを使用します。 このクラスの数字は 12分割のグリッドシステム に基づいており、1行を12としたときに、各コンテンツが何列分の幅を占めるかを決定します。

具体的な例

  • acms-col-6 → 12列中6列分を使用(2列表示)
  • acms-col-sm-4 → 画面幅が smサイズ以上のとき、12列中4列分を使用(3列表示)
  • acms-col-md-3 → 画面幅が mdサイズ以上のとき、12列中3列分を使用(4列表示)

つまり、画面のサイズによって自動的に 2列 → 3列 → 4列 とレイアウトが変化します。



クラス名 デフォルトのブレイクポイント
acms-col- 全て
acms-col-sm- 480px以上〜
acms-col-md- 768px以上〜
acms-col-lg- 1024px以上〜
acms-col-xl- 1440px以上〜

この設定では、制作者には理解できたとしても、サイトを運用している担当者には難しいと感じるのではないでしょうか。

実際の実装

モジュールフィールドに、各画面の幅の時のカラム数のカスタムフィールドを作る事で実装が可能になります。



さらに工夫を

カスタム設定で設定した SELECTcol-base , col-sm , col-md )を元に、表示設定の「ループ内クラス」{entry:loop.class} を自動で書くことができる JavaScript を用意しました。こうすることでテンプレートは標準的な記述のままで済ますことができ、独自に設定したい時にも対応が可能となります。

モジュールフィールドの設定

  1. <select name="col-base" class="acms-admin-form-width-mini">
  2. <option value="acms-col-12" {col-base:selected#acms-col-12}>1カラム</option>
  3. <option value="acms-col-6" {col-base:selected#acms-col-6}>2カラム</option>
  4. <option value="acms-col-4" {col-base:selected#acms-col-4}>3カラム</option>
  5. <option value="acms-col-3" {col-base:selected#acms-col-3}>4カラム</option>
  6. </select>
  7. <input type="hidden" name="field[]" value="col-base" />
  8.  
  9. <select name="col-sm" class="acms-admin-form-width-mini">
  10. <option value="" {col-sm:selected#}>未設定 (480px以上)</option>
  11. <option value="acms-col-sm-12" {col-sm:selected#acms-col-sm-12}>1カラム</option>
  12. <option value="acms-col-sm-6" {col-sm:selected#acms-col-sm-6}>2カラム</option>
  13. <option value="acms-col-sm-4" {col-sm:selected#acms-col-sm-4}>3カラム</option>
  14. <option value="acms-col-sm-3" {col-sm:selected#acms-col-sm-3}>4カラム</option>
  15. </select>
  16. <input type="hidden" name="field[]" value="col-sm" />
  17.  
  18. <select name="col-md" class="acms-admin-form-width-mini">
  19. <option value="" {col-md:selected#}>未設定 (768px以上)</option>
  20. <option value="acms-col-md-12" {col-md:selected#acms-col-md-12}>1カラム</option>
  21. <option value="acms-col-md-6" {col-md:selected#acms-col-md-6}>2カラム</option>
  22. <option value="acms-col-md-4" {col-md:selected#acms-col-md-4}>3カラム</option>
  23. <option value="acms-col-md-3" {col-md:selected#acms-col-md-3}>4カラム</option>
  24. </select>
  25. <input type="hidden" name="field[]" value="col-md" />
  26.  
  27. <span id="col-etc"></span>

今回の JavaScript

最初に INPUT name="entry_summary_loop_class" の値からカスタム設定の SELECT を調整します。その後、SELECT を修正すると INPUT のテキストを書き換え、INPUT 側をダイレクトに修正すると SELECT 側を書き換えるようにしています。

SELECT の OPTION に無い acms-col-2 などが INPUT に書かれた際には、自動的に SELECT に追加し、acms-col- , acms-col-sm- , acms-col-md- でない Class を書いた時には <span id="col-etc"></span> に表示するようにして追加で Class を書いていることも分かるようにしておきました。

  1. <script>
  2. ACMS.addListener("acmsReady", function() {
  3.  
  4. const loopClass = document.querySelector('input[name="entry_summary_loop_class"]');
  5.  
  6. const colBase = document.querySelector('select[name="col-base"]');
  7. const colSm = document.querySelector('select[name="col-sm"]');
  8. const colMd = document.querySelector('select[name="col-md"]');
  9. const colEtc = document.getElementById("col-etc");
  10.  
  11. function updateLoopClass() {
  12. const values = [colBase.value, colSm.value, colMd.value].filter(value => value !== "");
  13. loopClass.value = values.join(" ");
  14. }
  15.  
  16. function ensureOptionExists(select, value) {
  17. if (!value || select.querySelector(`option[value="${value}"]`)) return;
  18. const existingValues = new Set(Array.from(select.options).map(opt => opt.value));
  19. if (!existingValues.has(value)) {
  20. const option = document.createElement("option");
  21. option.value = value;
  22. option.textContent = value;
  23. option.setAttribute("data-custom", "true");
  24. select.appendChild(option);
  25. }
  26. }
  27.  
  28. function updateSelectValues() {
  29. const values = loopClass.value.split(" ").map(val => val.trim()).filter(val => val !== "");
  30. let baseValue = "", smValue = "", mdValue = "";
  31. let extraValues = [];
  32.  
  33. values.forEach(val => {
  34. if (/^acms-col-sm-\d+$/.test(val)) {
  35. smValue = val;
  36. } else if (/^acms-col-md-\d+$/.test(val)) {
  37. mdValue = val;
  38. } else if (/^acms-col-\d+$/.test(val)) {
  39. baseValue = val;
  40. } else {
  41. extraValues.push(val);
  42. }
  43. });
  44.  
  45. ensureOptionExists(colBase, baseValue);
  46. ensureOptionExists(colSm, smValue);
  47. ensureOptionExists(colMd, mdValue);
  48.  
  49. colBase.value = baseValue;
  50. colSm.value = smValue;
  51. colMd.value = mdValue;
  52.  
  53. colEtc.textContent = extraValues.length > 0 ? extraValues.join(", ") : "";
  54. }
  55.  
  56. colBase.addEventListener("change", updateLoopClass);
  57. colSm.addEventListener("change", updateLoopClass);
  58. colMd.addEventListener("change", updateLoopClass);
  59.  
  60. loopClass.addEventListener("input", updateSelectValues);
  61.  
  62. updateSelectValues();
  63. });
  64. </script>

a-blog cms 独自の実装ポイント

通常の管理画面のモジュールID の設定でテストを行っていた際には

  1. document.addEventListener("DOMContentLoaded", function () {
  2. ...
  3. });

のように記述して動作していましたが、モジュールIDの管理画面は表の表示画面の「編集」リンクからモーダル表示が可能です。その際には既に DOMコンテンツがロード済みという事になり、ここで書かれた JavaScript を動作させることができません。

そこで a-blog cms の方で用意されている ACMS.addListener("acmsReady" ... を利用します。詳しくは「組み込みJSのイベントハンドラ」をご覧ください。

  1. ACMS.addListener("acmsReady", function() {
  2. ...
  3. });

追加のカスタマイズを考える

今回の JavaScript には input[name="entry_summary_loop_class" と Entry_Summary のコンフィグのフィールドが書かれていますが、他のモジュールでも使う事を考慮するような実装も考えられます。

このカスタム設定の SELECT 部分 + JavaScript のファイルを別ファイルにし target_input を変数化します。

  1. @include("/admin/parts/loop-class.html", {"target_input": "entry_summary_loop_class"})

/admin/parts/loop-class.html の中の JavaScript 部分を

  1. const loopClass = document.querySelector('input[name="{{target_input}}"]');

のように指定する事で、他のモジュールでも同様に扱いたいことがあれば共用して利用することができるようにできます。

最後に

次のバージョン Ver. 3.2 で標準提供するテーマには {entry:loop.class} を活用したテーマにするようにしようと考えています。

また private/config.system.default.yaml

  1. entry_summary_loop_class : acms-col-6 acms-col-sm-4

のような設定を追加を検討します。どのような初期値がいいかは検討しないといけませんが、こうする事で標準テーマで表示設定にある項目が機能する事になります。

動作サンプルを掲載しておきます。


この記事は、developer.a-blogcms.jp にて正式に公開しています。

本ブログでは、検索エンジンが正規の公開先を認識できるよう Canonical設定 を適用し、重複コンテンツとしてインデックスされないよう、noindex設定 を行っています。

https://developer.a-blogcms.jp/blog/custom/summary-entry-loop-class.html

関連記事

この記事のハッシュタグ から関連する記事を表示しています。
エントリー編集画面の UI について見直しを考えてみる

エントリー編集画面の UI について見直しを考えてみる

a-blog cms バージョン管理 UI について見直しを考えてみる

バージョン管理 UI について見直しを考えてみる

編集者権限でバナーモジュールをフロント側で編集できるような UI を使えるような実装を考える

a-blog cms の TimePicker をアナログ時計UI にする方法

a-blog cms 2.10 エントリーのタグ入力画面を元に戻す方法

a-blog cms のベンチマークモードの活用法