touch up welcome page

This commit is contained in:
danielgrippi 2011-10-19 18:44:09 -07:00
parent c49c76e2a6
commit ed473e0f46
9 changed files with 474 additions and 24 deletions

View file

@ -97,6 +97,7 @@ gem 'haml', '3.1.2'
gem 'mobile-fu' gem 'mobile-fu'
gem 'sass', '3.1.7' gem 'sass', '3.1.7'
gem 'will_paginate', '3.0.pre2' gem 'will_paginate', '3.0.pre2'
gem 'client_side_validations'
# web # web

View file

@ -139,6 +139,7 @@ GEM
uuidtools uuidtools
childprocess (0.2.2) childprocess (0.2.2)
ffi (~> 1.0.6) ffi (~> 1.0.6)
client_side_validations (3.1.0)
closure-compiler (1.1.4) closure-compiler (1.1.4)
cloudfiles (1.4.10) cloudfiles (1.4.10)
mime-types (>= 1.16) mime-types (>= 1.16)
@ -486,6 +487,7 @@ DEPENDENCIES
capybara (~> 0.3.9) capybara (~> 0.3.9)
carrierwave (= 0.5.2) carrierwave (= 0.5.2)
chef (= 0.9.12) chef (= 0.9.12)
client_side_validations
cloudfiles (= 1.4.10) cloudfiles (= 1.4.10)
cucumber-api-steps cucumber-api-steps
cucumber-rails (= 0.3.2) cucumber-rails (= 0.3.2)

View file

@ -1,43 +1,45 @@
= javascript_include_tag "validation" = javascript_include_tag "validation"
= javascript_tag "Diaspora.Page = 'InvitationsEdit';" = javascript_tag "Diaspora.Page = 'InvitationsEdit';"
:javascript
$(function() {
$("#user_username").focus();
});
.span-7.append-1.prepend-3 .span-7.append-1.prepend-3
%br %br
%br %br
%br %br
%br %br
%h2 %h1
= t('.your_account_awaits') = t('users.getting_started.welcome')
%h3.accept_invitation_text %h3.accept_invitation_text
= t('.accept_your_invitation') = t('.accept_your_invitation')
- flash.each do |name, msg| - flash.each do |name, msg|
%p{:class => "login_#{name}"}= msg %p{:class => "login_#{name}"}= msg
= image_tag 'diaspora_collage.png', :style => "margin-left:-50px;"
.span-10 .span-10
%br %br
%br %br
%br %br
%br %br
= form_for(resource, :as => resource_name, :url => invitation_path(resource_name), :html => {:method => :put, :class => 'accept_invitation_form'}) do |f| = form_for(resource, :as => resource_name, :url => invitation_path(resource_name), :html => {:method => :put, :class => 'accept_invitation_form'}, :validate => true) do |f|
%p %fieldset
= f.label :username , t('username') .clearfix
%b
= t('username')
= f.text_field :username, :title => t('registrations.new.enter_username') = f.text_field :username, :title => t('registrations.new.enter_username')
%p .clearfix
= f.label :email , t('email') %b
= t('email')
= f.email_field :email, :title => t('registrations.new.enter_email') = f.email_field :email, :title => t('registrations.new.enter_email')
%p .clearfix
= f.label :password , t('password') %b
= t('password')
= f.password_field :password, :title => t('registrations.new.enter_password') = f.password_field :password, :title => t('registrations.new.enter_password')
%p .clearfix
= f.label :password_confirmation , t('password_confirmation') %b
= t('password_confirmation')
= f.password_field :password_confirmation, :title => t('registrations.new.enter_password_again') = f.password_field :password_confirmation, :title => t('registrations.new.enter_password_again')
= f.hidden_field :invitation_token = f.hidden_field :invitation_token
.submit_field .submit_field
= f.submit t('registrations.new.create_my_account') = f.submit t('registrations.new.create_my_account'), :class => 'in_aspects'

View file

@ -10,6 +10,7 @@ javascripts:
jquery: jquery:
- public/javascripts/vendor/jquery162.min.js - public/javascripts/vendor/jquery162.min.js
main: main:
- public/javascripts/rails.validations.js
- public/javascripts/rails.js - public/javascripts/rails.js
- public/javascripts/vendor/jquery.tipsy.js - public/javascripts/vendor/jquery.tipsy.js
- public/javascripts/vendor/jquery.hotkeys.js - public/javascripts/vendor/jquery.hotkeys.js

View file

