Prerequisite
Ruby 2.6.3
Rails 5.2.4
PostgreSQL 11.5
Prakata
Sebenarnya saya kurang memahami harus memberikan judul seperti apa untuk catatan kali ini.
Tapi, mudah-mudahan judul yang saya berikan saat ini, dapat mewakili isi dari tulisan yang ingin saya dokumentasikan.
Mungkin saya akan mulai dengan memberikan ilustrasi gambar.
Nah, sudah sedikit terbayang kan.
Jadi, dropdown menu tersebut mempunyai fungsi seperti ini:
- Dapat mengganti language dan currency di setiap halaman.
- Berfungsi pada user maupun guest
- Apabila guest melakukan registrasi, maka language atau currency yang mereka pilih juga akan ikut tersimpan.
Permasalahan
Awalnya saya berfikir, “Bagaimana bisa mengganti language dan currency yang merupakan field atau entitas dari tabel users pada setiap halaman?”
Selama ini, saya hanya melakukan action edit pada controller dan view yang bersangkutan dengan dimana field atau entitas tersebut berada.
Misal, pada halaman edit user. Di dalamnya terdapat input field untuk mengganti language dan currency.
Sedangkan, fungsi navbar ini harus dapat melakukan update data saat language dan currency dipilih pada semua halaman di dalam web.
Bagi Junior Rails Developer seperti saya yang masih anak kemarin sore, ini merupakan hal yang sangat baru.
Saking bersemangatnya, rasanya seperti ada tablet Redoxon yang di larutkan di dalam dada. Wkwkwk
Pemecahan Masalah
Sebagai sekenario, saya sudah memiliki tabel users
dengan field locales:string
untuk menyimpan data language dan field rate:string
untuk menyimpan data currency.
Schema
1
2
3
4
5
6
7
8
9
10
create_table "users", force: :cascade do |t|
# ...
# ...
# ...
t.string "locale", default: "en"
t.string "rate", default: "MYR"
# ...
# ...
# ...
end
Controller
Karena locale
dan rate
terdapat pada tabel users
maka kita akan menambahkan kedua controller tersebut untuk users.
Saya akan menggunakan struktur seperti yang pada catatan sebelum-sebelumnya, yaitu “Controller Namespaes and Routing”.1 dengan tujuan untuk memisahkan Admin dan User.
Oh ya, pada catatan kali ini, application_controller.rb
yang akan menghandle users. Tidak seperti catatan sebelumnya yang saya pisahkan. Tujuannya agar proses pencatatan menjadi lebih ringkas. Wkwkwk
Masukkan kedua controller yang akan kita buat ke dalam direktori users/
agar nantinya mudah untuk di-maintain.
Kira-kira seperti ini struktur file dan direktorinya.
├─ app/
│ ├─ assets/
│ ├─ channels/
│ ├─ controllers/
│ │ ├─ admins/
│ │ ├─ concerns/
│ │ ├─ users/
│ │ │ ├─ locales_controller.rb
│ │ │ └─ rates_controller.rb
│ │ ├─ admins_controller.rb
│ │ └─ application_controller.rb
│ ├─ ...
│ ...
├─ ...
...
Berikut ini isi dari file-file tersebut.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
class Users::LocalesController < ApplicationController
before_action :authenticate_user!
def update
if current_user.update(locale_param)
redirect_to request.referer, notice: t('navbar.locale_updated')
else
redirect_to request.referer, notice: t('navbar.locale_failed_updated')
end
end
private
def locale_param
params.permit(:locale)
end
end
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
class Users::RatesController < ApplicationController
before_action :authenticate_user!
def update
if current_user.update(rate_param)
redirect_back request.referer, notice: t('navbar.rate_updated')
else
redirect_back request.referer, notice: t('navbar.rate_failed_updated')
end
end
private
def rate_param
params.permit(:rate)
end
end
Oh iya, karena catatan kali ini berhubungan dengan bahasa (locale), maka saya akan menyinggung sedikit penggunaan Rails Internationalization (I18n) API 2 untuk fungsi language preferences.
Pada dua controller di atas. Saya sudah menggunakan helper i18n pada object :notice
, t('...')
, yang nantinya akan memberikan notifikasi apakah proses perubahan berhasil atau tidak.
Oke langsung saja, saya buat dulu file en.yml
.
1
2
3
4
5
6
en:
navbar:
rate_updated: "Your currency rate has been updated"
rate_failed_update: "Failed to update your currency rate"
locale_updated: "Your default language has been updated"
locale_failed_update: "Failed to update your default language"
Karena saya juga menggunakan bahasa Mandarin, maka saya akan buat juga file ch.yml
.
1
2
3
4
5
6
en:
navbar:
rate_updated: "您的货币汇率已更新"
rate_failed_update: "无法更新您的货币汇率"
locale_updated: "您的默认语言已更新"
locale_failed_update: "无法更新您的默认语言"
Selanjutnya, saya akan mendifinisikan dimana translation load path, permit available locale, dan default locale yang digunakan di config/initializers/locale.rb
.
1
2
3
4
5
6
7
8
# Where the I18n library should search for translation files
I18n.load_path += Dir[Rails.root.join('lib', 'locale', '*.{rb,yml}')]
# Permitted locales available for the application
I18n.available_locales = ['en', 'ch']
# Set default locale to something other than :en
I18n.default_locale = 'en'
Selanjutnya, saya akan mengkonfigurasi locale pada application_controller.rb
untuk menangani semua permintaan terhadap locale.
Oh ya, sekalian untuk rate preferences juga.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
class ApplicationController < ActionController::Base
before_action :set_rate
around_action :set_locale
private
def set_rate
@rate = params[:rate]
end
def set_locale(&action)
locale = current_user.try(:locale) || params[:locale] || I18n.default_locale
I18n.with_locale(locale, &action)
end
def default_url_options
current_user ? {locale: I18n.locale} : {locale: I18n.locale, rate: @rate}
end
end
current_user
adalah object yang disediakan oleh Devise gem untuk user yang sudah loginmelakukan login.
Saya menggunakan helper default_url_options
untuk membuat url form menjadi lebih mudah dibaca.
http://localhost:3000/en/users
Kalau saya tidak menggunakan helper tersebut, maka url secara default akan seperti ini.
http://localhost:3000/users?locale=en
Nah, pasti akan lebih memilih url form yang atas.
Route
Kemudian pada bagian routing, tinggal mengikuti controller.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
Rails.application.routes.draw do
scope "(:locale)", locale: /#{I18n.available_locales.join("|")}/ do
# ...
# ...
namespace :users do
# ...
# ...
resources :rates, only: %w[update]
resources :locales, only: %w[update]
end
end
namespace :admins do
# ...
# ...
end
end
Blok scope "(:locale)", locale: ...
dimaksudkan untuk membuat url form menjadi seperti yang saya sebutkan di atas.
Sehingga blok-blok routing yang lain, harus dimasukkan ke dalam blok ini agar memiliki url form yang sama.
Karena Admin tidak memerlukan url form yang bagus, maka saya keluarkan saja dari blok tersebut.
Selanjutnya, tinggal membuat view template.
View
Seperti biasa, stylesheet pada catatan ini hanya sebagai contoh dan merupakan dummy. Jadi, akan tidak sesuai dengan yang ada pada ilustrasi gambar.
Hal ini dimaksudkan agar kode hanya terkonsentrasi pada blok erb yang berhubungan dengan topik yang sedang saya bahas.
Seperti yang tertulis pada judul, saya akan membuat menu ini pada navigation bar yang biasanya ada pada posisi atas dari halaman.
Struktur file dan direktorinya seperti ini.
├─ app/
│ ├─ assets/
│ ├─ channels/
│ ├─ controllers/
│ ├─ helpers/
│ ├─ jobs/
│ ├─ mailers/
│ ├─ models/
│ └─ views/
│ ├─ admins/
│ ├─ layouts/
│ │ ├─ admins/
│ │ ├─ users/
│ │ │ ├─ navbar/
│ │ │ │ ├─ _locale.html.erb
│ │ │ │ └─ _rate.html.erb
│ │ │ ├─ _flash_message.html.erb
│ │ │ └─ _navbar.html.erb
│ │ ├─ admins.html.erb
│ │ ├─ admins_devise.html.erb
│ │ ├─ application.html.erb
│ │ ├─ mailer.html.erb
│ │ └─ mailer.text.erb
│ └─ users/
├─ ...
...
Saya mulai dari membuat render partial dari navigation bar pada application.html.erb
.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<!DOCTYPE html>
<html>
<head>
<title>BANDITHIJO.COM</title>
<%= csrf_meta_tags %>
<%= csp_meta_tag %>
<meta name="viewport" content="width=device-width, initial-scale=1">
<%= stylesheet_link_tag 'application', media: 'all', 'data-turbolinks-track': 'reload' %>
<%= javascript_include_tag 'application', 'data-turbolinks-track': 'reload' %>
</head>
<body>
<%= render 'layouts/users/navbar' %>
<%= render 'layouts/users/flash_message' %>
<%= yield %>
</body>
</html>
Berikut ini isi dari file _navbar.html.erb
.
Saya membuat render partial untuk blok language dan currency.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
<nav class="navbar navbar-expand-xl">
<!-- Right Menu -->
<div class="collapse navbar-collapse" id="navbar4">
<ul class="navbar-nav ml-auto justify-content-end flex-grow-1 font-family-medium">
<li class="nav-item d-flex">
<!-- Community menu -->
</li>
<li class="nav-item d-flex">
<!-- Promo menu -->
</li>
<li class="nav-item d-flex">
<!-- Help menu -->
</li>
<!-- Dropdown Language Icon -->
<li class="nav-item d-flex dropdown">
<!-- For user -->
<% if user_signed_in? %>
<% unless current_user.locale == "ch" %>
<a class="nav-link align-self-center" href="#" id="dropdown-language" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
<%= image_tag("label/flag/us.svg", width: "20", height: "20") %>
ENG
<span class="icon-dropdown align-middle"></span>
</a>
<% else %>
<a class="nav-link align-self-center" href="#" id="dropdown-language" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
<%= image_tag("label/flag/ch.svg", width: "20", height: "20") %>
CNY
<span class="icon-dropdown"></span>
</a>
<% end %>
<!-- For Guest -->
<% else %>
<% unless current_page?(locale: "ch") %>
<a class="nav-link align-self-center" href="#" id="dropdown-language" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
<%= image_tag("label/flag/us.svg", width: "20", height: "20") %>
ENG
<span class="icon-dropdown"></span>
</a>
<% else %>
<a class="nav-link align-self-center" href="#" id="dropdown-language" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
<%= image_tag("label/flag/ch.svg", width: "20", height: "20") %>
CNY
<span class="icon-dropdown"></span>
</a>
<% end %>
<% end %>
<!-- END Dropdown Language Icon -->
<!-- Content of Dropdown List Language and Currency -->
<div class="dropdown-menu border-0 rounded">
<div class="row no-gutters">
<!-- Language Preferences -->
<%= render 'layouts/users/navbar/locale' %>
<!-- END Language Preferences -->
<!-- Currency Preferences -->
<%= render 'layouts/users/navbar/rate' %>
<!-- END Currency Preferences -->
</div>
</div>
</li>
<!-- END Dropdown Language Icon -->
<!-- Dropdown User Menu -->
...
...
</ul>
</div>
<!-- END Right Menu -->
</nav>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
<div class="col-md-auto border-bottom">
<span>Language</span>
<!-- Change language for user -->
<% if user_signed_in? %>
<%= button_to users_locale_path(current_user, locale: "en"), method: :put, class: "dropdown-item", style: "outline:none" do %>
<% if current_user.locale == "en" %>
<i class="icon-check-selected"></i>
<% else %>
<i class="icon-check-none"></i>
<% end %>
<span class="text-normal">
<%= image_tag("label/flag/us.svg", class: "mr-1", width: "20", height: "20") %>
English
</span>
<% end %>
<%= button_to users_locale_path(current_user, locale: "ch"), method: :put, class: "dropdown-item", style: "outline:none" do %>
<% if current_user.locale == "ch" %>
<i class="icon-check-selected"></i>
<% else %>
<i class="icon-check-none"></i>
<% end %>
<span class="text-normal">
<%= image_tag("label/flag/ch.svg", class: "mr-1", width: "20", height: "20") %>
Chinese
</span>
<% end %>
<!-- Change language for guest -->
<% else %>
<%= link_to({locale: "en"}, class: "dropdown-item #{"active" if current_page?(locale: "en") || current_page?(locale: "")}") do %>
<%= image_tag("label/flag/us.svg", class: "mr-1", width: "20", height: "20") %>
English
<% end %>
<%= link_to({locale: "ch"}, class: "dropdown-item #{"active" if current_page?(locale: "ch")}") do %>
<%= image_tag("label/flag/ch.svg", class: "mr-1", width: "20", height: "20") %>
Chinese
<% end %>
<% end %>
</div>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
<div class="col-md-auto px-2">
<span><%= t("navbar_menu.currency_title") %></span>
<!-- Change currency for user -->
<% if user_signed_in? %>
<%= button_to users_rate_path(current_user), params: {:rate => "MYR"}, method: :put, class: "dropdown-item", style: "outline:none" do %>
<% if current_user.rate == "MYR" %>
<i class="icon-check-selected"></i>
<% else %>
<i class="icon-check-none"></i>
<% end %>
MYR
<span class="text-normal">
Malaysian Ringgit
</span>
<% end %>
<%= button_to users_rate_path(current_user), params: {:rate => "USD"}, method: :put, class: "dropdown-item", style: "outline:none" do %>
<% if current_user.rate == "USD" %>
<i class="icon-check-selected"></i>
<% else %>
<i class="icon-check-none"></i>
<% end %>
USD
<span class="text-normal">
US Dollar
</span>
<% end %>
<%= button_to users_rate_path(current_user), params: {:rate => "CNY"}, method: :put, class: "dropdown-item", style: "outline:none" do %>
<% if current_user.rate == "CNY" %>
<i class="icon-check-selected"></i>
<% else %>
<i class="icon-check-none"></i>
<% end %>
RMB
<span class="text-normal">
Chinese Yuan
</span>
<% end %>
<!-- Change currency for guest -->
<% else %>
<%= link_to({rate: "MYR"}, class: "dropdown-item #{(params[:rate] == 'USD' || params[:rate] == 'CNY') ? '' : 'active'}") do %>
MYR
<span class="text-normal">
Malaysian Ringgit
</span>
<% end %>
<%= link_to({rate: "USD"}, class: "dropdown-item #{params[:rate] == 'USD' ? 'active' : ''}") do %>
USD
<span class="text-normal">
US Dollar
</span>
<% end %>
<%= link_to({rate: "CNY"}, class: "dropdown-item #{params[:rate] == 'CNY' ? 'active' : ''}") do %>
RMB
<span class="text-normal">
Chinese Yuan
</span>
<% end %>
<% end %>
</div>
Selanjutnya, isi dari file _flash_message.html.erb
1
2
3
4
5
6
<% flash.each do |name, msg| %>
<div class="alert bg-<%= name == 'error' ? 'secondary' : 'primary' %> text-center text-white">
<i class="fa fa-exclamation-triangle mr-1"></i><%= msg %>
<span class="close-button fa fa-times fa-2x" aria-hidden="true" data-dismiss="alert"></span>
</div>
<% end %>
Selesai!
Mudah kan?
Kelihatannya saja banyak. Mungkin dikesempatan yang lain akan saya sederhanakan lagi.
Untuk kali ini, seperti ini dulu.
Mudah-mudahan dapat bermanfaat buat teman-teman.
Terima kasih.
(^_^)
Referensi
-
guides.rubyonrails.org/routing.html#controller-namespaces-and-routing
Diakses tanggal: 2019/12/21 -
guides.rubyonrails.org/i18n.html
Diakses tanggal: 2019/12/21
Lisensi
Atribusi-NonKomersial-BerbagiSerupa 4.0 Internasional (CC BY-NC-SA 4.0)
Penulis
My journey kicks off from reading textbooks as a former Medical Student to digging bugs as a Software Engineer – a delightful rollercoaster of career twists. Embracing failure with the grace of a Cat avoiding water, I've seamlessly transitioned from Stethoscope to Keyboard. Armed with ability for learning and adapting faster than a Heart Beat, I'm on a mission to turn Code into a Product.
- Rizqi Nur Assyaufi