前端学Ruby:全栈论坛(地宫)项目二

简介: 前端学Ruby:全栈论坛(地宫)项目二

各个模型建立

笔者是前端出身,对数据库的理解仅限于用 node + mysql (mongodb)做过微型博客。除此之外,数据库的知识点就无了,以下写的不好的,多多担待

文章模型与用户模型结合

文章模型与用户模型的结合,一个人必须要先登录后才能写文章,其次,一个人可以有很多文章,但当他注销后,文章就没了

先在 article model 中创建一个 user_id,将它指向 user model

rails g migration add_user_id_to_articles user_id:integer:index

app/models/article.rb 中加上:

class Article < ApplicationRecord
    belong_to :user
end

app/models/user.rb 中加上:

class User < ApplicationRecord
  # 意为一个人有很多文章,当人不在时,文章也就没了
  + has_many :articles, dependent: :destroy
end

这时,在文章详情页,可以通过 @article.user 来获取这篇文章对应的用户信息:

<h2><%= @article.title %></h2>
<p><%= @article.content %></p>
<p>Written by <%= @article.user.name %></p>

当然,如果你想获取一个用户所写的所有文章,则是在个人页,找到用户后,就能展示:

<% @user.articles.each do |article| %>
    <h2><%= article.title %></h2>
    <p><%= article.body %></p>
<% end %>

转换日期

将 create_at 转换为 ”March 28, 2023“ 这种格式

用 Ruby 的 strftime 方法

<%= @article.created_at.strftime("%B %d, %Y") %>
  • %B  表示月份的全名
  • %d 表示日期(两位数)
  • %Y 表示四位数的年份

建立评论model

建立 comment model

rails g model Comment body:text article:references user:references

迁移数据库

rails db:migrate

在建立 model 时,models/comment 就 belongs_to 文章和用户,即

class Comment < ApplicationRecord
  belongs_to :article
  belongs_to :user
end

所以我们需要在文章模型和用户模型中都加一下拥有多个评论

class User < ApplicationRecord
 ...
   has_many :articles, dependent: :destroy
  + has_many :comments, dependent: :destroy
end
class Article < ApplicationRecord
    belongs_to :user
    + has_many :comments, dependent: :destroy
end

Comment 模型和 Article 和 User 模型已经关联好了

现在我们创建 comment 控制器

rails g controller comment

rails 会帮忙生成controller、view、helper 等文件,这里我们只用到app/controllers/comments_controller,在应用中,我们的文章页面下会有评论,所以不单独做页面

我们前往config/routes.rb ,在 articles 下新增 resources :comments

resources :articles do
    + resources :comments
  end

这是符合 restful 风格的,如果严格一点,再加上 only: [:create, :destroy],只允许创建和删除,其他的接口不开放。回到最重要的 comments_controller 处,我们需要新增 create 和 destroy 方法,这里笔者尝试了一段时间不得解,还好借助 chatgpt 帮忙渡过,真乃神器

class CommentsController < ApplicationController
    before_action :authenticate_user!
    before_action :set_article!, only: %i[create destroy]
    def create
        @comment = @article.comments.create(comment_params)
        redirect_to article_path(@article)
    end
    def destroy
        @comment = @article.comments.find(params[:id])
        @comment.destroy
        redirect_to article_path(@article)
    end
    private
    def set_article!
        @article = Article.find(params[:article_id])
    end
    def comment_params
        params.require(:comment).permit(:body).merge(user: current_user)
    end
end

其中 @comment = @article.comments.create(comment_params) 这行代码很有趣,读起来像英文,在文章的 comment 中创建一个评论,其中 comment_params 中有 merge(user: current_user) 意为当前用户

Relationship 模型

一个用户可以关注别人,可以取关别人,别人也可以关注他,也可以去管他。用户之间的关注是多对多,笔者解释不了为什么再建一个表来关联两个用户,也许是性能,也许是结构,总之,笔者失败过,稚嫩的脸庞上多过一道泪痕

我们没必要创建 Relationship model 文件,直接创建迁移文件即可:

rails g migration CreateRelationship

修改迁移文件

