以下概述 Action View 中可用的一些最常用的輔助方法。它是一個很好的起點,但也建議查看完整的 API 文件,因為它更詳細地涵蓋了所有輔助方法。
1 格式化
1.1 日期
這些輔助方法有助於以符合上下文的人類可讀形式顯示日期和/或時間元素。
1.1.1 distance_of_time_in_words
報告兩個 Time
或 Date
物件或整數(以秒為單位)之間的大約時間距離。如果您想要更詳細的近似值,請將 include_seconds
設定為 true。
distance_of_time_in_words(Time.current, 15.seconds.from_now)
# => less than a minute
distance_of_time_in_words(Time.current, 15.seconds.from_now, include_seconds: true)
# => less than 20 seconds
我們使用 Time.current
而不是 Time.now
,因為它會根據 Rails 中設定的時區傳回目前時間,而 Time.now
則會根據伺服器的時區傳回 Time 物件。
如需更多資訊,請參閱 distance_of_time_in_words
API 文件。
1.1.2 time_ago_in_words
報告 Time
或 Date
物件,或整數(以秒為單位)與 Time.current
之間的大約時間距離。
time_ago_in_words(3.minutes.from_now) # => 3 minutes
如需更多資訊,請參閱 time_ago_in_words
API 文件。
1.2 數字
一組用於將數字轉換為格式化字串的方法。提供用於電話號碼、貨幣、百分比、精確度、位置表示法和檔案大小的方法。
1.2.1 number_to_currency
將數字格式化為貨幣字串(例如,$13.65)。
number_to_currency(1234567890.50) # => $1,234,567,890.50
如需更多資訊,請參閱 number_to_currency
API 文件。
1.2.2 number_to_human
以美觀的方式列印(格式化和近似)數字,使其更容易被使用者閱讀;對於可能變得非常大的數字很有用。
number_to_human(1234) # => 1.23 Thousand
number_to_human(1234567) # => 1.23 Million
如需更多資訊,請參閱 number_to_human
API 文件。
1.2.3 number_to_human_size
將大小(以位元組為單位)格式化為更容易理解的表示形式;對於向使用者報告檔案大小很有用。
number_to_human_size(1234) # => 1.21 KB
number_to_human_size(1234567) # => 1.18 MB
如需更多資訊,請參閱 number_to_human_size
API 文件。
1.2.4 number_to_percentage
將數字格式化為百分比字串。
number_to_percentage(100, precision: 0) # => 100%
如需更多資訊,請參閱 number_to_percentage
API 文件。
1.2.5 number_to_phone
將數字格式化為電話號碼(預設為美國)。
number_to_phone(1235551234) # => 123-555-1234
如需更多資訊,請參閱 number_to_phone
API 文件。
1.2.6 number_with_delimiter
使用分隔符號格式化具有分組千位的數字。
number_with_delimiter(12345678) # => 12,345,678
如需更多資訊,請參閱 number_with_delimiter
API 文件。
1.2.7 number_with_precision
使用指定的 precision
等級格式化數字,預設值為 3。
number_with_precision(111.2345) # => 111.235
number_with_precision(111.2345, precision: 2) # => 111.23
如需更多資訊,請參閱 number_with_precision
API 文件。
1.3 文字
一組用於篩選、格式化和轉換字串的方法。
1.3.1 excerpt
給定 text
和 phrase
,excerpt
會搜尋並提取第一次出現的 phrase
,以及由 radius
決定的請求周圍文字。如果結果的開始/結束與文字的開始/結束不一致,則會預先/附加省略標記。
excerpt("This is a very beautiful morning", "very", separator: " ", radius: 1)
# => ...a very beautiful...
excerpt("This is also an example", "an", radius: 8, omission: "<chop> ")
#=> <chop> is also an example
如需更多資訊,請參閱 excerpt
API 文件。
1.3.2 pluralize
根據數字的值傳回單數或複數形式的單字。
pluralize(1, "person") # => 1 person
pluralize(2, "person") # => 2 people
pluralize(3, "person", plural: "users") # => 3 users
如需更多資訊,請參閱 pluralize
API 文件。
1.3.3 truncate
將給定的 text
截斷為給定的 length
。如果文字被截斷,則會將省略標記附加到結果中,使其總長度不超過 length
。
truncate("Once upon a time in a world far far away")
# => "Once upon a time in a world..."
truncate("Once upon a time in a world far far away", length: 17)
# => "Once upon a ti..."
truncate("one-two-three-four-five", length: 20, separator: "-")
# => "one-two-three..."
truncate("And they found that many people were sleeping better.", length: 25, omission: "... (continued)")
# => "And they f... (continued)"
truncate("<p>Once upon a time in a world far far away</p>", escape: false)
# => "<p>Once upon a time in a wo..."
如需更多資訊,請參閱 truncate
API 文件。
1.3.4 word_wrap
將文字包裝成不超過 line_width
寬度的行。
word_wrap("Once upon a time", line_width: 8)
# => "Once\nupon a\ntime"
如需更多資訊,請參閱 word_wrap
API 文件。
2 表單
與單獨使用標準 HTML 元素相比,表單輔助方法簡化了使用模型的工作。它們提供了一系列針對基於您的模型產生表單量身定制的方法。某些方法對應於特定類型的輸入,例如文字欄位、密碼欄位、選取下拉式選單等等。提交表單後,表單內的輸入會分組到 params 物件中,並傳送回控制器。
您可以在 Action View 表單輔助方法指南中了解更多關於表單輔助方法的資訊。
3 導覽
一組用於建構依賴路由子系統的連結和 URL 的方法。
3.1 button_to
產生一個表單,該表單會提交到傳遞的 URL。此表單有一個提交按鈕,其值為 name
。
<%= button_to "Sign in", sign_in_path %>
會輸出以下 HTML
<form method="post" action="/sessions" class="button_to">
<input type="submit" value="Sign in" />
</form>
請參閱 button_to
API 文件以獲取更多資訊。
3.2 current_page?
如果目前的請求 URL 與給定的 options
匹配,則返回 true。
<% if current_page?(controller: 'profiles', action: 'show') %>
<strong>Currently on the profile page</strong>
<% end %>
請參閱 current_page?
API 文件以獲取更多資訊。
3.3 link_to
連結到在底層由 url_for
衍生的 URL。它通常用於為 RESTful 資源建立連結,尤其是在將模型作為參數傳遞給 link_to
時。
link_to "Profile", @profile
# => <a href="/profiles/1">Profile</a>
link_to "Book", @book # given a composite primary key [:author_id, :id]
# => <a href="/books/2_1">Book</a>
link_to "Profiles", profiles_path
# => <a href="/profiles">Profiles</a>
link_to nil, "https://example.com"
# => <a href="https://example.com">https://example.com</a>
link_to "Articles", articles_path, id: "articles", class: "article__container"
# => <a href="/articles" class="article__container" id="articles">Articles</a>
如果您的連結目標無法放入 name 參數中,您可以使用程式碼區塊。
<%= link_to @profile do %>
<strong><%= @profile.name %></strong> -- <span>Check it out!</span>
<% end %>
它會輸出以下 HTML
<a href="/profiles/1">
<strong>David</strong> -- <span>Check it out!</span>
</a>
請參閱 link_to
API 文件以獲取更多資訊。
3.4 mail_to
為指定的電子郵件地址產生一個 mailto
連結標籤。您還可以指定連結文字、額外的 HTML 選項以及是否對電子郵件地址進行編碼。
mail_to "john_doe@gmail.com"
# => <a href="mailto:john_doe@gmail.com">john_doe@gmail.com</a>
mail_to "me@john_doe.com", cc: "me@jane_doe.com",
subject: "This is an example email"
# => <a href="mailto:"me@john_doe.com?cc=me@jane_doe.com&subject=This%20is%20an%20example%20email">"me@john_doe.com</a>
請參閱 mail_to
API 文件以獲取更多資訊。
3.5 url_for
返回為提供的 options
設定的 URL。
url_for @profile
# => /profiles/1
url_for [ @hotel, @booking, page: 2, line: 3 ]
# => /hotels/1/bookings/1?line=3&page=2
url_for @post # given a composite primary key [:blog_id, :id]
# => /posts/1_2
4 清理
一組用於清除文字中不需要的 HTML 元素的方法。這些輔助方法對於幫助確保僅呈現安全且有效的 HTML/CSS 特別有用。它也可以透過在您的視圖中呈現使用者輸入之前,對其進行逸出或移除潛在的惡意內容,來防止 XSS 攻擊。
此功能在內部由 rails-html-sanitizer gem 提供支援。
4.1 sanitize
sanitize
方法將會 HTML 編碼所有標籤,並去除所有未明確允許的屬性。
sanitize @article.body
如果傳遞了 :attributes
或 :tags
選項,則僅允許提及的屬性和標籤,其他都不允許。
sanitize @article.body, tags: %w(table tr td), attributes: %w(id class style)
要變更多個用途的預設值,例如,將表格標籤新增至預設值
# config/application.rb
class Application < Rails::Application
config.action_view.sanitized_allowed_tags = %w(table tr td)
end
請參閱 sanitize
API 文件以獲取更多資訊。
4.2 sanitize_css
清理 CSS 程式碼區塊,尤其是在 HTML 內容中遇到 style 屬性時。當處理使用者產生的內容或包含 style 屬性的動態內容時,sanitize_css
特別有用。
下面的 sanitize_css
方法將會移除不允許的樣式。
sanitize_css("background-color: red; color: white; font-size: 16px;")
請參閱 sanitize_css
API 文件以獲取更多資訊。
4.3 strip_links
從文字中去除所有連結標籤,僅留下連結文字。
strip_links("<a href='https://rubyonrails.org'>Ruby on Rails</a>")
# => Ruby on Rails
strip_links("emails to <a href='mailto:me@email.com'>me@email.com</a>.")
# => emails to me@email.com.
strip_links("Blog: <a href='http://myblog.com/'>Visit</a>.")
# => Blog: Visit.
請參閱 strip_links
API 文件以獲取更多資訊。
4.4 strip_tags
從 HTML 中去除所有 HTML 標籤,包括註解和特殊字元。
strip_tags("Strip <i>these</i> tags!")
# => Strip these tags!
strip_tags("<b>Bold</b> no more! <a href='more.html'>See more</a>")
# => Bold no more! See more
strip_links('<<a href="https://example.org">malformed & link</a>')
# => <malformed & link
請參閱 strip_tags
API 文件以獲取更多資訊。
5 資產
一組用於產生 HTML 的方法,該 HTML 將視圖連結到資產,例如影像、JavaScript 檔案、樣式表和資訊訂閱。
預設情況下,Rails 會連結到目前主機 public 資料夾中的這些資產,但是您可以透過在應用程式設定中設定 config.asset_host
,通常在 config/environments/production.rb
中設定,來指示 Rails 從專用的資產伺服器連結到資產。
例如,假設您的資產主機是 assets.example.com
config.asset_host = "assets.example.com"
則 image_tag
的對應 URL 將為
image_tag("rails.png")
# => <img src="//assets.example.com/images/rails.png" />
5.1 audio_tag
產生一個 HTML 音訊標籤,其中包含來源,可以是單一標籤(用於字串來源),或是巢狀來源標籤(用於多個來源)。sources
可以是完整路徑、您 public 音訊目錄中的檔案,或是 Active Storage 附件。
audio_tag("sound")
# => <audio src="/audios/sound"></audio>
audio_tag("sound.wav", "sound.mid")
# => <audio><source src="/audios/sound.wav" /><source src="/audios/sound.mid" /></audio>
audio_tag("sound", controls: true)
# => <audio controls="controls" src="/audios/sound"></audio>
在內部,audio_tag
使用來自 AssetUrlHelpers 的 audio_path
來建置音訊路徑。
請參閱 audio_tag
API 文件以獲取更多資訊。
5.2 auto_discovery_link_tag
返回一個連結標籤,瀏覽器和資訊訂閱讀取器可以使用它來自動偵測 RSS、Atom 或 JSON 資訊訂閱。
auto_discovery_link_tag(:rss, "http://www.example.com/feed.rss", { title: "RSS Feed" })
# => <link rel="alternate" type="application/rss+xml" title="RSS Feed" href="http://www.example.com/feed.rss" />
請參閱 auto_discovery_link_tag
API 文件以獲取更多資訊。
5.3 favicon_link_tag
返回一個由資產管線管理的網站圖示的連結標籤。source
可以是完整路徑,或是存在於您資產目錄中的檔案。
favicon_link_tag
# => <link href="/assets/favicon.ico" rel="icon" type="image/x-icon" />
請參閱 favicon_link_tag
API 文件以獲取更多資訊。
5.4 image_tag
返回來源的 HTML 影像標籤。source
可以是完整路徑,或是存在於您 app/assets/images
目錄中的檔案。
image_tag("icon.png")
# => <img src="/assets/icon.png" />
image_tag("icon.png", size: "16x10", alt: "Edit Article")
# => <img src="/assets/icon.png" width="16" height="10" alt="Edit Article" />
在內部,image_tag
使用來自 AssetUrlHelpers 的 image_path
來建置影像路徑。
請參閱 image_tag
API 文件以獲取更多資訊。
5.5 javascript_include_tag
為提供的每個來源返回一個 HTML script 標籤。您可以傳遞存在於您 app/assets/javascripts
目錄中的 JavaScript 檔案的檔案名稱(.js
副檔名是選用的),以包含到目前頁面中,或者您可以傳遞相對於您文件根目錄的完整路徑。
javascript_include_tag("common")
# => <script src="/assets/common.js"></script>
javascript_include_tag("common", async: true)
# => <script src="/assets/common.js" async="async"></script>
一些最常見的屬性是 async
和 defer
,其中 async
將允許並行載入腳本,並在盡可能的情況下盡快進行剖析和評估,而 defer
將指示該腳本應在文件剖析後執行。
在內部,javascript_include_tag
使用來自 AssetUrlHelpers 的 javascript_path
來建置腳本路徑。
請參閱 javascript_include_tag
API 文件以獲取更多資訊。
5.6 picture_tag
返回來源的 HTML 圖片標籤。它支援傳遞字串、陣列或區塊。
picture_tag("icon.webp", "icon.png")
這會產生以下 HTML
<picture>
<source srcset="/assets/icon.webp" type="image/webp" />
<source srcset="/assets/icon.png" type="image/png" />
<img src="/assets/icon.png" />
</picture>
請參閱 picture_tag
API 文件以獲取更多資訊。
5.7 preload_link_tag
返回瀏覽器可以用來預載來源的連結標籤。來源可以是資產管線管理的資源路徑、完整路徑或 URI。
preload_link_tag("application.css")
# => <link rel="preload" href="/assets/application.css" as="style" type="text/css" />
請參閱 preload_link_tag
API 文件以獲取更多資訊。
5.8 stylesheet_link_tag
為指定為引數的來源返回樣式表連結標籤。如果您未指定副檔名,則會自動附加 .css
。
stylesheet_link_tag("application")
# => <link href="/assets/application.css" rel="stylesheet" />
stylesheet_link_tag("application", media: "all")
# => <link href="/assets/application.css" media="all" rel="stylesheet" />
media
用於指定連結的媒體類型。最常見的媒體類型是 all
、screen
、print
和 speech
。
在內部,stylesheet_link_tag
使用來自 AssetUrlHelpers 的 stylesheet_path
來建置樣式表路徑。
請參閱 stylesheet_link_tag
API 文件以獲取更多資訊。
5.9 video_tag
產生一個 HTML 影片標籤,其中包含來源,可以是單一標籤(用於字串來源),或是巢狀來源標籤(用於多個來源)。sources
可以是完整路徑、您 public 影片目錄中的檔案,或是 Active Storage 附件。
video_tag("trailer")
# => <video src="/videos/trailer"></video>
video_tag(["trailer.ogg", "trailer.flv"])
# => <video><source src="/videos/trailer.ogg" /><source src="/videos/trailer.flv" /></video>
video_tag("trailer", controls: true)
# => <video controls="controls" src="/videos/trailer"></video>
在內部,video_tag
使用來自 AssetUrlHelpers 的 video_path
來建置影片路徑。
請參閱 video_tag
API 文件以獲取更多資訊。
6 JavaScript
一組用於在視圖中處理 JavaScript 的方法。
6.1 escape_javascript
為 JavaScript 片段逸出歸位字元和單引號和雙引號。您可以使用此方法來取得一串文字,並確保當瀏覽器嘗試剖析它時,它不包含任何無效字元。
例如,如果您有一個包含雙引號的問候語的部分,您可以逸出問候語以在 JavaScript 警示中使用。
<%# app/views/users/greeting.html.erb %>
My name is <%= current_user.name %>, and I'm here to say "Welcome to our website!"
<script>
var greeting = "<%= escape_javascript render('users/greeting') %>";
alert(`Hello, ${greeting}`);
</script>
這將正確地逸出引號,並在警示方塊中顯示問候語。
請參閱 escape_javascript
API 文件以獲取更多資訊。
6.2 javascript_tag
返回一個包裝所提供程式碼的 JavaScript 標籤。您可以傳遞一個選項雜湊來控制 <script>
標籤的行為。
javascript_tag("alert('All is good')", type: "application/javascript")
<script type="application/javascript">
//<![CDATA[
alert('All is good')
//]]>
</script>
您也可以使用區塊,而不是將內容作為引數傳遞。
<%= javascript_tag type: "application/javascript" do %>
alert("Welcome to my app!")
<% end %>
請參閱 javascript_tag
API 文件以獲取更多資訊。
7 替代標籤
一組以程式設計方式產生 HTML 標籤的方法。
7.1 tag
使用給定的 name
和 options
產生一個獨立的 HTML 標籤。
每個標籤都可以使用
tag.some_tag_name(optional content, options)
其中標籤名稱可以是例如 br
、div
、section
、article
或任何標籤。
例如,以下是一些常見用法
tag.h1 "All titles fit to print"
# => <h1>All titles fit to print</h1>
tag.div "Hello, world!"
# => <div>Hello, world!</div>
此外,您可以傳遞選項以將屬性新增至產生的標籤。
tag.section class: %w( kitties puppies )
# => <section class="kitties puppies"></section>
此外,可以使用 data
選項將 HTML data-*
屬性傳遞給 tag
輔助方法,其中包含子屬性的鍵/值對雜湊。然後,子屬性會轉換為 data-*
屬性,這些屬性會以破折號分隔,以便與 JavaScript 良好運作。
tag.div data: { user_id: 123 }
# => <div data-user-id="123"></div>
請參閱 tag
API 文件以獲取更多資訊。
7.2 token_list
返回從提供的引數建置的權杖字串。此方法也別名為 class_names
。
token_list("cats", "dogs")
# => "cats dogs"
token_list(nil, false, 123, "", "foo", { bar: true })
# => "123 foo bar"
mobile, alignment = true, "center"
token_list("flex items-#{alignment}", "flex-col": mobile)
# => "flex items-center flex-col"
class_names("flex items-#{alignment}", "flex-col": mobile) # using the alias
# => "flex items-center flex-col"
8 擷取區塊
一組方法,可讓您擷取產生的標記,這些標記可用於樣板或版面配置檔案的其他部分。
它提供一種透過 capture
將區塊擷取到變數中的方法,以及一種透過 content_for
擷取標記區塊以用於版面配置的方法。
8.1 capture
capture
方法可讓您將樣板的一部分擷取到變數中。
<% @greeting = capture do %>
<p>Welcome! The date and time is <%= Time.current %></p>
<% end %>
然後,您可以在樣板、版面配置或輔助方法中的任何位置使用此變數。
<html>
<head>
<title>Welcome!</title>
</head>
<body>
<%= @greeting %>
</body>
</html>
capture 的傳回值是區塊產生的字串。
@greeting
# => "Welcome to my shiny new web page! The date and time is 2018-09-06 11:09:16 -0500"
請參閱 capture
API 文件以獲取更多資訊。
8.2 content_for
呼叫 content_for
會將一段標記儲存在一個識別符中,以便稍後使用。您可以在其他模板、輔助模組或佈局中,透過將識別符作為參數傳遞給 yield
來後續呼叫已儲存的內容。
一個常見的用例是在 content_for
區塊中設定頁面的標題。
您可以在特殊頁面的視圖中定義 content_for
區塊,然後在佈局中 yield
它。對於其他沒有使用 content_for
區塊的頁面,它會保持空白,導致沒有任何內容被 yield。
<%# app/views/users/special_page.html.erb %>
<% content_for(:html_title) { "Special Page Title" } %>
<%# app/views/layouts/application.html.erb %>
<html>
<head>
<title><%= content_for?(:html_title) ? yield(:html_title) : "Default Title" %></title>
</head>
</html>
您會注意到在上面的範例中,我們使用 content_for?
述詞方法來有條件地呈現標題。此方法會檢查是否已使用 content_for
擷取任何內容,讓您可以根據視圖中的內容來調整佈局的某些部分。
此外,您可以在輔助模組中使用 content_for
。
# app/helpers/title_helper.rb
module TitleHelper
def html_title
content_for(:html_title) || "Default Title"
end
end
現在,您可以在佈局中呼叫 html_title
來檢索儲存在 content_for
區塊中的內容。如果正在呈現的頁面(例如 special_page
的情況)上設定了 content_for
區塊,它將會顯示標題。否則,它將會顯示預設文字「Default Title」。
content_for
在快取中會被忽略。因此,您不應將它用於將被快取片段化的元素。
您可能在想 capture
和 content_for
之間有什麼區別?
capture
用於將一段標記擷取到一個變數中,而 content_for
用於將一段標記儲存在一個識別符中,以便稍後使用。在內部,content_for
實際上會呼叫 capture
。然而,關鍵差異在於它們在多次調用時的行為。
content_for
可以重複呼叫,按照它們被提供的順序,將針對特定識別符接收到的區塊串連起來。每次後續的呼叫都會簡單地添加到已儲存的內容中。相反地,capture
只會返回區塊的內容,而不會追蹤任何先前的調用。
請參閱 content_for
API 文件 以獲取更多資訊。
9 效能
9.1 效能測試
將 benchmark
區塊包裝在耗時的操作或可能的瓶頸周圍,以取得該操作的時間讀數。
<% benchmark "Process data files" do %>
<%= expensive_files_operation %>
<% end %>
這會在日誌中添加類似 Process data files (0.34523)
的內容,然後您可以使用它來比較最佳化程式碼時的時間。
這個輔助方法是 Active Support 的一部分,並且也可用於控制器、輔助方法、模型等。
請參閱 benchmark
API 文件 以獲取更多資訊。
9.2 快取
您可以快取視圖的片段,而不是整個動作或頁面。此技術對於快取選單、新聞主題列表、靜態 HTML 片段等很有用。它允許將視圖邏輯的片段包裝在快取區塊中,並在下次請求傳入時從快取儲存區中提供。
cache
方法會接收一個區塊,其中包含您想要快取的內容。
例如,您可以透過將應用程式佈局的頁尾包裝在 cache
區塊中來快取它。
<% cache do %>
<%= render "application/footer" %>
<% end %>
您也可以根據模型實例進行快取,例如,您可以透過將 article
物件傳遞給 cache
方法來快取頁面上的每個文章。這會單獨快取每個文章。
<% @articles.each do |article| %>
<% cache article do %>
<%= render article %>
<% end %>
<% end %>
當您的應用程式收到此頁面的第一個請求時,Rails 會寫入一個具有唯一索引鍵的新快取項目。索引鍵看起來像這樣
views/articles/index:bea67108094918eeba32cd4a6f786301/articles/1
請參閱 Fragment Caching
和 cache
API 文件 以獲取更多資訊。
10 其他
10.1 atom_feed
Atom Feeds 是基於 XML 的檔案格式,用於聯合內容,並且可以被 feed 閱讀器中的使用者使用來瀏覽內容,或被搜尋引擎使用來幫助發現關於您網站的額外資訊。
這個輔助方法讓建立 Atom feed 變得容易,並且主要用於建立 XML 的 Builder 模板中。這是一個完整的使用範例
# config/routes.rb
resources :articles
# app/controllers/articles_controller.rb
def index
@articles = Article.all
respond_to do |format|
format.html
format.atom
end
end
# app/views/articles/index.atom.builder
atom_feed do |feed|
feed.title("Articles Index")
feed.updated(@articles.first.created_at)
@articles.each do |article|
feed.entry(article) do |entry|
entry.title(article.title)
entry.content(article.body, type: "html")
entry.author do |author|
author.name(article.author_name)
end
end
end
end
請參閱 atom_feed
API 文件 以獲取更多資訊。
10.2 debug
返回一個用 pre
標籤包裝的物件的 YAML 表示形式。這會建立一種非常易讀的方式來檢查物件。
my_hash = { "first" => 1, "second" => "two", "third" => [1, 2, 3] }
debug(my_hash)
<pre class="debug_dump">---
first: 1
second: two
third:
- 1
- 2
- 3
</pre>
請參閱 debug
API 文件 以獲取更多資訊。