WordPress

提供:GNU social JP Wiki
2024年2月28日 (水) 16:24時点におけるGnusocialjp (トーク | 投稿記録)による版 (WPCode)

About

WordPressは世界の多くのウェブサイトで使用されているCMSだ。PHPで記述されている。

WordPressの外観や挙動はPHPのプログラムを書いてカスタマイズできる。また、プラグイン機構を持ち、世界中の開発者によるプラグインを導入することで簡単にウェブサイトを拡張できる。

Install

WordPress.comなどのホスティングサービスを使用することもできるが、レンタルサーバーやVPSに設置することが好ましい。

ActivityPub

WordPressはActivityPubを実装している。WordPressへの投稿時に分散SNSにも投稿される、分散SNSからWordPressの投稿にコメントできるなどの機能がある。

WordPressの公式ホスティングサービスであるWordPress.comもActivityPubに対応している。分散SNSの中でも巨大なサーバーと言え、無視できない存在だ。

Function

Comment

Ref:

このあたりの関数で定義されている。フィルターを挟むことで変更できる模様。

PV

WordPressでPV数を表示させる方法がいくつかある。

WP-PostViewsプラグインのほうが柔軟性がある? (WordPressで記事ランキングを設置するならWP-PostViewsが最適なのでは? | kanae-design-works「WP-PostViews」と「WordPress Popular Posts」の違い | クロジカ)。

WordPressの分析プラグインはDBにアクセ格納するから重くなるらしい

Jetpackの統計情報から閲覧数を取得表示することができる。これが一番手っ取り早い気がする。

<p>
<?php // 全体、月別、週別、日別のPV表示
  global $post;
  $views_all = 0;
  $views_monthly = 0;
  $views_weekly = 0;
  $views_daily = 0;
  //Jetpack利用時
  if (is_jetpack_stats_module_active()) {
    $jetpack_views = stats_get_csv( 'postviews', array('days' => -1, 'limit' => 1, 'post_id' => $post->ID ));
    if (isset($jetpack_views[0]['views'])) {
      $views_all = $jetpack_views[0]['views'];
    }
    $jetpack_views = stats_get_csv( 'postviews', array('days' => 30, 'limit' => 1, 'post_id' => $post->ID ));
    if (isset($jetpack_views[0]['views'])) {
      $views_monthly = $jetpack_views[0]['views'];
    }
    $jetpack_views = stats_get_csv( 'postviews', array('days' => 7, 'limit' => 1, 'post_id' => $post->ID ));
    if (isset($jetpack_views[0]['views'])) {
      $views_weekly = $jetpack_views[0]['views'];
    }
    $jetpack_views = stats_get_csv( 'postviews', array('days' => 1, 'limit' => 1, 'post_id' => $post->ID ));
    if (isset($jetpack_views[0]['views'])) {
      $views_daily = $jetpack_views[0]['views'];
    }
  }
  echo '<span class="daily">日:<span class="pv-count">', $views_daily, '</span></span>';
  echo '<span class="weekly">週:<span class="pv-count">', $views_weekly, '</span></span>';
  echo '<span class="monthly">月:<span class="pv-count">', $views_monthly, '</span></span>';
  echo '<span class="all">全体:<span class="pv-count">', $views_all, '</span></span>';
?>
</p>

上記コードをウィジェットに貼り付けたらうまくいった。

Language

WordPressで多言語対応サイトも構築できる。

情報源:

いろいろ方式があるが、サブディレクトリー形式が手軽な感じがする。

英語で情報発信できると、かなりの数のユーザーにアプローチできる。単に機械翻訳にするだけでも検討する価値はあるだろう。

広告ブロッカー対策

出典: 【WordPress編】広告ブロッカーからサイトコンテンツと収益を守る! | SEの良心

[CHP Ads Block Detector] というプラグインで実現できるらしい。

プラグイン: WordPressの広告ブロッカー対策CHP Ads Block Detector | GNU social JP Web」の通り、簡単にできた。

Widget内でのPHP実行

WordPressのウィジェットでPHPコードを実行する方法 – FOXWP

function widget_text_exec_php( $widget_text ) {
    if( strpos( $widget_text, '<' . '?' ) !== false ) {
        ob_start();
        eval( '?>' . $widget_text );
        $widget_text = ob_get_contents();
        ob_end_clean();
    }
    return $widget_text;
}
add_filter( 'widget_text', 'widget_text_exec_php', 99 );

