allow administrators to change the text displayed for each content in the back-office (documentation is coming)

This commit is contained in:
did 2011-06-01 23:43:12 +02:00
parent 4618b35318
commit 91119bf54e
18 changed files with 134 additions and 21 deletions

View File

@ -7,7 +7,7 @@ module Admin::ContentTypesHelper
@content_types = current_site.content_types.ordered. @content_types = current_site.content_types.ordered.
limit(:contents => Locomotive.config.lastest_items_nb). limit(:contents => Locomotive.config.lastest_items_nb).
only(:name, :slug, :highlighted_field_name, :content_custom_fields_version, :order_by).to_a only(:name, :slug, :highlighted_field_name, :content_custom_fields_version, :order_by, :serialized_item_template).to_a
if @content_type && @content_type.persisted? && @content_types.index(@content_type) >= MAX_DISPLAYED_CONTENTS if @content_type && @content_type.persisted? && @content_types.index(@content_type) >= MAX_DISPLAYED_CONTENTS
@content_types.delete(@content_type) @content_types.delete(@content_type)
@ -67,4 +67,23 @@ module Admin::ContentTypesHelper
end end
end end
def content_label_for(content)
if content._parent.item_template.nil?
content._label # default one
else
assigns = {
'site' => current_site,
'content' => content.to_liquid
}
registers = {
:controller => self,
:site => current_site,
:current_admin => current_admin
}
preserve(content._parent.item_template.render(::Liquid::Context.new({}, assigns, registers)))
end
end
end end

View File

@ -40,7 +40,9 @@ module Admin::CustomFieldsHelper
end end
def options_for_association_target def options_for_association_target
current_site.content_types.collect { |c| [c.name, c.content_klass.to_s] } current_site.content_types.collect do |c|
c.persisted? ? [c.name, c.content_klass.to_s] : nil
end.compact
end end
def options_for_has_one(field) def options_for_has_one(field)

View File

@ -2,6 +2,9 @@ class ContentType
include Locomotive::Mongoid::Document include Locomotive::Mongoid::Document
## extensions ##
include Extensions::ContentType::ItemTemplate
## fields ## ## fields ##
field :name field :name
field :description field :description

View File

@ -0,0 +1,57 @@
module Extensions
module ContentType
module ItemTemplate
extend ActiveSupport::Concern
included do
field :raw_item_template
field :serialized_item_template, :type => Binary
before_validation :serialize_item_template
validate :item_template_must_be_valid
end
module InstanceMethods
def item_template
@item_template ||= Marshal.load(read_attribute(:serialized_item_template).to_s) rescue nil
end
protected
def serialize_item_template
if self.new_record? || self.raw_item_template_changed?
@item_parsing_errors = []
begin
self._parse_and_serialize_item_template
rescue ::Liquid::SyntaxError => error
@item_parsing_errors << I18n.t(:liquid_syntax, :error => error.to_s, :scope => [:errors, :messages])
end
end
end
def _parse_and_serialize_item_template
item_template = ::Liquid::Template.parse(self.raw_item_template, {})
self.serialized_item_template = BSON::Binary.new(Marshal.dump(item_template))
end
def item_template_must_be_valid
@item_parsing_errors.try(:each) { |msg| self.errors.add :item_template, msg }
end
# def item_template
# self.read_attribute(:default_item_template) || self.default_item_template
# end
#
# def default_item_template
# '{{ content.highlighted_field_value }}'
# end
end
end
end
end

View File

@ -36,9 +36,9 @@ module Extensions
begin begin
self._parse_and_serialize_template self._parse_and_serialize_template
rescue ::Liquid::SyntaxError => error rescue ::Liquid::SyntaxError => error
@parsing_errors << I18n.t(:liquid_syntax, :fullpath => self.fullpath, :error => error.to_s,:scope => [:errors, :messages]) @parsing_errors << I18n.t(:liquid_syntax, :fullpath => self.fullpath, :error => error.to_s, :scope => [:errors, :messages, :page])
rescue ::Locomotive::Liquid::PageNotFound => error rescue ::Locomotive::Liquid::PageNotFound => error
@parsing_errors << I18n.t(:liquid_extend, :fullpath => self.fullpath,:scope => [:errors, :messages]) @parsing_errors << I18n.t(:liquid_extend, :fullpath => self.fullpath, :scope => [:errors, :messages, :page])
end end
end end
end end

