Le suivant attempt_login
La méthode est appelée à l'aide d'Ajax après l'envoi d'un formulaire de connexion.
class AccessController < ApplicationController
[...]
def attempt_login
authorized_user = User.authenticate(params[:username], params[:password])
if authorized_user
session[:user_id] = authorized_user.id
session[:username] = authorized_user.username
flash[:notice] = "Hello #{authorized_user.name}."
redirect_to(:controller => 'jobs', :action => 'index')
else
[...]
end
end
end
Le problème est que redirect_to
ne fonctionne pas.
Comment résoudriez-vous cela?
Enfin, je viens de remplacer
redirect_to(:controller => 'jobs', :action => 'index')
avec ça:
render :js => "window.location = '/jobs/index'"
et ça marche bien!
Il existe un moyen très simple de conserver le flash pour la prochaine demande. Dans votre contrôleur, faites quelque chose comme
flash[:notice] = 'Your work was awesome! A Unicorn is born!'
flash.keep(:notice)
render js: "window.location = '#{root_path}'"
Le flash.keep
s'assurera que le flash est conservé pour la prochaine demande. Alors quand le root_path
est rendu, il affichera le message flash donné. Rails est génial :)
Je pense que c'est un peu mieux:
render js: "window.location.pathname='#{jobs_path}'"
Dans l'une de mes applications, j'utilise JSON pour continuer les données de redirection et de message flash. Cela ressemblerait à quelque chose comme ceci:
class AccessController < ApplicationController
...
def attempt_login
...
if authorized_user
if request.xhr?
render :json => {
:location => url_for(:controller => 'jobs', :action => 'index'),
:flash => {:notice => "Hello #{authorized_user.name}."}
}
else
redirect_to(:controller => 'jobs', :action => 'index')
end
else
# Render login screen with 422 error code
render :login, :status => :unprocessable_entity
end
end
end
Et un exemple simple de jQuery serait:
$.ajax({
...
type: 'json',
success: functon(data) {
data = $.parseJSON(data);
if (data.location) {
window.location.href = data.location;
}
if (data.flash && data.flash.notice) {
// Maybe display flash message, etc.
}
},
error: function() {
// If login fails, sending 422 error code sends you here.
}
})
Combiner la meilleure des réponses:
...
if request.xhr?
flash[:notice] = "Hello #{authorized_user.name}."
flash.keep(:notice) # Keep flash notice around for the redirect.
render :js => "window.location = #{jobs_path.to_json}"
else
...
def redirect_to(options = {}, response_status = {})
super(options, response_status)
if request.xhr?
# empty to prevent render duplication exception
self.status = nil
self.response_body = nil
path = location
self.location = nil
render :js => "window.location = #{path.to_json}"
end
end
Je ne voulais pas modifier les actions de mon contrôleur, j'ai donc trouvé ce hack:
class ApplicationController < ActionController::Base
def redirect_to options = {}, response_status = {}
super
if request.xhr?
self.status = 200
self.response_body = "<html><body><script>window.location.replace('#{location}')</script></body></html>"
end
end
end