class CreateRelationship < ActiveRecord::Migration[7.0]
  def change
    create_table :relationships do |t|
      t.integer :follower_id
      t.integer :following_id
      t.timestamps
    end
    change_column_null :relationships, :follower_id, true
    change_column_null :relationships, :following_id, true
    add_index :relationships, :follower_id
    add_index :relationships, :following_id
  end
end

迁移数据

rails db:migrate

因为关注是和用户有关,所以我们前往models/user 模型,加入 relationships 与 user 的关联

has_many :articles, dependent: :destroy
  has_many :comments, dependent: :destroy
 + has_and_belongs_to_many :following,
 + class_name: 'User',
 + join_table: 'relationships',
 + foreign_key: 'follower_id',
 + association_foreign_key: 'following_id'
 + has_and_belongs_to_many :followers,
 + class_name: 'User',
 + join_table: 'relationships',
 + foreign_key: 'following_id',
 + association_foreign_key: 'follower_id'

模型建好了,接着弄 config/routes,文档 上写的很清楚,他是在 profiles 路由下的动作,所以我们修改:

- get '/:name', to: 'profile#show', as: :profile
+  scope :profiles do
+    get ':username', to: 'profiles#show'
+    post ':username/follow', to: 'profiles#follow'
+    delete ':username/follow', to: 'profiles#unfollow'
+  end

前往视图层:

<% if current_user.following?(@article.user) %>
    <%= button_to unfollow_user_path(@article.user.username), method: :delete, remote: true,
     form_class: "d-inline-block", class: "btn btn-sm btn-outline-secondary", id: "unfollow-button" do %>
        取消关注 <%= @article.user.username %>
    <% end %>
<% else %>
    <%= button_to follow_user_path(@article.user.username), method: :post, remote: true,
    form_class: "d-inline-block", class: "btn btn-sm btn-outline-secondary", id: "follow-button" do %>
        <i class="fa-solid fa-plus"></i>&nbsp;关注 <%= @article.user.username %>
    <% end %>
<% end %>

在上述示例中,我们通过 button_to 方法创建了一个链接,当用户点击该链接时,会向 follow_user_path 路径发送 POST 请求,并将 remote 参数设置为 true,以便在不刷新整个页面的情况下完成请求(ajax请求)

在  profiles 控制器中定义 followunfollow 动作,用于处理关注和取消关注事件,同时返回 JS 视图

class ProfilesController < ApplicationController
    before_action :authenticate_user!, except: [:show]
    before_action :set_profile
    def show
    end
    def follow
        current_user.follow @user
        respond_to do |format|
            format.js
        end
    end
    def unfollow
        current_user.unfollow @user
        respond_to do |format|
            format.js
        end
    end
    private
    def set_profile
        @user = User.find_by_username(params[:username])
    end
end

其中,视图层中的following? 方法和控制器层的 followunfollow 方法我们都去user 模型中定义

...
    def following?(other_user)
        following.include?(other_user)
    end
    def follow(user)
        following << user unless following.include? user   
    end
    def unfollow(user)
        following.delete(user)
    end
...

这里,笔者没有弄出 format.js ,因为加上后也没有效果,如果机会,会补上这块,也就是当点击关注后,接口请求成功后页面弹出 已关注,取消关注后,页面弹出已取消

like 模型

按照上述的经验,我们知道了,如果是多对多,就需要建立一个中间表来存储两者之间的关系。如果要做某个用户给某篇文章点赞呢?也属于多对多关系,

基于 articles 和 user 模型建立新模型 Like:

# 创建 migration 文件
rails g model Like article:references user:references
# 运行 migration
rails db:migrate

前往config/routes:

resources :articles do
    resources :comments, only: [:create, :destroy]
    member do
      post 'like'
      delete 'unlike'
    end
end

再去 app/models/article.rb 模型中,新增方法

class Article < ApplicationRecord
    belongs_to :user
    has_many :comments, dependent: :destroy
    + has_many :likes, dependent: :destroy
    + def liked_by?(user)
    +    likes.where(user_id: user.id).exists?
    + end
end

再去控制器新增 like 和 unlike 方法

before_action :set_article, only: %i[ show edit update destroy like unlike ]
 def like
    unless @article.liked_by?(current_user)
      @like = @article.likes.create(user_id: current_user.id)
    end
    respond_to do |format|
        format.js
    end
  end
  def unlike
    if @article.liked_by?(current_user)
      @like = @article.likes.find_by(user_id: current_user.id)
      @like.destroy
    end
    respond_to do |format|
        format.js
    end
  end