View File

@ -13,6 +13,10 @@
= f.foldable_inputs :name => :presentation, :class => 'switchable' do = f.foldable_inputs :name => :presentation, :class => 'switchable' do
= f.input :highlighted_field_name, :as => :select, :collection => options_for_highlighted_field(f.object, 'contents'), :include_blank => false = f.input :highlighted_field_name, :as => :select, :collection => options_for_highlighted_field(f.object, 'contents'), :include_blank => false
= f.input :group_by_field_name, :as => :select, :collection => options_for_group_by_field(f.object, 'contents') = f.input :group_by_field_name, :as => :select, :collection => options_for_group_by_field(f.object, 'contents')
= f.custom_input :raw_item_template, :css => 'small-code' do
%code{ :class => 'html' }
= f.text_area :raw_item_template, :class => 'small'
= f.errors_on :item_template
= f.foldable_inputs :name => :options, :class => 'switchable' do = f.foldable_inputs :name => :options, :class => 'switchable' do
= f.input :order_by, :as => :select, :collection => options_for_order_by(f.object, 'contents'), :include_blank => false = f.input :order_by, :as => :select, :collection => options_for_order_by(f.object, 'contents'), :include_blank => false

View File

@ -6,7 +6,7 @@
%li.content{ :id => "content-#{content._id}" } %li.content{ :id => "content-#{content._id}" }
%em %em
%strong %strong
= link_to content.send(@content_type.highlighted_field_name), edit_admin_content_path(@content_type.slug, content) = link_to content_label_for(content), edit_admin_content_path(@content_type.slug, content)
.more .more
%span %span
!= t('admin.contents.index.updated_at') != t('admin.contents.index.updated_at')

View File

@ -49,6 +49,7 @@ javascripts:
- public/javascripts/admin/plugins/tiny_mce/tinymce.js - public/javascripts/admin/plugins/tiny_mce/tinymce.js
- public/javascripts/admin/contents.js - public/javascripts/admin/contents.js
content_types: content_types:
- public/javascripts/admin/plugins/codemirror/codemirror.min.js
- public/javascripts/admin/content_types.js - public/javascripts/admin/content_types.js
site: site:
- public/javascripts/admin/site.js - public/javascripts/admin/site.js

View File

@ -32,9 +32,12 @@ de:
protected_page: "Du kannst keine Index oder 404 Seiten löschen" protected_page: "Du kannst keine Index oder 404 Seiten löschen"
extname_changed: "Die neue Datei hat nicht die originale Dateiendung" extname_changed: "Die neue Datei hat nicht die originale Dateiendung"
array_too_short: "ist zu kurz (minimale Element-Zahl ist %{count})" array_too_short: "ist zu kurz (minimale Element-Zahl ist %{count})"
liquid_syntax: "Liquid Syntax-Fehler, bitte überprüfe die Syntax ('%{error}'/'%{fullpath}')" liquid_syntax: "Liquid Syntax-Fehler, bitte überprüfe die Syntax ('%{error}')"
liquid_extend: "Die Seite '%{fullpath}' verwendet eine Vorlage, die gar nicht existiert"
invalid_theme_file: "darf nicht leer sein oder ist keine zip-Datei" invalid_theme_file: "darf nicht leer sein oder ist keine zip-Datei"
page:
liquid_syntax: "Liquid Syntax-Fehler, bitte überprüfe die Syntax ('%{error}'/'%{fullpath}')"
liquid_extend: "Die Seite '%{fullpath}' verwendet eine Vorlage, die gar nicht existiert"
attributes: attributes:
defaults: defaults:

View File

@ -11,9 +11,11 @@ en:
protected_page: "You can not remove index or 404 pages" protected_page: "You can not remove index or 404 pages"
extname_changed: "New file does not have the original extension" extname_changed: "New file does not have the original extension"
array_too_short: "is too small (minimum element number is %{count})" array_too_short: "is too small (minimum element number is %{count})"
liquid_syntax: "Liquid Syntax error ('%{error}' on '%{fullpath}')" liquid_syntax: "Liquid Syntax error ('%{error}')"
liquid_extend: "The page '%{fullpath}' extends a template which does not exist"
invalid_theme_file: "can't be blank or isn't a zip file" invalid_theme_file: "can't be blank or isn't a zip file"
page:
liquid_syntax: "Liquid Syntax error ('%{error}' on '%{fullpath}')"
liquid_extend: "The page '%{fullpath}' extends a template which does not exist"
attributes: attributes:
defaults: defaults:

View File

@ -32,9 +32,11 @@ fr:
protected_page: "Vous ne pouvez pas supprimer les pages index et 404" protected_page: "Vous ne pouvez pas supprimer les pages index et 404"
extname_changed: "Nouveau fichier n'a pas l'extension original" extname_changed: "Nouveau fichier n'a pas l'extension original"
array_too_short: "est trop petit (le nombre minimum d'éléments est %{count})" array_too_short: "est trop petit (le nombre minimum d'éléments est %{count})"
liquid_syntax: "Erreur de syntaxe dans les sections de page, veuillez vérifier la syntaxe ('%{error}'/'%{fullpath}')" liquid_syntax: "Erreur de syntaxe ('%{error}')"
liquid_extend: "La page '%{fullpath}' étend le contenu d'une page qui n'existe pas"
invalid_theme_file: "doit être rempli ou n'est pas un fichier zip" invalid_theme_file: "doit être rempli ou n'est pas un fichier zip"
page:
liquid_syntax: "Erreur de syntaxe dans les sections de page, veuillez vérifier la syntaxe ('%{error}'/'%{fullpath}')"
liquid_extend: "La page '%{fullpath}' étend le contenu d'une page qui n'existe pas"
attributes: attributes:
defaults: defaults:

View File

@ -38,9 +38,11 @@ it:
protected_page: "Non è possibile eliminare le pagine index e 404" protected_page: "Non è possibile eliminare le pagine index e 404"
extname_changed: "Il nuovo file non ha l'estesione originale" extname_changed: "Il nuovo file non ha l'estesione originale"
array_too_short: "troppo corto (il numero minimo di elementi è %{count})" array_too_short: "troppo corto (il numero minimo di elementi è %{count})"
liquid_syntax: "Errore nella sintassi Liquid ('%{error}' in '%{fullpath}')" liquid_syntax: "Errore nella sintassi Liquid ('%{error}')"
liquid_extend: "La pagina '%{fullpath}' estende un template che non esiste"
invalid_theme_file: "non può essere vuoto o non è un file zip" invalid_theme_file: "non può essere vuoto o non è un file zip"
page:
liquid_syntax: "Errore nella sintassi Liquid ('%{error}' in '%{fullpath}')"
liquid_extend: "La pagina '%{fullpath}' estende un template che non esiste"
attributes: attributes:
defaults: defaults:

View File

@ -30,9 +30,11 @@ pt-BR:
protected_page: "Você não pode remover a página inicial e a do erro 404" protected_page: "Você não pode remover a página inicial e a do erro 404"
extname_changed: "Novo arquivos não tem a mesma extensão que o original" extname_changed: "Novo arquivos não tem a mesma extensão que o original"
array_too_short: "é muito pequeno (mínimo de elementos é %{count})" array_too_short: "é muito pequeno (mínimo de elementos é %{count})"
liquid_syntax: "Erro de sintaxe do Liquid, por favor verifique a sintaxe" liquid_syntax: "Erro de sintaxe do Liquid, por favor verifique a sintaxe ('%{error}')"
liquid_extend: "A página extende um template que não existe."
invalid_theme_file: "não pode ser vazio ou não é um arquivo zip" invalid_theme_file: "não pode ser vazio ou não é um arquivo zip"
page:
liquid_syntax: "Erro de sintaxe do Liquid, por favor verifique a sintaxe"
liquid_extend: "A página extende um template que não existe."
date: date:

View File

