首頁 > 軟體

Typecho外掛實現新增文章目錄的方法詳解

2023-02-22 06:00:29

我的長博文不少,比較影響閱讀體驗,有必要新增一個文章目錄功能。相比 Wordpress, Typecho 的外掛就比較少了。我想找一個像掘金那樣為文章新增目錄的外掛,沒一個合適的。此類教學也不是很多,而且差不多都是前臺 JavaScript 來實現的,感覺這樣不如後臺實現來的好。

注意:我使用的是Joe主題7.3,其他主題檔案路徑可能不一樣。

新增文章標題錨點

1.宣告 createAnchor 函數

core/functions.php 中新增如下程式碼:

// 新增文章標題錨點
function createAnchor($obj) {
  global $catalog;
  global $catalog_count;
  $catalog = array();
  $catalog_count = 0;
  $obj = preg_replace_callback('/<h([1-4])(.*?)>(.*?)</h1>/i', function($obj) {
    global $catalog;
    global $catalog_count;
    $catalog_count ++;
    $catalog[] = array('text' => trim(strip_tags($obj[3])), 'depth' => $obj[1], 'count' => $catalog_count);
    return '<h'.$obj[1].$obj[2].' id="cl-'.$catalog_count.'">'.$obj[3].'</h'.$obj[1].'>';
  }, $obj);
  return $obj;
}

也可以在標題元素內新增 <a> 標籤,然後該標籤新增 id 屬性。

createAnchor 函數主要是通過正規表示式替換文章標題H1~H4來新增錨點,接下來我們需要呼叫它。

2.呼叫函數

同樣在 core/core.php 中的 themeInit 方法最後一行之前新增如下程式碼:

if ($self->is('single')) {
  $self->content = createAnchor($self->content);
}

現在可以檢視一下文章詳情頁面的原始碼。文章的 H1~H4 元素應該新增了諸如 cl-1cl-2 之類的 id 屬性值。具體啥名不是關鍵,好記就行。

顯示文章目錄

1.宣告 getCatalog 函數

core/functions.php 中新增如下程式碼:

// 顯示文章目錄
function getCatalog() {  
  global $catalog;
  $str = '';
  if ($catalog) {
    $str = '<ul class="list">'."n";
    $prev_depth = '';
    $to_depth = 0;
    foreach($catalog as $catalog_item) {
      $catalog_depth = $catalog_item['depth'];
      if ($prev_depth) {
        if ($catalog_depth == $prev_depth) {
          $str .= '</li>'."n";
        } elseif ($catalog_depth > $prev_depth) {
          $to_depth++;
          $str .= '<ul class="sub-list">'."n";
        } else {
          $to_depth2 = ($to_depth > ($prev_depth - $catalog_depth)) ? ($prev_depth - $catalog_depth) : $to_depth;
          if ($to_depth2) {
            for ($i=0; $i<$to_depth2; $i++) {
              $str .= '</li>'."n".'</ul>'."n";
              $to_depth--;
            }
          }
          $str .= '</li>';
        }
      }
      $str .= '<li class="item"><a class="link" href="#cl-'.$catalog_item['count'].'" rel="external nofollow"  title="'.$catalog_item['text'].'">'.$catalog_item['text'].'</a>';
      $prev_depth = $catalog_item['depth'];
    }
    for ($i=0; $i<=$to_depth; $i++) {
      $str .= '</li>'."n".'</ul>'."n"; 
    }
    $str = '<section class="toc">'."n".'<div class="title">文章目錄</div>'."n".$str.'</section>'."n";
  }
  echo $str;
}

getCatalog 方法通過遞迴 $catalog 陣列生成文章目錄,接下來我們需要呼叫它。

2.函數

最好將放在右側邊欄中。為此在 public/aside.php 中新增如下程式碼:

<?php if ($this->is('post')) getCatalog(); ?>

注意:只有文章才使用目錄,獨立頁面那些不需要,所以加了判斷。Typecho 有一些神奇的 is 語法可以方便二次開發,可以存取它的官網檔案瞭解更多。

現在點選右側的文章目錄,可以捲動到相應的文章小標題位置了。

新增文章目錄樣式

可以看到,當前的文章目錄還比較醜陋,我們來美化一下。在 assets/css/joe.post.min.scss 中新增如下 SCSS 程式碼:

.joe_aside {
  .toc {
    position: sticky;
    top: 20px;
    width: 250px;
    background: var(--background);
    border-radius: var(--radius-wrap);
    box-shadow: var(--box-shadow);
    overflow: hidden;

    .title {
      display: block;
      border-bottom: 1px solid var(--classA);
      font-size: 16px;
      font-weight: 500;
      height: 45px;
      line-height: 45px;
      text-align: center;
      color: var(--theme);
    }

    .list {
      padding-top: 10px;
      padding-bottom: 10px;
      max-height: calc(100vh - 80px);
      overflow: auto;

      .link {
        display: block;
        padding: 8px 16px;
        border-left: 4px solid transparent;
        color: var(--main);
        text-decoration: none;
        white-space: nowrap;
        overflow: hidden;
        text-overflow: ellipsis;

        &:hover {
          background-color: var(--classC);
        }

        &.active {
          border-left-color: var(--theme);
        }
      }
    }
  }
}

為了方便操作,將 .toc 設定成 position: sticky; 實現了吸頂定位。考慮到文章目錄可能很多,為 .toc 列表新增了 overflow: auto;,如程式碼第 3 ~ 4 行。

由於 .joe_header(主題檔頭)也使用了吸頂定位,導致和文章目錄有遮擋,所有加了 has_toc .joe_header 來取消頁面主題檔頭的吸頂功能,如下程式碼:

.has_toc {
  .joe_header {
    position: relative;
  }
}

定位到文章

要顯示文章目錄當前選中項的狀態,需要用到 JavaScript 給選中項新增一個 active 樣式。在 assets/js/joe.post_page.js 中新增如下程式碼:

var headings = $('.joe_detail__article').find('h1, h2, h3, h4');
var links = $('.toc .link');
var tocList = document.querySelector('.tocr > .list');
var itemHeight = $('.toc .item').height();
var distance = tocList.scrollHeight - tocList.clientHeight;
var timer = 0;
// 是否自動捲動
var autoScrolling = true;

function setItemActive(id) {
  links.removeClass('active');
  var link = links.filter("[href='#" + id + "']")
  link.addClass('active');
}

function onChange() {
  autoScrolling = true;
  if (location.hash) {
    id = location.hash.substr(1);
    var heading = headings.filter("[id='" + id + "']");
    var top = heading.offset().top - 15;
    window.scrollTo({ top: top })
    setItemActive(id)
  }
}
window.addEventListener('hashchange', onChange);
// hash沒有改變時手動呼叫一次
onChange();

由於佈局和捲動動畫的影響,導致錨點定位有點偏差。我們再 setItemActive 函數中用 scrollToscrollIntoView 來糾正。另外,我們希望有錨點的連結可以直接定位,因此監聽了 hashchange 事件。點選文章目錄測試一下定位,再手動鍵入錨點測試一下,應該都沒啥問題。

定位到目錄

目前可以從文章目錄定位到文章標題了,是單向定位,雙向定位還需要實現捲動文章內容時定位到文章目錄的當前項。正如我們馬上能想到的,需要監聽 windowscroll 事件,如下程式碼:

function onScroll() {
  if (timer) {
    clearTimeout(timer);
  }
  timer = setTimeout(function () {
    var top = $(window).scrollTop();
    var count = headings.length;
    for (var i = 0; i < count; i++) {
      var j = i;
      // 捲動和點選時 index 相差 1,需要 autoScrolling 來區分
      if (i > 0 && !autoScrolling) {
        j = i - 1;
      }
      var headingTop = $(headings[i]).offset().top;
      var listTop = distance * i / count
      // 判斷卷軸捲動距離是否大於當前捲動項可捲動距離
      if (headingTop > top) {
        var id = $(headings[j]).attr('id');
        setItemActive(id);
        // 如果目錄列表有滑條,使被選中的下一元素可見
        if (listTop > 0) {
          // 向上捲動
          if (listTop < itemHeight) {
            listTop -= itemHeight;
          } else {
            listTop += itemHeight;
          }
          $(tocList).scrollTop(listTop)
        }
        break;
      } else if (i === count - 1) {
        // 特殊處理最後一個元素
        var id = $(headings[i]).attr('id');
        setItemActive(id);
        if (listTop > 0) {
          $(tocList).scrollTop(distance)
        }
      }
    }
    autoScrolling = false;
  }, 100);
}

$(window).on('scroll', onScroll);

首先,在 onScroll 事件處理常式中遍歷標題陣列 headings, 如果卷軸捲動距離 top 大於當前標題項 item 可捲動距離 headingTop,再呼叫 setItemActive 函數,傳入當前的標題項的 id 來判斷文章目錄啟用狀態。

如果目錄列表有滑條,呼叫 jQuery 的 scrollTop 方法捲動目錄列表滑條,使被選中目錄項的上下元素可見,

現在文章目錄基本上可用了,也還美觀,後續可以考慮優化再封裝成一個外掛。

吐槽一下:Joe 主題太依賴jQuery了,修改起來費勁 ::(汗)。

到此這篇關於Typecho外掛實現新增文章目錄的方法詳解的文章就介紹到這了,更多相關Typecho新增文章目錄內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com