其实,这个和 follow 很像,都是多对多的

标签模型

创建标签模型,它属于文章模型

建立一个多对多关系,一篇文章有多个标签,一个标签下有多篇文章

# 创建 Tag model
rails g model Tag name:string
# 修改 Article 模型文件。在 app/models/article.rb 文件中,添加以下代码
class Article < ApplicationRecord
  has_and_belongs_to_many :tags
end
# 修改 Tag 模型文件。在 app/models/tag.rb 文件中,添加以下代码
class Tag < ApplicationRecord
  has_and_belongs_to_many :articles
end
# 创建 articles_tags 关系表
rails g migration CreateJoinTableArticlesTags articles tags
# 运行 migration
rails db:migrate

如此,我们就建立起了多对多的关系

代码方面笔者踩了一下坑,首先要在models/article层注入:

# 用于 view 层
def tag_list
    tags.map(&:name).join(",")
end
# 用于 controller 层
def sync_tags(tag_list)
    tagArr = JSON.parse(tag_list)
    tagArr.each do |tag_name|
        tag = Tag.find_or_create_by(name: tag_name)
        tags << tag
    end
end

前往 controllers/articles_controller.rb 注入:

def create
  @article = current_user.articles.new(article_params.except(:tag_list))
    respond_to do |format|
      if @article.save
        @article.sync_tags(article_params[:tag_list])
        ...
      else
        ...
      end
    end
end
def article_params
    # 新增 tag_list 变量
 params.require(:article).permit(:title, :description, :body, :tag_list)
end

再回到views/articles 层,在 body 下加入相关 tag 代码

...
<div class="form-group mt-3">
    <%= f.hidden_field :tag_list, id: 'tag-input' %>
    <input 
      id="tag-field"
      class="form-control" 
      type="text"
      placeholder="输入标签" 
      onkeydown="addToList(event)"
      >
    <div class="tag-list mt-1" id="tag-list">
    </div>
</div>

当然,还有 js 代码,就不贴了,逻辑是,输入标签后回车,生成一个标签

受欢迎的标签,我们要通过查询来找到前十的

# 获取最受欢迎的十大标签
tag_counts = Tag.joins(:articles_tags).group(:tag_id).order('count_all desc').limit(10).count
popular_tag_ids = tag_counts.keys
@popular_tags = Tag.where(id: popular_tag_ids).sort_by { |t| popular_tag_ids.index(t.id) }

查询功能

既然喜欢刺激,那就进行到底

既然做到这个份上了,那就把剩下的功能给补齐,这也是笔者最菜的地方——ORM

先补上slug,在文章详情中,我们是通过 id 来查询文章,这样不安全。我们可以用随机字符串,这里我们使用标题来作为我们查询点,专业术语叫“slug”,指「字符串转换成合法的URL路径的过程」

先在 artilce model 中增加字段,然后再迁移数据

# 创建 migration 文件
rails g migration addSlugToArticle slug:string
# 修改 migration 文件,添加搜索索引
class AddSlugToArticle < ActiveRecord::Migration[7.0]
  def change
    add_column :articles, :slug, :string
  end
  + add_index :articles, :slug
end
# 运行 migration
rails db:migrate

前往conf/routes,在resources :articles 后加上 param: :slug

+ resources :articles, param: :slug do
    resources :comments, only: [:create, :destroy]
    member do
      post 'like'
      delete 'unlike'
    end
  end

将类似<%= link_to article ...%> 的地方改成 <%= link_to article_path(article.slug) ,至于 sync_tags,我们因为有修改标签的操作,所以有标签时,更新原来的标签列表,但是笔者说过,操作数据库或者说 rails 相关的 api 接触的太少,所以笔者先把标签清空,再将新的标签放进去,也许会影响性能,但又有什么办法

def sync_tags(tag_list)
    tagArr = JSON.parse(tag_list)
    # 如果已经有标签,删除原有标签
    if tags.any?
        tags.destroy_all
    end
    tagArr.each do |tag_name|
        tag = Tag.find_or_create_by(name: tag_name)
        tags << tag
    end
end