上記コードをfunction.phpに配置すると、テキストウィジェット内でPHPコード<?php ?>を実行できる模様。

Linkage

WordPressの記事・投稿を他のSNSに自動投稿する方法がいくつかある。

There has been a critical error on this website. Please check your site admin email inbox for instructions.

プラグインを無効にしようと

How to deactivate all plugins when not able to access the administrative menus?

shortcode

Ref:

WordPressにはショートコードというのがある。角括弧 [shortcode][/shortcode] のような形式。本来ならHTML/JavaScriptしか使えない投稿本文に、PHP側の処理を仕込める。

新規作成する場合、書式がだいたい決まっている。

    function callback_function($atts, $content = null, $tag ) {
      extract(shortcode_atts(array(
        //属性
        'パラメータ1' => 'デフォルトの値1',
        'パラメータ2' => 'デフォルトの値2',
        ・・・
      ), $atts));
      //ショートコードの処理(動作)の記述
      return $value;
    }
    add_shortcode('my_shortcode', 'callback_function');

こえrでmy_shortcodeというショートコードが使用可能になる。

Reference

Ref: Reference | Developer.WordPress.org.

Functions

shortcode_atts
shortcode_atts( array $pairs, array $atts, string $shortcode = ” ): array

ユーザーが指定した属性と、使用可能な属性のデフォルト値を結合する。使用不能な属性を無視してくれる。

  • $pairs: 使用可能な属性とそのデフォルト値。
  • $atts: ショートコードでユーザーが指定した属性。
  • $shortcode: ショートコードの名前。これを指定するとshortcode_atts_$shortcode フィルターで外部から属性をフィルターできる。通常は使わない。

WPCode

Shortcodeの定義はくせがある。

[Insert Method]=[Shortcode] を選ぶと [wpcode id="10405"] のようなショートコードが定義される。これを [wpcode id="10405"][/wpcode]のようなブロックで使う。

Code Previewには、関数の本文を指定して、表示する場合echoなどで出力させる。

無料版だとshortcodeの名前を指定できない。

Auto Insertで自分で定義したらどうだろう?それで問題ない。

About

PMPro。有料会員サービスを実現するプラグイン。

ソースコード内のincludes配下の関数は自分でも使用可能。

Free

PMProは基本的に無料で使える模様 (PMPro: Free WordPress Membership Plugin)。サポートを販売している扱いの模様。ログアウトしていないと表示されない。

Stranger Studios」がGitHubのリポジトリー。ここに全部のアドオンがある。ここからzipファイル類をダウンロードすれば使える。ただし、一部のプラグインは開発者の経験が必要なため、ライセンスキーの購入をsupportとして販売している。開発経験が一切ない人にはGitHubからのインストールもけっこう面倒くさいからそれでうまくGPLの条件を満たしているのだと思われる。

実行自体はライセンスの有無によらず可能。ただし、ライセンスの有無でStripeの決済手数料が変わる。

Paid Memberships Pro (@paidmembershipspro) – WordPress user profile | WordPress.org」でも公開しているが、これは全部ではない。サードパーティープラグインとの連携用が大半。

金額変更

定期購読で販売していて、登録後に金額を変更したいことがある。

これは基本的にできない。定期購読は決済サービス側で登録時の金額で固定されるため。

金額を変更したければ、一度キャンセルして、再度チェックアウト・購入してもらうしかない。

既存会員に対しては、更新ではなく、期限を設けて強制的に一度退会させるのがいい。

Restriction

membership short code

Ref: Hide bottom register form when it has membership code block.

Source: paid-memberships-pro/shortcodes/membership.php at dev · strangerstudios/paid-memberships-pro

PMProの基本的なコンテンツ制限の概念がある。

基本は投稿全体を非表示で、excerptで冒頭部分を一部表示これが基本。membershipショートコードはこの逆で、基本全表示で一部だけ非表示。

そもそも仕組みが違う。

だから、カテゴリーで管理していて、一部だけ非表示にしたかったら、末尾にmoreブロックを配置して、membeshipショートコードも使う必要がある。二重に使わないといけない。けっこう面倒くさい。

Custom Message for Non-Members on Protected Content」がメッセージの変更方法。ただ、制限のロジックを変更できるわけではない。別のフィルターだと思う。

