da-suspenders 1.0.0

Sign up to get free protection for your applications and to get access to all the features.
data/.gitignore ADDED
@@ -0,0 +1,4 @@
1
+ *.swp
2
+ pkg
3
+ test_project
4
+ .rake_tasks
data/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License
2
+
3
+ Copyright (c) 2010 Mike Burns
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,62 @@
1
+ DA-Suspenders is a base Rails application that you can upgrade.
2
+
3
+ This is a fork of [thoughtbot's original Suspenders](https://github.com/thoughtbot/suspenders/), customized to our needs.
4
+
5
+
6
+ ## Installation
7
+
8
+ First install the da-suspenders gem:
9
+
10
+ gem install da-suspenders
11
+
12
+ Then run:
13
+
14
+ da-suspenders create projectname
15
+
16
+ This will create a Rails 3 app in `projectname'. This script creates a new git repository. It is not meant to be used against an existing repo.
17
+
18
+ Suspenders uses [Trout](https://github.com/thoughtbot/trout) to make it easier to maintain a base version of special files (like Gemfile) in Suspenders.
19
+
20
+ Whenever you want to get the latest and greatest Suspenders' Gemfile, run:
21
+
22
+ trout update Gemfile
23
+
24
+
25
+ ## Gemfile
26
+
27
+ To see the latest and greatest gems, look at DA-Suspenders'
28
+ [template/trout/Gemfile](https://github.com/die-antwort/da-suspenders/blob/master/template/trout/Gemfile), which will be copied into your projectname/Gemfile.
29
+
30
+ It includes application gems like:
31
+
32
+ * [Compass](https://github.com/chriseppstein/compass), a [Sass-based](http://sass-lang.com/) CSS Meta-Framework
33
+ * [Formtastic](https://github.com/justinfrench/formtastic) for better forms
34
+ * [Hoptoad Notifier](https://github.com/thoughtbot/hoptoad_notifier) for exception notification
35
+ * [Paperclip](https://github.com/thoughtbot/paperclip) for file uploads
36
+ * [Sprockets](https://github.com/sstephenson/sprockets) for JavaScript dependency management and concatenation
37
+ * Our fork of [jRails](https://github.com/die-antwort/jrails), which brings you [jQuery](https://github.com/jquery/jquery) and [jQuery UI](https://github.com/jquery/jquery-ui) via Sprockets
38
+ * [AppConfig](https://github.com/die-antwort/app_config) for simple application configuration
39
+
40
+
41
+ And testing gems like:
42
+
43
+ * [Cucumber, Capybara, and Akephalos](http://robots.thoughtbot.com/post/1658763359/thoughtbot-and-the-holy-grail) for integration testing, including Javascript behavior
44
+ * [RSpec](https://github.com/rspec/rspec) for awesome, readable isolation testing
45
+ * [Factory Girl](https://github.com/thoughtbot/factory_girl) for easier creation of test data
46
+ * [Timecop](https://github.com/jtrupiano/timecop) for dealing with time
47
+
48
+
49
+ ## Other goodies
50
+
51
+ DA-Suspenders also comes with:
52
+
53
+ * Additional staging environment.
54
+ * German localization and timezone setting.
55
+ * Rails' flashes set up and in application layout.
56
+
57
+ See [template/files](https://github.com/die-antwort/da-suspenders/blob/master/template/trout) to see what is installed via Trout.
58
+
59
+
60
+ ## Credits
61
+
62
+ DA-Suspenders is created by [DIE ANTWORT](http://www.die-antwort.eu), based on Suspenders created and funded by [thoughtbot, inc](http://thoughtbot.com/community).
data/Rakefile ADDED
@@ -0,0 +1,40 @@
1
+ require "bundler"
2
+ require "cucumber/rake/task"
3
+
4
+ Bundler::GemHelper.install_tasks
5
+
6
+ #############################################################################
7
+ #
8
+ # Testing functions
9
+ #
10
+ #############################################################################
11
+
12
+ Cucumber::Rake::Task.new
13
+
14
+ TEST_PROJECT = "test_project"
15
+
16
+ namespace :test do
17
+ desc "A full suspenders app's test suite"
18
+ task :full => ['test_project:generate', 'cucumber', 'test_project:destroy']
19
+ end
20
+
21
+ namespace :test_project do
22
+ desc 'Suspend a new project. Pass REPO=... to change the Suspenders repo.'
23
+ task :generate do
24
+ FileUtils.rm_rf(TEST_PROJECT)
25
+ sh 'ruby', 'bin/da-suspenders', 'create', TEST_PROJECT, ENV['REPO'].to_s
26
+ end
27
+
28
+ desc 'Remove a suspended project'
29
+ task :destroy do
30
+ FileUtils.cd TEST_PROJECT
31
+ sh "rake db:drop RAILS_ENV=development"
32
+ sh "rake db:drop RAILS_ENV=test"
33
+ FileUtils.cd '..'
34
+ FileUtils.rm_rf TEST_PROJECT
35
+ end
36
+ end
37
+
38
+ desc 'Run the test suite'
39
+ task :default => ['test:full']
40
+
data/bin/da-suspenders ADDED
@@ -0,0 +1,23 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require File.expand_path(File.dirname(__FILE__) + "/../lib/create")
4
+ require File.expand_path(File.dirname(__FILE__) + "/../lib/errors")
5
+
6
+ include DaSuspenders::Errors
7
+
8
+ def usage
9
+ "Usage: #{File.basename(__FILE__)} create new_project_name"
10
+ end
11
+
12
+ case ARGV[0]
13
+ when 'create', '--create'
14
+ begin
15
+ DaSuspenders::Create.run!(ARGV[1])
16
+ rescue DaSuspenders::InvalidInput => e
17
+ error_with_message(e.message)
18
+ end
19
+ when 'help', '--help'
20
+ puts usage
21
+ else
22
+ error_with_message "Unknown subcommand: #{ARGV[0]}\n#{usage}"
23
+ end
@@ -0,0 +1,26 @@
1
+ require File.expand_path("../lib/da-suspenders/version", __FILE__)
2
+
3
+ Gem::Specification.new do |s|
4
+ s.name = 'da-suspenders'
5
+ s.version = DaSuspenders::VERSION
6
+ s.platform = Gem::Platform::RUBY
7
+ s.authors = ["thoughtbot", "Stefan Daschek"]
8
+ s.email = 'stefan@die-antwort.eu'
9
+ s.homepage = 'http://github.com/die-antwort/da-suspenders'
10
+ s.summary = "Generate a Rails app using DIE ANTWORT's best practices."
11
+ s.description = "DIE ANTWORT's fork of thoughtbot's original Suspenders. Suspenders is an upgradeable base Rails project."
12
+
13
+ s.required_rubygems_version = ">= 1.3.6"
14
+
15
+ s.add_dependency('rails', '>= 3.0.4')
16
+ s.add_dependency('bundler', '>= 1.0.10')
17
+ s.add_dependency('trout', '>= 0.3.0')
18
+
19
+ s.files = `git ls-files`.split("\n")
20
+ s.executables = `git ls-files`.split("\n").map{|f| f =~ /^bin\/(.*)/ ? $1 : nil}.compact
21
+ s.test_files = s.files.select{ |path| path =~ /^features/ }
22
+ s.require_path = 'lib'
23
+
24
+ s.rdoc_options = ["--charset=UTF-8"]
25
+ s.extra_rdoc_files = %w[README.md LICENSE]
26
+ end
@@ -0,0 +1,18 @@
1
+ Feature: Running cucumber in the generated project
2
+
3
+ Scenario: jQuery and Modernizr work in the generated project
4
+ Given I drop and create the required databases
5
+ And I generate "scaffold user username:string"
6
+ And I run the rake task "db:migrate"
7
+ And I create a file named "features/js_test.feature" with:
8
+ """
9
+ Feature: Javascript
10
+ @javascript
11
+ Scenario: Test jQuery and Modernizr
12
+ When I go to "the users page"
13
+ Then the javascript expression "$('html').hasClass('js')" should return "true"
14
+ And the javascript expression "$('html').hasClass('no-js')" should return "false"
15
+ """
16
+ When I run the rake task "cucumber"
17
+ Then I see a successful response in the shell
18
+
@@ -0,0 +1,30 @@
1
+ When 'I run the rake task "$task_name"' do |task_name|
2
+ Dir.chdir('test_project') do
3
+ system("rake #{task_name}")
4
+ end
5
+ end
6
+
7
+ When 'I generate "$generator_with_args"' do |generator_with_args|
8
+ Dir.chdir('test_project') do
9
+ system("rails generate #{generator_with_args}")
10
+ end
11
+ end
12
+
13
+ When 'I create a file named "$filename" with:' do |filename, content|
14
+ File.open("test_project/#{filename}", "w") do |file|
15
+ file.write(content)
16
+ end
17
+ end
18
+
19
+ Then 'I see a successful response in the shell' do
20
+ $?.to_i.should == 0
21
+ end
22
+
23
+ When 'I drop and create the required databases' do
24
+ Dir.chdir('test_project') do
25
+ system("rake db:drop RAILS_ENV=test")
26
+ system("rake db:drop")
27
+ system("rake db:create RAILS_ENV=test")
28
+ system("rake db:create")
29
+ end
30
+ end
@@ -0,0 +1 @@
1
+ require 'spec/expectations'
data/lib/create.rb ADDED
@@ -0,0 +1,55 @@
1
+ # Methods needed to create a project.
2
+
3
+ require 'rubygems'
4
+ require File.expand_path(File.dirname(__FILE__) + "/errors")
5
+
6
+ module DaSuspenders
7
+ class Create
8
+ attr_accessor :project_path
9
+
10
+ def self.run!(project_path)
11
+ creator = self.new(project_path)
12
+ creator.create_project!
13
+ end
14
+
15
+ def initialize(project_path)
16
+ self.project_path = project_path
17
+ validate_project_path
18
+ validate_project_name
19
+ end
20
+
21
+ def create_project!
22
+ exec(<<-COMMAND)
23
+ rails new #{project_path} \
24
+ --template="#{template}" \
25
+ --database=mysql \
26
+ --skip-test-unit \
27
+ --skip-prototype
28
+ COMMAND
29
+ end
30
+
31
+ private
32
+
33
+ def validate_project_name
34
+ project_name = File.basename(project_path)
35
+ unless project_name =~ /^[a-z0-9_]+$/
36
+ raise InvalidInput.new("Project name must only contain [a-z0-9_]")
37
+ end
38
+ end
39
+
40
+ def validate_project_path
41
+ base_directory = Dir.pwd
42
+ full_path = File.join(base_directory, project_path)
43
+
44
+ # This is for the common case for the user's convenience; the race
45
+ # condition can still occur.
46
+ if File.exists?(full_path)
47
+ raise InvalidInput.new("Project directory (#{project_path}) already exists")
48
+ end
49
+ end
50
+
51
+ def template
52
+ File.expand_path(File.dirname(__FILE__) + "/../template/da-suspenders.rb")
53
+ end
54
+ end
55
+ end
@@ -0,0 +1,3 @@
1
+ module DaSuspenders
2
+ VERSION = "1.0.0"
3
+ end
data/lib/errors.rb ADDED
@@ -0,0 +1,12 @@
1
+ # User errors
2
+
3
+ module DaSuspenders
4
+ class InvalidInput < Exception; end
5
+
6
+ module Errors
7
+ def error_with_message(msg)
8
+ STDERR.puts msg
9
+ exit 1
10
+ end
11
+ end
12
+ end
@@ -0,0 +1,166 @@
1
+ require "fileutils"
2
+ require "pathname"
3
+ require "trout"
4
+
5
+ template_root = File.expand_path(File.join(File.dirname(__FILE__)))
6
+ source_paths << File.join(template_root, "files")
7
+
8
+ # Helpers
9
+
10
+ def action_mailer_host(rails_env, host)
11
+ inject_into_file(
12
+ "config/environments/#{rails_env}.rb",
13
+ "\n\n config.action_mailer.default_url_options = { :host => '#{host}' }",
14
+ :before => "\nend"
15
+ )
16
+ end
17
+
18
+ def origin
19
+ "git://github.com/die-antwort/da-suspenders.git"
20
+ end
21
+
22
+ def trout(destination_path)
23
+ parent = Pathname.new(destination_path).parent
24
+ FileUtils.mkdir_p(parent) unless File.exists?(parent)
25
+ run "trout checkout --source-root=template/trout #{destination_path} #{origin}"
26
+ end
27
+
28
+ # Actions
29
+
30
+ def create_gemfile_and_install_gems
31
+ say "Creating Gemfile and installing gems (this may take a while)", :yellow
32
+ trout "Gemfile"
33
+ run "bundle install"
34
+ end
35
+
36
+ def add_staging_environment
37
+ say "Adding staging environment", :yellow
38
+ run "cp config/environments/production.rb config/environments/staging.rb"
39
+ end
40
+
41
+ def setup_action_mailer
42
+ say "Setting up action mailer config", :yellow
43
+ action_mailer_host "development", "localhost:3000"
44
+ action_mailer_host "test", "example.com"
45
+ action_mailer_host "staging", "#{app_name}.dev.die-antwort.eu"
46
+ action_mailer_host "production", "FIXME"
47
+ end
48
+
49
+ def setup_database
50
+ say "Setting up database config", :yellow
51
+ template "mysql_database.yml.erb", "config/database.yml", :force => true
52
+ rake "db:create"
53
+ end
54
+
55
+ def setup_german_locale
56
+ say "Setting up german locale", :yellow
57
+ trout "config/locales/de.yml"
58
+ gsub_file 'config/application.rb', '# config.i18n.default_locale = :de', 'config.i18n.default_locale = :de'
59
+ end
60
+
61
+ def setup_viennese_timezone
62
+ say "Setting up viennese timezone", :yellow
63
+ gsub_file 'config/application.rb', "# config.time_zone = 'Central Time (US & Canada)'", "config.time_zone = 'Vienna'"
64
+ end
65
+
66
+ def update_generators_config
67
+ say "Updating config for generators", :yellow
68
+ generators_config = <<-RUBY
69
+ config.generators do |generate|
70
+ generate.stylesheets false
71
+ generate.test_framework :rspec
72
+ end
73
+ RUBY
74
+ inject_into_class "config/application.rb", "Application", generators_config
75
+ end
76
+
77
+ def create_application_layout_and_views
78
+ say "Creating application layout and shared views", :yellow
79
+ trout "app/views/layouts/application.html.erb"
80
+ trout "app/views/shared/_flashes.html.erb"
81
+ end
82
+
83
+ def install_misc_support_files
84
+ say "Installing miscellaneous support files", :yellow
85
+ trout "config/initializers/errors.rb"
86
+ trout "app/helpers/body_class_helper.rb"
87
+ end
88
+
89
+ def install_app_config
90
+ say "Installing app_config", :yellow
91
+ generate "app_config:install staging"
92
+ end
93
+
94
+ def install_compass
95
+ say "Installing compass", :yellow
96
+ run "compass init rails . --sass-dir app/stylesheets --css-dir public/stylesheets -q"
97
+ inside 'app/stylesheets' do
98
+ remove_file 'ie.scss'
99
+ remove_file 'print.scss'
100
+ remove_file 'screen.scss'
101
+ empty_directory 'content'
102
+ empty_directory 'vendor'
103
+ end
104
+ trout "app/stylesheets/application.scss"
105
+ trout "app/stylesheets/ie.scss"
106
+ end
107
+
108
+ def install_formtastic
109
+ say "Installing formtastic", :yellow
110
+ generate "formtastic:install"
111
+ remove_file "public/stylesheets/formtastic.css"
112
+ remove_file "public/stylesheets/formtastic-changes.css"
113
+ trout "lib/templates/erb/scaffold/_form.html.erb"
114
+ end
115
+
116
+ def install_sprockets_and_jquery
117
+ say "Installing sprockets, jQuery, and some other javascripts", :yellow
118
+ plugin 'sprockets-rails', :git => 'git://github.com/gmoeck/sprockets-rails.git'
119
+ plugin 'jrails', :git => 'git://github.com/die-antwort/jrails.git'
120
+ route 'SprocketsApplication.routes(self)'
121
+ remove_dir 'public/javascripts'
122
+ remove_file "app/javascripts/application.js"
123
+ trout "app/javascripts/application.js"
124
+ trout "vendor/sprockets/modernizr/src/modernizr.js"
125
+ trout "vendor/sprockets/jquery-ujs/src/rails.js"
126
+ end
127
+
128
+ def install_rspec_and_cucumber
129
+ say "Installing rspec and cucumber", :yellow
130
+ generate "rspec:install"
131
+ generate "cucumber:install", "--rspec --capybara"
132
+ inject_into_file "features/support/env.rb",
133
+ %{Capybara.save_and_open_page_path = "tmp"\n} +
134
+ %{Capybara.javascript_driver = :akephalos\n},
135
+ :before => %{Capybara.default_selector = :css}
136
+ # replace_in_file "features/support/env.rb",
137
+ # %r{require .*capybara_javascript_emulation.*},
138
+ # ''
139
+ copy_file "factory_girl_steps.rb", "features/step_definitions/factory_girl_steps.rb"
140
+ trout "features/step_definitions/js_steps.rb"
141
+ end
142
+
143
+ def cleanup
144
+ say "Cleaning up", :yellow
145
+ remove_file 'README'
146
+ remove_file 'public/index.html'
147
+ remove_file 'public/images/rails.png'
148
+ end
149
+
150
+ create_gemfile_and_install_gems
151
+ add_staging_environment
152
+ setup_database
153
+ setup_german_locale
154
+ setup_viennese_timezone
155
+ update_generators_config
156
+ create_application_layout_and_views
157
+ install_misc_support_files
158
+ install_app_config
159
+ install_compass
160
+ install_formtastic
161
+ install_sprockets_and_jquery
162
+ install_rspec_and_cucumber
163
+ cleanup
164
+
165
+ say "Rails app #{app_name} has been created successully!", :blue
166
+ say "Remember to run 'rails generate hoptoad' with your API key.", :blue
@@ -0,0 +1 @@
1
+ require 'factory_girl/step_definitions'
@@ -0,0 +1,39 @@
1
+ development:
2
+ adapter: mysql
3
+ encoding: utf8
4
+ reconnect: false
5
+ database: <%= app_name %>_dev
6
+ pool: 5
7
+ username: root
8
+ password: root
9
+ host: localhost
10
+
11
+ test:
12
+ adapter: mysql
13
+ encoding: utf8
14
+ reconnect: false
15
+ database: <%= app_name %>_test
16
+ pool: 5
17
+ username: root
18
+ password: root
19
+ host: localhost
20
+
21
+ staging:
22
+ adapter: mysql
23
+ encoding: utf8
24
+ reconnect: false
25
+ database: <%= app_name %>_staging
26
+ pool: 5
27
+ username: <%= app_name %>_s
28
+ password:
29
+ host: localhost
30
+
31
+ production:
32
+ adapter: mysql
33
+ encoding: utf8
34
+ reconnect: false
35
+ database: <%= app_name %>
36
+ pool: 5
37
+ username: <%= app_name %>
38
+ password:
39
+ host: localhost
@@ -0,0 +1,30 @@
1
+ source :rubygems
2
+
3
+ gem "rails", "3.0.4"
4
+ gem "mysql"
5
+
6
+ gem "app_config", :git => "git://github.com/die-antwort/app_config.git"
7
+ gem "compass", "~> 0.10.6"
8
+ gem "formtastic", ">= 1.2.1"
9
+ gem "hoptoad_notifier"
10
+ gem "sprockets", "~> 1.0.2"
11
+ gem "will_paginate"
12
+
13
+ # Uncomment if you want to use any of those
14
+ #gem "paperclip"
15
+
16
+ # RSpec needs to be in :development group to expose generators
17
+ # and rake tasks without having to type RAILS_ENV=test.
18
+ group :development, :test do
19
+ gem "rspec-rails", "~> 2.4.0"
20
+ end
21
+
22
+ group :test do
23
+ gem "akephalos", :git => "git://github.com/die-antwort/akephalos.git"
24
+ gem "cucumber-rails"
25
+ gem "factory_girl_rails"
26
+ gem "capybara"
27
+ gem "database_cleaner"
28
+ gem "launchy"
29
+ gem "timecop"
30
+ end
@@ -0,0 +1,6 @@
1
+ module BodyClassHelper
2
+ def body_class
3
+ qualified_controller_name = controller.controller_path.gsub('/','-')
4
+ "#{qualified_controller_name} #{controller.action_name}"
5
+ end
6
+ end
@@ -0,0 +1,10 @@
1
+ //= require <modernizr>
2
+ //= require <jquery.min>
3
+ //= require <rails>
4
+
5
+ // activate if you want to use jrails:
6
+ // = require <jquery-ui.min>
7
+ // = require <jrails>
8
+
9
+ // activate if you want to localize jQuery UI Datepicker:
10
+ // = require <jquery-ui-i18n.min>
@@ -0,0 +1,14 @@
1
+ $ie: false !default;
2
+
3
+ @import 'compass/reset';
4
+ @import 'compass/utilities';
5
+ @import 'compass/css3';
6
+
7
+ // Use following syntax for IE-specific (<= 7) rules:
8
+ // .foobar {
9
+ // float: left;
10
+ // margin-left: 20px;
11
+ // @if $ie {
12
+ // margin-left: 10px;
13
+ // }
14
+ // }
@@ -0,0 +1,2 @@
1
+ $ie: true;
2
+ @import 'application';
@@ -0,0 +1,16 @@
1
+ <!DOCTYPE html>
2
+ <html lang="de" class="no-js">
3
+ <head>
4
+ <title>Testapp</title>
5
+ <link href="/stylesheets/application.css" type="text/css" rel="stylesheet">
6
+ <!--[if lte IE 7]><link href="/public/stylesheets/ie7.css" type="text/css" rel="stylesheet"><![endif]-->
7
+ <%= sprockets_include_tag %>
8
+ <%= javascript_tag "$.ajaxSetup({ async: false });" if Rails.env.test?%>
9
+ <%= csrf_meta_tag %>
10
+ </head>
11
+ <body class="<%= body_class %>">
12
+ <%= render :partial => "shared/flashes" %>
13
+ <%= yield %>
14
+ </body>
15
+ </html>
16
+ <!-- Created with care by DIE ANTWORT · Büro für Informationstechnik GmbH -->
@@ -0,0 +1,5 @@
1
+ <div id="flash">
2
+ <% flash.each do |key, value| -%>
3
+ <div id="flash_<%= key %>"><%= value %></div>
4
+ <% end -%>
5
+ </div>
@@ -0,0 +1,26 @@
1
+ require 'net/smtp'
2
+ # Example:
3
+ # begin
4
+ # some http call
5
+ # rescue *HTTP_ERRORS => error
6
+ # notify_hoptoad error
7
+ # end
8
+
9
+ HTTP_ERRORS = [Timeout::Error,
10
+ Errno::EINVAL,
11
+ Errno::ECONNRESET,
12
+ EOFError,
13
+ Net::HTTPBadResponse,
14
+ Net::HTTPHeaderSyntaxError,
15
+ Net::ProtocolError]
16
+
17
+ SMTP_SERVER_ERRORS = [TimeoutError,
18
+ IOError,
19
+ Net::SMTPUnknownError,
20
+ Net::SMTPServerBusy,
21
+ Net::SMTPAuthenticationError]
22
+
23
+ SMTP_CLIENT_ERRORS = [Net::SMTPFatalError,
24
+ Net::SMTPSyntaxError]
25
+
26
+ SMTP_ERRORS = SMTP_SERVER_ERRORS + SMTP_CLIENT_ERRORS
@@ -0,0 +1,149 @@
1
+ # Based on original translation by Clemens Kofler (clemens@railway.at)
2
+ de:
3
+ date:
4
+ formats:
5
+ default: "%d.%m.%Y"
6
+ short: "%e. %b"
7
+ long: "%e. %B %Y"
8
+ only_day: "%e"
9
+
10
+ day_names: [Sonntag, Montag, Dienstag, Mittwoch, Donnerstag, Freitag, Samstag]
11
+ abbr_day_names: [So, Mo, Di, Mi, Do, Fr, Sa]
12
+ month_names: [~, Jänner, Februar, März, April, Mai, Juni, Juli, August, September, Oktober, November, Dezember]
13
+ abbr_month_names: [~, Jän, Feb, Mär, Apr, Mai, Jun, Jul, Aug, Sep, Okt, Nov, Dez]
14
+ order: [ :day, :month, :year ]
15
+
16
+ time:
17
+ formats:
18
+ default: "%A, %d. %B %Y, %H:%M Uhr"
19
+ short: "%d. %B, %H:%M Uhr"
20
+ long: "%A, %d. %B %Y, %H:%M Uhr"
21
+ time: "%H:%M"
22
+
23
+ am: "vormittags"
24
+ pm: "nachmittags"
25
+
26
+ datetime:
27
+ distance_in_words:
28
+ half_a_minute: 'eine halbe Minute'
29
+ less_than_x_seconds:
30
+ one: 'weniger als eine Sekunde'
31
+ other: 'weniger als %{count} Sekunden'
32
+ x_seconds:
33
+ one: 'eine Sekunde'
34
+ other: '%{count} Sekunden'
35
+ less_than_x_minutes:
36
+ one: 'weniger als eine Minute'
37
+ other: 'weniger als %{count} Minuten'
38
+ x_minutes:
39
+ one: 'eine Minute'
40
+ other: '%{count} Minuten'
41
+ about_x_hours:
42
+ one: 'etwa eine Stunde'
43
+ other: 'etwa %{count} Stunden'
44
+ x_days:
45
+ one: 'ein Tag'
46
+ other: '%{count} Tage'
47
+ about_x_months:
48
+ one: 'etwa ein Monat'
49
+ other: 'etwa %{count} Monate'
50
+ x_months:
51
+ one: 'ein Monat'
52
+ other: '%{count} Monate'
53
+ almost_x_years:
54
+ one: 'fast ein Jahr'
55
+ other: 'fast %{count} Jahre'
56
+ about_x_years:
57
+ one: 'etwa ein Jahr'
58
+ other: 'etwa %{count} Jahre'
59
+ over_x_years:
60
+ one: 'mehr als ein Jahr'
61
+ other: 'mehr als %{count} Jahre'
62
+ prompts:
63
+ second: "Sekunden"
64
+ minute: "Minuten"
65
+ hour: "Stunden"
66
+ day: "Tag"
67
+ month: "Monat"
68
+ year: "Jahr"
69
+
70
+ number:
71
+ format:
72
+ precision: 2
73
+ separator: ','
74
+ delimiter: '.'
75
+ currency:
76
+ format:
77
+ unit: '€'
78
+ format: '%n%u'
79
+ separator:
80
+ delimiter:
81
+ precision:
82
+ percentage:
83
+ format:
84
+ delimiter: ""
85
+ precision:
86
+ format:
87
+ delimiter: ""
88
+ human:
89
+ format:
90
+ delimiter: ""
91
+ precision: 1
92
+ storage_units:
93
+ # Storage units output formatting.
94
+ # %u is the storage unit, %n is the number (default: 2 MB)
95
+ format: "%n %u"
96
+ units:
97
+ byte:
98
+ one: "Byte"
99
+ other: "Bytes"
100
+ kb: "KB"
101
+ mb: "MB"
102
+ gb: "GB"
103
+ tb: "TB"
104
+
105
+ support:
106
+ array:
107
+ words_connector: ", "
108
+ two_words_connector: " und "
109
+ last_word_connector: " und "
110
+ select:
111
+ prompt: "Bitte wählen:"
112
+
113
+ activemodel:
114
+ errors:
115
+ template:
116
+ header:
117
+ one: "Konnte %{model} nicht speichern: ein Fehler."
118
+ other: "Konnte %{model} nicht speichern: %{count} Fehler."
119
+ body: "Bitte überprüfen Sie die folgenden Felder:"
120
+
121
+ activerecord:
122
+ errors:
123
+ template:
124
+ header:
125
+ one: "Konnte %{model} nicht speichern: ein Fehler."
126
+ other: "Konnte %{model} nicht speichern: %{count} Fehler."
127
+ body: "Bitte überprüfen Sie die folgenden Felder:"
128
+
129
+ messages:
130
+ inclusion: "ist kein gültiger Wert"
131
+ exclusion: "ist nicht verfügbar"
132
+ invalid: "ist nicht gültig"
133
+ confirmation: "stimmt nicht mit der Bestätigung überein"
134
+ accepted: "muss akzeptiert werden"
135
+ empty: "muss ausgefüllt werden"
136
+ blank: "muss ausgefüllt werden"
137
+ too_long: "ist zu lang (nicht mehr als %{count} Zeichen)"
138
+ too_short: "ist zu kurz (nicht weniger als %{count} Zeichen)"
139
+ wrong_length: "hat die falsche Länge (muss genau %{count} Zeichen haben)"
140
+ taken: "ist bereits vergeben"
141
+ not_a_number: "ist keine Zahl"
142
+ greater_than: "muss größer als %{count} sein"
143
+ greater_than_or_equal_to: "muss größer oder gleich %{count} sein"
144
+ equal_to: "muss genau %{count} sein"
145
+ less_than: "muss kleiner als %{count} sein"
146
+ less_than_or_equal_to: "muss kleiner oder gleich %{count} sein"
147
+ odd: "muss ungerade sein"
148
+ even: "muss gerade sein"
149
+ record_invalid: "Gültigkeitsprüfung ist fehlgeschlagen: %{errors}"
@@ -0,0 +1,3 @@
1
+ Then 'the javascript expression "$expression" should return "$result"' do |expression, result|
2
+ page.evaluate_script(expression).to_s.should == result
3
+ end
@@ -0,0 +1,13 @@
1
+ <%%= semantic_form_for(@<%= singular_table_name %>) do |f| %>
2
+ <%%= f.semantic_errors %>
3
+
4
+ <%% f.inputs do %>
5
+ <% for attribute in attributes -%>
6
+ <%%= f.input :<%= attribute.name %> %>
7
+ <% end -%>
8
+ <%% end %>
9
+
10
+ <%% f.buttons do %>
11
+ <%%= f.commit_button %>
12
+ <%% end %>
13
+ <%% end %>
@@ -0,0 +1,169 @@
1
+ /**
2
+ * Unobtrusive scripting adapter for jQuery
3
+ *
4
+ * Requires jQuery 1.4.3 or later.
5
+ * https://github.com/rails/jquery-ujs
6
+ */
7
+
8
+ (function($) {
9
+ // Make sure that every Ajax request sends the CSRF token
10
+ function CSRFProtection(fn) {
11
+ var token = $('meta[name="csrf-token"]').attr('content');
12
+ if (token) fn(function(xhr) { xhr.setRequestHeader('X-CSRF-Token', token) });
13
+ }
14
+ if ($().jquery == '1.5') { // gruesome hack
15
+ var factory = $.ajaxSettings.xhr;
16
+ $.ajaxSettings.xhr = function() {
17
+ var xhr = factory();
18
+ CSRFProtection(function(setHeader) {
19
+ var open = xhr.open;
20
+ xhr.open = function() { open.apply(this, arguments); setHeader(this) };
21
+ });
22
+ return xhr;
23
+ };
24
+ }
25
+ else $(document).ajaxSend(function(e, xhr) {
26
+ CSRFProtection(function(setHeader) { setHeader(xhr) });
27
+ });
28
+
29
+ // Triggers an event on an element and returns the event result
30
+ function fire(obj, name, data) {
31
+ var event = new $.Event(name);
32
+ obj.trigger(event, data);
33
+ return event.result !== false;
34
+ }
35
+
36
+ // Submits "remote" forms and links with ajax
37
+ function handleRemote(element) {
38
+ var method, url, data,
39
+ dataType = element.attr('data-type') || ($.ajaxSettings && $.ajaxSettings.dataType);
40
+
41
+ if (element.is('form')) {
42
+ method = element.attr('method');
43
+ url = element.attr('action');
44
+ data = element.serializeArray();
45
+ // memoized value from clicked submit button
46
+ var button = element.data('ujs:submit-button');
47
+ if (button) {
48
+ data.push(button);
49
+ element.data('ujs:submit-button', null);
50
+ }
51
+ } else {
52
+ method = element.attr('data-method');
53
+ url = element.attr('href');
54
+ data = null;
55
+ }
56
+
57
+ $.ajax({
58
+ url: url, type: method || 'GET', data: data, dataType: dataType,
59
+ // stopping the "ajax:beforeSend" event will cancel the ajax request
60
+ beforeSend: function(xhr, settings) {
61
+ if (settings.dataType === undefined) {
62
+ xhr.setRequestHeader('accept', '*/*;q=0.5, ' + settings.accepts.script);
63
+ }
64
+ return fire(element, 'ajax:beforeSend', [xhr, settings]);
65
+ },
66
+ success: function(data, status, xhr) {
67
+ element.trigger('ajax:success', [data, status, xhr]);
68
+ },
69
+ complete: function(xhr, status) {
70
+ element.trigger('ajax:complete', [xhr, status]);
71
+ },
72
+ error: function(xhr, status, error) {
73
+ element.trigger('ajax:error', [xhr, status, error]);
74
+ }
75
+ });
76
+ }
77
+
78
+ // Handles "data-method" on links such as:
79
+ // <a href="/users/5" data-method="delete" rel="nofollow" data-confirm="Are you sure?">Delete</a>
80
+ function handleMethod(link) {
81
+ var href = link.attr('href'),
82
+ method = link.attr('data-method'),
83
+ csrf_token = $('meta[name=csrf-token]').attr('content'),
84
+ csrf_param = $('meta[name=csrf-param]').attr('content'),
85
+ form = $('<form method="post" action="' + href + '"></form>'),
86
+ metadata_input = '<input name="_method" value="' + method + '" type="hidden" />';
87
+
88
+ if (csrf_param !== undefined && csrf_token !== undefined) {
89
+ metadata_input += '<input name="' + csrf_param + '" value="' + csrf_token + '" type="hidden" />';
90
+ }
91
+
92
+ form.hide().append(metadata_input).appendTo('body');
93
+ form.submit();
94
+ }
95
+
96
+ function disableFormElements(form) {
97
+ form.find('input[data-disable-with]').each(function() {
98
+ var input = $(this);
99
+ input.data('ujs:enable-with', input.val())
100
+ .val(input.attr('data-disable-with'))
101
+ .attr('disabled', 'disabled');
102
+ });
103
+ }
104
+
105
+ function enableFormElements(form) {
106
+ form.find('input[data-disable-with]').each(function() {
107
+ var input = $(this);
108
+ input.val(input.data('ujs:enable-with')).removeAttr('disabled');
109
+ });
110
+ }
111
+
112
+ function allowAction(element) {
113
+ var message = element.attr('data-confirm');
114
+ return !message || (fire(element, 'confirm') && confirm(message));
115
+ }
116
+
117
+ function requiredValuesMissing(form) {
118
+ var missing = false;
119
+ form.find('input[name][required]').each(function() {
120
+ if (!$(this).val()) missing = true;
121
+ });
122
+ return missing;
123
+ }
124
+
125
+ $('a[data-confirm], a[data-method], a[data-remote]').live('click.rails', function(e) {
126
+ var link = $(this);
127
+ if (!allowAction(link)) return false;
128
+
129
+ if (link.attr('data-remote') != undefined) {
130
+ handleRemote(link);
131
+ return false;
132
+ } else if (link.attr('data-method')) {
133
+ handleMethod(link);
134
+ return false;
135
+ }
136
+ });
137
+
138
+ $('form').live('submit.rails', function(e) {
139
+ var form = $(this), remote = form.attr('data-remote') != undefined;
140
+ if (!allowAction(form)) return false;
141
+
142
+ // skip other logic when required values are missing
143
+ if (requiredValuesMissing(form)) return !remote;
144
+
145
+ if (remote) {
146
+ handleRemote(form);
147
+ return false;
148
+ } else {
149
+ // slight timeout so that the submit button gets properly serialized
150
+ setTimeout(function(){ disableFormElements(form) }, 13);
151
+ }
152
+ });
153
+
154
+ $('form input[type=submit], form button[type=submit], form button:not([type])').live('click.rails', function() {
155
+ var button = $(this);
156
+ if (!allowAction(button)) return false;
157
+ // register the pressed submit button
158
+ var name = button.attr('name'), data = name ? {name:name, value:button.val()} : null;
159
+ button.closest('form').data('ujs:submit-button', data);
160
+ });
161
+
162
+ $('form').live('ajax:beforeSend.rails', function(event) {
163
+ if (this == event.target) disableFormElements($(this));
164
+ });
165
+
166
+ $('form').live('ajax:complete.rails', function(event) {
167
+ if (this == event.target) enableFormElements($(this));
168
+ });
169
+ })( jQuery );
@@ -0,0 +1,30 @@
1
+ /*
2
+ * Modernizr v1.6
3
+ * http://www.modernizr.com
4
+ *
5
+ * Developed by:
6
+ * - Faruk Ates http://farukat.es/
7
+ * - Paul Irish http://paulirish.com/
8
+ *
9
+ * Copyright (c) 2009-2010
10
+ * Dual-licensed under the BSD or MIT licenses.
11
+ * http://www.modernizr.com/license/
12
+ */
13
+ window.Modernizr=function(i,e,u){function s(a,b){return(""+a).indexOf(b)!==-1}function D(a,b){for(var c in a)if(j[a[c]]!==u&&(!b||b(a[c],E)))return true}function n(a,b){var c=a.charAt(0).toUpperCase()+a.substr(1);c=(a+" "+F.join(c+" ")+c).split(" ");return!!D(c,b)}function S(){f.input=function(a){for(var b=0,c=a.length;b<c;b++)L[a[b]]=!!(a[b]in h);return L}("autocomplete autofocus list placeholder max min multiple pattern required step".split(" "));f.inputtypes=function(a){for(var b=0,c,k=a.length;b<
14
+ k;b++){h.setAttribute("type",a[b]);if(c=h.type!=="text"){h.value=M;if(/^range$/.test(h.type)&&h.style.WebkitAppearance!==u){l.appendChild(h);c=e.defaultView;c=c.getComputedStyle&&c.getComputedStyle(h,null).WebkitAppearance!=="textfield"&&h.offsetHeight!==0;l.removeChild(h)}else/^(search|tel)$/.test(h.type)||(c=/^(url|email)$/.test(h.type)?h.checkValidity&&h.checkValidity()===false:h.value!=M)}N[a[b]]=!!c}return N}("search tel url email datetime date month week time datetime-local number range color".split(" "))}
15
+ var f={},l=e.documentElement,E=e.createElement("modernizr"),j=E.style,h=e.createElement("input"),M=":)",O=Object.prototype.toString,q=" -webkit- -moz- -o- -ms- -khtml- ".split(" "),F="Webkit Moz O ms Khtml".split(" "),v={svg:"http://www.w3.org/2000/svg"},d={},N={},L={},P=[],w,Q=function(a){var b=document.createElement("style"),c=e.createElement("div");b.textContent=a+"{#modernizr{height:3px}}";(e.head||e.getElementsByTagName("head")[0]).appendChild(b);c.id="modernizr";l.appendChild(c);a=c.offsetHeight===
16
+ 3;b.parentNode.removeChild(b);c.parentNode.removeChild(c);return!!a},o=function(){var a={select:"input",change:"input",submit:"form",reset:"form",error:"img",load:"img",abort:"img"};return function(b,c){c=c||document.createElement(a[b]||"div");b="on"+b;var k=b in c;if(!k){c.setAttribute||(c=document.createElement("div"));if(c.setAttribute&&c.removeAttribute){c.setAttribute(b,"");k=typeof c[b]=="function";if(typeof c[b]!="undefined")c[b]=u;c.removeAttribute(b)}}return k}}(),G={}.hasOwnProperty,R;R=
17
+ typeof G!=="undefined"&&typeof G.call!=="undefined"?function(a,b){return G.call(a,b)}:function(a,b){return b in a&&typeof a.constructor.prototype[b]==="undefined"};d.flexbox=function(){var a=e.createElement("div"),b=e.createElement("div");(function(k,g,r,x){g+=":";k.style.cssText=(g+q.join(r+";"+g)).slice(0,-g.length)+(x||"")})(a,"display","box","width:42px;padding:0;");b.style.cssText=q.join("box-flex:1;")+"width:10px;";a.appendChild(b);l.appendChild(a);var c=b.offsetWidth===42;a.removeChild(b);
18
+ l.removeChild(a);return c};d.canvas=function(){var a=e.createElement("canvas");return!!(a.getContext&&a.getContext("2d"))};d.canvastext=function(){return!!(f.canvas&&typeof e.createElement("canvas").getContext("2d").fillText=="function")};d.webgl=function(){var a=e.createElement("canvas");try{if(a.getContext("webgl"))return true}catch(b){}try{if(a.getContext("experimental-webgl"))return true}catch(c){}return false};d.touch=function(){return"ontouchstart"in i||Q("@media ("+q.join("touch-enabled),(")+
19
+ "modernizr)")};d.geolocation=function(){return!!navigator.geolocation};d.postmessage=function(){return!!i.postMessage};d.websqldatabase=function(){return!!i.openDatabase};d.indexedDB=function(){for(var a=-1,b=F.length;++a<b;){var c=F[a].toLowerCase();if(i[c+"_indexedDB"]||i[c+"IndexedDB"])return true}return false};d.hashchange=function(){return o("hashchange",i)&&(document.documentMode===u||document.documentMode>7)};d.history=function(){return!!(i.history&&history.pushState)};d.draganddrop=function(){return o("drag")&&
20
+ o("dragstart")&&o("dragenter")&&o("dragover")&&o("dragleave")&&o("dragend")&&o("drop")};d.websockets=function(){return"WebSocket"in i};d.rgba=function(){j.cssText="background-color:rgba(150,255,150,.5)";return s(j.backgroundColor,"rgba")};d.hsla=function(){j.cssText="background-color:hsla(120,40%,100%,.5)";return s(j.backgroundColor,"rgba")||s(j.backgroundColor,"hsla")};d.multiplebgs=function(){j.cssText="background:url(//:),url(//:),red url(//:)";return/(url\s*\(.*?){3}/.test(j.background)};d.backgroundsize=
21
+ function(){return n("backgroundSize")};d.borderimage=function(){return n("borderImage")};d.borderradius=function(){return n("borderRadius","",function(a){return s(a,"orderRadius")})};d.boxshadow=function(){return n("boxShadow")};d.textshadow=function(){return e.createElement("div").style.textShadow===""};d.opacity=function(){var a=q.join("opacity:.5;")+"";j.cssText=a;return s(j.opacity,"0.5")};d.cssanimations=function(){return n("animationName")};d.csscolumns=function(){return n("columnCount")};d.cssgradients=
22
+ function(){var a=("background-image:"+q.join("gradient(linear,left top,right bottom,from(#9f9),to(white));background-image:")+q.join("linear-gradient(left top,#9f9, white);background-image:")).slice(0,-17);j.cssText=a;return s(j.backgroundImage,"gradient")};d.cssreflections=function(){return n("boxReflect")};d.csstransforms=function(){return!!D(["transformProperty","WebkitTransform","MozTransform","OTransform","msTransform"])};d.csstransforms3d=function(){var a=!!D(["perspectiveProperty","WebkitPerspective",
23
+ "MozPerspective","OPerspective","msPerspective"]);if(a)a=Q("@media ("+q.join("transform-3d),(")+"modernizr)");return a};d.csstransitions=function(){return n("transitionProperty")};d.fontface=function(){var a,b=e.head||e.getElementsByTagName("head")[0]||l,c=e.createElement("style"),k=e.implementation||{hasFeature:function(){return false}};c.type="text/css";b.insertBefore(c,b.firstChild);a=c.sheet||c.styleSheet;b=k.hasFeature("CSS2","")?function(g){if(!(a&&g))return false;var r=false;try{a.insertRule(g,
24
+ 0);r=!/unknown/i.test(a.cssRules[0].cssText);a.deleteRule(a.cssRules.length-1)}catch(x){}return r}:function(g){if(!(a&&g))return false;a.cssText=g;return a.cssText.length!==0&&!/unknown/i.test(a.cssText)&&a.cssText.replace(/\r+|\n+/g,"").indexOf(g.split(" ")[0])===0};f._fontfaceready=function(g){g(f.fontface)};return b('@font-face { font-family: "font"; src: "font.ttf"; }')};d.video=function(){var a=e.createElement("video"),b=!!a.canPlayType;if(b){b=new Boolean(b);b.ogg=a.canPlayType('video/ogg; codecs="theora"');
25
+ b.h264=a.canPlayType('video/mp4; codecs="avc1.42E01E"')||a.canPlayType('video/mp4; codecs="avc1.42E01E, mp4a.40.2"');b.webm=a.canPlayType('video/webm; codecs="vp8, vorbis"')}return b};d.audio=function(){var a=e.createElement("audio"),b=!!a.canPlayType;if(b){b=new Boolean(b);b.ogg=a.canPlayType('audio/ogg; codecs="vorbis"');b.mp3=a.canPlayType("audio/mpeg;");b.wav=a.canPlayType('audio/wav; codecs="1"');b.m4a=a.canPlayType("audio/x-m4a;")||a.canPlayType("audio/aac;")}return b};d.localstorage=function(){try{return"localStorage"in
26
+ i&&i.localStorage!==null}catch(a){return false}};d.sessionstorage=function(){try{return"sessionStorage"in i&&i.sessionStorage!==null}catch(a){return false}};d.webWorkers=function(){return!!i.Worker};d.applicationcache=function(){return!!i.applicationCache};d.svg=function(){return!!e.createElementNS&&!!e.createElementNS(v.svg,"svg").createSVGRect};d.inlinesvg=function(){var a=document.createElement("div");a.innerHTML="<svg/>";return(a.firstChild&&a.firstChild.namespaceURI)==v.svg};d.smil=function(){return!!e.createElementNS&&
27
+ /SVG/.test(O.call(e.createElementNS(v.svg,"animate")))};d.svgclippaths=function(){return!!e.createElementNS&&/SVG/.test(O.call(e.createElementNS(v.svg,"clipPath")))};for(var H in d)if(R(d,H)){w=H.toLowerCase();f[w]=d[H]();P.push((f[w]?"":"no-")+w)}f.input||S();f.crosswindowmessaging=f.postmessage;f.historymanagement=f.history;f.addTest=function(a,b){a=a.toLowerCase();if(!f[a]){b=!!b();l.className+=" "+(b?"":"no-")+a;f[a]=b;return f}};j.cssText="";E=h=null;i.attachEvent&&function(){var a=e.createElement("div");
28
+ a.innerHTML="<elem></elem>";return a.childNodes.length!==1}()&&function(a,b){function c(p){for(var m=-1;++m<r;)p.createElement(g[m])}function k(p,m){for(var I=p.length,t=-1,y,J=[];++t<I;){y=p[t];m=y.media||m;J.push(k(y.imports,m));J.push(y.cssText)}return J.join("")}var g="abbr|article|aside|audio|canvas|details|figcaption|figure|footer|header|hgroup|mark|meter|nav|output|progress|section|summary|time|video".split("|"),r=g.length,x=RegExp("<(/*)(abbr|article|aside|audio|canvas|details|figcaption|figure|footer|header|hgroup|mark|meter|nav|output|progress|section|summary|time|video)",
29
+ "gi"),T=RegExp("\\b(abbr|article|aside|audio|canvas|details|figcaption|figure|footer|header|hgroup|mark|meter|nav|output|progress|section|summary|time|video)\\b(?!.*[;}])","gi"),z=b.createDocumentFragment(),A=b.documentElement,K=A.firstChild,B=b.createElement("style"),C=b.createElement("body");B.media="all";c(b);c(z);a.attachEvent("onbeforeprint",function(){for(var p=-1;++p<r;)for(var m=b.getElementsByTagName(g[p]),I=m.length,t=-1;++t<I;)if(m[t].className.indexOf("iepp_")<0)m[t].className+=" iepp_"+
30
+ g[p];K.insertBefore(B,K.firstChild);B.styleSheet.cssText=k(b.styleSheets,"all").replace(T,".iepp_$1");z.appendChild(b.body);A.appendChild(C);C.innerHTML=z.firstChild.innerHTML.replace(x,"<$1bdo")});a.attachEvent("onafterprint",function(){C.innerHTML="";A.removeChild(C);K.removeChild(B);A.appendChild(z.firstChild)})}(this,document);f._enableHTML5=true;f._version="1.6";l.className=l.className.replace(/\bno-js\b/,"")+" js";l.className+=" "+P.join(" ");return f}(this,this.document);
metadata ADDED
@@ -0,0 +1,136 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: da-suspenders
3
+ version: !ruby/object:Gem::Version
4
+ prerelease: false
5
+ segments:
6
+ - 1
7
+ - 0
8
+ - 0
9
+ version: 1.0.0
10
+ platform: ruby
11
+ authors:
12
+ - thoughtbot
13
+ - Stefan Daschek
14
+ autorequire:
15
+ bindir: bin
16
+ cert_chain: []
17
+
18
+ date: 2011-02-14 00:00:00 +01:00
19
+ default_executable:
20
+ dependencies:
21
+ - !ruby/object:Gem::Dependency
22
+ name: rails
23
+ prerelease: false
24
+ requirement: &id001 !ruby/object:Gem::Requirement
25
+ requirements:
26
+ - - ">="
27
+ - !ruby/object:Gem::Version
28
+ segments:
29
+ - 3
30
+ - 0
31
+ - 4
32
+ version: 3.0.4
33
+ type: :runtime
34
+ version_requirements: *id001
35
+ - !ruby/object:Gem::Dependency
36
+ name: bundler
37
+ prerelease: false
38
+ requirement: &id002 !ruby/object:Gem::Requirement
39
+ requirements:
40
+ - - ">="
41
+ - !ruby/object:Gem::Version
42
+ segments:
43
+ - 1
44
+ - 0
45
+ - 10
46
+ version: 1.0.10
47
+ type: :runtime
48
+ version_requirements: *id002
49
+ - !ruby/object:Gem::Dependency
50
+ name: trout
51
+ prerelease: false
52
+ requirement: &id003 !ruby/object:Gem::Requirement
53
+ requirements:
54
+ - - ">="
55
+ - !ruby/object:Gem::Version
56
+ segments:
57
+ - 0
58
+ - 3
59
+ - 0
60
+ version: 0.3.0
61
+ type: :runtime
62
+ version_requirements: *id003
63
+ description: DIE ANTWORT's fork of thoughtbot's original Suspenders. Suspenders is an upgradeable base Rails project.
64
+ email: stefan@die-antwort.eu
65
+ executables:
66
+ - da-suspenders
67
+ extensions: []
68
+
69
+ extra_rdoc_files:
70
+ - README.md
71
+ - LICENSE
72
+ files:
73
+ - .gitignore
74
+ - LICENSE
75
+ - README.md
76
+ - Rakefile
77
+ - bin/da-suspenders
78
+ - da-suspenders.gemspec
79
+ - features/running_cucumber.feature
80
+ - features/step_definitions/shell.rb
81
+ - features/support/env.rb
82
+ - lib/create.rb
83
+ - lib/da-suspenders/version.rb
84
+ - lib/errors.rb
85
+ - template/da-suspenders.rb
86
+ - template/files/factory_girl_steps.rb
87
+ - template/files/mysql_database.yml.erb
88
+ - template/trout/Gemfile
89
+ - template/trout/app/helpers/body_class_helper.rb
90
+ - template/trout/app/javascripts/application.js
91
+ - template/trout/app/stylesheets/application.scss
92
+ - template/trout/app/stylesheets/ie.scss
93
+ - template/trout/app/views/layouts/application.html.erb
94
+ - template/trout/app/views/shared/_flashes.html.erb
95
+ - template/trout/config/initializers/errors.rb
96
+ - template/trout/config/locales/de.yml
97
+ - template/trout/features/step_definitions/js_steps.rb
98
+ - template/trout/lib/templates/erb/scaffold/_form.html.erb
99
+ - template/trout/vendor/sprockets/jquery-ujs/src/rails.js
100
+ - template/trout/vendor/sprockets/modernizr/src/modernizr.js
101
+ has_rdoc: true
102
+ homepage: http://github.com/die-antwort/da-suspenders
103
+ licenses: []
104
+
105
+ post_install_message:
106
+ rdoc_options:
107
+ - --charset=UTF-8
108
+ require_paths:
109
+ - lib
110
+ required_ruby_version: !ruby/object:Gem::Requirement
111
+ requirements:
112
+ - - ">="
113
+ - !ruby/object:Gem::Version
114
+ segments:
115
+ - 0
116
+ version: "0"
117
+ required_rubygems_version: !ruby/object:Gem::Requirement
118
+ requirements:
119
+ - - ">="
120
+ - !ruby/object:Gem::Version
121
+ segments:
122
+ - 1
123
+ - 3
124
+ - 6
125
+ version: 1.3.6
126
+ requirements: []
127
+
128
+ rubyforge_project:
129
+ rubygems_version: 1.3.6
130
+ signing_key:
131
+ specification_version: 3
132
+ summary: Generate a Rails app using DIE ANTWORT's best practices.
133
+ test_files:
134
+ - features/running_cucumber.feature
135
+ - features/step_definitions/shell.rb
136
+ - features/support/env.rb