订阅功能

到现在,我们已经完成了一个小论坛的基本雏形,现在,补上论坛中最重要的一点,订阅

def feed
    user =  User.find(current_user.following_ids)
    @articles = Article.order(created_at: :desc).where(user:user).includes(:user)
end

分页功能

分页应该有很多 gem 库,从Rails 谈谈 Rails 中的分页 - 简易版 知道两个库,kaminari 和  pagy 。两者相比, kaminari 更简单,pagy 复杂一点但性能更好,这里我以 kaminari 为例继续我的论坛项目

先加上 gem

gem 'kaminari'

再安装它

bundle

生成默认配置

rails g kaminari:config

此时会生成 config/initializers/kaminari_config.rb ,我们修改配置

# frozen_string_literal: true
Kaminari.configure do |config|
  config.default_per_page = 5 # 修改它,默认为25,将其修改为5做测试用
  # config.max_per_page = nil
  # config.window = 4
  # config.outer_window = 0
  # config.left = 0
  # config.right = 0
  # config.page_method_name = :page
  # config.param_name = :page
  # config.max_pages = nil
  # config.params_on_first_page = false
end

在 controller 层修改

def index
- @articles = Article.order(created_at: :desc).includes(:user)
+ @articles = Article.page(params[:page]).order(created_at: :desc).includes(:user)
end

在 view 层加入

<% @articles.each do |article| %>
  <%= render article %>
<% end %>
+<div class="text-center">
+  <%= paginate @articles %>
+</div>

如下所示:

使用kaminari

但是样式还是默认样式,我们用 bootstrap5,所以尽量也用相关的UI,于是在 RubyToolbox 上找到了 bootstrap5-kaminari-views ,按照 demo 使用

<div class="text-center">
  +<%= paginate @articles, theme: 'bootstrap-5',
  +pagination_class: "flex-wrap justify-content-center" %>
</div>

样式是好了,但还是是英文的,所以还需要按照 i18n,所以还要安装 kaminari-i18n,安装好 kaminari-i18n,UI 就成了我们想要的样子了

![加上i18n以及bootstrap5之后的kaminari](D:\Documents\PicGo Files\image-20230408074857093.png)

再次部署

我们还是在 fly.io 中部署,分两步,一是将项目重新部署下,二是迁入数据

# 实例化应用
fly launch
# 部署应用
fly deploy
# 打开应用
fly open

如此,我们能看到页面,但是因为创建的数据库没导入,所以会报错,我们需要迁入数据

# 进入控制台
flyctl ssh console
# 迁入数据
bin/rails db:migrate

这时, https://underground-palace.fly.dev 就能正常访问

Logo设计

在项目初期阶段,完全不用担心 logo 的事情,没人会在意你,你要做的就是做个可以看的logo贴上去,如果在 logo 上花费太多时间,得不偿失

笔者习惯在 favicon.io 中找 emoji 来做logo,这次也是,看到合适的,下载,然后把文件拉到 public 中即可

后记

我当然知道,如果要做一个完整的项目,以上这些是不够的,还要有更考究的UI、交互,还要加上搜索,静态资源的中文化、错误提示的中文化等等。但,那又怎么样呢