ソースコードを参考に自分で実装するしかないだろう。

message

Custom Message for Non-Members on Protected Content」に基本的な表示内容やスタイルの設定方法がある。

<div class="pmpro_content_message">
//custom message here
</div>

表示部分は上記要素になるのでこのクラスでスタイルをあてる。

.pmpro_content_message {
	background: #F1F1F1;
	margin: 0 auto;
	max-width: 700px;
	padding: 15px;
}

他に以下のフィルターが該当するメッセージ表示。

  • add_filter( 'pmpro_non_member_text_filter', 'my_pmpro_non_member_text_filter', 5 );
  • add_filter( 'pmpro_not_logged_in_text_filter', 'my_pmpro_not_logged_in_text_filter', 5 );

例えば、会員限定の文字数表示などをしたければ、自前でやるしかない。基本は自分で上記フィルターで上書きするのがよい。

function my_pmpro_non_member_text_filter( $text ) {
	global $post, $post_membership_levels_names;
	$count = mb_strlen( $post->post_content, 'UTF-8' ) - mb_strlen( get_the_excerpt(), 'UTF-8' );
	$access = pmpro_has_membership_access( $post->ID, null, true );
	$text = "<p>残り{$count}文字。続きは" . implode( '/', $access[2] ) . '会員限定。</p><p>Free=0/Bronze=220/Silver=1100/Gold=2200円。</p>'
		. '<p><a href="https://web.gnusocial.jp/member/level/">会員登録</a>/'
		. '<a href="//web.gnusocial.jp/member/login/">ログイン</a> (<a href="//web.gnusocial.jp/about/member/">About Member</a>)。</p>';
	return $text;
}
add_filter( 'pmpro_non_member_text_filter', 'my_pmpro_non_member_text_filter', 5 );
add_filter( 'pmpro_not_logged_in_text_filter', 'my_pmpro_non_member_text_filter', 5 );

Style

Custom Message for Non-Members on Protected Content」でペイウォールの表示内容・スタイルを制御できる。

ペイウォール (.pmpro_content_message) のスタイルは会員登録の誘発に重要。

有名サービスの例は上記。登録・購入ボタンを大きく押しやすくしている。a要素の他button要素でも実装されている。ここはいろいろ考えてもいいかもしれない。

Category

membershipショートコードやAddon Packagesではカテゴリーやタグでの会員限定に対応していない。これがちょっと困る。既存のカテゴリー、タグでの会員対応の把握方法を確認する。

includes/contents.phpのpmpro_has_membership_accessのコードが参考になる。

	if(isset($mypost->post_type) && $mypost->post_type == "post")
	{
		// Get the categories for this post.
		$post_terms = wp_get_post_categories( $mypost->ID );

		// Get the tags for this post.
		$post_terms = array_merge( $post_terms, wp_get_post_tags( $mypost->ID, array('fields' => 'ids' ) ) );

		if( ! $post_terms )
		{
			//just check for entries in the memberships_pages table
			$sqlQuery = "SELECT m.id, m.name FROM $wpdb->pmpro_memberships_pages mp LEFT JOIN $wpdb->pmpro_membership_levels m ON mp.membership_id = m.id WHERE mp.page_id = '" . esc_sql( $mypost->ID ) . "'";
		}
		else
		{
			//are any of the post categories associated with membership levels? also check the memberships_pages table
			$sqlQuery = "(SELECT m.id, m.name FROM $wpdb->pmpro_memberships_categories mc LEFT JOIN $wpdb->pmpro_membership_levels m ON mc.membership_id = m.id WHERE mc.category_id IN(" . implode(",", array_map( 'intval', $post_terms ) ) . ") AND m.id IS NOT NULL) UNION (SELECT m.id, m.name FROM $wpdb->pmpro_memberships_pages mp LEFT JOIN $wpdb->pmpro_membership_levels m ON mp.membership_id = m.id WHERE mp.page_id = '" . esc_sql( $mypost->ID ) . "')";
		}
	}
	else
	{
		//are any membership levels associated with this page?
		$sqlQuery = "SELECT m.id, m.name FROM $wpdb->pmpro_memberships_pages mp LEFT JOIN $wpdb->pmpro_membership_levels m ON mp.membership_id = m.id WHERE mp.page_id = '" . esc_sql( $mypost->ID ) . "'";
	}

    $post_membership_levels = $wpdb->get_results($sqlQuery);

