5
0
Fork 0
mirror of https://0xacab.org/sutty/sutty synced 2024-11-22 15:06:22 +00:00
panel/app/controllers/posts_controller.rb

331 lines
9.2 KiB
Ruby
Raw Permalink Normal View History

2019-03-26 15:32:20 +00:00
# frozen_string_literal: true
2019-07-03 23:25:23 +00:00
# Controlador para artículos
2018-01-29 22:19:10 +00:00
class PostsController < ApplicationController
include StrongParamsHelper
2019-07-03 23:25:23 +00:00
before_action :authenticate_usuarie!
before_action :service_for_direct_upload, only: %i[new edit]
2018-01-29 22:19:10 +00:00
# TODO: Traer los comunes desde ApplicationController
breadcrumb -> { current_usuarie.email }, :edit_usuarie_registration_path
breadcrumb 'sites.index', :sites_path, match: :exact
breadcrumb -> { site.title }, -> { site_posts_path(site, locale: locale) }, match: :exact
2020-05-23 15:38:03 +00:00
# Las URLs siempre llevan el idioma actual o el de le usuarie
def default_url_options
{ locale: locale }
2020-05-23 15:38:03 +00:00
end
2024-05-17 18:24:08 +00:00
# @todo Mover a tu propio scope
def new_array
@value = pluck_param(:value)
@name = pluck_param(:name)
id = pluck_param(:id)
2024-05-17 18:24:08 +00:00
headers['HX-Trigger-After-Swap'] = 'htmx:resetForm'
render layout: false
end
def new_array_value
@value = pluck_param(:value)
2024-05-17 18:24:08 +00:00
render layout: false
end
2024-05-17 20:58:44 +00:00
def new_related_post
@uuid = pluck_param(:value)
2024-05-17 20:58:44 +00:00
@indexed_post = site.indexed_posts.find_by!(post_id: @uuid)
render layout: false
end
2024-06-03 20:52:53 +00:00
def new_has_one
@uuid = pluck_param(:value)
2024-06-03 20:52:53 +00:00
@indexed_post = site.indexed_posts.find_by!(post_id: @uuid)
render layout: false
end
# El formulario de un Post, si pasamos el UUID, estamos editando, sino
# estamos creando.
def form
uuid = pluck_param(:uuid, optional: true)
locale
@post =
if uuid.present?
site.indexed_posts.find_by!(post_id: uuid).post
else
# @todo Usar la base de datos
site.posts(lang: locale).build(layout: pluck_param(:layout))
end
swap_modals
render layout: false
end
# Genera un modal completo
#
# @todo recibir el atributo anterior
# @param :uuid [String] UUID del post (opcional)
# @param :layout [String] El layout a cargar (opcional)
def modal
uuid = pluck_param(:uuid, optional: true)
locale
# @todo hacer que si el uuid no existe se genera un post, para poder
# pasar el uuid sabiendolo
@post =
if uuid.present?
site.indexed_posts.find_by!(post_id: uuid).post
else
# @todo Usar la base de datos
site.posts(lang: locale).build(layout: pluck_param(:layout))
end
swap_modals
render layout: false
end
2018-01-29 22:19:10 +00:00
def index
authorize Post
2020-05-23 15:38:03 +00:00
# XXX: Cada vez que cambiamos un Post tocamos el sitio con lo que es
# más simple saber si hubo cambios.
2021-10-21 16:25:05 +00:00
return unless stale?([current_usuarie, site, filter_params])
# Todos los artículos de este sitio para el idioma actual
@posts = site.indexed_posts.where(locale: locale)
@posts = @posts.page(filter_params.delete(:page)) if site.pagination
2021-10-21 16:25:05 +00:00
# De este tipo
@posts = @posts.where(layout: filter_params[:layout]) if filter_params[:layout]
# Que estén dentro de la categoría
@posts = @posts.in_category(filter_params[:category]) if filter_params[:category]
# Aplicar los parámetros de búsqueda
@posts = @posts.search(locale, filter_params[:q]) if filter_params[:q].present?
# A los que este usuarie tiene acceso
@posts = PostPolicy::Scope.new(current_usuarie, @posts).resolve
# Filtrar los posts que les invitades no pueden ver
@usuarie = site.usuarie? current_usuarie
2022-07-15 17:17:46 +00:00
@site_stat = SiteStat.new(site)
2018-01-29 22:19:10 +00:00
end
2018-01-30 15:20:19 +00:00
def show
authorize post
breadcrumb post.title.value, ''
fresh_when post
2018-01-30 15:20:19 +00:00
end
2018-01-31 20:29:27 +00:00
# Genera una previsualización del artículo.
def preview
authorize post
render html: post.render
end
def new
2018-09-28 17:15:09 +00:00
authorize Post
@post = site.posts(lang: locale).build(layout: pluck_param(:layout))
breadcrumb I18n.t('loaf.breadcrumbs.posts.new', layout: @post.layout.humanized_name.downcase), ''
end
def create
2018-09-28 17:15:09 +00:00
authorize Post
service = PostService.new(site: site,
2019-08-13 23:33:57 +00:00
usuarie: current_usuarie,
params: params)
@post = service.create_or_update
2018-12-14 15:12:17 +00:00
2024-06-03 20:52:53 +00:00
if post.persisted?
site.touch
forget_content
end
# @todo Enviar la creación a otro endpoint para evitar tantas
# condiciones.
if htmx?
2024-06-03 20:52:53 +00:00
if post.persisted?
triggers = { 'notification:show' => { 'id' => pluck_param(:saved, optional: true) } }
swap_modals(triggers)
2024-06-03 20:52:53 +00:00
@value = post.title.value
@uuid = post.uuid.value
@name = pluck_param(:name)
2024-05-28 16:15:45 +00:00
render render_path_from_attribute, layout: false
else
headers['HX-Retarget'] = "##{pluck_param(:form)}"
headers['HX-Reswap'] = 'outerHTML'
render 'posts/form', layout: false, post: post, site: site, **params.permit(:form, :base, :dir, :locale)
end
2024-06-03 20:52:53 +00:00
elsif post.persisted?
redirect_to site_post_path(site, post)
else
render 'posts/new'
end
end
2018-01-31 20:29:27 +00:00
def edit
authorize post
breadcrumb post.title.value, site_post_path(site, post, locale: locale), match: :exact
breadcrumb 'posts.edit', ''
2018-01-31 20:29:27 +00:00
end
# Este endpoint se encarga de actualizar el post. Si el post se edita
# desde el formulario principal, re-renderizamos el formulario si hay
# errores o enviamos a otro lado al guardar.
#
# Si los datos llegaron por HTMX, hay que regenerar el formulario
# y reemplazarlo en su modal (?) o responder con su tarjeta para
# reemplazarla donde sea que esté.
#
# @todo la re-renderización del formulario no es necesaria si tenemos
# validación client-side.
2018-01-31 20:29:27 +00:00
def update
authorize post
2018-09-28 17:15:09 +00:00
service = PostService.new(site: site,
post: post,
2019-08-16 23:12:22 +00:00
usuarie: current_usuarie,
params: params)
2018-09-28 17:35:40 +00:00
2019-08-16 23:12:22 +00:00
if service.update.persisted?
site.touch
forget_content
end
if htmx?
if post.persisted?
triggers = { 'notification:show' => pluck_param(:saved, optional: true) }
swap_modals(triggers)
@value = post.title.value
@uuid = post.uuid.value
if (result_id = pluck_param(:result_id, optional: true))
headers['HX-Retarget'] = "##{result_id}"
headers['HX-Reswap'] = 'outerHTML'
@indexed_post = site.indexed_posts.find_by_post_id(post.uuid.value)
render 'posts/new_related_post', layout: false
# @todo Confirmar que esta ruta no esté transitada
else
@name = pluck_param(:name)
render render_path_from_attribute, layout: false
end
else
headers['HX-Retarget'] = "##{params.require(:form)}"
headers['HX-Reswap'] = 'outerHTML'
render 'posts/form', layout: false, post: post, site: site, **params.permit(:form, :base, :dir, :locale)
end
elsif post.persisted?
redirect_to site_post_path(site, post)
2018-02-02 22:20:31 +00:00
else
render 'posts/edit'
end
2018-01-31 20:29:27 +00:00
end
2019-05-30 17:33:51 +00:00
# Eliminar artículos
def destroy
authorize post
2019-05-30 17:33:51 +00:00
service = PostService.new(site: site,
post: post,
2019-08-16 23:12:22 +00:00
usuarie: current_usuarie,
params: params)
2019-05-30 17:33:51 +00:00
2019-08-16 23:12:22 +00:00
# TODO: Notificar si se pudo o no
service.destroy
site.touch
redirect_to site_posts_path(site, locale: post.lang.value)
2019-05-30 17:33:51 +00:00
end
2019-10-18 20:35:09 +00:00
# Reordenar los artículos
def reorder
authorize site
2019-10-18 20:35:09 +00:00
service = PostService.new(site: site,
2019-10-18 20:35:09 +00:00
usuarie: current_usuarie,
params: params)
service.reorder
site.touch
2021-05-10 18:18:04 +00:00
redirect_to site_posts_path(site, locale: site.default_locale)
2019-10-18 20:35:09 +00:00
end
2020-03-19 18:31:29 +00:00
# Devuelve el idioma solicitado a través de un parámetro, validando
2020-05-23 15:38:03 +00:00
# que el sitio soporte ese idioma, de lo contrario devuelve el idioma
# actual.
#
# TODO: Debería devolver un error o mostrar una página donde se
# solicite a le usuarie crear el nuevo idioma y que esto lo agregue al
# _config.yml del sitio en lugar de mezclar idiomas.
def locale
@locale ||= site&.locales&.find(-> { site&.default_locale }) do |l|
l.to_s == params[:locale]
2020-05-23 15:38:03 +00:00
end
2020-03-19 18:31:29 +00:00
end
# Instruye al editor a olvidarse el contenido del artículo. Usar
# cuando hayamos guardado la información correctamente.
def forget_content
flash[:js] = { target: 'editor', action: 'forget-content', keys: (params[:storage_keys] || []).to_json }
end
private
# Los parámetros de filtros que vamos a mantener en todas las URLs,
# solo los que no estén vacíos.
#
# @return [Hash]
def filter_params
@filter_params ||= params.permit(:q, :category, :layout, :page).to_hash.select do |_, v|
v.present?
end.transform_keys(&:to_sym)
end
def post
@post ||= site.posts(lang: locale).find(params[:post_id] || params[:id])
end
# Recuerda el nombre del servicio de subida de archivos
def service_for_direct_upload
session[:service_name] = site.name.to_sym
end
# @param triggers [Hash] Otros disparadores
def swap_modals(triggers = {})
params.permit(:show, :hide).each_pair do |key, value|
2024-06-03 20:52:53 +00:00
triggers["modal:#{key}"] = { id: value } if value.present?
end
headers['HX-Trigger'] = triggers.to_json if triggers.present?
end
2024-05-28 16:15:45 +00:00
# @return [String]
def render_path_from_attribute
case pluck_param(:attribute)
2024-05-28 16:15:45 +00:00
when 'new_has_many' then 'posts/new_has_many_value'
when 'new_belongs_to' then 'posts/new_belongs_to_value'
2024-05-28 19:27:39 +00:00
when 'new_has_and_belongs_to_many' then 'posts/new_has_many_value'
2024-06-03 20:52:53 +00:00
when 'new_has_one' then 'posts/new_has_one_value'
2024-05-28 16:15:45 +00:00
else 'nothing'
end
end
2018-01-29 22:19:10 +00:00
end