はじめに
相互フォロワー限定で記事を投稿する機能を追加しようと思ったのですが、私の調べ方が足りなかったのかなかなか似たような機能の記事が見当たらず時間がかかったので記録します。 初心者故拙いところも多々ありますので何か間違っているところがあればご指摘いただけると嬉しいです。前提
・フォロー機能の実装 こちらの記事を参考にさせていただきました。 フォロー機能にはacts_as_followerというgemを必ず使用してください。 後述のfollowed_by?メソッドはこのgemによって定義されていますので、その他のフォロー機能の記事を参考にした方はacts_as_followerのfollowed_by?部分がどう定義しているか確認して各々で定義してあげてください (※私はそれが結局わからず別のフォロー機能記事を参考にしていましたが、こちらのgemに乗り換えました。)・マイページ機能の実装
・公開非公開機能の実装
こちらの記事を参考にさせていただきました。
当記事では公開設定をプルダウンを利用して進めていきます。
・投稿機能の実装
当記事ではpostモデルを使用しております。モデル名は各自で変更してください。
実装
・投稿画面<%= f.select :privacy, [["公開", 1], ["非公開", 2], ["相互限定", 3]] %>
プルダウン部分に「相互限定」の公開範囲を追加します。
編集機能がある人はそこにも同じように追加します。
・タイムライン
今回はタブ機能を用いて
・投稿一覧
・フォローしているユーザーと自分の投稿一覧
を分けて表示できるように実装していきます。
タブ機能はこちらの記事の「ボタンが幅に合わせて伸縮するデザイン」を使用させていただきました。
<% if user_signed_in? %>
<div class="tab-wrap">
<input id="TAB-01" type="radio" name="TAB" class="tab-switch" checked="checked" /><label class="tab-label" for="TAB-01">Posts all</label>
<div class="tab-content">
<% @posts.each do |t| %>
<% if t.privacy=="1" %>
投稿
<% end %>
<% end %>
</div>
<input id="TAB-02" type="radio" name="TAB" class="tab-switch" /><label class="tab-label" for="TAB-02">Follow</label>
<div class="tab-content">
<% @posts.each do |t| %>
<% if (t.user.followed_by? current_user) || current_user == t.user %>
<% if t.privacy=="3"&&(current_user.followed_by? t.user) && (t.user.followed_by? current_user) || t.privacy=="1" %>
投稿
<% end %>
<% end %>
<% end %>
</div>
</div>
<% else %>
<% @posts.each do |t| %>
<% if t.privacy=="1" %>
投稿
<% end %>
<% end %>
<% end %>
※投稿部分は各自で表示させたいものを入れてください
解説
if文ではユーザーのログイン状態によって分岐しています。
・ユーザーがログインしているとき、Tab1ではすべての公開記事を表示、Tab2ではフォローしているユーザーと自分の投稿のみを表示
(相互フォローしているユーザーがいる場合は限定公開の記事も表示する)
・ユーザーがログインしていないとき、すべての公開記事を表示
<% if (t.user.followed_by? current_user) || current_user == t.user %>
この部分でcurent_userがフォローしている投稿、またはcurrent_userの投稿を表示するように指示しています。
・マイページ
<% if current_user.id == @user.id %>
<div class="tab-wrap">
<input id="TAB-01" type="radio" name="TAB" class="tab-switch" checked="checked" /><label class="tab-label" for="TAB-01">Release</label>
<div class="tab-content">
<% @user.posts.each do |t| %>
<% if t.privacy=="1" %>
投稿
<% end %>
<% end %>
</div>
<input id="TAB-02" type="radio" name="TAB" class="tab-switch" /><label class="tab-label" for="TAB-02">Follow</label>
<div class="tab-content">
<% @user.posts.each do |t| %>
<% if t.privacy=="3"&&t.user.id==current_user.id %>
投稿
<% end %>
<% end %>
</div>
<input id="TAB-03" type="radio" name="TAB" class="tab-switch" /><label class="tab-label" for="TAB-03">Private</label>
<div class="tab-content">
<% @user.posts.each do |t| %>
<% if t.privacy=="2"&&t.user.id==current_user.id %>
投稿
<% end %>
<% end %>
</div>
</div>
<% elsif (current_user.followed_by? @user) && (@user.followed_by? current_user) %>
<div class="tab-wrap">
<input id="TAB-01" type="radio" name="TAB" class="tab-switch" checked="checked" /><label class="tab-label" for="TAB-01">Release</label>
<div class="tab-content">
<% @user.posts.each do |t| %>
<% if t.privacy=="1" %>
投稿
<% end %>
<% end %>
</div>
<input id="TAB-02" type="radio" name="TAB" class="tab-switch" /><label class="tab-label" for="TAB-02">Follow</label>
<div class="tab-content">
<% @user.posts.each do |t| %>
<% if t.privacy=="3" %>
投稿
<% end %>
<% end %>
</div>
<input id="TAB-03" type="radio" name="TAB" class="tab-switch" /><label class="tab-label" for="TAB-03">Likes</label>
<div class="tab-content">
<% @user.liked_posts.each do |t| %>
<% if t.privacy=="1" || t.privacy=="3"&& (current_user.followed_by? @user) && (@user.followed_by? current_user) %>
投稿
<% end %>
<% end %>
</div>
</div>
<% else %>
<div class="tab-wrap">
<input id="TAB-01" type="radio" name="TAB" class="tab-switch" checked="checked" /><label class="tab-label" for="TAB-01">Release</label>
<div class="tab-content">
<% @user.posts.each do |t| %>
<% if t.privacy=="1" %>
投稿
<% end %>
<% end %>
</div>
<input id="TAB-02" type="radio" name="TAB" class="tab-switch" /><label class="tab-label" for="TAB-02">Likes</label>
<div class="tab-content">
<% @user.liked_posts.each do |t| %>
<% if t.privacy=="1" %>
投稿
<% end %>
<% end %>
</div>
</div>
<% end %>
※いいね機能を実装していない場合は適宜該当部分を外してください。
解説
・current_userが表示されているユーザーページ本人であるか
・表示されているユーザーページ本人とcurrent_userが相互フォロー状態か
・それ以外か
で分岐させます。
ここでは、公開範囲が全体公開の記事、もしくは公開範囲が限定公開かつcurrent_userが記事の投稿主と相互フォロー状態であるときを指定します。
そうすることで、いいねしている投稿が限定公開の記事であっても、相互フォローの人以外はユーザーページのいいね一覧から見られなくなります。
終わりに
なかなか似たような機能を探すのに手間取って初心者には実装にかなり時間がかかりましたが、たくさんの記事や助けてくださる方に恵まれてなんとか実装することができました。 私のような完全初心者で似たような機能を実装しようと考えている人のために、そしてなにより未来の自分のためにも備忘録として記録しておきます。参考記事
https://qiita.com/bindingpry/items/e15f38fb2ca298012684