ساخت اپلیکیشن اینستاگرام با Ruby on Rails (بخش اول) – از صفر تا صد


در این نوشته روش ساخت یک اپلیکیشن مشابه اینستاگرام با استفاده از Ruby on Rails را توضیح میدهیم. مواردی که با مطالعه این راهنما یاد میگیرید را در ادامه فهرست کردهایم:
- چگونه یک اپلیکیشن Rails جدید آغاز کنیم؟
- طراحی سیستم از صفر
- درک معماری MVC (مدل-ویو-کنترلر)
- مدل: مهاجرت (Migration) به Active Record، اعتبارسنجی، فراخوانی، ارتباط، و اینترفیس کوئری.
- ویو: تابعهای کمکی Layout ،Partial و Form
- کنترلر: Action-ها، پارامترهای Strong
- مسیریابی Rails
- ذخیرهسازی Active برای آپلود فایلها
- استفاده از بوتاسترپ، Devise ،Kaminari gem در اپلیکیشنهای Rails.
در این مقاله با مباحث زیر آشنا خواهیم شد:
- مجموعههای فناوری
- کسب بینشی از MVC در Ruby on Rails
- ایجاد اپلیکیشنهای Rails جدید
- استفاده از پایگاه داده PostgreSQL و بوتاسترپ برای اپلیکیشن
- فرایندهای Sign Up ،Sign In و Sign Out برای کاربران با استفاده از Devise gem
- ایجاد صفحه پروفایل کاربری
- ویرایش/بهروزرسانی پروفایل کاربر
مجموعه فناوریهای مورد نیاز در این مقاله به شرح زیر هستند:
بکاند:
- زبان برنامه نویسی روبی نسخه 2.4
- Rails 5.2.x
- Database: Postgres 9.6
فرانتاند:
- HTML, CSS, Javascript, jQuery
- (Bootstrap (3.x or 4.x
MVC (مدل-ویو-کنترلر) در Ruby on Rails
MVC یک الگوی معماری برای اپلیکیشنهای نرمافزاری است. این معماری برای طراحی وب اپلیکیشنها بسیار محبوب است. MVC در PHP به عنوان چارچوبی رایج برای توسعه وب کاربرد دارد. این معماری موجب جداسازی اپلیکیشن به دستههای زیر میشود:
- مدلها (Active Record): به مدیریت داده و منطق تجاری اپلیکیشن میپردازد.
- ویو ها (ActionView): به مدیریت شیءهای اینترفیس و ارائه میپردازد.
- کنترلرها (ActionController): بین مدل و ویو قرار میگیرد و با دریافت ورودی کاربر تصمیم میگیرد که با آن چه بکند.
چرخه درخواست-پاسخ در Rails
اگر بخواهیم این فرایند را به صورت فهرستی نشان دهیم، به صورت زیر خواهد بود:
- کاربر مرورگرش را باز میکند، یک URL وارد میکند و اینتر را میزند. زمانی که یک کاربر اینتر را میزند، مرورگر درخواستی به آن URL میفرستد.
- این درخواست به روتر Rails میرسد (config/routes.rb).
- روتر، URL را به کنترلر و اکشن صحیح نگاشت میکند تا این درخواست مدیریت شود.
- اکشن مربوطه درخواست را دریافت کرده و از مدل درخواست میکند که دادههای مربوطه از پایگاه داده واکشی شوند.
- مدل، دادهها را به اکشن کنترلر بازمیگرداند.
- اکشن کنترلر دادهها را به ویو بازمیگرداند.
- ویو صفحه را به صورت HTML رندر میکند.
- کنترلر HTML را به مرورگر بازمیگرداند. سپس صفحه بارگذاری میشوند و کاربر میتواند آن را ببیند.
ایجاد اپلیکیشنهای Rails جدید
برای ایجاد یک اپلیکیشن جدید در Rails باید مراحل زیر را طی کنیم:
نصب Rails
برای نصب Rails باید از دستور gem install استفاده کنیم که از سوی RubyGems ارائه شده است:
gem install rails -v 5.2.1
پس از نصب Rails اقدام به بررسی نسخه نصب شده میکنیم:
rails --version => Rails 5.2.1
اگر چیزی مانند Rails 5.2.1 بازگشت یابد، میتوانیم به کار خود ادامه داده و اپلیکیشنهای جدید Rails ایجاد کنیم.
نصب PostgreSQL
در سیستم عامل Mac OSX میتوان سرور و کلاینت PostgreSQL را از Homebrew نصب کرده:
brew install postgresql
سرویس Postgresql به صورت زیر آغاز میشود:
brew services start postgresql
ایجاد اپلیکیشن جدید Rails
rails new instagram --version=5.2.1
پس از ایجاد اپلیکیشن اینستاگرام به پوشه آن مراجعه کنید:
cd Instagram
Gems را برای اپلیکیشنمان ایجاد کنید:
bundle install
با دستور زیر نیز وبسرور را راهاندازی کنید:
rails server
برای مشاهده اپلیکیشن میتوانید مرورگر خود را باز کرده و به آدرس http://localhost:3000/ مراجعه کنید. در این مرحله صفحه پیشفرض Rails را مشاهده خواهید کرد:
جهت متوقف کردن سرور نیز میتوانید دکمههای Ctrl+G را در ترمینال وارد کنید.
ایجاد صفحه اصلی
- یک کنترلر با یک اکشن ایجاد کنید.
- یک مسیر اضافه کنید.
یک کنترلر به نام Home را با اکشن index ایجاد کنید:
rails g controller Home index
Rails برخی فایلها و یک مسیر برای شما ایجاد میکند:
create app/controllers/home_controller.rb route get 'home/index' invoke erb create app/views/home create app/views/home/index.html.erb invoke test_unit create test/controllers/home_controller_test.rb invoke helper create app/helpers/home_helper.rb invoke test_unit invoke assets invoke coffee create app/assets/javascripts/home.coffee invoke scss create app/assets/stylesheets/home.scss
فایل app/views/home/index.html.erb را باز کرده و کد موجود را با کد زیر جایگزین کنید:
<h1>This is my home page</h1>
وبسرور را ریاستارت کنید. به این منظور میتوانید ابتدا با دکمههای Ctrl+G سرور را متوقف کرده و سپس با دستور rails server آن را آغاز کنید و در مرورگر خود به آدرس http://localhost:3000/home/index بروید. بدین ترتیب پیام «This is my home page» را که در فایل iews/home/index.html.erb قرار دادهاید، مشاهده میکنید. اینک فایل config/routes.rb را باز کنید:
Rails.application.routes.draw do get 'home/index' # For details on the DSL available within this file, see http://guides.rubyonrails.org/routing.html end
و خط کدی که به صورت root 'home#index' است را اضافه نمایید. اینک کل کد باید به صورت زیر در آمده باشد:
Rails.application.routes.draw do get 'home/index' root to: ‘home#index’ end
- بخش 'get 'home/index به Rails اعلام میکند که درخواستها به آدرس http://localhost:3000/home/index را به اکشن index در کنترلر Home هدایت کند.
- بخش 'root 'home#index به Rails اعلام میکند که درخواستها به ریشه اپلیکیشن را به اکشن index در کنترلر home هدایت کند.
اینک وبسرور را ریاستارت کرده و به آدرس http://localhost:3000 مراجعه کنید تا با پیامی به صورت «This is my home page» مواجه شوید. شما میتوانید همه مسیرهای کنونی اپلیکیشن را با دستور زیر مشاهده کنید:
rails routes
استفاده از PostgreSQL در اپلیکیشن Rails
برای استفاده از PostgreSQL در اپلیکیشنهای Rails باید یک gem به نام pg را به Gemfile اضافه کنیم:
gem ‘pg’
اینک برای نصب gem به نام pg دستور bundle install را اجرا کنید.
پیکربندی پایگاه داده (config/database.yml)
default: &default adapter: postgresql pool: <%= ENV.fetch(“RAILS_MAX_THREADS”) { 5 } %> timeout: 5000 development: <<: *default database: development_instagram test: <<: *default database: test_instagram production: <<: *default database: production_instagram
ایجاد پایگاه داده
برای ایجاد پایگاه داده برای اپلیکیشن میتوانید از دستور مهاجرت (migration) استفاده کنید:
rails db:create >> Created database ‘development_instagram’ >> Created database ‘test_instagram’
نصب بوتاسترپ برای اپلیکیشنهای Rails
«بوتاسترپ» (Bootstrap) یک کیت ابزار اوپنسورس برای توسعه صفحههایی با استفاده از HTML ،CSS و JS است. این فریمورک به ایجاد سریع نمونههای اولیه از ایدهها یا ساخت کل اپلیکیشن با استفاده از متغیرهای Saas و mixin ها، سیستم شبکهای واکنشگرا، کامپوننتهای پیشساخته گسترده و افزونههای قدرتمند برای بهکارگیری jQuery کمک میکند.
ما برای یکپارچهسازی بوتاسترپ با اپلیکیشنهای Rails از bootstrap-rubygem استفاده میکنیم. به این منظور bootstrap را به Gemfile خود اضافه کنید:
gem ‘bootstrap’, ‘~> 4.1.1’
دستور bundle install را برای نصب bootstrap gem و ریاستارت کردن سرور اجرا کنید تا فایلها از طریق pipline در دسترس قرار گیرند.
پیکربندی روی application.css یا همان فایل app/assets/stylesheets/application.css
- تغییر نام application.css به application.scss
- سپس همه گزارههای *= require و *= require_tree را از فایل Sass حذف کنید. در عوض از import@ برای ایمپورت کردن فایلهای Sass استفاده میکنیم.
- ایمپورت کردن سبکهای Bootstrap در application.scss:
@import "bootstrap";
پیکربندی روی application.js
بخش جاوا اسکریپت بوتاسترپ به jQuety وابسته است. از آنجا که ما از Rails 5.1+ استفاده میکنیم، gem به نام jquery-rails را به Gemfile خود اضافه کنید:
gem 'jquery-rails'
وابستگیهای بوتاسترپ را اضافه کرده و به صورت application.js بوتاسترپ کنید:
//= require jquery3 //= require popper //= require bootstrap
طرحبندی اپلیکیشن
در ادامه قصد داریم اپلیکیشن را در 3 بخش اصلی طرحبندی کنیم:
- نوار ناوبری
- محتوای اصلی
- فوتر
این طرحبندی چیزی مانند تصویر زیر خواهد بود:
Rails از فایل طرحبندی پیشفرض در مسیر app/views/layouts/application.html.erb استفاده میکند.
کد HTML را به طرحبندی اضافه کنید (layouts/application.html.erb/):
<!DOCTYPE html> <html> <head> <title>Instuigram</title> <%= csrf_meta_tags %> <%= stylesheet_link_tag 'application', media: 'all', 'data-turbolinks-track': 'reload' %> <%= javascript_include_tag 'application', 'data-turbolinks-track': 'reload' %> </head> <body> <!--Navigation Bar--> <nav class="navbar navbar-light bg-light d-flex"> <div class="container"> <div class="logo"> <a class="navbar-brand" href="#"> <%= fa_stacked_icon "camera inverse", base: "circle"%> Instagram </a> </div> <div class="m-auto"> <form class="form-inline search-form"> <input class="form-control mr-sm-2" type="search" placeholder="@search" aria-label="Search"> </form> </div> <div class="user-section"> <a><i class="fa fa-compass"></i></a> <a><i class="fa fa-heart"></i></a> <a><i class="fa fa-user"></i></a> </div> </div> </nav> <!--Main content--> <main class="container" style="min-height: 500px;"> <%= yield %> </main> <!--Footer--> <footer class="container footer"> <nav class="navbar d-flex"> <ul> <li>ABOUT US </li> <li>SUPPORT</li> <li>PRESS</li> <li>API</li> <li>JOBS</li> <li>PRIVACY</li> <li>TERMS</li> <li>DIRECTORY</li> <li>PROFILES</li> <li>HASHTAGS</li> <li>LANGUAGE</li> </ul> <p> © 2018 INSTAGRAM </p> </nav> </footer> </body> </html>
استفاده از Awesome Icon
ما در اپلیکیشن خود از برخی آیکونهای Font Awesome Icon استفاده میکنیم. برای استفاده آسان از این آیکونها باید بتوانیم gem به نام font-awesome-rails استفاده کنیم.
کد زیر را به فایل زیر اضافه کنید:
gem 'font-awesome-rails'
و دستور bundle install را اجرا کنید. اینک font-awesome را به فایل application.scss اضافه کنید.
@import "font-awesome";
کد CSS را برای طرحبندی (application.scss) اضافه کنید.
.container{ max-width: 1000px; } .navbar{ border-bottom: 1px solid rgba(0,0,0,.0975); } .user-section{ a{ padding-left: 15px; } i.fa{ font-size: 22px; color: #000000d4; } } .search-form{ input{ font-size: 13px; text-align: center; } } .footer{ font-size: 12px; padding-top: 30px; ul{ padding-left: 0; > li{ display: inline; padding-right: 15px; color: #003569; } } p{ color: #999; } }
کاربران: Sign Up ،Sign In و Sign Out
در این بخش ما قصد داریم از gem به نام devise برای ایجاد کارکردهای sign up، sign in و sign out برای کاربران استفاده کنیم. Devise یک راهحل احراز هویت انعطافپذیر برای Rails است. این راهحل از 10 ماژول زیر تشکیل یافته است:
- Database Authenticatable: این ماژول یک رمز عبور را هش کرده و در پایگاه داده برای اعتبار سنجی و احراز هویت کاربر در زمان ورود به اپلیکیشن استفاده میکند. احراز هویت میتواند از طریق درخواستهای POST یا احراز هویت مقدماتی HTTP صورت بگیرد.
- Omniauthable: پشتیبانی از (OmniAuth (https://github.com/omniauth/omniauth را ارائه میکند.
- Confirmable: ایمیلهایی با دستور تأیید ارسال کرده و بررسی میکند که آیا یک حساب کاربری قبلاً در طی ورود کاربر تأیید شده یا نه.
- Recoverable: رمز عبور کاربر را ریست کرده و دستورالعمل تنظیم مجدد آن را به ایمیلش ارسال میکند.
- Registerable: ثبت نام کاربر را از طریق فرایند registration مدیریت کرده و به کاربران امکان ویرایش و حذف حسابهایشان را نیز میدهد.
- Rememberable: ایجاد و پاکسازی توکن برای بهخاطرسپاری کاربر از روی کوکیهای ذخیره شده را مدیریت میکند.
- Trackable: تعداد ورودها، زمانها و آدرسهای IP را ردگیری میکند.
- Timeoutable: نشستهایی که در دوره زمانی مشخصی فعال نبودهاند را منقضی میکند.
- Validatable: اعتبارسنجیهای ایمیل و رمز عبور را انجام میدهد. این گزینه اختیاری است و میتوان آن را سفارشیسازی کرد، به طوری که میتوان تابعهای خاص برای اعتبار سنجی تعریف کرد.
- Lockable: یک حساب را پس از تعداد مشخصی از تلاشهای شکستخورده برای ورود قفل میکند. چنین حسابهایی را میتوان پس از گذشت زمان مشخصی از حالت قفل خارج کرد.
نصب Devise Gem
برای نصب devise باید کد زیرا را به Gemfile اضافه کنید:
gem 'devise'
سپس دستور bundle install را وارد کرده و generator را اجرا کنید:
rails generate devise:install
ما 2 فایل زیر را به صورت خودکار تولید کردهایم:
- config/initializers/devise.rb
- config/locales/devise.en.yml
به منظور راه اندازی گزینههای پیشفرض URL برای ارسال کننده ایمیل Devise در محیط توسعه، یعنی فایل config/environments/development.rb، کد زیر را وارد کنید:
config.action_mailer.default_url_options = { host: ‘localhost’, port: 3000 }
در محیط Production مقدار :host باید به میزان واقعی اپلیکیشن اشاره کند.
تولید مدل کاربر
به این منظور باید از دستور زیر استفاده شود:
rails generate devise User invoke active_record create db/migrate/20180722043305_devise_create_users.rb create app/models/user.rb invoke test_unit create test/models/user_test.rb create test/fixtures/users.yml insert app/models/user.rb route devise_for:users
پس از اجرای دستور فوق یک فایل برای ایجاد کاربر، یک فایل در app/models به نام user.rb و یک مسیر برای کاربر و فایل تست ایجاد میشوند.
فایل مدل یعنی app/models/user.rb را باز کنید تا ماژولهای پیشفرض devise را که در مدل کاربر وجود دارند ببینید.
devise:database_authenticatable,:registerable, :recoverable,:rememberable,:trackable,:validatable
سپس دستور زیر را اجرا کنید:
rails db:migrate
اینک باید اپلیکیشن را پس از تغییر دادن گزینههای پیکربندی Devise ریاستارت کنید. در غیر این صورت با خطاهای عجیبی مواجه خواهید شد، برای نمونه ممکن است کاربران نتوانند وارد اپلیکیشن شوند یا تابعهای کمکی مسیر تعریفنشده باشند.
افزودن لینکهای ورود و خروج
در نوار ناوبری سمت راست باید لینکهای sign_in و sign_out را به صورت زیر اضافه کنید. دقت کنید که کد HTML زیر را در طرحبندی (application.html.erb):
<a><i class=”fa fa-user”></i></a>
با کد زیر عوض کنید:
<% if user_signed_in? %> <a href="<%= destroy_user_session_path%>" data-method="delete"> <i class="fa fa-user"></i> </a> <% else%> <a href="<%= new_user_session_path%>"> <i class="fa fa-sign-in"></i> </a> <% end%>
توضیح برخی عناصر به صورت زیر است:
- ?user_signed_in: یک تابع کمکی devise است که کاربر وارد شده را تأیید میکند.
- destroy_user_session_path: مسیر خروجی است، method::delete متد پیشفرض HTML مورد استفاده برای خارج شدن یک منبع (کاربر) است.
- new_user_session_path: مسیر خروجی است.
زمانی که روی آیکون خروج کلیک کنید، با ویویی مانند زیر مواجه میشوید:
اما پیش از این که بتوانید دوباره وارد شوید باید یک حساب جدید با کلیک روی لینک Sign up در زیر دکمه Log in ایجاد کنید. ویوی این صفحه ثبت نام به صورت زیر است:
با پر کردن آدرس ایمیل و رمز عبور میتوانید ثبت نام کنید و سپس به صفحه ورود اپلیکیشن بروید.
زمانی که ثبت نام موفق بود، آیکون ورود با آیکون کاربر در نوار ناوبری تعویض میشود و اگر روی آیکون کاربر کلیک کنید میتوانید از اپلیکیشن خارج شوید.
ایجاد صفحه پروفایل کاربر
جدول کاربر ستونهایی به صورت username ،name ،email ،password ،website، bio ،phone و gender است. کاربر از قبل دارای ستونهای ایمیل و رمز عبور است و اینک باید فیلدهای دیگر را با استفاده از migration اضافه کنیم:
rails g migration AddMoreFieldsToUsers username:string name:string website:string bio:text phone:integer gender:string
migration را اجرا کنید:
rails db:migrate == 20180813140820 AddMoreFieldsToUsers: migrating ============================= — add_column(:users,:username,:string) -> 0.0333s — add_column(:users,:name,:string) -> 0.0006s == 20180813140820 AddMoreFieldsToUsers: migrated (0.0363s) ====================
ایجاد کنترلر کاربر
برای ایجاد صفحه پروفایل کاربر نخستین گام این است که یک کنترلر به نام UsersController را با استفاده از دستور زیر ایجاد کنیم:
rails generate controller Users
سپس فایل app/controllers/users_controller.rb را در ویرایشگر باز کنید تا کدی مانند زیر را ببینید:
class UsersController < ApplicationController end
در مرحله بعد یک اکشن show به UsersController اضافه میکنیم:
class UsersController < ApplicationController def show end end
سپس یک ویوی متناظر به صورت زیر ایجاد میکنیم:
app/views/users/show.html.erb
در نهایت اکشن show را به مسیرها (config/routes.rb) اضافه میکنیم:
Rails.application.routes.draw do ... resources:users, only: [:show] end
بخش [resources:users, only: [:show به Rails اعلام میکند که درخواستهایی که به آدرس http://localhost:3000/users/:id دریافت میکند را به اکشن show کنترلر کاربر با id به صورت ID جاری کاربر هدایت کند. اگر ID کاربر جاری برابر با 1 باشد، مسیر به صفحه پروفایل کاربر به صورت زیر خواهد بود:
http://localhost:3000/users/1
بهروزرسانی لینک پروفایل کاربر در نوار ناوبری
در سمت راست نوار ناوبری، لینک logout را با لینک پروفایل کاربر تعویض میکنیم. این بدان معنی است که وقتی روی آیکون کاربر کلیک میکنیم باید به صفحه پروفایل کاربری برود. پس از بهروزرسانی کد HTML به صورت زیر درمیآید:
<!-- app/views/layouts/application.html.erb --> <% if user_signed_in? %> <a href=”<%= user_path(current_user)%>”> <i class=”fa fa-user”></i> </a> <% else %> <a href=”<%= new_user_session_path%>”> <i class=”fa fa-sign-in”></i> </a> <% end %>
ایجاد UI برای صفحه پروفایل کاربر
برای طرحبندی صفحه پروفایل کاربر دو بخش را در نظر داریم: یکی اطلاعات پایه و دیگر پستها.
- اطلاعات پایه: شامل آواتار، نام، نام کاربری، پستها، فالوورها، فالووینگها.
- پستها: شامل تصاویر کاربر.
بخش 1: اطلاعات پایه کاربر

این بخش شامل دو ستون است که ستون سمت چپ آواتار کاربر و ستون سمت راست شامل اطلاعات کاربر است. برای طرحبندی این وضعیت از ردیفها و ستونها استفاده میکنیم و کد HTML مربوطه به صورت زیر است:
# app/views/users/show.html.erb <div class="profile row"> <div class="col-md-4 avatar"> <!-- LEFT: AVATAR HERE --> </div> <div class="col-md-8 basic-info"> <!-- RIGHT: USER INFORMATION HERE --> </div> </div>
از آنجا که ما هیچ دادهای برای برخی فیلدهای اطلاعات کاربر مانند نام، تعداد پست، فالوور، فالووینگ و تصاویر نداریم، از این رو باید از برخی اطلاعات ساختگی برای ساخت UI استفاده کنیم تا بعداً بتوانیم آنها را بهروزرسانی کنیم. پس از افزودن کامپوننتهای دیگر در این بخش، کد HTML به صورت زیر درمیآید:
<div class="user-profile"> <div class="profile row"> <div class="col-md-4 avatar"> <div class="wrapper"> <img src="https://instagram.fsgn5-1.fna.fbcdn.net/vp/3ef315d2478029b529d3c3f38daadeb3/5C12B330/t51.2885-19/s150x150/21980557_344791629305453_6312020290528346112_n.jpg"> </div> </div> <div class="col-md-8 basic-info"> <div class="user-wrapper"> <h2 class="username">luanotes</h2> <a class="edit-profile" href="<%= edit_user_path(current_user) %>"> <button>Edit Profile</button> </a> </div> <ul class="posts"> <li><span>12</span> posts</li> <li><span>195</span> followers</li> <li><span>345</span> following</li> </ul> <h2 class="name">Luan Nguyen</h2> <a class="website">luanotes.com</a> </div> </div> </div>
برای ایجاد CSS برای این ویو، یک فایل به نام assets/stylesheets/users.scss میسازیم که شامل سبکهای مورد نیاز باری ین صفحه است و آن را در فایل application.scss ایمپورت میکنیم:
@import "users";
کدهای CSS زیر را به فایل users.scss اضافه کنید:
.profile .avatar{ .wrapper{ width: 140px; height: 140px; margin: 40px auto; } .wrapper img{ width: 100%; border-radius: 50%; } } .profile .basic-info{ padding: 40px 0; .user-wrapper{ display: flex; } .username{ font-size: 30px; line-height: 40px; font-weight: 200; } .edit-profile{ margin-left: 20px; margin-top: 5px; } .edit-profile button{ background-color: transparent; border-radius: 5px; border: 1px solid #dbdbdb; color: #262626; font-size: 14px; padding: 4px 10px; font-weight: 500; } ul.posts{ padding-left: 0; > li{ display: inline; padding-right: 40px; font-size: 15px; span{ color: #262626; font-weight: 600; } } } .name{ font-size: 15px; font-weight: 600; } .website{ color: #003569; text-decoration: none; font-size: 15px; font-weight: 600; } }
اینک ویوی ما به صورت زیر درمیآید:

بخش 2: پستهای کاربر
در این مرحله 4 برگه زیر به صورت POSTS ،IGTV ،SAVED، و TAGGED را مانند تصویر زیر اضافه میکنیم:
<div class="user-tabs"> <a class="tab active" href=""> <i class="fa fa-th"></i> POSTS </a> <a class="tab" href=""> <i class="fa fa-tv"></i> IGTV </a> <a class="tab" href=""> <i class="fa fa-bookmark"></i> SAVED </a> <a class="tab" href=""> <i class="fa fa-tag"></i> TAGGED </a> </div>
و کد سبک دهی آنها نیز به صورت زیر است:
.user-tabs{ border-top: 1px solid #efefef; display: flex; justify-content: center; a.tab{ height: 35px; margin-right: 50px; line-height: 45px; color: #999; font-size: 12px; font-weight: 500; text-align: center; i{ padding-right: 1px; } &:hover{ text-decoration: none; } } a.active{ border-top: 1px solid #262626; color: #262626; } }
بخش تصاویر شامل سه ستون خواهد بود که در هر ردیف سه تصویر به نمایش درمیآید:
کد HTML:
<div class="user-images"> <div class="wrapper"> <img src="your_image_url"> </div> <div class="wrapper"> <img src="your_image_url"> </div> <div class="wrapper"> <img src="your_image_url"> </div> </div>
ما از تکنیک Flexbox در CSS برای طرحبندی این تصاویر استفاده کردیم که کد آن به صورت زیر است:
.user-images{ display: flex; flex-wrap: wrap; justify-content: space-between; margin: 0 20px; .wrapper{ width: 280px; height: 280px; margin: 15px; img{ width: 100%; border-radius: 4%; } } }
در نهایت صفحه پروفایل کاربر به صورت زیر درمیآید:

اینک ما توانستهایم صفحه پروفایل کاربر را ایجاد کنیم و در گام بعدی صفحه ویرایش/بهروزرسانی پروفایل کاربر را اضافه میکنیم.
ویرایش/بهروزرسانی کاربر
مراحل ایجاد صفحه ویرایش کاربر را میتوان در 4 گام زیر خلاصه کرد:
- مرحله 1: افزودن صفحه ویرایش کاربر
- مرحله 2: طرحبندی صفحه ویرایش کاربر
- مرحله 3: افزودن لینکهای ناوبری به سمت چپ صفحه
- مرحله 4: افزودن فرم ویرایش پروفایل به سمت راست صفحه
UI صفحه ویرایش پروفایل به صورت زیر است:

مرحله 1: افزودن صفحه ویرایش کاربر
در گام نخست باید یک اکشن با نام edit در کنترلر UsersController اضافه کنیم:
class UsersController < ApplicationController ... def edit end end
سپس یک ویوی متناظر به نام app/views/users/edit.html.erb اضافه میکنیم. پس از آن اکشن edit را به مسیرها اضافه میکنیم:
resources:users, only: [:show,:edit]
اینک میتوانید مسیر صفحه ویرایش کاربر را با وارد کردن دستور زیر مشاهده کنید:
rake routes ... ... edit_user GET /users/:id/edit(.:format) users#edit
بخش [resources:users, only: [...,:edit به Rails اعلام میکند که درخواستهای دریافتی به آدرس http://localhost:3000/users/:id/edi را به اکشن edit کنترلر کاربر هدایت کنید. لینک ویرایش پروفایل را به دکمه Edit Profile در صفحه پروفایل کاربر اضافه کنید:
<a class="edit-profile" href="<%= edit_user_path(current_user) %>"> <button>Edit Profile</button> </a>
current_user یک تابع کمکی devise است که کاربر جاری آن خارج شده است.
مرحله 2: طرحبندی صفحه ویرایش پروفایل
ما طرحبندی صفحه ویرایش کاربر را با 2 ستون انجام دادیم. ستون چپ action-ها و ستون راست جزییات متناظر هستند.

کد HTML این صفحه به صورت زیر است:
<div class="user-edit-page"> <div class="actions"> <!-- Left column --> </div> <div class="details"> <!-- Right column --> </div> </div>
کد CSS آن نیز با استفاده از تکنیک flex به صورت زیر خواهد بود:
.user-edit-page{ display: flex; margin-top: 60px; min-height: 500px; .actions{ width: 220px; border: 1px solid #dbdbdb; } .details{ flex: 1; border: 1px solid #dbdbdb; border-left:none; } }
مرحله 3: افزودن ناوبری به سمت چپ صفحه
ما از کامپوننتهای navs بوتاسترپ 4 با چینش عمودی در این بخش استفاده میکنیم. کد HTML آن به صورت زیر است:
<div class="user-edit-page"> <div class="actions"> <div class="nav flex-column nav-pills" role="tablist" aria-orientation="vertical"> <a class="nav-link active" data-toggle="pill" href="#v-pills-home" role="tab" aria-selected="true">Edit Profile</a> <a class="nav-link" data-toggle="pill">Change Password</a> <a class="nav-link" data-toggle="pill">Authorized Applications</a> <a class="nav-link" data-toggle="pill">Email and SMS</a> <a class="nav-link" data-toggle="pill">Manage Contacts</a> <a class="nav-link" data-toggle="pill">Privacy and Security</a> </div> </div> <div class="details tab-content"> <div class="tab-pane fade show active" id="v-pills-home" role="tabpanel" aria-labelledby="v-pills-home-tab"> <!-- Form edit profile --> </div> </div> </div>
همچنین کد CSS برای بخش اکشنها به صورت زیر است:
.actions{ width: 220px; border: 1px solid #dbdbdb; .nav-link{ font-size: 14px; background-color: white; color: black; border-radius: 0; padding: 12px 0 12px 30px; &:hover{ cursor: pointer; } } .nav-pills .nav-link.active{ border-left: 2px solid; font-weight: 600; } }
اینک UI این بخش به صورت زیر در آمده است:

مرحله 4: افزودن فرم ویرایش کاربر به سمت راست صفحه
یک خط کد برای یافتن کاربر جاری به اکشن edit در articles_controller.rb اضافه میکنیم:
def edit @user = User.find(params[:id]) End
در ستون سمت راست یک فرم برای بهروزرسانی کاربر جاری اضافه میکنیم. ما از تابع کمکی form_with در Rails برای ایجاد فرم استفاده میکنیم. فرم ویرایش کاربر به صورت زیر است:
<%= form_with model: current_user, local: true, html: {class: "form-horizontal form-edit-user"} do |f| %> <div class="form-group row"> <div class="col-sm-3 col-form-label"> <img class="avatar" src="https://instagram.fsgn5-1.fna.fbcdn.net/vp/3ef315d2478029b529d3c3f38daadeb3/5C12B330/t51.2885-19/s150x150/21980557_344791629305453_6312020290528346112_n.jpg"> </div> <div class="col-sm-9"> <p class="username"><%=current_user.username %></p> <a class="change-photo" href="">Change Profile Photo</a> </div> </div> <div class="form-group row"> <%= f.label :name, class: 'col-sm-3 col-form-label' %> <div class="col-sm-9"> <%= f.text_field :name, class: 'form-control' %> </div> </div> <div class="form-group row"> <%= f.label :username, class: 'col-sm-3 col-form-label' %> <div class="col-sm-9"> <%= f.text_field :username, class: 'form-control' %> </div> </div> <div class="form-group row"> <%= f.label :website, class: 'col-sm-3 col-form-label' %> <div class="col-sm-9"> <%= f.text_field :website, class: 'form-control' %> </div> </div> <div class="form-group row"> <%= f.label :bio, class: 'col-sm-3 col-form-label' %> <div class="col-sm-9"> <%= f.text_area :bio, class: 'form-control' %> </div> </div> <div class="form-group row"> <%= f.label :email, class: 'col-sm-3 col-form-label' %> <div class="col-sm-9"> <%= f.text_field :email, class: 'form-control' %> </div> </div> <div class="form-group row"> <%= f.label :phone, class: 'col-sm-3 col-form-label' %> <div class="col-sm-9"> <%= f.text_field :phone, class: 'form-control' %> </div> </div> <div class="form-group row"> <%= f.label :gender, class: 'col-sm-3 col-form-label' %> <div class="col-sm-9"> <%= f.text_field :gender, class: 'form-control' %> </div> </div> <div class="form-group row"> <div class="col-sm-3"> </div> <div class="col-sm-9"> <%= f.submit "Submit", class: 'btn btn-primary' %> </div> </div> <% end %>
کد CSS نیز به صورت زیر خواهد بود:
.form-edit-user{ padding: 30px 100px 10px 50px; .form-group, input{ font-size: 14px; color: black; } input, textarea{ border: 1px solid #efefef; } .col-form-label{ text-align: right; font-weight: 600; } .avatar{ height: 38px; width: 38px; border-radius: 50%; } .username{ font-size: 20px; line-height: 22px; margin-bottom: 0; } .change-photo{ color: #3897f0; text-decoration: none; font-size: 13px; } input[type='submit']{ color: white; } }
اینک UI فرم به صورت زیر آمده است:
زمانی که اطلاعات را در فرم وارد کرده و دکمه submit را برای پردازش بهروزرسانی کاربر بزنید، با خطای عدم مطابقت هیچ مسیر مانند تصویر زیر مواجه میشوید:
این خطا به این جهت رخ میدهد که تا به این جا هیچ مسیری برای کاربر تعریف نکردهایم. اینک باید یک مسیر برای update کاربر به routes.rb اضافه کنیم:
resources:users, only: [:show,:edit,:update]
اینک اگر مجدداً فرم را پر کرده و دکمه submit را بزنیم، با یک خطای دیگر به صورت زیر مواجه میشویم:
اکشن update نمیتواند پیدا شود، بنابراین باید یک اکشن update در UserController به صورت زیر ایجاد کنیم:
def update current_user.update(params[:user]) redirect_to current_user end
ما در اکشن update سعی میکنیم کاربر جاری را بر اساس پارامترهایی که در فرم ویرایش پر شدهاند، بهروزرسانی کنیم. منظور از current_user همان کاربری است که در حال حاضر به اپلیکیشن وارد شده است. اگر بار دیگر تلاش کنیم دکمه submit را بزنیم، با خطایی مانند زیر مواجه میشویم:

Rails از برخی خصوصیات پشتیبانی میکند که به نوشتن اپلیکیشنهای امن کمک میکند. این خصوصیت strong parameters نامیده میشوند که ما را ملزم میکنند تا تعریف کنیم کدام پارامترها در اکشنهای کنترلرها مجاز هستند. ما باید پارامترهای خود را در لیست سفید قرار دهیم تا از انتسابهای جمعی اشتباه جلوگیری کنیم.
بنابراین فعلاً برای اصلاح ForbiddenAttributesError باید از پارامترهای strong پیش از بهروزرسانی کاربر بهره بگیریم. برای بهرهگیری از پارامترهای strong در این مثال باید از متدهای require و permit به صورت زیر استفاده کنیم:
params.require(:user).permit(:username,:name,:website,:bio,:email,:phone,:gender)
این بدان معنی است که پارامترهای username, name, website, bio, email, phone و gender به عنوان پارامترهای معتبر، مجاز هستند. اکنون اکشن update به صورت زیر در آمده است:
def update current_user.update(user_params) redirect_to current_user end private def user_params params.require(:user).permit(:username,:name,:website, :bio,:email,:phone,:gender) End
اینک باید به فرم ویرایش بازگردیم و دوباره فرم را ارائه کنیم. این بار فرم کار میکند و بهروزرسانی کاربر موفق خواهد بود و سپس به صفحه پروفایل کاربر هدایت میشویم.
برای این که بتوانیم تغییر اطلاعات را پس از بهروزرسانی اطلاعات کاربر به سهولت ببینیم، باید به صفحه پروفایل کاربر (users/show.html.erb) بازگردیم تا برخی دادههای جعلی را با وارد کردن دادههای واقعی کاربر عوض کنیم. کد HTML به صورت زیر است:
<div class="col-md-8 basic-info"> <div class="user-wrapper"> <h2 class="username"><%= current_user.username %></h2> ... </div> ... <h2 class="name"><%= current_user.name %></h2> <%=link_to current_user.website, current_user.website, class: 'website' %> </div>
سخن پایانی
در این مقاله به توضیح گام به گام روش ساخت یک اپلیکیشن Rails به کمک PostgreSQL و Bootstrap پرداختیم. همچنین درکی از مفهوم MVC در Rails کسب کردیم. با استفاده از devise gem اقدام به ساخت تابعهای احراز هویت کردیم و در نهایت تابعهایی برای مشاهده و بهروزرسانی پروفایل کاربر نوشتیم. در مقاله بعدی جزییاتی در مورد (Active Record (CRUD، اعتبارسنجی، ارتباط، خصوصیت Active Storage و استفاده از Kaminari gem برای صفحهبندی ارائه میکنیم.
در صورت تمایل برای مطالعه بخش دوم این مطلب، روی لینک زیر کلیک کنید:
اگر این مطلب برای شما مفید بوده است، آموزشهای زیر نیز به شما پیشنهاد میشوند:
- آموزش روبی (Ruby) | راهنمای کامل و رایگان برای شروع به کار — به زبان ساده
- مجموعه آموزشهای برنامهنویسی
- مجموعه آموزشهای طراحی و توسعه پروژه های وب
- ساختمان های داده در روبی (Ruby) — درخت های دودویی
- تنظیم سرورهای محیط توزیع نهایی (Production) برای وب اپلیکیشن — راهنمای مقدماتی
- داده های ساختگی (Dummy data) در روبی، پرل و پایتون — راهنمایی از صفر تا صد
==