相关文章
|
2月前
|
Cloud Native 前端开发 JavaScript
前端开发者必看:不懂云原生你就OUT了!揭秘如何用云原生技术提升项目部署与全栈能力
【10月更文挑战第23天】随着云计算的发展,云原生逐渐成为技术热点。前端开发者了解云原生有助于提升部署与运维效率、实现微服务化、掌握全栈开发能力和利用丰富技术生态。本文通过示例代码介绍云原生在前端项目中的应用,帮助开发者更好地理解其重要性。
95 0
|
2月前
|
监控 前端开发 数据可视化
3D架构图软件 iCraft Editor 正式发布 @icraft/player-react 前端组件, 轻松嵌入3D架构图到您的项目,实现数字孪生
@icraft/player-react 是 iCraft Editor 推出的 React 组件库,旨在简化3D数字孪生场景的前端集成。它支持零配置快速接入、自定义插件、丰富的事件和方法、动画控制及实时数据接入,帮助开发者轻松实现3D场景与React项目的无缝融合。
180 8
3D架构图软件 iCraft Editor 正式发布 @icraft/player-react 前端组件, 轻松嵌入3D架构图到您的项目,实现数字孪生
|
3月前
|
JavaScript 前端开发 Docker
前端全栈之路Deno篇(二):几行代码打包后接近100M?别慌,带你掌握Deno2.0的安装到项目构建全流程、剖析构建物并了解其好处
在使用 Deno 构建项目时,生成的可执行文件体积较大,通常接近 100 MB,而 Node.js 构建的项目体积则要小得多。这是由于 Deno 包含了完整的 V8 引擎和运行时,使其能够在目标设备上独立运行,无需额外安装依赖。尽管体积较大,但 Deno 提供了更好的安全性和部署便利性。通过裁剪功能、使用压缩工具等方法,可以优化可执行文件的体积。
172 3
前端全栈之路Deno篇(二):几行代码打包后接近100M?别慌,带你掌握Deno2.0的安装到项目构建全流程、剖析构建物并了解其好处
|
2月前
|
前端开发 测试技术
前端工程化的分支策略要如何与项目的具体情况相结合?
前端工程化的分支策略要紧密结合项目的实际情况,以实现高效的开发、稳定的版本控制和顺利的发布流程。
31 1
|
2月前
|
前端开发 Unix 测试技术
揭秘!前端大牛们如何高效管理项目,确保按时交付高质量作品!
【10月更文挑战第30天】前端开发项目涉及从需求分析到最终交付的多个环节。本文解答了如何制定合理项目计划、提高团队协作效率、确保代码质量和应对项目风险等问题,帮助你学习前端大牛们的项目管理技巧,确保按时交付高质量的作品。
45 2
|
3月前
|
JavaScript 前端开发 测试技术
前端全栈之路Deno篇(五):如何快速创建 WebSocket 服务端应用 + 客户端应用 - 可能是2025最佳的Websocket全栈实时应用框架
本文介绍了如何使用Deno 2.0快速构建WebSocket全栈应用,包括服务端和客户端的创建。通过一个简单的代码示例,展示了Deno在WebSocket实现中的便捷与强大,无需额外依赖,即可轻松搭建具备基本功能的WebSocket应用。Deno 2.0被认为是最佳的WebSocket全栈应用JS运行时,适合全栈开发者学习和使用。
149 7
|
3月前
|
前端开发 JavaScript 中间件
前端全栈之路Deno篇(四):Deno2.0如何快速创建http一个 restfulapi/静态文件托管应用及oak框架介绍
Deno 是由 Node.js 创始人 Ryan Dahl 开发的新一代 JavaScript 和 TypeScript 运行时,旨在解决 Node.js 的设计缺陷,具备更强的安全性和内置的 TypeScript 支持。本文介绍了如何使用 Deno 内置的 `Deno.serve` 快速创建 HTTP 服务,并详细讲解了 Oak 框架的安装和使用方法,包括中间件、路由和静态文件服务等功能。Deno 和 Oak 的结合使得创建 RESTful API 变得高效且简便,非常适合快速开发和部署现代 Web 应用程序。
133 2
|
3月前
|
JavaScript 前端开发 Serverless
前端全栈之路Deno篇:Deno2.0与Bun对比,谁更胜一筹?可能Deno目前更适合serverless业务
在前端全栈开发中,Deno 2.0 和 Bun 作为新兴的 JavaScript 运行时,各自展现了不同的优势。Deno 2.0 重视安全性和多平台兼容性,尤其是对 Windows 的良好支持和原生 TypeScript 支持;而 Bun 则以卓越的性能和简便的开发体验著称,适合快速迭代的小型项目。两者在不同场景下各具特色,Deno 更适合企业级应用和serverless,Bun 则适用于追求速度的项目。
291 1
|
3月前
|
前端开发 安全 API
前端全栈之路Deno篇(三):一次性搞懂和学会用Deno 2.0 的权限系统详解和多种权限配置权限声明方式
本文深入解析了 Deno 2.0 的权限系统,涵盖主包和第三方包的权限控制机制,探讨了通过命令行参数、权限 API 和配置文件等多种权限授予方式,并提供了代码示例和运行指导,帮助开发者有效管理权限,提升应用安全性。
|
8月前
|
JSON 数据格式 Ruby