@ -0,0 +1,14 @@
# ClientSideValidations Initializer
#require 'client_side_validations/simple_form' if defined?(::SimpleForm)
#require 'client_side_validations/formtastic' if defined?(::Formtastic)
# Uncomment the following block if you want each input field to have the validation messages attached.
ActionView::Base.field_error_proc = Proc.new do |html_tag, instance|
unless html_tag =~ /^<label/
%{<div class="field_with_errors">#{html_tag}<label for="#{instance.send(:tag_id)}" class="message">#{instance.error_message.first}</label></div>}.html_safe
else
%{<div class="field_with_errors">#{html_tag}</div>}.html_safe
end
end

View file

@ -630,7 +630,7 @@ en:
registrations: registrations:
new: new:
create_my_account: "Create my account" create_my_account: "Create my account!"
join_the_movement: "Join the movement!" join_the_movement: "Join the movement!"
sign_up_today: "Sign up today" sign_up_today: "Sign up today"

Binary file not shown.

After

Width:  |  Height:  |  Size: 148 KiB

View file

@ -0,0 +1,404 @@
/*!
* Rails 3 Client Side Validations - v3.1.0
* https://github.com/bcardarlela/client_side_validations
*
* Copyright (c) 2011 Brian Cardarella
* Licensed under the MIT license
* http://www.opensource.org/licenses/mit-license.php
*/
(function($) {
$.fn.validate = function() {
return this.filter('form[data-validate]').each(function() {
var form = $(this);
var settings = window[form.attr('id')];
// Set up the events for the form
form
.submit(function() { return form.isValid(settings.validators); })
.bind('ajax:beforeSend', function() { return form.isValid(settings.validators); })
// Callbacks
.bind('form:validate:after', function(eventData) { clientSideValidations.callbacks.form.after( form, eventData); })
.bind('form:validate:before', function(eventData) { clientSideValidations.callbacks.form.before(form, eventData); })
.bind('form:validate:fail', function(eventData) { clientSideValidations.callbacks.form.fail( form, eventData); })
.bind('form:validate:pass', function(eventData) { clientSideValidations.callbacks.form.pass( form, eventData); })
// Set up the events for each validatable form element
.find('[data-validate]:input:not(:radio)')
.live('focusout', function() { $(this).isValid(settings.validators); })
.live('change', function() { $(this).data('changed', true); })
// Callbacks
.live('element:validate:after', function(eventData) { clientSideValidations.callbacks.element.after( $(this), eventData); })
.live('element:validate:before', function(eventData) { clientSideValidations.callbacks.element.before($(this), eventData); })
.live('element:validate:fail', function(eventData, message) {
var element = $(this);
clientSideValidations.callbacks.element.fail(element, message, function() {
addError(element, message);
}, eventData) })
.live('element:validate:pass', function(eventData) {
var element = $(this);
clientSideValidations.callbacks.element.pass(element, function() {
removeError(element);
}, eventData) })
// Checkboxes - Live events don't support filter
.end().find('[data-validate]:checkbox')
.live('click', function() { $(this).isValid(settings.validators); })
// Inputs for confirmations
.end().find('[id*=_confirmation]').each(function() {
var confirmationElement = $(this),
element = form.find('#' + this.id.match(/(.+)_confirmation/)[1] + '[data-validate]:input');
if (element[0]) {
$('#' + confirmationElement.attr('id'))
.live('focusout', function() {
element.data('changed', true).isValid(settings.validators);
})
.live('keyup', function() {
element.data('changed', true).isValid(settings.validators);
})
}
});
var addError = function(element, message) {
clientSideValidations.formBuilders[settings.type].add(element, settings, message);
}
var removeError = function(element) {
clientSideValidations.formBuilders[settings.type].remove(element, settings);
}
});
}
$.fn.isValid = function(validators) {
if ($(this[0]).is('form')) {
return validateForm($(this[0]), validators);
} else {
return validateElement($(this[0]), validators[this[0].name]);
}
}
var validateForm = function(form, validators) {
var valid = true;
form.trigger('form:validate:before').find('[data-validate]:input').each(function() {
if (!$(this).isValid(validators)) { valid = false; }
});
if (valid) {
form.trigger('form:validate:pass');
} else {
form.trigger('form:validate:fail');
}
form.trigger('form:validate:after');
return valid;
}
var validateElement = function(element, validators) {
element.trigger('element:validate:before');
if (element.data('changed') !== false) {
var valid = true;
element.data('changed', false);
// Because 'length' is defined on the list of validators we cannot call jQuery.each on
// the clientSideValidations.validators.all() object
for (kind in clientSideValidations.validators.all()) {
if (validators[kind] && (message = clientSideValidations.validators.all()[kind](element, validators[kind]))) {
element.trigger('element:validate:fail', message).data('valid', false);
valid = false;
break;
}
}
if (valid) { element.data('valid', null); element.trigger('element:validate:pass'); }
}
element.trigger('element:validate:after');
return element.data('valid') === false ? false : true;
}
// Main hook
// If new forms are dynamically introduced into the DOM the .validate() method
// must be invoked on that form
$(function() { $('form[data-validate]').validate(); })
})(jQuery);
var clientSideValidations = {
validators: {
all: function() { return jQuery.extend({}, clientSideValidations.validators.local, clientSideValidations.validators.remote) },
local: {
presence: function(element, options) {
if (/^\s*$/.test(element.val() || "")) {
return options.message;
}
},
acceptance: function(element, options) {
switch (element.attr('type')) {
case 'checkbox':
if (!element.attr('checked')) {
return options.message;
}
break;
case 'text':
if (element.val() != (options.accept || '1')) {
return options.message;
}
break;
}
},
format: function(element, options) {
if ((message = this.presence(element, options)) && options.allow_blank == true) {
return;
} else if (message) {
return message;
} else {
if (options['with'] && !options['with'].test(element.val())) {
return options.message;
} else if (options['without'] && options['without'].test(element.val())) {
return options.message;
}
}
},
numericality: function(element, options) {
if (!/^-?(?:\d+|\d{1,3}(?:,\d{3})+)(?:\.\d*)?$/.test(element.val())) {
return options.messages.numericality;
}
if (options.only_integer && !/^\d+$/.test(element.val())) {
return options.messages.only_integer;
}
var CHECKS = { greater_than: '>', greater_than_or_equal_to: '>=',
equal_to: '==', less_than: '<', less_than_or_equal_to: '<=' }
for (var check in CHECKS) {
if (options[check] != undefined && !(new Function("return " + element.val() + CHECKS[check] + options[check])())) {
return options.messages[check];
}
}
if (options.odd && !(parseInt(element.val()) % 2)) {
return options.messages.odd;
}
if (options.even && (parseInt(element.val()) % 2)) {
return options.messages.even;
}
},
length: function(element, options) {
var blankOptions = {};
if (options.is) {
blankOptions.message = options.messages.is;
} else if (options.minimum) {
blankOptions.message = options.messages.minimum;
}
if ((message = this.presence(element, blankOptions)) && options.allow_blank == true) {
return;
} else if (message) {
return message;
} else {
var CHECKS = { is: '==', minimum: '>=', maximum: '<=' }
var tokenizer = options.js_tokenizer || "split('')";
var tokenized_length = new Function("element", "return (element.val()." + tokenizer + " || '').length;")(element);
for (var check in CHECKS) {
if (options[check] && !(new Function("return " + tokenized_length + CHECKS[check] + options[check])())) {
return options.messages[check];
}
}
}
},
exclusion: function(element, options) {
if ((message = this.presence(element, options)) && options.allow_blank == true) {
return;
} else if (message) {
return message;
} else {
if (options['in']) {
for (var i = 0; i < options['in'].length; i++) {
if (options['in'][i] == element.val()) {
return options.message;
}
}
} else if (options['range']) {
var lower = options['range'][0],
upper = options['range'][1];
if (element.val() >= lower && element.val() <= upper) {
return options.message;
}
}
}
},
inclusion: function(element, options) {
if ((message = this.presence(element, options)) && options.allow_blank == true) {
return;
} else if (message) {
return message;
} else {
if (options['in']) {
for (var i = 0; i < options['in'].length; i++) {
if (options['in'][i] == element.val()) {
return;
}
}
return options.message;
} else if (options['range']) {
var lower = options['range'][0],
upper = options['range'][1];
if (element.val() >= lower && element.val() <= upper) {
return;
} else {
return options.message;
}
}
}
},
confirmation: function(element, options) {
if (element.val() != jQuery('#' + element.attr('id') + '_confirmation').val()) {
return options.message;
}
}
},
remote: {
uniqueness: function(element, options) {
var data = {};
data['case_sensitive'] = !!options.case_sensitive;
if (options.id) {
data['id'] = options.id;
}
if (options.scope) {
data.scope = {}
for (key in options.scope) {
var scoped_element = jQuery('[name="' + element.attr('name').replace(/\[\w+]$/, '[' + key + ']' + '"]'));
if (scoped_element[0] && scoped_element.val() != options.scope[key]) {
data.scope[key] = scoped_element.val();
scoped_element.unbind('change.' + element.id).bind('change.' + element.id, function() { element.trigger('change'); element.trigger('focusout'); });
} else {
data.scope[key] = options.scope[key];
}
}
}
// Kind of a hack but this will isolate the resource name and attribute.
// e.g. user[records_attributes][0][title] => records[title]
// e.g. user[record_attributes][title] => record[title]
// Server side handles classifying the resource properly
if (/_attributes]/.test(element.attr('name'))) {
var name = element.attr('name').match(/\[\w+_attributes]/g).pop().match(/\[(\w+)_attributes]/).pop();
name += /(\[\w+])$/.exec(element.attr('name'))[1];
} else {
var name = element.attr('name');
}
// Override the name if a nested module class is passed
if (options['class']) {
name = options['class'] + '[' + name.split('[')[1]
}
data[name] = element.val();
if (jQuery.ajax({
url: '/validators/uniqueness',
data: data,
async: false
}).status == 200) {
return options.message;
}
}
}
},
formBuilders: {
'ActionView::Helpers::FormBuilder': {
add: function(element, settings, message) {
if (element.data('valid') !== false && jQuery('label.message[for="' + element.attr('id') + '"]')[0] == undefined) {
var inputErrorField = jQuery(settings.input_tag),
labelErrorField = jQuery(settings.label_tag),
label = jQuery('label[for="' + element.attr('id') + '"]:not(.message)');
if (element.attr('autofocus')) { element.attr('autofocus', false) };
element.before(inputErrorField);
inputErrorField.find('span#input_tag').replaceWith(element);
inputErrorField.find('label.message').attr('for', element.attr('id'));
labelErrorField.find('label.message').attr('for', element.attr('id'));
label.replaceWith(labelErrorField);
labelErrorField.find('label#label_tag').replaceWith(label);
}
jQuery('label.message[for="' + element.attr('id') + '"]').text(message);
},
remove: function(element, settings) {
var errorFieldClass = jQuery(settings.input_tag).attr('class'),
inputErrorField = element.closest('.' + errorFieldClass),
label = jQuery('label[for="' + element.attr('id') + '"]:not(.message)'),
labelErrorField = label.closest('.' + errorFieldClass);
if (inputErrorField[0]) {
inputErrorField.find('#' + element.attr('id')).detach();
inputErrorField.replaceWith(element);
label.detach();
labelErrorField.replaceWith(label);
}
}
},
'SimpleForm::FormBuilder': {
add: function(element, settings, message) {
if (element.data('valid') !== false) {
var wrapper = element.closest(settings.wrapper_tag);
wrapper.addClass(settings.wrapper_error_class);
var errorElement = $('<' + settings.error_tag + ' class="' + settings.error_class + '">' + message + '</' + settings.error_tag + '>');
wrapper.append(errorElement);
} else {
element.parent().find(settings.error_tag + '.' + settings.error_class).text(message);
}
},
remove: function(element, settings) {
var wrapper = element.closest(settings.wrapper_tag + '.' + settings.wrapper_error_class);
wrapper.removeClass(settings.wrapper_error_class);
var errorElement = wrapper.find(settings.error_tag + '.' + settings.error_class);
errorElement.remove();
}
},
'Formtastic::FormBuilder': {
add: function(element, settings, message) {
if (element.data('valid') !== false) {
var wrapper = element.closest('li');
wrapper.addClass('error');
var errorElement = $('<p class="' + settings.inline_error_class + '">' + message + '</p>');
wrapper.append(errorElement);
} else {
element.parent().find('p.' + settings.inline_error_class).text(message);
}
},
remove: function(element, settings) {
var wrapper = element.closest('li.error');
wrapper.removeClass('error');
var errorElement = wrapper.find('p.' + settings.inline_error_class);
errorElement.remove();
}
},
'NestedForm::Builder': {
add: function(element, settings, message) {
clientSideValidations.formBuilders['ActionView::Helpers::FormBuilder'].add(element, settings, message);
},
remove: function(element, settings, message) {
clientSideValidations.formBuilders['ActionView::Helpers::FormBuilder'].remove(element, settings, message);
}
}
},
callbacks: {
element: {
after: function(element, eventData) { },
before: function(element, eventData) { },
fail: function(element, message, addError, eventData) { addError() },
pass: function(element, removeError, eventData) { removeError() }
},
form: {
after: function(form, eventData) { },
before: function(form, eventData) { },
fail: function(form, eventData) { },
pass: function(form, eventData) { }
}
}
};

View file

@ -3462,3 +3462,29 @@ ul#getting_started
:align center :align center
:border :border
:bottom 1px solid #ddd :bottom 1px solid #ddd
.field_with_errors
:position relative
input
@include box-shadow(0,0,8px,lighten(#D00,30%))
:border 1px solid #D00 !important
.field_with_errors .message
:color #D00
:font-size 12px
:position absolute
:top 4px
:right 10px
:left auto
.accept_invitation_form fieldset
@include border-radius(3px)
:background
:color #fff
:color rgba(255,255,255,0.95)
.submit_field
:margin
:top 15px