普通にSQLで取得している。このへんのコードを流用すれば対応できる気はする。ただ、ちょっと面倒くさい。

Analytics

PMProの流入元の記事を分析・特定する方法がある。

GA4を使った一般的なサイト分析になる。有料プラグインでもできる。

Sell

Single

WooCommerceと連携したり、PMProの有料プラグインを使うとか。他にも方法を以前調査したの後日追記する。

理想としては、単品販売と、会員を両立したい。2個以上買うなら会員のほうがいいみたいな。

単品販売の方法。有料アドオン、LearnDash、後はWooCommerce連携など。

Addon Packagesがいい。PMProのAddon Packagesは、まずあるメンバーシップレベルのユーザーに購入権利を与えて、そこから購入する形。

あくまで単品販売で、定期販売はできない。記事別に金額の入力欄が表示されてそれで単品販売する形。悪くない。こちらの理想に近い。単品販売することにした記事は買うしかない。あるメンバーレベルなら買わなくても見れるというようにはできなさそう。いや。1レベルだけ全アクセス可能はできる。

Course

LearnCashはちょっといまいち。外部のプラグインとの連携。この機能自体はPMProのあるレベルが、LernDashのあるコースにアクセス可能にするだけだから、あまり意味ない。PMProだけでも似たようなことはできる。

Addon Packages

About

記事の単品販売用にAddon Packagesがあるのでこれを試す。このプラグインは将来的にPMPro本体に取り込まれるもので、一時的な解決用という位置づけ。

Usage

インストールしたらpost/page画面に金額設定画面が表示されるので、ここで金額をいれて、[Require Membership] も指定すると単品販売できる。

カテゴリーでのレベル指定は無効。一番レベル低いFreeなどを1個指定すれば、自動的に全部に適用される。レベルの指定が若干使いにくい。カテゴリーで指定できれば楽だった。やむを得ない。PMPro 3で取り込まれたりしないかな。

しないでしょうね。コンテンツの部分非表示と同じで、ページ単位で会員レベルを指定する必要があるのが作りとしていまいち。だがこれに従うしかないか。基本は変えない運用にすればいいのか。

Configuration
  • define( 'PMPROAP_EXP_DAYS', 0 );: 有効期限の設定デフォルト0 (無期限)。基本はデフォルトでOK。
  • apply_filters( 'pmproap_all_access_levels', array(), $user_id, $post_id );: 単品購入を無視してアクセスできる神レベル的なものを指定できる。
  • apply_filters( 'pmproap_text_level_id', $text_level_id, $post_id, $user_id, $post_levels );: 未ログインユーザー向けの新規購入時に必要なレベルID。
  • apply_filters( 'pmproap_supported_post_types', array( 'page', 'post' ) );
  • do_action( 'pmproap_action_add_to_package', $user_id, $post_id );
  • do_action( 'pmproap_action_remove_from_package', $user_id, $post_id );

pmproap_all_access_levels を使うと、単品購入を無視してアクセスできる神レベル的なものを指定できる。

例えば、最上位だけ除外するとか。金額でレベル指定できればなおよかったがしかたない。

/*
	Set levels as "all access levels" so members of these levels will be able to view all Addon Packages.
	Requires Paid Memberships Pro and the pmpro-addon-packages plugin.
*/
function my_pmproap_all_access_levels($levels, $user_id, $post_id)
{
	//I'm just adding the level, but I could do some calculation based on the user and post id to programatically give access to content
	$levels = array(16);	
	return $levels;
}
add_filter("pmproap_all_access_levels", "my_pmproap_all_access_levels", 10, 3);
Message

pmpro_non_member_text_filter/pmpro_not_logged_in_text_filterをpmproap_pmpro_text_filterで上書きしている。この実装を参考に、自分で元のフィルターを上書きすれば、表示文を変更できる。

<div class="pmpro_content_message"><p>This content requires that you purchase additional access. The price is ¥220 or free for our Gold members.</p><p><a href="https://web.gnusocial.jp/member/checkout/?level=1&amp;ap=10307">Purchase this Content (¥220)</a> <a href="https://web.gnusocial.jp/member/level/">Choose a Membership Level</a></p></div>

Addon Package有無の両方の考慮が必要でちょっと複雑になる。

Shortcode

[pmpro_addon_packages] のショートコードを使うと、単品販売の記事一覧を出力できる。