clean translations (a part of it moved to the CustomFields gem) + refactor the custom fields form (mustache) as well as enhance the ui/look&feel + fix some bugs related to the creation/edition of content types and contents

This commit is contained in:
did 2011-03-05 00:29:40 +01:00
parent 2022b08b83
commit a2ea522964
30 changed files with 823 additions and 285 deletions

View File

@ -12,13 +12,11 @@ module Admin
end end
def create def create
@content = @content_type.contents.create(params[:content]) create! { edit_admin_content_url(@content_type.slug, @content.id) }
respond_with(@content, :location => edit_admin_content_url(@content_type.slug, @content))
end end
def update def update
update! { edit_admin_content_url(@content_type.slug, @content) } update! { edit_admin_content_url(@content_type.slug, @content.id) }
end end
def sort def sort

View File

@ -39,6 +39,14 @@ module Admin::BaseHelper
end end
end end
def collection_to_js(collection, options = {})
js = collection.collect { |object| object.to_json }
options_to_js = options.to_json.gsub(/^\{/, '').gsub(/\}$/, '')
"new Object({ \"collection\": [#{js.join(', ')}], #{options_to_js} })"
end
def growl_message def growl_message
if not flash.empty? if not flash.empty?
%{ %{

View File

@ -2,7 +2,7 @@ module Admin::CustomFieldsHelper
def options_for_field_kind def options_for_field_kind
options = %w{string text category boolean date file}.map do |kind| options = %w{string text category boolean date file}.map do |kind|
[t("admin.custom_fields.kind.#{kind}"), kind] [t("custom_fields.kind.#{kind}"), kind]
end end
end end

View File

@ -9,7 +9,7 @@ class AssetCollection
## associations ## ## associations ##
referenced_in :site referenced_in :site
embeds_many :assets embeds_many :assets, :validate => false
## behaviours ## ## behaviours ##
custom_fields_for :assets custom_fields_for :assets

View File

@ -43,20 +43,6 @@ class ContentInstance
self._visible || self._visible.nil? self._visible || self._visible.nil?
end end
def aliased_attributes # TODO: move it to the custom_fields gem
hash = { :created_at => self.created_at, :updated_at => self.updated_at }
self.custom_fields.each do |field|
case field.kind
when 'file' then hash[field._alias] = self.send(field._name.to_sym).url
else
hash[field._alias] = self.send(field._name.to_sym)
end
end
hash
end
def errors_to_hash def errors_to_hash
Hash.new.replace(self.errors) Hash.new.replace(self.errors)
end end

View File

@ -14,12 +14,15 @@ class ContentType
## associations ## ## associations ##
referenced_in :site referenced_in :site
embeds_many :contents, :class_name => 'ContentInstance' do embeds_many :contents, :class_name => 'ContentInstance', :validate => false do
def visible def visible
@target.find_all { |c| c.visible? } @target.find_all { |c| c.visible? }
end end
end end
## named scopes ##
scope :first_by_slug, lambda { |slug| where(:slug => slug) }
## indexes ## ## indexes ##
index [[:site_id, Mongo::ASCENDING], [:slug, Mongo::ASCENDING]] index [[:site_id, Mongo::ASCENDING], [:slug, Mongo::ASCENDING]]

View File

@ -8,15 +8,11 @@
= semantic_form_for @account, :url => admin_accounts_url do |f| = semantic_form_for @account, :url => admin_accounts_url do |f|
= f.foldable_inputs :name => :information do = f.foldable_inputs :name => :information do
= f.input :name, :required => false = f.input :name
= f.foldable_inputs :name => :credentials do = f.foldable_inputs :name => :credentials do
= f.input :email, :required => false = f.input :email
= f.input :password, :input_html => { :autocomplete => "off" }
= f.custom_input :password, :label => :new_password do = f.input :password_confirmation, :input_html => { :autocomplete => "off" }
= f.password_field :password
= f.custom_input :password_confirmation, :label => :new_password_confirmation do
= f.password_field :password_confirmation
= render 'admin/shared/form_actions', :back_url => edit_admin_current_site_url, :button_label => :create = render 'admin/shared/form_actions', :back_url => edit_admin_current_site_url, :button_label => :create

View File

@ -2,7 +2,7 @@
= form.inputs :name => title || :attributes do = form.inputs :name => title || :attributes do
- form.object.custom_fields.each do |field| - form.object.custom_fields.each do |field|
- required = highlighted_field_name == field._name - required = highlighted_field_name == field._name || field.required
- if field.string? - if field.string?
= form.input field._alias.to_sym, :label => field.label, :hint => field.hint, :required => required = form.input field._alias.to_sym, :label => field.label, :hint => field.hint, :required => required

View File

@ -7,4 +7,9 @@
= g.inputs :name => :attributes do = g.inputs :name => :attributes do
= g.input :_alias = g.input :_alias
= g.input :hint = g.input :hint
= g.input :text_formatting, :as => 'select', :collection => options_for_text_formatting, :include_blank => false = g.input :text_formatting, :as => 'select', :collection => options_for_text_formatting, :include_blank => false, :wrapper_html => { :style => 'display: none' }
.popup-actions
%p
%button.button.light{ :type => 'submit' }
%span= t('admin.shared.form_actions.update')

View File

@ -1,61 +1,49 @@
- collection_name = "#{collection_name.singularize}_custom_fields" - collection_name = "#{collection_name.singularize}_custom_fields"
- custom_fields = form.object.send(collection_name.to_sym) - custom_fields = form.object.send(collection_name.to_sym)
- ordered_custom_fields = form.object.send(:"ordered_#{collection_name}") - ordered_custom_fields = form.object.send(:"ordered_#{collection_name}")
- field_klass = "#{form.object.class.name}#{collection_name.classify}".gsub(/CustomField$/, 'Field').constantize
= form.foldable_inputs :name => :custom_fields, :class => 'editable-list fields' do = form.foldable_inputs :name => :custom_fields, :class => 'editable-list fields' do
- ordered_custom_fields.each do |field|
= form.fields_for collection_name.to_sym, field, :child_index => field._index do |g| %script{ :type => 'text/x-mustache-template', :name => 'template', :'data-base-input-name' => "#{form.object.class.name.underscore}[#{collection_name}_attributes]" }
%li{ :class => "item added #{'new' if form.object.new_record?} #{'error' unless field.errors.empty?}"} %li{ :class => "item {{behaviour_flag}} {{new_record_flag}} {{errors_flag}} {{required_flag}}" }
%span.handle %span.handle
= image_tag 'admin/form/icons/drag.png' = image_tag 'admin/form/icons/drag.png'
= g.hidden_field :position, :class => 'position' {{#if_existing_record}}
%input{ :name => '{{base_name}}[id]', :value => '{{{id}}}', :type => 'hidden', :'data-field' => 'id' }
%input{ :name => '{{base_name}}[_destroy]', :value => '0', :type => 'hidden', :'data-field' => '_destroy' }
{{/if_existing_record}}
= g.hidden_field :_alias, :class => 'alias' %input{ :name => '{{base_name}}[position]', :value => '{{{position}}}', :type => 'hidden', :'data-field' => 'position' }
= g.hidden_field :hint, :class => 'hint' %input{ :name => '{{base_name}}[_alias]', :value => '{{{_alias}}}', :type => 'hidden', :'data-field' => '_alias' }
= g.hidden_field :text_formatting, :class => 'text-formatting' %input{ :name => '{{base_name}}[hint]', :value => '{{{hint}}}', :type => 'hidden', :'data-field' => 'hint' }
= g.text_field :label, :class => 'label' %input{ :name => '{{base_name}}[text_formatting]', :value => '{{{text_formatting}}}', :type => 'hidden', :'data-field' => 'text_formatting' }
%input{ :name => '{{base_name}}[label]', :value => '{{{label}}}', :type => 'text', :'data-field' => 'label' }
— —
%em= t("admin.custom_fields.kind.#{field.kind.downcase}") %em {{kind_name}}
= g.select :kind, options_for_field_kind = select_tag '{{base_name}}[kind]', options_for_select(options_for_field_kind), :'data-field' => 'kind'
   
%span.actions %input{ :name => '{{base_name}}[required]', :value => '0', :type => 'hidden', :'data-field' => 'hidden_required' }
= link_to image_tag('admin/form/pen.png'), '#edit-custom-field', :class => 'edit first' %input{ :name => '{{base_name}}[required]', :'{{required_checked}}' => '{{required_checked}}', :value => '1', :type => 'checkbox', :'data-field' => 'required', :id => '{{base_dom_id}}_required' }
= link_to image_tag('admin/form/icons/trash.png'), '#', :class => 'remove', :confirm => t('admin.messages.confirm')
= form.fields_for collection_name.to_sym, custom_fields.build(:label => 'field name', :_alias => ''), :child_index => '-1' do |g| %label{ :for => "{{{base_dom_id}}}_required" }= t('.is_required')
%li{ :class => 'item template' }
%span.handle
= image_tag 'admin/form/icons/drag.png'
= g.hidden_field :position, :class => 'position'
= g.hidden_field :_alias, :class => 'alias'
= g.hidden_field :hint, :class => 'hint'
= g.hidden_field :text_formatting, :class => 'text-formatting'
= g.text_field :label, :class => 'string label void'
—
%em
= g.select :kind, options_for_field_kind
 
%span.actions %span.actions
= link_to image_tag('admin/form/pen.png'), '#edit-custom-field', :class => 'edit first' = link_to image_tag('admin/form/pen.png'), '#edit-custom-field', :class => 'edit first'
= link_to image_tag('admin/form/icons/trash.png'), '#', :class => 'remove', :confirm => t('admin.messages.confirm') = link_to image_tag('admin/form/icons/trash.png'), '#', :class => 'remove', :confirm => t('admin.messages.confirm')
%button{ :class => 'button light add', :type => 'button' } %button{ :class => 'button light add', :type => 'button' }
%span!= t('admin.buttons.new_item') %span!= t('admin.buttons.new_item')
%script{ :type => 'text/javascript', :name => 'data' }
!= collection_to_js(ordered_custom_fields, :template => field_klass.new(:label => t('.default_label'), :_alias => '', :kind => 'string').to_hash)

View File

@ -18,10 +18,8 @@
= f.foldable_inputs :name => :credentials do = f.foldable_inputs :name => :credentials do
= f.input :email = f.input :email
= f.custom_input :password, :label => :new_password do = f.input :password, :input_html => { :autocomplete => "off" }
= f.password_field :password, :autocomplete => "off" = f.input :password_confirmation, :input_html => { :autocomplete => "off" }
= f.custom_input :password_confirmation, :label => :new_password_confirmation do
= f.password_field :password_confirmation, :autocomplete => "off"
= f.foldable_inputs :name => :sites, :class => 'sites off' do = f.foldable_inputs :name => :sites, :class => 'sites off' do
- @account.sites.each do |site| - @account.sites.each do |site|

View File

@ -6,7 +6,6 @@
- else - else
= link_to current_site.name, admin_pages_url, :class => 'single' = link_to current_site.name, admin_pages_url, :class => 'single'
#global-actions-bar #global-actions-bar
!= t('.welcome', :name => link_to(current_admin.name, edit_admin_my_account_url)) != t('.welcome', :name => link_to(current_admin.name, edit_admin_my_account_url))
%span= '|' %span= '|'

View File

@ -25,6 +25,7 @@ javascripts:
- public/javascripts/admin/application.js - public/javascripts/admin/application.js
custom_fields: custom_fields:
- public/javascripts/admin/plugins/fancybox.js - public/javascripts/admin/plugins/fancybox.js
- public/javascripts/admin/plugins/mustache.js
- public/javascripts/admin/custom_fields.js - public/javascripts/admin/custom_fields.js
edit_custom_fields: edit_custom_fields:
- public/javascripts/admin/plugins/tiny_mce/tinymce.js - public/javascripts/admin/plugins/tiny_mce/tinymce.js
@ -33,6 +34,7 @@ javascripts:
- public/javascripts/admin/custom_fields/category.js - public/javascripts/admin/custom_fields/category.js
asset_collections: asset_collections:
- public/javascripts/admin/plugins/fancybox.js - public/javascripts/admin/plugins/fancybox.js
- public/javascripts/admin/plugins/mustache.js
- public/javascripts/admin/custom_fields.js - public/javascripts/admin/custom_fields.js
- public/javascripts/admin/asset_collections.js - public/javascripts/admin/asset_collections.js
assets: assets:

View File

@ -51,13 +51,6 @@ de:
custom_fields: custom_fields:
edit: edit:
title: Benutzerdefinierte Felder bearbeiten title: Benutzerdefinierte Felder bearbeiten
kind:
string: Einfache Texteingabe
text: Text
category: Auswahlbox
boolean: Checkbox
date: Datum
file: Datei
text_formatting: text_formatting:
none: Keine none: Keine
html: HTML html: HTML

View File

@ -51,13 +51,6 @@ en:
custom_fields: custom_fields:
edit: edit:
title: Editing custom field title: Editing custom field
kind:
string: Simple Input
text: Text
category: Select
boolean: Checkbox
date: Date
file: File
text_formatting: text_formatting:
none: None none: None
html: HTML html: HTML
@ -70,6 +63,9 @@ en:
custom_form: custom_form:
edit_categories: Edit options edit_categories: Edit options
delete_file: Delete file delete_file: Delete file
index:
is_required: is required
default_label: Field name
sessions: sessions:
new: new:

View File

@ -51,13 +51,6 @@ fr:
custom_fields: custom_fields:
edit: edit:
title: Editer champ personnalisé title: Editer champ personnalisé
kind:
string: Texte
text: Zone de texte
category: Liste déroulante
boolean: Case à cocher
date: Date
file: Fichier
text_formatting: text_formatting:
none: Aucun none: Aucun
html: HTML html: HTML
@ -70,6 +63,9 @@ fr:
custom_form: custom_form:
edit_categories: Editer options edit_categories: Editer options
delete_file: Supprimer fichier delete_file: Supprimer fichier
index:
is_required: est obligatoire
default_label: Nom du champ
sessions: sessions:
new: new:

View File

@ -51,13 +51,6 @@ pt-BR:
custom_fields: custom_fields:
edit: edit:
title: Editando campo customizado title: Editando campo customizado
kind:
string: Texto Simples
text: Texto
category: Caixa de Seleção
boolean: Checkbox
date: Data
file: Arquivo
text_formatting: text_formatting:
none: Nenhum none: Nenhum
html: HTML html: HTML

View File

@ -48,7 +48,7 @@ de:
other: other:
body: "Inhalte kommen hier rein" body: "Inhalte kommen hier rein"
activemodel: mongoid:
attributes: attributes:
page: page:
title: Titel title: Titel

View File

@ -46,16 +46,20 @@ fr:
title: "Page non trouvée" title: "Page non trouvée"
body: "Contenu de la page d'erreur 404" body: "Contenu de la page d'erreur 404"
other: other:
body: "Le contenu va ici" body: "{% extends 'parent' %}"
activemodel: mongoid:
attributes: attributes:
page: page:
title: Titre title: Titre
parent: Parent parent: Dossier parent
parent_id: Dossier parent
slug: Raccourci slug: Raccourci
listed: Menu
templatized: Templatisée templatized: Templatisée
published: Publiée published: Publiée
redirect: Redirection
redirect_url: Url de redirection
cache_strategy: Cache cache_strategy: Cache
content_type: content_type:
name: Nom name: Nom
@ -72,11 +76,11 @@ fr:
name: Nom name: Nom
source: Fichier source: Fichier
account: account:
email: Email email: E-mail
name: Nom name: Nom
language: Langue language: Langue
new_password: "Nouveau mot de passe" password: "Nouveau mot de passe"
new_password_confirmation: "Confirmation nouveau mot de passe" password_confirmation: "Confirmation nouveau mot de passe"
snippet: snippet:
body: Code body: Code
slug: Raccourci slug: Raccourci
@ -90,6 +94,12 @@ fr:
restricted_access: Activer ? restricted_access: Activer ?
access_login: Identifiant access_login: Identifiant
access_password: "Mot de passe" access_password: "Mot de passe"
custom_fields:
field:
_alias: Alias
hint: Aide
required: Requis ?
text_formatting: Formattage
pagination: pagination:
previous: "« Précédent" previous: "« Précédent"

View File

@ -52,7 +52,7 @@ pt-BR:
other: other:
body: "Conteúdo vai aqui" body: "Conteúdo vai aqui"
activemodel: mongoid:
attributes: attributes:
page: page:
title: Título title: Título
@ -211,4 +211,3 @@ pt-BR:
words_connector: ", " words_connector: ", "
two_words_connector: " e " two_words_connector: " e "
last_word_connector: " e " last_word_connector: " e "

View File

@ -41,16 +41,6 @@ fr:
reset: Remettre à zéro reset: Remettre à zéro
content_type: content_type:
api_accounts: Comptes à notifier api_accounts: Comptes à notifier
page:
title: Titre
parent_id: Dossier
slug: Raccourci URL
published: Publiée
templatized: Templatisée
listed: Menu
redirect: Redirection
redirect_url: Url de redirection
cache_strategy: Cache
hints: hints:
page: page:
@ -78,5 +68,6 @@ fr:
samples: "Si activé, les contenus et les média seront aussi copiés lors de l'import" samples: "Si activé, les contenus et les média seront aussi copiés lors de l'import"
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
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

@ -5,10 +5,11 @@ x pull requests #31 et #32
- bugs - bugs
- editable_elements slug becomes nil - editable_elements slug becomes nil
- uploading videos http://groups.google.com/group/carrierwave/browse_thread/thread/6e211d98f1ff4bc0/51717c2167695ca2?lnk=gst&q=version#51717c2167695ca2 - uploading videos http://groups.google.com/group/carrierwave/browse_thread/thread/6e211d98f1ff4bc0/51717c2167695ca2?lnk=gst&q=version#51717c2167695ca2
- editable_elements not updated (doesn't use index anymore) x editable_elements not updated (doesn't use index anymore)
- custom_fields not deleted (doesn't use index anymore) x custom_fields not deleted (doesn't use index anymore)
- editable_elements: inheritable: false (Mattias) - editable_elements: inheritable: false (Mattias)
- 2 different sites on the same main domain (one in www, the other one in something else) (Raphael Costa) - 2 different sites on the same main domain (one in www, the other one in something else) (Raphael Costa)
- password / new_password
BACKLOG: BACKLOG:

View File

@ -56,6 +56,10 @@ module Locomotive
# Devise # Devise
Devise.mailer_sender = self.config.mailer_sender Devise.mailer_sender = self.config.mailer_sender
# Load all the dynamic classes (custom fields)
ContentType.all.collect(&:fetch_content_klass)
AssetCollection.all.collect(&:fetch_asset_klass)
end end
def self.logger(message) def self.logger(message)

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.0 KiB

After

Width:  |  Height:  |  Size: 345 B

View File

@ -16,15 +16,15 @@ $(document).ready(function() {
// api enabled ? // api enabled ?
console.log('subscribing...'); // console.log('subscribing...');
$.subscribe('toggle.content_type_api_enabled.checked', function(event, data) { $.subscribe('toggle.content_type_api_enabled.checked', function(event, data) {
console.log('checked'); // console.log('checked');
$('#content_type_api_accounts_input').show(); $('#content_type_api_accounts_input').show();
}, []); }, []);
$.subscribe('toggle.content_type_api_enabled.unchecked', function(event, data) { $.subscribe('toggle.content_type_api_enabled.unchecked', function(event, data) {
console.log('unchecked'); // console.log('unchecked');
$('#content_type_api_accounts_input').hide(); $('#content_type_api_accounts_input').hide();
}, []); }, []);

View File

@ -1,155 +1,360 @@
$(document).ready(function() { $(document).ready(function() {
$('fieldset.fields').parents('form').submit(function() { var wrapper = $('fieldset.editable-list.fields');
$('fieldset.fields li.template input, fieldset.fields li.template select').attr('disabled', 'disabled'); var list = wrapper.find('ol');
var template = wrapper.find('script[name=template]').html();
var baseInputName = wrapper.find('script[name=template]').attr('data-base-input-name');
var data = eval(wrapper.find('script[name=data]').html());
var index = 0;
var domFieldVal = function(domField, fieldName, val) {
var el = domField.find('input[data-field=' + fieldName + '], select[data-field=' + fieldName + ']');
return (typeof(val) == 'undefined' ? el.val() : el.val(val));
}
var domBoxAttr = function(fieldName) {
return $('#fancybox-wrap form').find('input[name=custom_fields_field[' + fieldName + ']], select[name=custom_fields_field[' + fieldName + ']]');
}
var domBoxAttrVal = function(fieldName, val) {
return (typeof(val) == 'undefined' ? domBoxAttr(fieldName).val() : domBoxAttr(fieldName).val(val));
}
/* ___ Register all the different events when a field is added (destroy, edit details, ...etc) ___ */
var registerFieldEvents = function(field, domField) {
// select
domField.find('em').click(function() {
$(this).hide();
$(this).next().show();
}); });
var defaultValue = $('fieldset.fields li.template input[type=text]').val(); domField.find('select').each(function() {
var selectOnChange = function(select) { var select = $(this);
select.hover(function() {
clearTimeout($.data(select, 'timer'));
},
function() {
$.data(select, 'timer', setTimeout(function() {
select.hide(); select.hide();
select.prev() select.prev().show();
}, 1000));
}).change(function() {
selectOnChange(select);
});
});
// checkbox
domField.find('input[type=checkbox]').click(function() { domField.toggleClass('required'); });
// edit
domField.find('a.edit').click(function(e) {
var link = $(this);
var attributes = ['_alias', 'hint', 'text_formatting'];
$.fancybox({
titleShow: false,
content: $(link.attr('href')).parent().html(),
padding: 0,
onComplete: function() {
$('#fancybox-wrap .actions button[type=submit]').click(function(e) {
$.each(attributes, function(index, name) {
var val = domBoxAttrVal(name).trim();
if (val != '') domFieldVal(domField, name, val);
});
domBoxAttr('text_formatting').parent().hide();
$.fancybox.close();
e.preventDefault(); e.stopPropagation();
});
// copy current val to the form in the box
$.each(attributes, function(index, name) {
var val = domFieldVal(domField, name).trim();
if (val == '' && name == '_alias') val = makeSlug(domFieldVal(domField, 'label'));
domBoxAttrVal(name, val);
});
if (domFieldVal(domField, 'kind').toLowerCase() == 'text') domBoxAttr('text_formatting').parents('li').show();
}
});
e.preventDefault(); e.stopPropagation();
});
// remove
domField.find('a.remove').click(function(e) {
if (confirm($(this).attr('data-confirm'))) {
if (field.new_record)
domField.remove();
else
domField.hide().find('input[data-field=_destroy]').val(1);
refreshPosition();
}
e.preventDefault(); e.stopPropagation();
});
}
var registerFieldTemplateEvents = function(domField) {
// checkbox
domField.find('input[type=checkbox]').click(function() { domField.toggleClass('required'); });
var labelDom = domField.find('input[data-field=label]').focus(function() {
if ($(this).val() == data.template.label) $(this).val('');
}).focusout(function() {
if ($(this).val() == '') $(this).val(data.template.label);
});
var kindDom = domField.find('select[data-field=kind]');
var requiredDom = domField.find('input[data-field=required]');
// bind the "Add field" button
domField.find('button').click(function(e) {
var newField = $.extend({}, data.template);
newField._alias = '';
newField.label = labelDom.val().trim();
newField.kind = kindDom.val();
newField.required = requiredDom.is(':checked');
if (newField.label == '' || newField.label == data.template.label) return false;
// reset template values
labelDom.val(data.template.label);
kindDom.val(data.template.kind);
requiredDom.attr('checked', '');
domField.removeClass('required');
addField(newField, { refreshPosition: true });
e.preventDefault(); e.stopPropagation();
});
}
var refreshPosition = function() {
$.each(list.find('li.added:visible input[data-field=position]'), function(index) { $(this).val(index); });
}
var selectOnChange = function(select) {
select.hide().prev()
.show() .show()
.html(select[0].options[select[0].options.selectedIndex].text); .html(select[0].options[select[0].options.selectedIndex].text);
} }
var refreshPosition = function() { /* ___ Add a field in the list of fields ___ */
jQuery.each($('fieldset.fields li.added input.position'), function(index) { var addField = function(field, options) {
$(this).val(index); options = $.extend({
}); 'is_template': false,
'refreshPosition': false
}, options);
field = $.extend({
behaviour_flag: function() { return options.is_template ? 'template' : 'added' },
new_record_flag: function() { return this.new_record == true && options.is_template == false ? 'new' : '' },
errors_flag: function() { return this.errors.length > 0 ? 'error' : '' },
required_flag: function() { return this.required ? 'required' : ''; },
base_name: function() { return options.is_template ? '' : baseInputName + "[" + index + "]"; },
base_dom_id: function() { return options.is_template ? 'custom_field_template' : 'custom_field_' + index; },
required_checked: function() { return this.required ? 'checked' : ''; },
if_existing_record: function() { return this.new_record == false }
}, field);
var html = Mustache.to_html(template, field);
var domField = null;
if (options.is_template) {
domField = list.append(html).find('.template');
registerFieldTemplateEvents(domField);
} }
/* __ fields ___ */
$('fieldset.fields li.added select').each(function() {
var select = $(this)
.hover(function() {
clearTimeout(select.attr('timer'));
}, function() {
select.attr('timer', setTimeout(function() {
select.hide();
select.prev().show();
}, 1000));
})
.change(function() { selectOnChange(select); });
select.prev().click(function() {
$(this).hide();
select.show();
});
});
$('fieldset.fields li.template input[type=text]').focus(function() {
if ($(this).hasClass('void') && $(this).parents('li').hasClass('template'))
$(this).val('').removeClass('void');
});
$('fieldset.fields li.template button').click(function() {
var lastRow = $(this).parents('li.template');
var label = lastRow.find('input.label').val().trim();
if (label == '' || label == defaultValue) return false;
var newRow = lastRow.clone(true).removeClass('template').addClass('added new').insertBefore(lastRow);
var dateFragment = '[' + new Date().getTime() + ']';
newRow.find('input, select').each(function(index) {
$(this).attr('name', $(this).attr('name').replace('[-1]', dateFragment));
});
var select = newRow.find('select')
.val(lastRow.find('select').val())
.change(function() { selectOnChange(select); })
.hover(function() {
clearTimeout(select.attr('timer'));
}, function() {
select.attr('timer', setTimeout(function() {
select.hide();
select.prev().show();
}, 1000));
});
select.prev()
.html(select[0].options[select[0].options.selectedIndex].text)
.click(function() {
$(this).hide();
select.show();
});
// then "reset" the form
lastRow.find('input.label').val(defaultValue).addClass('void');
// warn the sortable widget about the new row
$("fieldset.fields ol").sortable('refresh');
refreshPosition();
});
$('fieldset.fields li a.remove').click(function(e) {
if (confirm($(this).attr('data-confirm'))) {
var parent = $(this).parents('li');
if (parent.hasClass('new'))
parent.remove();
else { else {
var field = parent.find('input.position') domField = list.find('> .template').before(html).prev('li');
field.attr('name', field.attr('name').replace('[position]', '[_destroy]'));
parent.hide().removeClass('added') registerFieldEvents(field, domField);
list.sortable('refresh');
if (options.refreshPosition) refreshPosition();
index++;
} }
refreshPosition(); domField.find('select').val(field.kind);
domField.find('em').html(domField.find('select option:selected').text());
} }
e.preventDefault(); /* ___ SETUP ___ */
e.stopPropagation(); var setup = function() {
});
// sortable list // sortable list
$("fieldset.fields ol").sortable({ list.sortable({
handle: 'span.handle', handle: 'span.handle',
items: 'li:not(.template)', items: 'li:not(.template)',
axis: 'y', axis: 'y',
update: refreshPosition update: refreshPosition
}); });
// edit in depth custom field // add the template field used to insert the new ones
$('fieldset.fields li.item span.actions a.edit').click(function() { addField(data.template, { is_template: true });
var link = $(this);
$.fancybox({
titleShow: false,
content: $(link.attr('href')).parent().html(),
onComplete: function() {
$('#fancybox-wrap form').submit(function(e) {
$.fancybox.close();
e.preventDefault();
e.stopPropagation();
});
var parent = link.parent(); // add the existing fields (if present)
for (var i = 0; i < data.collection.length; i++) {
if (parent.prevAll('select').val() == 'Text') { addField(data.collection[i]);
var formatting = parent.prevAll('.text-formatting').val(); }
$('#fancybox-wrap #custom_fields_field_text_formatting').val(formatting);
$('#fancybox-wrap #custom_fields_field_text_formatting_input').show();
} else {
$('#fancybox-wrap #custom_fields_field_text_formatting_input').hide();
} }
var alias = parent.prevAll('.alias').val().trim(); setup(); // <- let's the show begin
if (alias == '') alias = makeSlug(link.parent().prevAll('.label').val());
$('#fancybox-wrap #custom_fields_field__alias').val(alias);
var hint = parent.prevAll('.hint').val();
$('#fancybox-wrap #custom_fields_field_hint').val(hint);
},
onCleanup: function() {
var parent = link.parent();
var alias = $('#fancybox-wrap #custom_fields_field__alias').val().trim();
if (alias != '') parent.prevAll('.alias').val(alias);
var hint = $('#fancybox-wrap #custom_fields_field_hint').val().trim();
if (hint != '') parent.prevAll('.hint').val(hint);
var formatting = $('#fancybox-wrap #custom_fields_field_text_formatting').val();
parent.prevAll('.text-formatting').val(formatting);
}
})
});
}); });
// $(document).ready(function() {
//
// $('fieldset.fields').parents('form').submit(function() {
// $('fieldset.fields li.template input, fieldset.fields li.template select').attr('disabled', 'disabled');
// });
//
// var defaultValue = $('fieldset.fields li.template input[type=text]').val();
// var selectOnChange = function(select) {
// select.hide();
// select.prev()
// .show()
// .html(select[0].options[select[0].options.selectedIndex].text);
// }
//
// var refreshPosition = function() {
// jQuery.each($('fieldset.fields li.added input.position'), function(index) {
// $(this).val(index);
// });
// }
//
// /* __ fields ___ */
// $('fieldset.fields li.added select').each(function() {
// var select = $(this)
// .hover(function() {
// clearTimeout(select.attr('timer'));
// }, function() {
// select.attr('timer', setTimeout(function() {
// select.hide();
// select.prev().show();
// }, 1000));
// })
// .change(function() { selectOnChange(select); });
//
// select.prev().click(function() {
// $(this).hide();
// select.show();
// });
// });
//
// $('fieldset.fields li.template input[type=text]').focus(function() {
// if ($(this).hasClass('void') && $(this).parents('li').hasClass('template'))
// $(this).val('').removeClass('void');
// });
//
// $('fieldset.fields li.template button').click(function() {
// var lastRow = $(this).parents('li.template');
//
// var label = lastRow.find('input.label').val().trim();
// if (label == '' || label == defaultValue) return false;
//
// var newRow = lastRow.clone(true).removeClass('template').addClass('added new').insertBefore(lastRow);
//
// var dateFragment = '[' + new Date().getTime() + ']';
// newRow.find('input, select').each(function(index) {
// $(this).attr('name', $(this).attr('name').replace('[-1]', dateFragment));
// });
//
// var select = newRow.find('select')
// .val(lastRow.find('select').val())
// .change(function() { selectOnChange(select); })
// .hover(function() {
// clearTimeout(select.attr('timer'));
// }, function() {
// select.attr('timer', setTimeout(function() {
// select.hide();
// select.prev().show();
// }, 1000));
// });
// select.prev()
// .html(select[0].options[select[0].options.selectedIndex].text)
// .click(function() {
// $(this).hide();
// select.show();
// });
//
// // then "reset" the form
// lastRow.find('input.label').val(defaultValue).addClass('void');
//
// // warn the sortable widget about the new row
// $("fieldset.fields ol").sortable('refresh');
//
// refreshPosition();
// });
//
// $('fieldset.fields li a.remove').click(function(e) {
// if (confirm($(this).attr('data-confirm'))) {
// var parent = $(this).parents('li');
//
// if (parent.hasClass('new'))
// parent.remove();
// else {
// var field = parent.find('input.position')
// field.attr('name', field.attr('name').replace('[position]', '[_destroy]'));
//
// parent.hide().removeClass('added')
// }
//
// refreshPosition();
// }
//
// e.preventDefault();
// e.stopPropagation();
// });
//
// // sortable list
// $("fieldset.fields ol").sortable({
// handle: 'span.handle',
// items: 'li:not(.template)',
// axis: 'y',
// update: refreshPosition
// });
//
// // edit in depth custom field
// $('fieldset.fields li.item span.actions a.edit').click(function() {
// var link = $(this);
// $.fancybox({
// titleShow: false,
// content: $(link.attr('href')).parent().html(),
// onComplete: function() {
// $('#fancybox-wrap form').submit(function(e) {
// $.fancybox.close();
// e.preventDefault();
// e.stopPropagation();
// });
//
// var parent = link.parent();
//
// if (parent.prevAll('select').val() == 'Text') {
// var formatting = parent.prevAll('.text-formatting').val();
// $('#fancybox-wrap #custom_fields_field_text_formatting').val(formatting);
// $('#fancybox-wrap #custom_fields_field_text_formatting_input').show();
// } else {
// $('#fancybox-wrap #custom_fields_field_text_formatting_input').hide();
// }
//
// var alias = parent.prevAll('.alias').val().trim();
// if (alias == '') alias = makeSlug(link.parent().prevAll('.label').val());
// $('#fancybox-wrap #custom_fields_field__alias').val(alias);
//
// var hint = parent.prevAll('.hint').val();
// $('#fancybox-wrap #custom_fields_field_hint').val(hint);
// },
// onCleanup: function() {
// var parent = link.parent();
//
// var alias = $('#fancybox-wrap #custom_fields_field__alias').val().trim();
// if (alias != '') parent.prevAll('.alias').val(alias);
// var hint = $('#fancybox-wrap #custom_fields_field_hint').val().trim();
// if (hint != '') parent.prevAll('.hint').val(hint);
// var formatting = $('#fancybox-wrap #custom_fields_field_text_formatting').val();
// parent.prevAll('.text-formatting').val(formatting);
// }
// })
// });
// });

View File

@ -0,0 +1,331 @@
/*
mustache.js Logic-less templates in JavaScript
See http://mustache.github.com/ for more info.
*/
var Mustache = function() {
var Renderer = function() {};
Renderer.prototype = {
otag: "{{",
ctag: "}}",
pragmas: {},
buffer: [],
pragmas_implemented: {
"IMPLICIT-ITERATOR": true
},
context: {},
render: function(template, context, partials, in_recursion) {
// reset buffer & set context
if(!in_recursion) {
this.context = context;
this.buffer = []; // TODO: make this non-lazy
}
// fail fast
if(!this.includes("", template)) {
if(in_recursion) {
return template;
} else {
this.send(template);
return;
}
}
template = this.render_pragmas(template);
var html = this.render_section(template, context, partials);
if(in_recursion) {
return this.render_tags(html, context, partials, in_recursion);
}
this.render_tags(html, context, partials, in_recursion);
},
/*
Sends parsed lines
*/
send: function(line) {
if(line != "") {
this.buffer.push(line);
}
},
/*
Looks for %PRAGMAS
*/
render_pragmas: function(template) {
// no pragmas
if(!this.includes("%", template)) {
return template;
}
var that = this;
var regex = new RegExp(this.otag + "%([\\w-]+) ?([\\w]+=[\\w]+)?" +
this.ctag);
return template.replace(regex, function(match, pragma, options) {
if(!that.pragmas_implemented[pragma]) {
throw({message:
"This implementation of mustache doesn't understand the '" +
pragma + "' pragma"});
}
that.pragmas[pragma] = {};
if(options) {
var opts = options.split("=");
that.pragmas[pragma][opts[0]] = opts[1];
}
return "";
// ignore unknown pragmas silently
});
},
/*
Tries to find a partial in the curent scope and render it
*/
render_partial: function(name, context, partials) {
name = this.trim(name);
if(!partials || partials[name] === undefined) {
throw({message: "unknown_partial '" + name + "'"});
}
if(typeof(context[name]) != "object") {
return this.render(partials[name], context, partials, true);
}
return this.render(partials[name], context[name], partials, true);
},
/*
Renders inverted (^) and normal (#) sections
*/
render_section: function(template, context, partials) {
if(!this.includes("#", template) && !this.includes("^", template)) {
return template;
}
var that = this;
// CSW - Added "+?" so it finds the tighest bound, not the widest
var regex = new RegExp(this.otag + "(\\^|\\#)\\s*(.+)\\s*" + this.ctag +
"\n*([\\s\\S]+?)" + this.otag + "\\/\\s*\\2\\s*" + this.ctag +
"\\s*", "mg");
// for each {{#foo}}{{/foo}} section do...
return template.replace(regex, function(match, type, name, content) {
var value = that.find(name, context);
if(type == "^") { // inverted section
if(!value || that.is_array(value) && value.length === 0) {
// false or empty list, render it
return that.render(content, context, partials, true);
} else {
return "";
}
} else if(type == "#") { // normal section
if(that.is_array(value)) { // Enumerable, Let's loop!
return that.map(value, function(row) {
return that.render(content, that.create_context(row),
partials, true);
}).join("");
} else if(that.is_object(value)) { // Object, Use it as subcontext!
return that.render(content, that.create_context(value),
partials, true);
} else if(typeof value === "function") {
// higher order section
return value.call(context, content, function(text) {
return that.render(text, context, partials, true);
});
} else if(value) { // boolean section
return that.render(content, context, partials, true);
} else {
return "";
}
}
});
},
/*
Replace {{foo}} and friends with values from our view
*/
render_tags: function(template, context, partials, in_recursion) {
// tit for tat
var that = this;
var new_regex = function() {
return new RegExp(that.otag + "(=|!|>|\\{|%)?([^\\/#\\^]+?)\\1?" +
that.ctag + "+", "g");
};
var regex = new_regex();
var tag_replace_callback = function(match, operator, name) {
switch(operator) {
case "!": // ignore comments
return "";
case "=": // set new delimiters, rebuild the replace regexp
that.set_delimiters(name);
regex = new_regex();
return "";
case ">": // render partial
return that.render_partial(name, context, partials);
case "{": // the triple mustache is unescaped
return that.find(name, context);
default: // escape the value
return that.escape(that.find(name, context));
}
};
var lines = template.split("\n");
for(var i = 0; i < lines.length; i++) {
lines[i] = lines[i].replace(regex, tag_replace_callback, this);
if(!in_recursion) {
this.send(lines[i]);
}
}
if(in_recursion) {
return lines.join("\n");
}
},
set_delimiters: function(delimiters) {
var dels = delimiters.split(" ");
this.otag = this.escape_regex(dels[0]);
this.ctag = this.escape_regex(dels[1]);
},
escape_regex: function(text) {
// thank you Simon Willison
if(!arguments.callee.sRE) {
var specials = [
'/', '.', '*', '+', '?', '|',
'(', ')', '[', ']', '{', '}', '\\'
];
arguments.callee.sRE = new RegExp(
'(\\' + specials.join('|\\') + ')', 'g'
);
}
return text.replace(arguments.callee.sRE, '\\$1');
},
/*
find `name` in current `context`. That is find me a value
from the view object
*/
find: function(name, context) {
name = this.trim(name);
// Checks whether a value is thruthy or false or 0
function is_kinda_truthy(bool) {
return bool === false || bool === 0 || bool;
}
var value = context;
var path = name.split(/\./);
for(var i = 0; i < path.length; i++) {
name = path[i];
if(value && is_kinda_truthy(value[name])) {
value = value[name];
} else if(i == 0 && is_kinda_truthy(this.context[name])) {
value = this.context[name];
} else {
value = undefined;
}
}
if(typeof value === "function") {
return value.apply(context);
}
if(value !== undefined) {
return value;
}
// silently ignore unkown variables
return "";
},
// Utility methods
/* includes tag */
includes: function(needle, haystack) {
return haystack.indexOf(this.otag + needle) != -1;
},
/*
Does away with nasty characters
*/
escape: function(s) {
s = String(s === null ? "" : s);
return s.replace(/&(?!\w+;)|["'<>\\]/g, function(s) {
switch(s) {
case "&": return "&amp;";
case "\\": return "\\\\";
case '"': return '&quot;';
case "'": return '&#39;';
case "<": return "&lt;";
case ">": return "&gt;";
default: return s;
}
});
},
// by @langalex, support for arrays of strings
create_context: function(_context) {
if(this.is_object(_context)) {
return _context;
} else {
var iterator = ".";
if(this.pragmas["IMPLICIT-ITERATOR"]) {
iterator = this.pragmas["IMPLICIT-ITERATOR"].iterator;
}
var ctx = {};
ctx[iterator] = _context;
return ctx;
}
},
is_object: function(a) {
return a && typeof a == "object";
},
is_array: function(a) {
return Object.prototype.toString.call(a) === '[object Array]';
},
/*
Gets rid of leading and trailing whitespace
*/
trim: function(s) {
return s.replace(/^\s*|\s*$/g, "");
},
/*
Why, why, why? Because IE. Cry, cry cry.
*/
map: function(array, fn) {
if (typeof array.map == "function") {
return array.map(fn);
} else {
var r = [];
var l = array.length;
for(var i = 0; i < l; i++) {
r.push(fn(array[i]));
}
return r;
}
}
};
return({
name: "mustache.js",
version: "0.3.1-dev",
/*
Turns a template and view into HTML
*/
to_html: function(template, view, partials, send_fun) {
var renderer = new Renderer();
if(send_fun) {
renderer.send = send_fun;
}
renderer.render(template, view, partials);
if(!send_fun) {
return renderer.buffer.join("\n");
}
}
});
}();

View File

@ -45,7 +45,7 @@
background-image: url("/images/admin/form/big_item-popup.png"); background-image: url("/images/admin/form/big_item-popup.png");
} }
#fancybox-inner p { color:#8B8D9A; font-size:0.8em; } #fancybox-inner p { color: #8B8D9A; font-size: 0.8em; }
/* ___ asset picker ___ */ /* ___ asset picker ___ */
@ -63,6 +63,7 @@ div.asset-picker ul li .more { top: 8px; }
/* ___ custom fields ___ */ /* ___ custom fields ___ */
#edit-custom-field { #edit-custom-field {
padding: 10px;
width: 473px; width: 473px;
} }

View File

@ -266,12 +266,18 @@ form.formtastic fieldset.editable-list ol li.added select {
} }
form.formtastic fieldset.editable-list ol li.added em { form.formtastic fieldset.editable-list ol li.added em {
display: inline-block;
position: relative;
top: -1px;
color: #8b8d9a; color: #8b8d9a;
font-size: 0.9em; font-size: 0.9em;
font-style: italic; font-style: italic;
margin-left: 3px; margin-left: 3px;
border: 1px solid transparent;
padding: 2px 5px;
height: 18px;
line-height: 16px;
} }
form.formtastic fieldset.editable-list ol li.added em { border: 1px solid transparent; padding: 2px 5px; }
form.formtastic fieldset.editable-list ol li.added em:hover { form.formtastic fieldset.editable-list ol li.added em:hover {
background: #fffbe5; background: #fffbe5;
border: 1px dotted #efe4a5; border: 1px dotted #efe4a5;
@ -280,6 +286,11 @@ form.formtastic fieldset.editable-list ol li.added em:hover {
font-weight: bold; font-weight: bold;
} }
form.formtastic fieldset.editable-list ol li.added select,
form.formtastic fieldset.editable-list ol li.added em {
width: 120px;
}
form.formtastic fieldset.editable-list ol li.added input { form.formtastic fieldset.editable-list ol li.added input {
position: relative; position: relative;
top: -1px; top: -1px;
@ -325,7 +336,11 @@ form.formtastic fieldset.editable-list ol li.template {
padding-top: 10px; padding-top: 10px;
} }
form.formtastic fieldset.editable-list ol li.template input { form.formtastic fieldset.editable-list ol li.template em {
display: none;
}
form.formtastic fieldset.editable-list ol li.template input[type=text] {
display: inline; display: inline;
margin-left: 10px; margin-left: 10px;
padding: 4px; padding: 4px;
@ -340,6 +355,7 @@ form.formtastic fieldset.editable-list ol li.template input {
form.formtastic fieldset.editable-list ol li.template select { form.formtastic fieldset.editable-list ol li.template select {
display: inline; display: inline;
top: -1px;
} }
form.formtastic fieldset.editable-list ol li.template span.handle { form.formtastic fieldset.editable-list ol li.template span.handle {
@ -367,6 +383,25 @@ form.formtastic fieldset.editable-list ol li.template span.actions button span {
font-size: 0.8em; font-size: 0.8em;
} }
/* ___ custom fields: inherits from editable-list ___ */
form.formtastic fieldset.fields input[type=checkbox] {
margin: 0 0 0 20px;
width: 20px;
}
form.formtastic fieldset.fields label {
display: inline;
float: none;
padding: 0px;
font-weight: normal;
color: #8B8D9A;
}
form.formtastic fieldset.fields li.required label {
font-weight: bold;
}
/* ___ editable-list (content type validations) ___ */ /* ___ editable-list (content type validations) ___ */
form.formtastic fieldset.validations ol li.added em.key { form.formtastic fieldset.validations ol li.added em.key {