0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

【自動化】Yahooトップ画面で不快なニュース配信元の記事を消すChrome拡張

Last updated at Posted at 2025-02-20

こたつ記事とか釣りタイトルの記事で稼ぐしょうもないビジネスしている人はここでお引き取りください。
くれぐれもスクロールしないでください。くれぐれも。m(__)m


「釣りタイトル」で煽る記事、不快ですよね。
インフルエンサーのSNSでの発言を拾っているだけの「こたつ記事」、嫌いですよね。
大谷翔平は好きでも、誇大表現を使って、中身がぺらっぺらっの配信元うんざりしますよね?

こういった記事・配信元をYahooのTOPページから自動で非表示にするChrome拡張です。

非表示にしたい記事配信元(blockProviders)とキーワード(blockKeywords)に任意の値を設定するだけです。

個人レベルで精神衛生上もよいのはもちろん、社会的意義もありそうです。
社会的意義とは具体的に、低品質なメディアのトラフィックを減らすことで淘汰がすすんで、劣化する一方のWeb体験やネットメディアの質の向上が期待できます。

さらに、なんちゃって学者タレントやインフルエンサーがよくやる「炎上商法」(悪名は無名に勝る、アテンション経済)が経済的に報われない(むしろ負の効果)になれば、極論・暴論・汚い言葉が隅っこに追いやられて、感情的にならずに建設的な議論が交わされるようになることも期待できます。

手順

  • 後述するmanifest.jsoncontent.js をフォルダに保存(例: yahoo_news_filter/)
    Image from Gyazo
  • Google Chrome の chrome://extensions/ にアクセス
    Image from Gyazo
  • デベロッパーモード をオンにする。
    Image from Gyazo
  • パッケージ化されていない拡張機能を読み込む から yahoo_news_filter/ フォルダを選択
    Image from Gyazo

正常に読み込まれるとこうなります。
Image from Gyazo

これだけで、 Yahoo! JAPANのトップページを開くと、指定したワードを含む記事が非表示になります。

コード

manifest.json
{
  "manifest_version": 3,
  "name": "Yahoo News Filter",
  "version": "1.0",
  "description": "特定の単語を含むニュース記事を非表示にする",
  "permissions": ["scripting"],
  "host_permissions": ["https://www.yahoo.co.jp/"],
  "content_scripts": [
    {
      "matches": ["https://www.yahoo.co.jp/"],
      "js": ["content.js"],
      "run_at": "document_end"
    }
  ]
}
content.js
(() => {
    // フィルタリング対象のキーワード
    const blockKeywords = ["堀江", "成田"];
    const blockProviders = ["デイリー", "ABEMA", "プレジデント","PRESIDENT", "ゴールドオンライン", "FRIDAY"];

    // ブロック統計情報
    let providerBlockCounts = {};  // providerごとの件数
    let providerCounts = {};  // providerごとの件数
    let reasonBlockCounts = { "provider": 0, "title": 0 };  // 理由ごとの件数

    function hideArticles() {
        document.querySelectorAll("#Stream article").forEach(article => {
            const titleElement = article.querySelector("h1 span");
            const providerElement = article.querySelector("cite");

            const title = titleElement ? titleElement.innerText : "";
            const provider = providerElement ? providerElement.innerText : "";

            let blocked = false;
            let blockReason = "";

            // 条件1: providerName にブロック対象のキーワードが含まれる
            if (blockProviders.some(keyword => provider.includes(keyword))) {
                reasonBlockCounts["provider"]++;
                blockReason = "provider";
                blocked = true;
            }

            // 条件2: articleTitle にブロック対象のキーワードが含まれる
            if (blockKeywords.some(keyword => title.includes(keyword))) {
                reasonBlockCounts["title"]++;
                blockReason = "title";
                blocked = true;
            }

            // ブロックされた場合
            if (blocked) {
                article.style.display = "none";

                // provider ごとのブロック件数を更新
                if (provider in providerBlockCounts) {
                    providerBlockCounts[provider]++;
                } else {
                    providerBlockCounts[provider] = 1;
                }
            }
            // provider ごとの記事件数を更新
            if (provider in providerCounts) {
                providerCounts[provider]++;
            } else {
                providerCounts[provider] = 1;
            }
        });

        // 統計情報をログ出力
        console.log("=== ブロック統計情報 ===");
        console.log("■ Providerごとのブロック件数:", providerBlockCounts);
        console.log("■ ブロック理由ごとの件数:", reasonBlockCounts);
        console.log("配信元件数=", Object.keys(providerCounts).length)
        const sortedProviderCounts = Object.fromEntries(
            Object.entries(providerCounts).sort((a, b) => b[1] - a[1])
          );
        console.table(sortedProviderCounts);
    }
    
    // 初回実行
    hideArticles();

    // 定期的に監視(Ajaxで更新される可能性があるため)
    const observer = new MutationObserver(hideArticles);
    observer.observe(document.body, { childList: true, subtree: true });

})();

実行結果例(開発者ツールのコンソール)

Image from Gyazo

余計なあとがき

Chromeの公式にアップロードすると、人気が出た後に悲劇(売られてバックドアつくられたりする)につながるので、それぞれが自作して利用するのがよいのかなと思います。

このコードは、どの記事配信元を何件ブロックしたかもコンソールに表示します。
Yahooのアルゴリズムや契約(によって配信記事数が決定?)がちょっと見えて、なかなか興味深いですね。

あー、すっきり!!

ちなみに、開発者ツールのコンソールタブでログを見ているとわかりますが、Yahooは非同期で記事を読み込んでくるのですが、その動作にも対応(=非同期読み込み時に再実行)しています。

Image from Gyazo

おまけ

そーすこーd

以下の機能を追加したコードをGithubにあげました。
ご利用する場合は自己責任でお願いします。

Web黎明期のWebサイトってこんな感じでしょうか?

  • ヘッダー削除
  • サイドバー削除して記事部の幅を100%
    • 画像サイズを小さく(80x80→40x40)
      • これで一記事の高さが縮小されて表示される1ページあたりに表示される記事件数が倍くらいに
  • 「エンタメ」タブ削除

フィルター適用後のページ

Image from Gyazo

フィルター適用前のページ

Image from Gyazo

0
0
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
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?