/ YYYY / MM / Title-Slug URL-Struktur mit Friendly_Id Solution-Drosseln auf # edit
Basierend auf der Anleitung, die ich in meinem @ bek frühere Frage bei der Lösung meinesoriginales Problem beim Implementieren der URL-Struktur "/ YYYY / MM / Slug", Ich hoffe, dass ich Hilfe bei der Behebung eines Fehlers bekomme, der beim Versuch, einen Beitrag zu bearbeiten, angezeigt wird:
Keine Route entspricht [PATCH] "/ blog / 2015/09 / beispiel-post / blog / 2015/09 / beispiel-post"
Hier sind alle fraglichen Dateien (die auf demselben sehr einfachen Gerüst-Blog funktionieren):
$ rails new blog
[...]
$ cd blog
# (Add friendly_id to Gemfile & install)
$ rails generate friendly_id
$ rails generate scaffold post title content slug:string:uniq
[...]
$ rake db:migrate
routes.rbRails.application.routes.draw do
scope 'blog' do
get '', to: 'posts#index', as: 'posts'
post '', to: 'posts#create'
get '/new', to: 'posts#new', as: 'new_post'
get '/:year/:month/:id/edit', to: 'posts#edit', as: 'edit_post'
get '/:year/:month/:id', to: 'posts#show', as: 'post'
patch '/:id', to: 'posts#update'
put '/:id', to: 'posts#update'
delete '/:year/:month/:id', to: 'posts#destroy'
end
end
post.rbclass Post < ActiveRecord::Base
extend FriendlyId
friendly_id :title, use: :slugged
def year
created_at.localtime.strftime("%Y")
end
def month
created_at.localtime.strftime("%m")
end
end
posts_controller.rbclass PostsController < ApplicationController
before_action :set_post, only: [:show, :edit, :update, :destroy]
def index
@posts = Post.all
end
def show
@post = Post.friendly.find(params[:id])
end
def new
@post = Post.new
end
def edit
@post = Post.friendly.find(params[:id])
end
def create
@post = Post.new(post_params)
respond_to do |format|
if @post.save
format.html { redirect_to post_path(@post.year, @post.month, @post), notice: 'Post was successfully created.' }
format.json { render :show, status: :created, location: @post }
else
format.html { render :new }
format.json { render json: @post.errors, status: :unprocessable_entity }
end
end
end
def update
respond_to do |format|
if @post.update(post_params)
format.html { redirect_to post_path, notice: 'Post was successfully updated.' }
format.json { render :show, status: :ok, location: @post }
else
format.html { render :edit }
format.json { render json: @post.errors, status: :unprocessable_entity }
end
end
end
def destroy
@post.destroy
respond_to do |format|
format.html { redirect_to posts_url, notice: 'Post was successfully destroyed.' }
format.json { head :no_content }
end
end
private
# Use callbacks to share common setup or constraints between actions.
def set_post
@post = Post.friendly.find(params[:id])
end
# Never trust parameters from the scary internet, only allow the white list through.
def post_params
params.require(:post).permit(:title, :content, :slug)
end
end
end
posts_helper.rbmodule PostsHelper
def post_path(post)
"blog/#{post.year}/#{post.month}/#{post.slug}"
end
def edit_post_path(post)
"#{post.year}/#{post.month}/#{post.slug}/edit"
end
end
app / views / posts / index.html.erb<p id="notice"><%= notice %></p>
<h1>Listing Posts</h1>
<table>
<thead>
<tr>
<th>Title</th>
<th>Content</th>
<th>Slug</th>
<th colspan="3"></th>
</tr>
</thead>
<tbody>
<% @posts.each do |post| %>
<tr>
<td><%= post.title %></td>
<td><%= post.content %></td>
<td><%= post.slug %></td>
<td><%= link_to 'Show', post_path(post) %></td>
<td><%= link_to 'Edit', edit_post_path(post) %></td>
<td><%= link_to 'Destroy', post, method: :delete, data: { confirm: 'Are you sure?' } %></td>
</tr>
<% end %>
</tbody>
</table>
<br>
<%= link_to 'New Post', new_post_path %>
Zusammenfassend funktioniert Folgendes:
/ blog / index / blog / 2015/09 / beispiel-postErstelle einen neuen Beitrag Zerstöre einen Beitrag… Was nicht geht:
Bearbeiten eines Beitrags (die Bearbeitungsseite mit dem Formular wird gerendert, aber nach dem Senden Ihrer Änderungen wird der oben genannte Fehler angezeigt - und die Änderungen gelangen niemals in die DB.)Ich erkenne, dass dieses Duplizierungsproblem wahrscheinlich mit dieser Überschreibung von edit_post_path zusammenhängt, aber die folgenden Versuche, die korrekte PATCH-Route zu erzwingen, haben keine Auswirkungen:
Update PATCH route topatch '/:year/:month/:id', to: 'posts#update'
Nennen Sie die aktualisierte PATCH-Route zuas: 'patch'
und PATCH-Pfadüberschreibung zu posts_helper hinzufügen:
def patch_path(post)
"#{post.year}/#{post.month}/#{post.slug}"
end
Ändern Sie die Außerkraftsetzung auf:
def patch_path(post)
""
end
Namen aufheben und PATCH-Route in @ ändepatch '', to: 'posts#update'
Betrachtet man den posts_controller, sieht es nicht so aus, als ob das Problem vorliegt, da es nicht die Weiterleitung ist - und ich verstehe nicht, warum@post.update(post_params)
wäre problematisch:
def update
respond_to do |format|
if @post.update(post_params)
format.html { redirect_to @post, notice: 'Post was successfully updated.' }
format.json { render :show, status: :ok, location: @post }
else
format.html { render :edit }
format.json { render json: @post.errors, status: :unprocessable_entity }
end
end
end
So wie ich feststellen kann, geschieht die Duplizierung in der URL vor der PATCH-Aktion, wodurch wir zum EDIT-Ablauf zurückkehren. Die Duplizierung muss an PATCH übergeben werden, wo sie sich verschluckt. Ideen? Danke im Voraus
BEARBEITE
edit.html.erb
<h1>Editing Post</h1>
<%= render 'form' %>
<%= link_to 'Show', @post %> |
<%= link_to 'Back', posts_path %>
_form.html.erb
<%= form_for(@post) do |f| %>
<% if @post.errors.any? %>
<div id="error_explanation">
<h2><%= pluralize(@post.errors.count, "error") %> prohibited this post from being saved:</h2>
<ul>
<% @post.errors.full_messages.each do |message| %>
<li><%= message %></li>
<% end %>
</ul>
</div>
<% end %>
<div class="field">
<%= f.label :title %><br>
<%= f.text_field :title %>
</div>
<div class="field">
<%= f.label :content %><br>
<%= f.text_field :content %>
</div>
<div class="actions">
<%= f.submit %>
</div>
<% end %>