@ -38,6 +38,7 @@ en:
reset: Reset site reset: Reset site
default_site_template: "Use the default site template. Click <a href='#'>here</a> to upload a site template as a zip file instead." default_site_template: "Use the default site template. Click <a href='#'>here</a> to upload a site template as a zip file instead."
content_type: content_type:
raw_item_template: Item template
api_accounts: Notified Accounts api_accounts: Notified Accounts
account: account:
edit: edit:
@ -78,6 +79,7 @@ en:
samples: "If enabled, the import process will also copy contents and assets" samples: "If enabled, the import process will also copy contents and assets"
reset: "If enabled, all the data of your site will be destroyed before importing the new site" reset: "If enabled, all the data of your site will be destroyed before importing the new site"
content_type: content_type:
raw_item_template: "You can customize the text displayed for each item in the list. Simply use Liquid. Ex: {{ content.name }})"
api_enabled: "It is used to let people from outside to create new instances (example: messages in a contact form)" api_enabled: "It is used to let people from outside to create new instances (example: messages in a contact form)"
api_accounts: "A notification email will be sent to each of the accounts listed above when a new instance is created" api_accounts: "A notification email will be sent to each of the accounts listed above when a new instance is created"

View File

@ -40,6 +40,7 @@ fr:
samples: Copier contenu samples: Copier contenu
reset: Remettre à zéro reset: Remettre à zéro
content_type: content_type:
raw_item_template: Template d'affichage
api_accounts: Comptes à notifier api_accounts: Comptes à notifier
account: account:
edit: edit:
@ -81,5 +82,6 @@ fr:
reset: "Si activé, toutes les données de votre site seront détruites avant l'import du nouveau site" reset: "Si activé, toutes les données de votre site seront détruites avant l'import du nouveau site"
content_type: content_type:
slug: Nom utilisé dans les templates liquid afin d'accéder aux enregistrements de ce modèle slug: Nom utilisé dans les templates liquid afin d'accéder aux enregistrements de ce modèle
raw_item_template: "Personnaliser le texte affiché pour chaque élément de la liste. Utilisez simplement du code Liquid. Ex: {{ content.name }}"
api_enabled: "Utilisé pour autoriser la création de nouvelles instances de l'extérieur (ex.: les messages dans un formulaire de contact)" api_enabled: "Utilisé pour autoriser la création de nouvelles instances de l'extérieur (ex.: les messages dans un formulaire de contact)"
api_accounts: "Un email de notification sera envoyé à chaque compte listé ci-dessus lors de la création d'une nouvelle instance" api_accounts: "Un email de notification sera envoyé à chaque compte listé ci-dessus lors de la création d'une nouvelle instance"

View File

@ -2,7 +2,7 @@ module Locomotive
module Liquid module Liquid
module Drops module Drops
class Content < Base class Content < Base
delegate :meta_keywords, :meta_description, :to => "@source" delegate :meta_keywords, :meta_description, :to => '@source'
def before_method(meth) def before_method(meth)
return '' if @source.nil? return '' if @source.nil?

View File

@ -22,7 +22,7 @@ var addCodeMirrorEditor = function(type, el, parser) {
parser = (parser || 'Liquid') + 'Parser'; parser = (parser || 'Liquid') + 'Parser';
var editor = CodeMirror.fromTextArea(el.attr('id'), { var editor = CodeMirror.fromTextArea(el.attr('id'), {
height: "400px", height: el.hasClass('small') ? '60px' : '400px',
stylesheet: [ stylesheet: [
"/stylesheets/admin/plugins/codemirror/csscolors.css", "/stylesheets/admin/plugins/codemirror/csscolors.css",
"/stylesheets/admin/plugins/codemirror/xmlcolors.css", "/stylesheets/admin/plugins/codemirror/xmlcolors.css",

View File

@ -144,6 +144,18 @@ form.formtastic fieldset ol li.code p.inline-errors {
background-image: none; background-image: none;
} }
form.formtastic fieldset ol li.small-code code { margin: 0px; display: inline-block; width: 75%; }
form.formtastic fieldset ol li.small-code p.inline-errors {
position: relative;
display: block !important;
float: none;
margin-left: 20%;
width: 45%;
margin: 0.5em 0px 0px 20%;
left: 0px;
}
form.formtastic fieldset ol .more { text-align: right; width: auto; margin: 10px 20px 0 0; line-height: 0.6em; } form.formtastic fieldset ol .more { text-align: right; width: auto; margin: 10px 20px 0 0; line-height: 0.6em; }
form.formtastic fieldset ol .more a { text-decoration: none; color: #787A89; font-size: 0.7em; } form.formtastic fieldset ol .more a { text-decoration: none; color: #787A89; font-size: 0.7em; }
form.formtastic fieldset ol .more a:hover { text-decoration: underline; } form.formtastic fieldset ol .more a:hover { text-decoration: underline; }