目前我的路线如下:
resources :posts
我想覆盖'show'动作,以便我可以显示这样的网址:
posts/:id/:slug
我目前可以通过添加自定义match
路线来完成此操作:
resources :posts
match 'posts/:id/:slug' => 'posts#show'
但是,当我使用link_to
帮助程序时,它不会使用我的自定义显示路径。
<%= link_to 'show', post %> # renders /posts/123
如何定义我的节目路线,以便我仍然可以使用link_to
助手?
更新:正如您可以阅读以下答案,您可以覆盖“show”操作的路线,但它可能比它的价值更多。创建自定义路线更容易:
# config/routes.rb
match 'posts/:id/:slug' => 'posts#show', as: 'post_seo'
# app/views/posts/index.html.erb
<%= link_to post.title, post_seo_path(post.id, post.slug) %>
答案 0 :(得分:38)
您有两条指向posts#show
的路线(您应该可以通过运行rake routes
来确认这一点),并且您的链接使用了错误的路线。
当您致电link_to('show', post)
时,通过调用url_for(post)
生成链接的URL(最终,在途中通过其他几种方法后)调用post_path(post)
。由于您拨打posts#show
创建的resources(:posts)
路由名称为post
,因此post_path
生成的路由。
您目前还有针对展示,更新和销毁操作的不一致路线,这可能会在以后导致您遇到问题。
您可以通过将路线更改为以下内容来解决此问题:
resources :posts, :except => ['show', 'update', 'destroy']
get 'posts/:id/:slug' => 'posts#show', :as => 'post'
put 'posts/:id/:slug' => 'posts#update'
delete 'posts/:id/:slug' => 'posts#destroy'
不幸的是,您仍然无法使用link_to('show', post)
,因为它依赖于能够使用post.to_param
作为构建帖子路径所需的单个参数。您的自定义路由需要两个参数,id
和slug
。所以现在您的链接代码需要如下所示:
link_to 'show', post_path(post.id, post.slug)
您可以通过在post_path
中定义自己的post_url
和app/helpers/posts_helper.rb
帮助来解决该问题:
module PostsHelper
def post_path(post, options={})
post_url(post, options.merge(:only_path => true))
end
def post_url(post, options={})
url_for(options.merge(:controller => 'posts', :action => 'show',
:id => post.id, :slug => post.slug))
end
end
这意味着我们终于可以使用:
link_to 'show', post
如果这一切看起来都太多了,那么常见的替代方法是使用看起来更像posts/:id-:slug
的网址,在这种情况下,您可以坚持使用标准的RESTful路由并覆盖to_param
方法在Post
课程中:
def to_param
"#{id}-#{slug}"
end
您还需要做一些工作,将params[:id]
拆分为ID和slug,然后才能在显示,编辑,更新和销毁控制器操作中查找相关实例。
答案 1 :(得分:2)
resources :posts, except: :show do
get ":slug" => :show, as: "", on: :member
end
并定义帮助
def post_path post
"/posts/#{post.id}/#{post.slug}"
end
答案 2 :(得分:0)
分贝/迁移/ add_slug_to_articles.rb
add_column :articles, :slug, :string
add_index :articles, :slug
模型/ article.rb
class Article < ActiveRecord::Base
extend FriendlyId
friendly_id :name, use: :slugged
def should_generate_new_friendly_id?
new_record?
end
end
class Article < ActiveRecord::Base
extend FriendlyId
friendly_id :name, use: :history
end
http://railscasts.com/episodes/314-pretty-urls-with-friendlyid