├── .gitignore
├── .rspec
├── .ruby-gemset
├── .ruby-version
├── Gemfile
├── Gemfile.lock
├── README.md
├── Rakefile
├── app
├── assets
│ ├── images
│ │ └── rails.png
│ ├── javascripts
│ │ ├── application.js
│ │ ├── cors.js
│ │ ├── products.js
│ │ └── scaffold.js
│ └── stylesheets
│ │ ├── application.css.scss
│ │ └── layout.css.scss
├── controllers
│ ├── application_controller.rb
│ ├── aws_controller.rb
│ └── products_controller.rb
├── mailers
│ └── .gitkeep
├── models
│ ├── .gitkeep
│ └── product.rb
└── views
│ ├── application
│ └── _flash_messages.html.erb
│ ├── layouts
│ └── application.html.erb
│ └── products
│ └── new.html.erb
├── config.ru
├── config
├── application.rb
├── boot.rb
├── database.yml
├── environment.rb
├── environments
│ ├── development.rb
│ ├── production.rb
│ └── test.rb
├── initializers
│ ├── aws.rb
│ ├── backtrace_silencers.rb
│ ├── inflections.rb
│ ├── mime_types.rb
│ ├── session_store.rb
│ ├── simple_form.rb
│ ├── simple_form_bootstrap.rb
│ └── wrap_parameters.rb
├── locales
│ ├── en.yml
│ └── simple_form.en.yml
├── routes.rb
└── secrets.yml
├── db
├── migrate
│ └── 20130523082752_create_products.rb
├── schema.rb
└── seeds.rb
├── doc
└── README_FOR_APP
├── log
└── .gitkeep
├── public
├── 404.html
├── 422.html
├── 500.html
├── favicon.ico
├── img
│ ├── glyphicons-halflings-white.png
│ └── glyphicons-halflings.png
└── robots.txt
├── script
└── rails
├── spec
├── controllers
│ ├── aws_controller_spec.rb
│ └── products_controller_spec.rb
├── helpers
│ └── products_helper_spec.rb
└── spec_helper.rb
├── test
├── fixtures
│ ├── .gitkeep
│ └── products.yml
├── functional
│ └── .gitkeep
├── integration
│ └── .gitkeep
├── performance
│ └── browsing_test.rb
├── test_helper.rb
└── unit
│ ├── .gitkeep
│ └── product_test.rb
└── vendor
├── assets
├── javascripts
│ ├── .gitkeep
│ ├── bootstrap-filestyle.js
│ └── bootstrap.js
└── stylesheets
│ ├── .gitkeep
│ ├── bootstrap-responsive.css
│ └── bootstrap.css
└── plugins
└── .gitkeep
/.gitignore:
--------------------------------------------------------------------------------
1 | # See http://help.github.com/ignore-files/ for more about ignoring files.
2 | #
3 | # If you find yourself ignoring temporary files generated by your text editor
4 | # or operating system, you probably want to add a global ignore instead:
5 | # git config --global core.excludesfile ~/.gitignore_global
6 |
7 | # Ignore bundler config
8 | /.bundle
9 |
10 | # Ignore the default SQLite database.
11 | /db/*.sqlite3
12 |
13 | # Ignore all logfiles and tempfiles.
14 | /log/*.log
15 | /tmp
16 | export.txt
17 | .env
18 | Procfile
19 |
--------------------------------------------------------------------------------
/.rspec:
--------------------------------------------------------------------------------
1 | --color
2 |
--------------------------------------------------------------------------------
/.ruby-gemset:
--------------------------------------------------------------------------------
1 | s3cors
2 |
--------------------------------------------------------------------------------
/.ruby-version:
--------------------------------------------------------------------------------
1 | ruby-2.2.2
2 |
--------------------------------------------------------------------------------
/Gemfile:
--------------------------------------------------------------------------------
1 | source "https://rubygems.org"
2 |
3 | gem "rails", "4.2.3"
4 | gem "sass-rails"
5 | gem 'uglifier', '>= 1.3.0'
6 | gem "jquery-rails"
7 | gem 'turbolinks'
8 | gem 'jbuilder', '~> 2.0'
9 | gem 'sdoc', '~> 0.4.0', group: :doc
10 |
11 | gem 'bootstrap-sass', '~> 3.3.1'
12 | gem 'autoprefixer-rails'
13 |
14 | gem "coffee-rails"
15 | gem "bcrypt"
16 | gem "simple_form"
17 | gem "cognac"
18 |
19 | group :development, :test do
20 | gem "sqlite3"
21 | gem "rspec-rails"
22 | gem 'quiet_assets'
23 | end
24 |
25 | group :test do
26 | # Pretty printed test output
27 | gem "turn", :require => false
28 | gem "factory_girl_rails"
29 | end
--------------------------------------------------------------------------------
/Gemfile.lock:
--------------------------------------------------------------------------------
1 | GEM
2 | remote: https://rubygems.org/
3 | specs:
4 | actionmailer (4.2.3)
5 | actionpack (= 4.2.3)
6 | actionview (= 4.2.3)
7 | activejob (= 4.2.3)
8 | mail (~> 2.5, >= 2.5.4)
9 | rails-dom-testing (~> 1.0, >= 1.0.5)
10 | actionpack (4.2.3)
11 | actionview (= 4.2.3)
12 | activesupport (= 4.2.3)
13 | rack (~> 1.6)
14 | rack-test (~> 0.6.2)
15 | rails-dom-testing (~> 1.0, >= 1.0.5)
16 | rails-html-sanitizer (~> 1.0, >= 1.0.2)
17 | actionview (4.2.3)
18 | activesupport (= 4.2.3)
19 | builder (~> 3.1)
20 | erubis (~> 2.7.0)
21 | rails-dom-testing (~> 1.0, >= 1.0.5)
22 | rails-html-sanitizer (~> 1.0, >= 1.0.2)
23 | activejob (4.2.3)
24 | activesupport (= 4.2.3)
25 | globalid (>= 0.3.0)
26 | activemodel (4.2.3)
27 | activesupport (= 4.2.3)
28 | builder (~> 3.1)
29 | activerecord (4.2.3)
30 | activemodel (= 4.2.3)
31 | activesupport (= 4.2.3)
32 | arel (~> 6.0)
33 | activesupport (4.2.3)
34 | i18n (~> 0.7)
35 | json (~> 1.7, >= 1.7.7)
36 | minitest (~> 5.1)
37 | thread_safe (~> 0.3, >= 0.3.4)
38 | tzinfo (~> 1.1)
39 | ansi (1.5.0)
40 | arel (6.0.0)
41 | autoprefixer-rails (5.2.1)
42 | execjs
43 | json
44 | bcrypt (3.1.10)
45 | bootstrap-sass (3.3.5.1)
46 | autoprefixer-rails (>= 5.0.0.1)
47 | sass (>= 3.3.0)
48 | builder (3.2.2)
49 | coffee-rails (4.1.0)
50 | coffee-script (>= 2.2.0)
51 | railties (>= 4.0.0, < 5.0)
52 | coffee-script (2.4.1)
53 | coffee-script-source
54 | execjs
55 | coffee-script-source (1.9.1.1)
56 | cognac (0.1.1)
57 | activesupport (~> 4.2)
58 | diff-lcs (1.2.5)
59 | erubis (2.7.0)
60 | execjs (2.5.2)
61 | factory_girl (4.5.0)
62 | activesupport (>= 3.0.0)
63 | factory_girl_rails (4.5.0)
64 | factory_girl (~> 4.5.0)
65 | railties (>= 3.0.0)
66 | globalid (0.3.5)
67 | activesupport (>= 4.1.0)
68 | i18n (0.7.0)
69 | jbuilder (2.3.0)
70 | activesupport (>= 3.0.0, < 5)
71 | multi_json (~> 1.2)
72 | jquery-rails (4.0.4)
73 | rails-dom-testing (~> 1.0)
74 | railties (>= 4.2.0)
75 | thor (>= 0.14, < 2.0)
76 | json (1.8.3)
77 | loofah (2.0.2)
78 | nokogiri (>= 1.5.9)
79 | mail (2.6.3)
80 | mime-types (>= 1.16, < 3)
81 | mime-types (2.6.1)
82 | mini_portile (0.6.2)
83 | minitest (5.7.0)
84 | multi_json (1.11.1)
85 | nokogiri (1.6.6.2)
86 | mini_portile (~> 0.6.0)
87 | quiet_assets (1.1.0)
88 | railties (>= 3.1, < 5.0)
89 | rack (1.6.4)
90 | rack-test (0.6.3)
91 | rack (>= 1.0)
92 | rails (4.2.3)
93 | actionmailer (= 4.2.3)
94 | actionpack (= 4.2.3)
95 | actionview (= 4.2.3)
96 | activejob (= 4.2.3)
97 | activemodel (= 4.2.3)
98 | activerecord (= 4.2.3)
99 | activesupport (= 4.2.3)
100 | bundler (>= 1.3.0, < 2.0)
101 | railties (= 4.2.3)
102 | sprockets-rails
103 | rails-deprecated_sanitizer (1.0.3)
104 | activesupport (>= 4.2.0.alpha)
105 | rails-dom-testing (1.0.6)
106 | activesupport (>= 4.2.0.beta, < 5.0)
107 | nokogiri (~> 1.6.0)
108 | rails-deprecated_sanitizer (>= 1.0.1)
109 | rails-html-sanitizer (1.0.2)
110 | loofah (~> 2.0)
111 | railties (4.2.3)
112 | actionpack (= 4.2.3)
113 | activesupport (= 4.2.3)
114 | rake (>= 0.8.7)
115 | thor (>= 0.18.1, < 2.0)
116 | rake (10.4.2)
117 | rdoc (4.2.0)
118 | rspec-core (3.3.1)
119 | rspec-support (~> 3.3.0)
120 | rspec-expectations (3.3.0)
121 | diff-lcs (>= 1.2.0, < 2.0)
122 | rspec-support (~> 3.3.0)
123 | rspec-mocks (3.3.1)
124 | diff-lcs (>= 1.2.0, < 2.0)
125 | rspec-support (~> 3.3.0)
126 | rspec-rails (3.3.2)
127 | actionpack (>= 3.0, < 4.3)
128 | activesupport (>= 3.0, < 4.3)
129 | railties (>= 3.0, < 4.3)
130 | rspec-core (~> 3.3.0)
131 | rspec-expectations (~> 3.3.0)
132 | rspec-mocks (~> 3.3.0)
133 | rspec-support (~> 3.3.0)
134 | rspec-support (3.3.0)
135 | sass (3.4.15)
136 | sass-rails (5.0.3)
137 | railties (>= 4.0.0, < 5.0)
138 | sass (~> 3.1)
139 | sprockets (>= 2.8, < 4.0)
140 | sprockets-rails (>= 2.0, < 4.0)
141 | tilt (~> 1.1)
142 | sdoc (0.4.1)
143 | json (~> 1.7, >= 1.7.7)
144 | rdoc (~> 4.0)
145 | simple_form (3.1.0)
146 | actionpack (~> 4.0)
147 | activemodel (~> 4.0)
148 | sprockets (3.2.0)
149 | rack (~> 1.0)
150 | sprockets-rails (2.3.2)
151 | actionpack (>= 3.0)
152 | activesupport (>= 3.0)
153 | sprockets (>= 2.8, < 4.0)
154 | sqlite3 (1.3.10)
155 | thor (0.19.1)
156 | thread_safe (0.3.5)
157 | tilt (1.4.1)
158 | turbolinks (2.5.3)
159 | coffee-rails
160 | turn (0.9.6)
161 | ansi
162 | tzinfo (1.2.2)
163 | thread_safe (~> 0.1)
164 | uglifier (2.7.1)
165 | execjs (>= 0.3.0)
166 | json (>= 1.8.0)
167 |
168 | PLATFORMS
169 | ruby
170 |
171 | DEPENDENCIES
172 | autoprefixer-rails
173 | bcrypt
174 | bootstrap-sass (~> 3.3.1)
175 | coffee-rails
176 | cognac
177 | factory_girl_rails
178 | jbuilder (~> 2.0)
179 | jquery-rails
180 | quiet_assets
181 | rails (= 4.2.3)
182 | rspec-rails
183 | sass-rails
184 | sdoc (~> 0.4.0)
185 | simple_form
186 | sqlite3
187 | turbolinks
188 | turn
189 | uglifier (>= 1.3.0)
190 |
191 | BUNDLED WITH
192 | 1.10.3
193 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ## Step 1 - Configure S3 for CORS
2 |
3 | 1. Create an Amazon S3 Bucket
4 | 2. Click on Properties -> Permissions -> Edit CORS
5 | 3. Enter a CORS configuration, such as the following. For security purposes you should restrict the origin further.
6 |
7 | ```xml
8 |
9 |
10 |
11 | *
12 | PUT
13 | 3000
14 | *
15 |
16 |
17 | ```
18 |
19 | ## Step 2 - Set your AWS Keys and bucket
20 |
21 | The application can be configured via the following environment variables.
22 |
23 | - `export AWS_ACCESS_KEY_ID='your aws acccess key id'`
24 | - `export AWS_SECRET_ACCESS_KEY='your aws secret access key'`
25 | - `export AWS_S3_BUCKET='example.com'`
26 |
27 |
28 | ## General Flow
29 |
30 | This project uses client-side JavaScript and server-side Ruby. In general, the completed file-upload process follows these steps:
31 |
32 | 1. A file is selected for upload by the user in their web browser;
33 | 2. JavaScript is then responsible for making a request to your web application, which produces a temporary signature with which to sign the upload request;
34 | 3. The temporary signed request is returned to the browser in JSON format;
35 | 4. JavaScript then uploads the file directly to Amazon S3 using the signed request supplied by your Rails application.
36 |
37 | ## Code overview
38 |
39 | The application flow is as follows:
40 |
41 | 1. [Client] When the user chooses a file, a callback (defined in `app/assets/javascripts/products.js`) contacts the server to request a signed URL that can be used for uploading to S3.
42 | 2. [Server] AWSController generates a signed URL according to http://docs.aws.amazon.com/AmazonS3/latest/dev/RESTAuthentication.html and returns it using a JSON hash.
43 | 3. [Client] The `uploadToS3` function (defined in `app/assets/javascripts/cors.js`), takes the signed URL and the file, and PUTs the file onto S3.
44 | 4. [Client] Once the upload is finished, the URL of the uploaded file is appended to the form as a hidden field.
45 | 5. [Client] When the user submits the form, the file field is removed (to avoid uploading the file again). Name, price and the file URL are POSTed to the server.
46 | 6. [Server] A new product record is created.
47 |
48 | The most important files are are:
49 |
50 | - `app/controllers/aws_controller` - Given a filename and content type, returns a JSON of the form:
51 |
52 | ```JSON
53 | {
54 | "put_url": "The *signed* URL that must be used to make the request from the client side]",
55 | "file_url": "The url the file will be uploaded to. A random number is appended to the file name to avoid name collisions."
56 | }
57 | ```
58 | - `app/assets/javascripts/cors.js` - Performs the actual upload using an XMLHttpRequest. Required the signed URL, and the file to perform the request.
59 | - `app/assets/javascripts/products.js` - Handles UI callbacks, such as updating the progress bar. Also handles the file-upload callback. When the user chooses a file, the server is contacted to generate a signed URL, which is then passed to the CORS script described above to complete the upload.
--------------------------------------------------------------------------------
/Rakefile:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env rake
2 | # Add your own tasks in files placed in lib/tasks ending in .rake,
3 | # for example lib/tasks/capistrano.rake, and they will automatically be available to Rake.
4 |
5 | require File.expand_path('../config/application', __FILE__)
6 |
7 | Rails.application.load_tasks
--------------------------------------------------------------------------------
/app/assets/images/rails.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/app/assets/images/rails.png
--------------------------------------------------------------------------------
/app/assets/javascripts/application.js:
--------------------------------------------------------------------------------
1 | // This is a manifest file that'll be compiled into application.js, which will include all the files
2 | // listed below.
3 | //
4 | // Any JavaScript/Coffee file within this directory, lib/assets/javascripts, vendor/assets/javascripts,
5 | // or vendor/assets/javascripts of plugins, if any, can be referenced here using a relative path.
6 | //
7 | // It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the
8 | // the compiled file.
9 | //
10 | // WARNING: THE FIRST BLANK LINE MARKS THE END OF WHAT'S TO BE PROCESSED, ANY BLANK LINE SHOULD
11 | // GO AFTER THE REQUIRES BELOW.
12 | //
13 | //= require jquery
14 | //= require jquery_ujs
15 | //= require bootstrap
16 | //= require bootstrap-filestyle
17 | //
18 | //= require scaffold
19 | //= require products
20 | //= require cors
21 |
--------------------------------------------------------------------------------
/app/assets/javascripts/cors.js:
--------------------------------------------------------------------------------
1 | function createCORSRequest(method, url)
2 | {
3 | var xhr = new XMLHttpRequest();
4 | if ("withCredentials" in xhr)
5 | {
6 | xhr.open(method, url, true);
7 | }
8 | else if (typeof XDomainRequest != "undefined")
9 | {
10 | xhr = new XDomainRequest();
11 | xhr.open(method, url);
12 | }
13 | else
14 | {
15 | xhr = null;
16 | }
17 | return xhr;
18 | }
19 |
20 | /**
21 | * Use a CORS call to upload the given file to S3. Assumes the url
22 | * parameter has been signed and is accessible for upload.
23 | */
24 | function uploadToS3(file, url)
25 | {
26 | var xhr = createCORSRequest('PUT', url);
27 | if (!xhr)
28 | {
29 | setProgress(0, 'CORS not supported');
30 | }
31 | else
32 | {
33 | xhr.onload = function()
34 | {
35 | if(xhr.status == 200)
36 | {
37 | setProgress(100, 'Upload completed.');
38 | }
39 | else
40 | {
41 | setProgress(0, 'Upload error: ' + xhr.status);
42 | }
43 | };
44 |
45 | xhr.onerror = function()
46 | {
47 | setProgress(0, 'XHR error.');
48 | };
49 |
50 | xhr.upload.onprogress = function(e)
51 | {
52 | if (e.lengthComputable)
53 | {
54 | var percentLoaded = Math.round((e.loaded / e.total) * 100);
55 | setProgress(percentLoaded, percentLoaded == 100 ? 'Finalizing.' : 'Uploading.');
56 | }
57 | };
58 |
59 | xhr.setRequestHeader('Content-Type', file.type);
60 | //xhr.setRequestHeader('x-amz-acl', 'public-read');
61 | xhr.send(file);
62 | }
63 | }
64 |
--------------------------------------------------------------------------------
/app/assets/javascripts/products.js:
--------------------------------------------------------------------------------
1 | window.uploadFile = function(file) {
2 | return $.getJSON("/generate_signed_s3_url?filename=" + file.name.replace(/ /g,"_") + "&content_type=" + file.type, function(data) {
3 | uploadToS3(file, data.put_url);
4 | return $("#product_url").val(data.file_url);
5 | });
6 | };
7 |
8 | window.setProgress = function(progress, str) {
9 | $(".progress .bar").css("width", progress + "%");
10 | $("#product-upload-progress .text").text(str);
11 | if (progress === 100) {
12 | return $("#new_product_submit").show();
13 | }
14 | };
15 |
16 | $(function() {
17 | $("#product-upload-progress").hide();
18 | $("#new_product_submit").hide();
19 | $("#new_product_submit").click(function() {
20 | $("#product_file").remove();
21 | return true;
22 | });
23 | return $("#product_file").change(function(obj) {
24 | var file;
25 |
26 | file = obj.target.files[0];
27 | if (file.size > 500 * 1024 * 1024) {
28 | alert("File cannot be larger than 500MB.");
29 | return false;
30 | }
31 | $("#product-upload-progress").show();
32 | return uploadFile(file);
33 | });
34 | });
--------------------------------------------------------------------------------
/app/assets/javascripts/scaffold.js:
--------------------------------------------------------------------------------
1 | $(function() {
2 | $(":file").filestyle({
3 | icon: false
4 | });
5 | });
--------------------------------------------------------------------------------
/app/assets/stylesheets/application.css.scss:
--------------------------------------------------------------------------------
1 | /*
2 | * This is a manifest file that'll be compiled into application.css, which will include all the files
3 | * listed below.
4 | *
5 | * Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets,
6 | * or vendor/assets/stylesheets of plugins, if any, can be referenced here using a relative path.
7 | *
8 | * You're free to add application-wide styles to this file and they'll appear at the top of the
9 | * compiled file, but it's generally better to create a new file per style scope.
10 | *
11 | *= require_self
12 | */
13 |
14 | @import "bootstrap-sprockets";
15 | @import "bootstrap";
16 | @import "bootstrap-responsive";
17 |
18 | @import "layout";
--------------------------------------------------------------------------------
/app/assets/stylesheets/layout.css.scss:
--------------------------------------------------------------------------------
1 | body {
2 | padding: 10px;
3 | }
--------------------------------------------------------------------------------
/app/controllers/application_controller.rb:
--------------------------------------------------------------------------------
1 | class ApplicationController < ActionController::Base
2 | # Prevent CSRF attacks by raising an exception.
3 | # For APIs, you may want to use :null_session instead.
4 | protect_from_forgery with: :exception
5 | end
6 |
--------------------------------------------------------------------------------
/app/controllers/aws_controller.rb:
--------------------------------------------------------------------------------
1 | class AwsController < ApplicationController
2 |
3 | def generate_signed_s3_url
4 | # To avoid file collision, we prepend string to the file_name
5 | file_name = Cognac::CloudFile.generate(params[:filename])
6 | resource_end_point = Cognac::CloudFile.resource_end_point(ENV["AWS_S3_BUCKET"], file_name)
7 |
8 | options = Cognac::Signature.generate_options_for_build_s3_upload_url(ENV["AWS_S3_BUCKET"], file_name, params[:content_type])
9 | url = Cognac::Signature.build_s3_upload_url(resource_end_point, ENV["AWS_ACCESS_KEY_ID"], ENV["AWS_SECRET_ACCESS_KEY"], options)
10 |
11 | render :json => {:put_url => url, :file_url => resource_end_point}
12 | end
13 | end
14 |
--------------------------------------------------------------------------------
/app/controllers/products_controller.rb:
--------------------------------------------------------------------------------
1 | class ProductsController < ApplicationController
2 |
3 | def new
4 | @product = Product.new
5 | end
6 |
7 | def create
8 | @product = Product.new(permitted_params)
9 |
10 | if @product.save
11 | redirect_to root_path, :notice => "Upload was successful."
12 | else
13 | render :new
14 | end
15 | end
16 |
17 | private
18 |
19 | def permitted_params
20 | params.require(:product).permit(:name, :price, :url)
21 | end
22 | end
23 |
--------------------------------------------------------------------------------
/app/mailers/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/app/mailers/.gitkeep
--------------------------------------------------------------------------------
/app/models/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/app/models/.gitkeep
--------------------------------------------------------------------------------
/app/models/product.rb:
--------------------------------------------------------------------------------
1 | class Product < ActiveRecord::Base
2 |
3 | validates :name, :presence => true
4 | validates :price, :presence => true
5 | validates :url, :presence => true
6 |
7 | end
8 |
--------------------------------------------------------------------------------
/app/views/application/_flash_messages.html.erb:
--------------------------------------------------------------------------------
1 | <% if flash[:notice] %>
2 |
3 | <%= link_to "#", :class => "close", :data => {:dismiss => "alert"} do %>
4 | ×
5 | <% end %>
6 | <%= flash[:notice] %>
7 |
8 | <% end %>
9 |
10 | <% if flash[:error] %>
11 |
12 | <%= link_to "#", :class => "close", :data => {:dismiss => "alert"} do %>
13 | ×
14 | <% end %>
15 | <%= flash[:error] %>
16 |
17 | <% end %>
--------------------------------------------------------------------------------
/app/views/layouts/application.html.erb:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | S3CorsUploadRails
5 | <%= stylesheet_link_tag "application", :media => "all" %>
6 | <%= javascript_include_tag "application" %>
7 | <%= csrf_meta_tags %>
8 |
9 |
10 |
11 | <%= render "flash_messages" %>
12 | <%= yield %>
13 |
14 |
15 |
--------------------------------------------------------------------------------
/app/views/products/new.html.erb:
--------------------------------------------------------------------------------
1 | <%= simple_form_for @product, :html => {:class => "form-horizontal"} do |f| %>
2 |
3 | <%= f.input :name, :required => true, :input_html => {:required => "required"} %>
4 | <%= f.input :price, :required => true, :input_html => {:required => "required"} %>
5 | <%= f.hidden_field :url, :required => true %>
6 | <%= f.input :file do %>
7 | <%= f.file_field :files, :id => "product_file", :accept => ".pdf,.mobi,.epub" %>
8 |
9 |
10 |
19 | <% end %>
20 |
21 | <%= f.button :submit, :id => "new_product_submit", :class => "btn btn-primary" %>
22 |
23 | <% end %>
--------------------------------------------------------------------------------
/config.ru:
--------------------------------------------------------------------------------
1 | # This file is used by Rack-based servers to start the application.
2 |
3 | require ::File.expand_path('../config/environment', __FILE__)
4 | run Rails.application
5 |
--------------------------------------------------------------------------------
/config/application.rb:
--------------------------------------------------------------------------------
1 | require File.expand_path('../boot', __FILE__)
2 |
3 | require 'rails/all'
4 |
5 | # Require the gems listed in Gemfile, including any gems
6 | # you've limited to :test, :development, or :production.
7 | Bundler.require(*Rails.groups)
8 |
9 | module Blog
10 | class Application < Rails::Application
11 | # Settings in config/environments/* take precedence over those specified here.
12 | # Application configuration should go into files in config/initializers
13 | # -- all .rb files in that directory are automatically loaded.
14 |
15 | # Set Time.zone default to the specified zone and make Active Record auto-convert to this zone.
16 | # Run "rake -D time" for a list of tasks for finding time zone names. Default is UTC.
17 | # config.time_zone = 'Central Time (US & Canada)'
18 |
19 | # The default locale is :en and all translations from config/locales/*.rb,yml are auto loaded.
20 | # config.i18n.load_path += Dir[Rails.root.join('my', 'locales', '*.{rb,yml}').to_s]
21 | # config.i18n.default_locale = :de
22 |
23 | # Do not swallow errors in after_commit/after_rollback callbacks.
24 | config.active_record.raise_in_transactional_callbacks = true
25 | end
26 | end
27 |
--------------------------------------------------------------------------------
/config/boot.rb:
--------------------------------------------------------------------------------
1 | ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../../Gemfile', __FILE__)
2 |
3 | require 'bundler/setup' # Set up gems listed in the Gemfile.
4 |
--------------------------------------------------------------------------------
/config/database.yml:
--------------------------------------------------------------------------------
1 | # SQLite version 3.x
2 | # gem install sqlite3
3 | #
4 | # Ensure the SQLite 3 gem is defined in your Gemfile
5 | # gem 'sqlite3'
6 | development:
7 | adapter: sqlite3
8 | database: db/development.sqlite3
9 | pool: 5
10 | timeout: 5000
11 |
12 | # Warning: The database defined as "test" will be erased and
13 | # re-generated from your development database when you run "rake".
14 | # Do not set this db to the same as development or production.
15 | test:
16 | adapter: sqlite3
17 | database: db/test.sqlite3
18 | pool: 5
19 | timeout: 5000
20 |
21 | production:
22 | adapter: sqlite3
23 | database: db/production.sqlite3
24 | pool: 5
25 | timeout: 5000
26 |
--------------------------------------------------------------------------------
/config/environment.rb:
--------------------------------------------------------------------------------
1 | # Load the Rails application.
2 | require File.expand_path('../application', __FILE__)
3 |
4 | # Initialize the Rails application.
5 | Rails.application.initialize!
6 |
--------------------------------------------------------------------------------
/config/environments/development.rb:
--------------------------------------------------------------------------------
1 | Rails.application.configure do
2 | # Settings specified here will take precedence over those in config/application.rb.
3 |
4 | # In the development environment your application's code is reloaded on
5 | # every request. This slows down response time but is perfect for development
6 | # since you don't have to restart the web server when you make code changes.
7 | config.cache_classes = false
8 |
9 | # Do not eager load code on boot.
10 | config.eager_load = false
11 |
12 | # Show full error reports and disable caching.
13 | config.consider_all_requests_local = true
14 | config.action_controller.perform_caching = false
15 |
16 | # Don't care if the mailer can't send.
17 | config.action_mailer.raise_delivery_errors = false
18 |
19 | # Print deprecation notices to the Rails logger.
20 | config.active_support.deprecation = :log
21 |
22 | # Raise an error on page load if there are pending migrations.
23 | config.active_record.migration_error = :page_load
24 |
25 | # Debug mode disables concatenation and preprocessing of assets.
26 | # This option may cause significant delays in view rendering with a large
27 | # number of complex assets.
28 | config.assets.debug = true
29 |
30 | # Asset digests allow you to set far-future HTTP expiration dates on all assets,
31 | # yet still be able to expire them through the digest params.
32 | config.assets.digest = true
33 |
34 | # Adds additional error checking when serving assets at runtime.
35 | # Checks for improperly declared sprockets dependencies.
36 | # Raises helpful error messages.
37 | config.assets.raise_runtime_errors = true
38 |
39 | # Raises error for missing translations
40 | # config.action_view.raise_on_missing_translations = true
41 | end
42 |
--------------------------------------------------------------------------------
/config/environments/production.rb:
--------------------------------------------------------------------------------
1 | Rails.application.configure do
2 | # Settings specified here will take precedence over those in config/application.rb.
3 |
4 | # Code is not reloaded between requests.
5 | config.cache_classes = true
6 |
7 | # Eager load code on boot. This eager loads most of Rails and
8 | # your application in memory, allowing both threaded web servers
9 | # and those relying on copy on write to perform better.
10 | # Rake tasks automatically ignore this option for performance.
11 | config.eager_load = true
12 |
13 | # Full error reports are disabled and caching is turned on.
14 | config.consider_all_requests_local = false
15 | config.action_controller.perform_caching = true
16 |
17 | # Enable Rack::Cache to put a simple HTTP cache in front of your application
18 | # Add `rack-cache` to your Gemfile before enabling this.
19 | # For large-scale production use, consider using a caching reverse proxy like
20 | # NGINX, varnish or squid.
21 | # config.action_dispatch.rack_cache = true
22 |
23 | # Disable serving static files from the `/public` folder by default since
24 | # Apache or NGINX already handles this.
25 | config.serve_static_files = ENV['RAILS_SERVE_STATIC_FILES'].present?
26 |
27 | # Compress JavaScripts and CSS.
28 | config.assets.js_compressor = :uglifier
29 | # config.assets.css_compressor = :sass
30 |
31 | # Do not fallback to assets pipeline if a precompiled asset is missed.
32 | config.assets.compile = false
33 |
34 | # Asset digests allow you to set far-future HTTP expiration dates on all assets,
35 | # yet still be able to expire them through the digest params.
36 | config.assets.digest = true
37 |
38 | # `config.assets.precompile` and `config.assets.version` have moved to config/initializers/assets.rb
39 |
40 | # Specifies the header that your server uses for sending files.
41 | # config.action_dispatch.x_sendfile_header = 'X-Sendfile' # for Apache
42 | # config.action_dispatch.x_sendfile_header = 'X-Accel-Redirect' # for NGINX
43 |
44 | # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies.
45 | # config.force_ssl = true
46 |
47 | # Use the lowest log level to ensure availability of diagnostic information
48 | # when problems arise.
49 | config.log_level = :debug
50 |
51 | # Prepend all log lines with the following tags.
52 | # config.log_tags = [ :subdomain, :uuid ]
53 |
54 | # Use a different logger for distributed setups.
55 | # config.logger = ActiveSupport::TaggedLogging.new(SyslogLogger.new)
56 |
57 | # Use a different cache store in production.
58 | # config.cache_store = :mem_cache_store
59 |
60 | # Enable serving of images, stylesheets, and JavaScripts from an asset server.
61 | # config.action_controller.asset_host = 'http://assets.example.com'
62 |
63 | # Ignore bad email addresses and do not raise email delivery errors.
64 | # Set this to true and configure the email server for immediate delivery to raise delivery errors.
65 | # config.action_mailer.raise_delivery_errors = false
66 |
67 | # Enable locale fallbacks for I18n (makes lookups for any locale fall back to
68 | # the I18n.default_locale when a translation cannot be found).
69 | config.i18n.fallbacks = true
70 |
71 | # Send deprecation notices to registered listeners.
72 | config.active_support.deprecation = :notify
73 |
74 | # Use default logging formatter so that PID and timestamp are not suppressed.
75 | config.log_formatter = ::Logger::Formatter.new
76 |
77 | # Do not dump schema after migrations.
78 | config.active_record.dump_schema_after_migration = false
79 | end
80 |
--------------------------------------------------------------------------------
/config/environments/test.rb:
--------------------------------------------------------------------------------
1 | Rails.application.configure do
2 | # Settings specified here will take precedence over those in config/application.rb.
3 |
4 | # The test environment is used exclusively to run your application's
5 | # test suite. You never need to work with it otherwise. Remember that
6 | # your test database is "scratch space" for the test suite and is wiped
7 | # and recreated between test runs. Don't rely on the data there!
8 | config.cache_classes = true
9 |
10 | # Do not eager load code on boot. This avoids loading your whole application
11 | # just for the purpose of running a single test. If you are using a tool that
12 | # preloads Rails for running tests, you may have to set it to true.
13 | config.eager_load = false
14 |
15 | # Configure static file server for tests with Cache-Control for performance.
16 | config.serve_static_files = true
17 | config.static_cache_control = 'public, max-age=3600'
18 |
19 | # Show full error reports and disable caching.
20 | config.consider_all_requests_local = true
21 | config.action_controller.perform_caching = false
22 |
23 | # Raise exceptions instead of rendering exception templates.
24 | config.action_dispatch.show_exceptions = false
25 |
26 | # Disable request forgery protection in test environment.
27 | config.action_controller.allow_forgery_protection = false
28 |
29 | # Tell Action Mailer not to deliver emails to the real world.
30 | # The :test delivery method accumulates sent emails in the
31 | # ActionMailer::Base.deliveries array.
32 | config.action_mailer.delivery_method = :test
33 |
34 | # Randomize the order test cases are executed.
35 | config.active_support.test_order = :random
36 |
37 | # Print deprecation notices to the stderr.
38 | config.active_support.deprecation = :stderr
39 |
40 | # Raises error for missing translations
41 | # config.action_view.raise_on_missing_translations = true
42 | end
43 |
--------------------------------------------------------------------------------
/config/initializers/aws.rb:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/config/initializers/aws.rb
--------------------------------------------------------------------------------
/config/initializers/backtrace_silencers.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # You can add backtrace silencers for libraries that you're using but don't wish to see in your backtraces.
4 | # Rails.backtrace_cleaner.add_silencer { |line| line =~ /my_noisy_library/ }
5 |
6 | # You can also remove all the silencers if you're trying to debug a problem that might stem from framework code.
7 | # Rails.backtrace_cleaner.remove_silencers!
8 |
--------------------------------------------------------------------------------
/config/initializers/inflections.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # Add new inflection rules using the following format
4 | # (all these examples are active by default):
5 | # ActiveSupport::Inflector.inflections do |inflect|
6 | # inflect.plural /^(ox)$/i, '\1en'
7 | # inflect.singular /^(ox)en/i, '\1'
8 | # inflect.irregular 'person', 'people'
9 | # inflect.uncountable %w( fish sheep )
10 | # end
11 | #
12 | # These inflection rules are supported but not enabled by default:
13 | # ActiveSupport::Inflector.inflections do |inflect|
14 | # inflect.acronym 'RESTful'
15 | # end
16 |
--------------------------------------------------------------------------------
/config/initializers/mime_types.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # Add new mime types for use in respond_to blocks:
4 | # Mime::Type.register "text/richtext", :rtf
5 | # Mime::Type.register_alias "text/html", :iphone
6 |
--------------------------------------------------------------------------------
/config/initializers/session_store.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | Rails.application.config.session_store :cookie_store, key: '_s3-cors-upload-rails_session'
4 |
--------------------------------------------------------------------------------
/config/initializers/simple_form.rb:
--------------------------------------------------------------------------------
1 | # Use this setup block to configure all options available in SimpleForm.
2 | SimpleForm.setup do |config|
3 | # Wrappers are used by the form builder to generate a
4 | # complete input. You can remove any component from the
5 | # wrapper, change the order or even add your own to the
6 | # stack. The options given below are used to wrap the
7 | # whole input.
8 | config.wrappers :default, :class => :input,
9 | :hint_class => :field_with_hint, :error_class => :field_with_errors do |b|
10 | ## Extensions enabled by default
11 | # Any of these extensions can be disabled for a
12 | # given input by passing: `f.input EXTENSION_NAME => false`.
13 | # You can make any of these extensions optional by
14 | # renaming `b.use` to `b.optional`.
15 |
16 | # Determines whether to use HTML5 (:email, :url, ...)
17 | # and required attributes
18 | b.use :html5
19 |
20 | # Calculates placeholders automatically from I18n
21 | # You can also pass a string as f.input :placeholder => "Placeholder"
22 | b.use :placeholder
23 |
24 | ## Optional extensions
25 | # They are disabled unless you pass `f.input EXTENSION_NAME => :lookup`
26 | # to the input. If so, they will retrieve the values from the model
27 | # if any exists. If you want to enable the lookup for any of those
28 | # extensions by default, you can change `b.optional` to `b.use`.
29 |
30 | # Calculates maxlength from length validations for string inputs
31 | b.optional :maxlength
32 |
33 | # Calculates pattern from format validations for string inputs
34 | b.optional :pattern
35 |
36 | # Calculates min and max from length validations for numeric inputs
37 | b.optional :min_max
38 |
39 | # Calculates readonly automatically from readonly attributes
40 | b.optional :readonly
41 |
42 | ## Inputs
43 | b.use :label_input
44 | b.use :hint, :wrap_with => { :tag => :span, :class => :hint }
45 | b.use :error, :wrap_with => { :tag => :span, :class => :error }
46 | end
47 |
48 | # The default wrapper to be used by the FormBuilder.
49 | config.default_wrapper = :default
50 |
51 | # Define the way to render check boxes / radio buttons with labels.
52 | # Defaults to :nested for bootstrap config.
53 | # :inline => input + label
54 | # :nested => label > input
55 | config.boolean_style = :nested
56 |
57 | # Default class for buttons
58 | config.button_class = 'btn'
59 |
60 | # Method used to tidy up errors. Specify any Rails Array method.
61 | # :first lists the first message for each field.
62 | # Use :to_sentence to list all errors for each field.
63 | # config.error_method = :first
64 |
65 | # Default tag used for error notification helper.
66 | config.error_notification_tag = :div
67 |
68 | # CSS class to add for error notification helper.
69 | config.error_notification_class = 'alert alert-error'
70 |
71 | # ID to add for error notification helper.
72 | # config.error_notification_id = nil
73 |
74 | # Series of attempts to detect a default label method for collection.
75 | # config.collection_label_methods = [ :to_label, :name, :title, :to_s ]
76 |
77 | # Series of attempts to detect a default value method for collection.
78 | # config.collection_value_methods = [ :id, :to_s ]
79 |
80 | # You can wrap a collection of radio/check boxes in a pre-defined tag, defaulting to none.
81 | # config.collection_wrapper_tag = nil
82 |
83 | # You can define the class to use on all collection wrappers. Defaulting to none.
84 | # config.collection_wrapper_class = nil
85 |
86 | # You can wrap each item in a collection of radio/check boxes with a tag,
87 | # defaulting to :span. Please note that when using :boolean_style = :nested,
88 | # SimpleForm will force this option to be a label.
89 | # config.item_wrapper_tag = :span
90 |
91 | # You can define a class to use in all item wrappers. Defaulting to none.
92 | # config.item_wrapper_class = nil
93 |
94 | # How the label text should be generated altogether with the required text.
95 | # config.label_text = lambda { |label, required| "#{required} #{label}" }
96 |
97 | # You can define the class to use on all labels. Default is nil.
98 | config.label_class = 'control-label'
99 |
100 | # You can define the class to use on all forms. Default is simple_form.
101 | # config.form_class = :simple_form
102 |
103 | # You can define which elements should obtain additional classes
104 | # config.generate_additional_classes_for = [:wrapper, :label, :input]
105 |
106 | # Whether attributes are required by default (or not). Default is true.
107 | # config.required_by_default = true
108 |
109 | # Tell browsers whether to use default HTML5 validations (novalidate option).
110 | # Default is enabled.
111 | config.browser_validations = true
112 |
113 | # Collection of methods to detect if a file type was given.
114 | # config.file_methods = [ :mounted_as, :file?, :public_filename ]
115 |
116 | # Custom mappings for input types. This should be a hash containing a regexp
117 | # to match as key, and the input type that will be used when the field name
118 | # matches the regexp as value.
119 | # config.input_mappings = { /count/ => :integer }
120 |
121 | # Custom wrappers for input types. This should be a hash containing an input
122 | # type as key and the wrapper that will be used for all inputs with specified type.
123 | # config.wrapper_mappings = { :string => :prepend }
124 |
125 | # Default priority for time_zone inputs.
126 | # config.time_zone_priority = nil
127 |
128 | # Default priority for country inputs.
129 | # config.country_priority = nil
130 |
131 | # Default size for text inputs.
132 | # config.default_input_size = 50
133 |
134 | # When false, do not use translations for labels.
135 | # config.translate_labels = true
136 |
137 | # Automatically discover new inputs in Rails' autoload path.
138 | # config.inputs_discovery = true
139 |
140 | # Cache SimpleForm inputs discovery
141 | # config.cache_discovery = !Rails.env.development?
142 | end
143 |
--------------------------------------------------------------------------------
/config/initializers/simple_form_bootstrap.rb:
--------------------------------------------------------------------------------
1 | # Use this setup block to configure all options available in SimpleForm.
2 | SimpleForm.setup do |config|
3 | config.wrappers :bootstrap, :tag => 'div', :class => 'control-group', :error_class => 'error' do |b|
4 | b.use :html5
5 | b.use :placeholder
6 | b.use :label
7 | b.wrapper :tag => 'div', :class => 'controls' do |ba|
8 | ba.use :input
9 | ba.use :error, :wrap_with => { :tag => 'span', :class => 'help-inline' }
10 | ba.use :hint, :wrap_with => { :tag => 'p', :class => 'help-block' }
11 | end
12 | end
13 |
14 | config.wrappers :prepend, :tag => 'div', :class => "control-group", :error_class => 'error' do |b|
15 | b.use :html5
16 | b.use :placeholder
17 | b.use :label
18 | b.wrapper :tag => 'div', :class => 'controls' do |input|
19 | input.wrapper :tag => 'div', :class => 'input-prepend' do |prepend|
20 | prepend.use :input
21 | end
22 | input.use :hint, :wrap_with => { :tag => 'span', :class => 'help-block' }
23 | input.use :error, :wrap_with => { :tag => 'span', :class => 'help-inline' }
24 | end
25 | end
26 |
27 | config.wrappers :append, :tag => 'div', :class => "control-group", :error_class => 'error' do |b|
28 | b.use :html5
29 | b.use :placeholder
30 | b.use :label
31 | b.wrapper :tag => 'div', :class => 'controls' do |input|
32 | input.wrapper :tag => 'div', :class => 'input-append' do |append|
33 | append.use :input
34 | end
35 | input.use :hint, :wrap_with => { :tag => 'span', :class => 'help-block' }
36 | input.use :error, :wrap_with => { :tag => 'span', :class => 'help-inline' }
37 | end
38 | end
39 |
40 | # Wrappers for forms and inputs using the Twitter Bootstrap toolkit.
41 | # Check the Bootstrap docs (http://twitter.github.com/bootstrap)
42 | # to learn about the different styles for forms and inputs,
43 | # buttons and other elements.
44 | config.default_wrapper = :bootstrap
45 | end
46 |
--------------------------------------------------------------------------------
/config/initializers/wrap_parameters.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # This file contains settings for ActionController::ParamsWrapper which
4 | # is enabled by default.
5 |
6 | # Enable parameter wrapping for JSON. You can disable this by setting :format to an empty array.
7 | ActiveSupport.on_load(:action_controller) do
8 | wrap_parameters format: [:json] if respond_to?(:wrap_parameters)
9 | end
10 |
11 | # To enable root element in JSON for ActiveRecord objects.
12 | # ActiveSupport.on_load(:active_record) do
13 | # self.include_root_in_json = true
14 | # end
15 |
--------------------------------------------------------------------------------
/config/locales/en.yml:
--------------------------------------------------------------------------------
1 | # Sample localization file for English. Add more files in this directory for other locales.
2 | # See https://github.com/svenfuchs/rails-i18n/tree/master/rails%2Flocale for starting points.
3 |
4 | en:
5 | hello: "Hello world"
6 |
--------------------------------------------------------------------------------
/config/locales/simple_form.en.yml:
--------------------------------------------------------------------------------
1 | en:
2 | simple_form:
3 | "yes": 'Yes'
4 | "no": 'No'
5 | required:
6 | text: 'required'
7 | mark: '*'
8 | # You can uncomment the line below if you need to overwrite the whole required html.
9 | # When using html, text and mark won't be used.
10 | # html: '* '
11 | error_notification:
12 | default_message: "Please review the problems below:"
13 | # Labels and hints examples
14 | # labels:
15 | # defaults:
16 | # password: 'Password'
17 | # user:
18 | # new:
19 | # email: 'E-mail to sign in.'
20 | # edit:
21 | # email: 'E-mail.'
22 | # hints:
23 | # defaults:
24 | # username: 'User name to sign in.'
25 | # password: 'No special characters, please.'
26 |
27 |
--------------------------------------------------------------------------------
/config/routes.rb:
--------------------------------------------------------------------------------
1 | Rails.application.routes.draw do
2 |
3 | root :to => "products#new"
4 | resources :products
5 |
6 | get "generate_signed_s3_url" => "aws#generate_signed_s3_url"
7 |
8 | end
9 |
--------------------------------------------------------------------------------
/config/secrets.yml:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # Your secret key is used for verifying the integrity of signed cookies.
4 | # If you change this key, all old signed cookies will become invalid!
5 |
6 | # Make sure the secret is at least 30 characters and all random,
7 | # no regular words or you'll be exposed to dictionary attacks.
8 | # You can use `rake secret` to generate a secure secret key.
9 |
10 | # Make sure the secrets in this file are kept private
11 | # if you're sharing your code publicly.
12 |
13 | development:
14 | secret_key_base: aff6a0e568e022bd28f4112c3ca7bdcf6605079a78df0f2ec7410712434b509ecaddb2d40602fcce5e38304c043247d8429d2a5279330f94b915fbfe6053d453
15 |
16 | test:
17 | secret_key_base: ad3d15f0af35b0dfebebfcce513d07e3c1800e663266a56469ec909019c8d55d4b00b2fc987ac9ff8d982add98a95ac970de2566b237299b428ac38b7c494e62
18 |
19 | # Do not keep production secrets in the repository,
20 | # instead read values from the environment.
21 | production:
22 | secret_key_base: <%= ENV["SECRET_KEY_BASE"] %>
23 |
--------------------------------------------------------------------------------
/db/migrate/20130523082752_create_products.rb:
--------------------------------------------------------------------------------
1 | class CreateProducts < ActiveRecord::Migration
2 | def change
3 | create_table :products do |t|
4 | t.string :name
5 | t.decimal :price
6 | t.string :url
7 |
8 | t.timestamps
9 | end
10 | end
11 | end
12 |
--------------------------------------------------------------------------------
/db/schema.rb:
--------------------------------------------------------------------------------
1 | # encoding: UTF-8
2 | # This file is auto-generated from the current state of the database. Instead
3 | # of editing this file, please use the migrations feature of Active Record to
4 | # incrementally modify your database, and then regenerate this schema definition.
5 | #
6 | # Note that this schema.rb definition is the authoritative source for your
7 | # database schema. If you need to create the application database on another
8 | # system, you should be using db:schema:load, not running all the migrations
9 | # from scratch. The latter is a flawed and unsustainable approach (the more migrations
10 | # you'll amass, the slower it'll run and the greater likelihood for issues).
11 | #
12 | # It's strongly recommended that you check this file into your version control system.
13 |
14 | ActiveRecord::Schema.define(version: 20130523082752) do
15 |
16 | create_table "products", force: :cascade do |t|
17 | t.string "name"
18 | t.decimal "price"
19 | t.string "url"
20 | t.datetime "created_at"
21 | t.datetime "updated_at"
22 | end
23 |
24 | end
25 |
--------------------------------------------------------------------------------
/db/seeds.rb:
--------------------------------------------------------------------------------
1 | # This file should contain all the record creation needed to seed the database with its default values.
2 | # The data can then be loaded with the rake db:seed (or created alongside the db with db:setup).
3 | #
4 | # Examples:
5 | #
6 | # cities = City.create([{ name: 'Chicago' }, { name: 'Copenhagen' }])
7 | # Mayor.create(name: 'Emanuel', city: cities.first)
8 |
--------------------------------------------------------------------------------
/doc/README_FOR_APP:
--------------------------------------------------------------------------------
1 | Use this README file to introduce your application and point to useful places in the API for learning more.
2 | Run "rake doc:app" to generate API documentation for your models, controllers, helpers, and libraries.
3 |
--------------------------------------------------------------------------------
/log/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/log/.gitkeep
--------------------------------------------------------------------------------
/public/404.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | The page you were looking for doesn't exist (404)
5 |
17 |
18 |
19 |
20 |
21 |
22 |
The page you were looking for doesn't exist.
23 |
You may have mistyped the address or the page may have moved.
24 |
25 |
26 |
27 |
--------------------------------------------------------------------------------
/public/422.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | The change you wanted was rejected (422)
5 |
17 |
18 |
19 |
20 |
21 |
22 |
The change you wanted was rejected.
23 |
Maybe you tried to change something you didn't have access to.
24 |
25 |
26 |
27 |
--------------------------------------------------------------------------------
/public/500.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | We're sorry, but something went wrong (500)
5 |
17 |
18 |
19 |
20 |
21 |
22 |
We're sorry, but something went wrong.
23 |
24 |
25 |
26 |
--------------------------------------------------------------------------------
/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/public/favicon.ico
--------------------------------------------------------------------------------
/public/img/glyphicons-halflings-white.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/public/img/glyphicons-halflings-white.png
--------------------------------------------------------------------------------
/public/img/glyphicons-halflings.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/public/img/glyphicons-halflings.png
--------------------------------------------------------------------------------
/public/robots.txt:
--------------------------------------------------------------------------------
1 | # See http://www.robotstxt.org/wc/norobots.html for documentation on how to use the robots.txt file
2 | #
3 | # To ban all spiders from the entire site uncomment the next two lines:
4 | # User-Agent: *
5 | # Disallow: /
6 |
--------------------------------------------------------------------------------
/script/rails:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 | # This command will automatically be run when you run "rails" with Rails 3 gems installed from the root of your application.
3 |
4 | APP_PATH = File.expand_path('../../config/application', __FILE__)
5 | require File.expand_path('../../config/boot', __FILE__)
6 | require 'rails/commands'
7 |
--------------------------------------------------------------------------------
/spec/controllers/aws_controller_spec.rb:
--------------------------------------------------------------------------------
1 | require 'spec_helper'
2 |
3 | describe AwsController do
4 |
5 | end
6 |
--------------------------------------------------------------------------------
/spec/controllers/products_controller_spec.rb:
--------------------------------------------------------------------------------
1 | require 'spec_helper'
2 |
3 | describe ProductsController do
4 |
5 | end
6 |
--------------------------------------------------------------------------------
/spec/helpers/products_helper_spec.rb:
--------------------------------------------------------------------------------
1 | require 'spec_helper'
2 |
3 | # Specs in this file have access to a helper object that includes
4 | # the ProductsHelper. For example:
5 | #
6 | # describe ProductsHelper do
7 | # describe "string concat" do
8 | # it "concats two strings with spaces" do
9 | # helper.concat_strings("this","that").should == "this that"
10 | # end
11 | # end
12 | # end
13 | describe ProductsHelper do
14 | pending "add some examples to (or delete) #{__FILE__}"
15 | end
16 |
--------------------------------------------------------------------------------
/spec/spec_helper.rb:
--------------------------------------------------------------------------------
1 | # This file is copied to spec/ when you run 'rails generate rspec:install'
2 | ENV["RAILS_ENV"] ||= 'test'
3 | require File.expand_path("../../config/environment", __FILE__)
4 | require 'rspec/rails'
5 |
6 | # Requires supporting ruby files with custom matchers and macros, etc,
7 | # in spec/support/ and its subdirectories.
8 | Dir[Rails.root.join("spec/support/**/*.rb")].each {|f| require f}
9 |
10 | RSpec.configure do |config|
11 | # ## Mock Framework
12 | #
13 | # If you prefer to use mocha, flexmock or RR, uncomment the appropriate line:
14 | #
15 | # config.mock_with :mocha
16 | # config.mock_with :flexmock
17 | # config.mock_with :rr
18 |
19 | # Remove this line if you're not using ActiveRecord or ActiveRecord fixtures
20 | config.fixture_path = "#{::Rails.root}/spec/fixtures"
21 |
22 | # If you're not using ActiveRecord, or you'd prefer not to run each of your
23 | # examples within a transaction, remove the following line or assign false
24 | # instead of true.
25 | config.use_transactional_fixtures = true
26 |
27 | # If true, the base class of anonymous controllers will be inferred
28 | # automatically. This will be the default behavior in future versions of
29 | # rspec-rails.
30 | config.infer_base_class_for_anonymous_controllers = false
31 |
32 | # Run specs in random order to surface order dependencies. If you find an
33 | # order dependency and want to debug it, you can fix the order by providing
34 | # the seed, which is printed after each run.
35 | # --seed 1234
36 | config.order = "random"
37 | end
38 |
--------------------------------------------------------------------------------
/test/fixtures/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/test/fixtures/.gitkeep
--------------------------------------------------------------------------------
/test/fixtures/products.yml:
--------------------------------------------------------------------------------
1 | # Read about fixtures at http://api.rubyonrails.org/classes/ActiveRecord/Fixtures.html
2 |
3 | one:
4 | name: MyString
5 | price: 9.99
6 | url: MyString
7 |
8 | two:
9 | name: MyString
10 | price: 9.99
11 | url: MyString
12 |
--------------------------------------------------------------------------------
/test/functional/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/test/functional/.gitkeep
--------------------------------------------------------------------------------
/test/integration/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/test/integration/.gitkeep
--------------------------------------------------------------------------------
/test/performance/browsing_test.rb:
--------------------------------------------------------------------------------
1 | require 'test_helper'
2 | require 'rails/performance_test_help'
3 |
4 | class BrowsingTest < ActionDispatch::PerformanceTest
5 | # Refer to the documentation for all available options
6 | # self.profile_options = { :runs => 5, :metrics => [:wall_time, :memory]
7 | # :output => 'tmp/performance', :formats => [:flat] }
8 |
9 | def test_homepage
10 | get '/'
11 | end
12 | end
13 |
--------------------------------------------------------------------------------
/test/test_helper.rb:
--------------------------------------------------------------------------------
1 | ENV["RAILS_ENV"] = "test"
2 | require File.expand_path('../../config/environment', __FILE__)
3 | require 'rails/test_help'
4 |
5 | class ActiveSupport::TestCase
6 | # Setup all fixtures in test/fixtures/*.(yml|csv) for all tests in alphabetical order.
7 | #
8 | # Note: You'll currently still have to declare fixtures explicitly in integration tests
9 | # -- they do not yet inherit this setting
10 | fixtures :all
11 |
12 | # Add more helper methods to be used by all tests here...
13 | end
14 |
--------------------------------------------------------------------------------
/test/unit/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/test/unit/.gitkeep
--------------------------------------------------------------------------------
/test/unit/product_test.rb:
--------------------------------------------------------------------------------
1 | require 'test_helper'
2 |
3 | class ProductTest < ActiveSupport::TestCase
4 | # test "the truth" do
5 | # assert true
6 | # end
7 | end
8 |
--------------------------------------------------------------------------------
/vendor/assets/javascripts/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/vendor/assets/javascripts/.gitkeep
--------------------------------------------------------------------------------
/vendor/assets/javascripts/bootstrap-filestyle.js:
--------------------------------------------------------------------------------
1 | /*
2 | * bootstrap-filestyle
3 | * http://dev.tudosobreweb.com.br/bootstrap-filestyle/
4 | *
5 | * Copyright (c) 2013 Markus Vinicius da Silva Lima
6 | * Version 1.0.3
7 | * Licensed under the MIT license.
8 | */
9 | (function ($) {
10 | "use strict";
11 |
12 | var Filestyle = function (element, options) {
13 | this.options = options;
14 | this.$elementFilestyle = [];
15 | this.$element = $(element);
16 | };
17 |
18 | Filestyle.prototype = {
19 | clear: function () {
20 | this.$element.val('');
21 | this.$elementFilestyle.find(':text').val('');
22 | },
23 |
24 | destroy: function () {
25 | this.$element
26 | .removeAttr('style')
27 | .removeData('filestyle')
28 | .val('');
29 | this.$elementFilestyle.remove();
30 | },
31 |
32 | icon: function (value) {
33 | if (value === true) {
34 | if (!this.options.icon) {
35 | this.options.icon = true;
36 | this.$elementFilestyle.find('label').prepend(this.htmlIcon());
37 | }
38 | } else if (value === false) {
39 | if (this.options.icon) {
40 | this.options.icon = false;
41 | this.$elementFilestyle.find('i').remove();
42 | }
43 | } else {
44 | return this.options.icon;
45 | }
46 | },
47 |
48 | input: function (value) {
49 | if (value === true) {
50 | if (!this.options.input) {
51 | this.options.input = true;
52 | this.$elementFilestyle.prepend(this.htmlInput());
53 |
54 | var content = '',
55 | files = [];
56 | if (this.$element[0].files === undefined) {
57 | files[0] = {'name': this.$element[0].value};
58 | } else {
59 | files = this.$element[0].files;
60 | }
61 |
62 | for (var i = 0; i < files.length; i++) {
63 | content += files[i].name.split("\\").pop() + ', ';
64 | }
65 | if (content !== '') {
66 | this.$elementFilestyle.find(':text').val(content.replace(/\, $/g, ''));
67 | }
68 | }
69 | } else if (value === false) {
70 | if (this.options.input) {
71 | this.options.input = false;
72 | this.$elementFilestyle.find(':text').remove();
73 | }
74 | } else {
75 | return this.options.input;
76 | }
77 | },
78 |
79 | buttonText: function (value) {
80 | if (value !== undefined) {
81 | this.options.buttonText = value;
82 | this.$elementFilestyle.find('label span').html(this.options.buttonText);
83 | } else {
84 | return this.options.buttonText;
85 | }
86 | },
87 |
88 | classButton: function (value) {
89 | if (value !== undefined) {
90 | this.options.classButton = value;
91 | this.$elementFilestyle.find('label').attr({'class': this.options.classButton});
92 | if (this.options.classButton.search(/btn-inverse|btn-primary|btn-danger|btn-warning|btn-success/i) !== -1) {
93 | this.$elementFilestyle.find('label i').addClass('icon-white');
94 | } else {
95 | this.$elementFilestyle.find('label i').removeClass('icon-white');
96 | }
97 | } else {
98 | return this.options.classButton;
99 | }
100 | },
101 |
102 | classIcon: function (value) {
103 | if (value !== undefined) {
104 | this.options.classIcon = value;
105 | if (this.options.classButton.search(/btn-inverse|btn-primary|btn-danger|btn-warning|btn-success/i) !== -1) {
106 | this.$elementFilestyle.find('label').find('i').attr({'class': 'icon-white '+this.options.classIcon});
107 | } else {
108 | this.$elementFilestyle.find('label').find('i').attr({'class': this.options.classIcon});
109 | }
110 | } else {
111 | return this.options.classIcon;
112 | }
113 | },
114 |
115 | classInput: function (value) {
116 | if (value !== undefined) {
117 | this.options.classInput = value;
118 | this.$elementFilestyle.find(':text').addClass(this.options.classInput);
119 | } else {
120 | return this.options.classInput;
121 | }
122 | },
123 |
124 | htmlIcon: function () {
125 | if (this.options.icon) {
126 | var colorIcon = '';
127 | if (this.options.classButton.search(/btn-inverse|btn-primary|btn-danger|btn-warning|btn-success/i) !== -1) {
128 | colorIcon = ' icon-white ';
129 | }
130 |
131 | return ' ';
132 | } else {
133 | return '';
134 | }
135 | },
136 |
137 | htmlInput: function () {
138 | if (this.options.input) {
139 | return ' ';
140 | } else {
141 | return '';
142 | }
143 | },
144 |
145 | constructor: function () {
146 | var _self = this,
147 | html = '',
148 | id = this.$element.attr('id'),
149 | files = [];
150 |
151 | if (id === '' || !id) {
152 | id = 'filestyle-'+$('.bootstrap-filestyle').length;
153 | this.$element.attr({'id': id});
154 | }
155 |
156 | html = this.htmlInput()+
157 | ''+
158 | this.htmlIcon()+
159 | ''+this.options.buttonText+' '+
160 | ' ';
161 |
162 | this.$elementFilestyle = $(''+html+'
');
163 |
164 | // hidding input file and add filestyle
165 | this.$element
166 | .css({'position':'fixed','left':'-500px'})
167 | .after(this.$elementFilestyle);
168 |
169 | // Getting input file value
170 | this.$element.change(function () {
171 | var content = '';
172 | if (this.files === undefined) {
173 | files[0] = {'name': this.value};
174 | } else {
175 | files = this.files;
176 | }
177 |
178 | for (var i = 0; i < files.length; i++) {
179 | content += files[i].name.split("\\").pop() + ', ';
180 | }
181 |
182 | if (content !== '') {
183 | _self.$elementFilestyle.find(':text').val(content.replace(/\, $/g, ''));
184 | }
185 | });
186 |
187 | // Check if browser is Firefox
188 | if (window.navigator.userAgent.search(/firefox/i) > -1) {
189 | // Simulating choose file for firefox
190 | this.$elementFilestyle.find('label').click(function () {
191 | _self.$element.click();
192 | return false;
193 | });
194 | }
195 | }
196 | };
197 |
198 | var old = $.fn.filestyle;
199 |
200 | $.fn.filestyle = function (option, value) {
201 | var get = '',
202 | element = this.each(function () {
203 | if ($(this).attr('type') === 'file') {
204 | var $this = $(this),
205 | data = $this.data('filestyle'),
206 | options = $.extend({}, $.fn.filestyle.defaults, option, typeof option === 'object' && option);
207 |
208 | if (!data) {
209 | $this.data('filestyle', (data = new Filestyle(this, options)));
210 | data.constructor();
211 | }
212 |
213 | if (typeof option === 'string') {
214 | get = data[option](value);
215 | }
216 | }
217 | });
218 |
219 | if (typeof get !== undefined) {
220 | return get;
221 | } else {
222 | return element;
223 | }
224 | };
225 |
226 | $.fn.filestyle.defaults = {
227 | 'buttonText': 'Choose file',
228 | 'input': true,
229 | 'icon': true,
230 | 'classButton': 'btn',
231 | 'classInput': 'input-large',
232 | 'classIcon': 'icon-folder-open'
233 | };
234 |
235 | $.fn.filestyle.noConflict = function () {
236 | $.fn.filestyle = old;
237 | return this;
238 | };
239 |
240 | // Data attributes register
241 | $('.filestyle').each(function () {
242 | var $this = $(this),
243 | options = {
244 | 'buttonText': $this.attr('data-buttonText'),
245 | 'input': $this.attr('data-input') === 'false' ? false : true,
246 | 'icon': $this.attr('data-icon') === 'false' ? false : true,
247 | 'classButton': $this.attr('data-classButton'),
248 | 'classInput': $this.attr('data-classInput'),
249 | 'classIcon': $this.attr('data-classIcon')
250 | };
251 |
252 | $this.filestyle(options);
253 | });
254 |
255 | })(window.jQuery);
--------------------------------------------------------------------------------
/vendor/assets/javascripts/bootstrap.js:
--------------------------------------------------------------------------------
1 | /* ===================================================
2 | * bootstrap-transition.js v2.3.2
3 | * http://twitter.github.com/bootstrap/javascript.html#transitions
4 | * ===================================================
5 | * Copyright 2012 Twitter, Inc.
6 | *
7 | * Licensed under the Apache License, Version 2.0 (the "License");
8 | * you may not use this file except in compliance with the License.
9 | * You may obtain a copy of the License at
10 | *
11 | * http://www.apache.org/licenses/LICENSE-2.0
12 | *
13 | * Unless required by applicable law or agreed to in writing, software
14 | * distributed under the License is distributed on an "AS IS" BASIS,
15 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 | * See the License for the specific language governing permissions and
17 | * limitations under the License.
18 | * ========================================================== */
19 |
20 |
21 | !function ($) {
22 |
23 | "use strict"; // jshint ;_;
24 |
25 |
26 | /* CSS TRANSITION SUPPORT (http://www.modernizr.com/)
27 | * ======================================================= */
28 |
29 | $(function () {
30 |
31 | $.support.transition = (function () {
32 |
33 | var transitionEnd = (function () {
34 |
35 | var el = document.createElement('bootstrap')
36 | , transEndEventNames = {
37 | 'WebkitTransition' : 'webkitTransitionEnd'
38 | , 'MozTransition' : 'transitionend'
39 | , 'OTransition' : 'oTransitionEnd otransitionend'
40 | , 'transition' : 'transitionend'
41 | }
42 | , name
43 |
44 | for (name in transEndEventNames){
45 | if (el.style[name] !== undefined) {
46 | return transEndEventNames[name]
47 | }
48 | }
49 |
50 | }())
51 |
52 | return transitionEnd && {
53 | end: transitionEnd
54 | }
55 |
56 | })()
57 |
58 | })
59 |
60 | }(window.jQuery);/* ==========================================================
61 | * bootstrap-alert.js v2.3.2
62 | * http://twitter.github.com/bootstrap/javascript.html#alerts
63 | * ==========================================================
64 | * Copyright 2012 Twitter, Inc.
65 | *
66 | * Licensed under the Apache License, Version 2.0 (the "License");
67 | * you may not use this file except in compliance with the License.
68 | * You may obtain a copy of the License at
69 | *
70 | * http://www.apache.org/licenses/LICENSE-2.0
71 | *
72 | * Unless required by applicable law or agreed to in writing, software
73 | * distributed under the License is distributed on an "AS IS" BASIS,
74 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
75 | * See the License for the specific language governing permissions and
76 | * limitations under the License.
77 | * ========================================================== */
78 |
79 |
80 | !function ($) {
81 |
82 | "use strict"; // jshint ;_;
83 |
84 |
85 | /* ALERT CLASS DEFINITION
86 | * ====================== */
87 |
88 | var dismiss = '[data-dismiss="alert"]'
89 | , Alert = function (el) {
90 | $(el).on('click', dismiss, this.close)
91 | }
92 |
93 | Alert.prototype.close = function (e) {
94 | var $this = $(this)
95 | , selector = $this.attr('data-target')
96 | , $parent
97 |
98 | if (!selector) {
99 | selector = $this.attr('href')
100 | selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
101 | }
102 |
103 | $parent = $(selector)
104 |
105 | e && e.preventDefault()
106 |
107 | $parent.length || ($parent = $this.hasClass('alert') ? $this : $this.parent())
108 |
109 | $parent.trigger(e = $.Event('close'))
110 |
111 | if (e.isDefaultPrevented()) return
112 |
113 | $parent.removeClass('in')
114 |
115 | function removeElement() {
116 | $parent
117 | .trigger('closed')
118 | .remove()
119 | }
120 |
121 | $.support.transition && $parent.hasClass('fade') ?
122 | $parent.on($.support.transition.end, removeElement) :
123 | removeElement()
124 | }
125 |
126 |
127 | /* ALERT PLUGIN DEFINITION
128 | * ======================= */
129 |
130 | var old = $.fn.alert
131 |
132 | $.fn.alert = function (option) {
133 | return this.each(function () {
134 | var $this = $(this)
135 | , data = $this.data('alert')
136 | if (!data) $this.data('alert', (data = new Alert(this)))
137 | if (typeof option == 'string') data[option].call($this)
138 | })
139 | }
140 |
141 | $.fn.alert.Constructor = Alert
142 |
143 |
144 | /* ALERT NO CONFLICT
145 | * ================= */
146 |
147 | $.fn.alert.noConflict = function () {
148 | $.fn.alert = old
149 | return this
150 | }
151 |
152 |
153 | /* ALERT DATA-API
154 | * ============== */
155 |
156 | $(document).on('click.alert.data-api', dismiss, Alert.prototype.close)
157 |
158 | }(window.jQuery);/* ============================================================
159 | * bootstrap-button.js v2.3.2
160 | * http://twitter.github.com/bootstrap/javascript.html#buttons
161 | * ============================================================
162 | * Copyright 2012 Twitter, Inc.
163 | *
164 | * Licensed under the Apache License, Version 2.0 (the "License");
165 | * you may not use this file except in compliance with the License.
166 | * You may obtain a copy of the License at
167 | *
168 | * http://www.apache.org/licenses/LICENSE-2.0
169 | *
170 | * Unless required by applicable law or agreed to in writing, software
171 | * distributed under the License is distributed on an "AS IS" BASIS,
172 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
173 | * See the License for the specific language governing permissions and
174 | * limitations under the License.
175 | * ============================================================ */
176 |
177 |
178 | !function ($) {
179 |
180 | "use strict"; // jshint ;_;
181 |
182 |
183 | /* BUTTON PUBLIC CLASS DEFINITION
184 | * ============================== */
185 |
186 | var Button = function (element, options) {
187 | this.$element = $(element)
188 | this.options = $.extend({}, $.fn.button.defaults, options)
189 | }
190 |
191 | Button.prototype.setState = function (state) {
192 | var d = 'disabled'
193 | , $el = this.$element
194 | , data = $el.data()
195 | , val = $el.is('input') ? 'val' : 'html'
196 |
197 | state = state + 'Text'
198 | data.resetText || $el.data('resetText', $el[val]())
199 |
200 | $el[val](data[state] || this.options[state])
201 |
202 | // push to event loop to allow forms to submit
203 | setTimeout(function () {
204 | state == 'loadingText' ?
205 | $el.addClass(d).attr(d, d) :
206 | $el.removeClass(d).removeAttr(d)
207 | }, 0)
208 | }
209 |
210 | Button.prototype.toggle = function () {
211 | var $parent = this.$element.closest('[data-toggle="buttons-radio"]')
212 |
213 | $parent && $parent
214 | .find('.active')
215 | .removeClass('active')
216 |
217 | this.$element.toggleClass('active')
218 | }
219 |
220 |
221 | /* BUTTON PLUGIN DEFINITION
222 | * ======================== */
223 |
224 | var old = $.fn.button
225 |
226 | $.fn.button = function (option) {
227 | return this.each(function () {
228 | var $this = $(this)
229 | , data = $this.data('button')
230 | , options = typeof option == 'object' && option
231 | if (!data) $this.data('button', (data = new Button(this, options)))
232 | if (option == 'toggle') data.toggle()
233 | else if (option) data.setState(option)
234 | })
235 | }
236 |
237 | $.fn.button.defaults = {
238 | loadingText: 'loading...'
239 | }
240 |
241 | $.fn.button.Constructor = Button
242 |
243 |
244 | /* BUTTON NO CONFLICT
245 | * ================== */
246 |
247 | $.fn.button.noConflict = function () {
248 | $.fn.button = old
249 | return this
250 | }
251 |
252 |
253 | /* BUTTON DATA-API
254 | * =============== */
255 |
256 | $(document).on('click.button.data-api', '[data-toggle^=button]', function (e) {
257 | var $btn = $(e.target)
258 | if (!$btn.hasClass('btn')) $btn = $btn.closest('.btn')
259 | $btn.button('toggle')
260 | })
261 |
262 | }(window.jQuery);/* ==========================================================
263 | * bootstrap-carousel.js v2.3.2
264 | * http://twitter.github.com/bootstrap/javascript.html#carousel
265 | * ==========================================================
266 | * Copyright 2012 Twitter, Inc.
267 | *
268 | * Licensed under the Apache License, Version 2.0 (the "License");
269 | * you may not use this file except in compliance with the License.
270 | * You may obtain a copy of the License at
271 | *
272 | * http://www.apache.org/licenses/LICENSE-2.0
273 | *
274 | * Unless required by applicable law or agreed to in writing, software
275 | * distributed under the License is distributed on an "AS IS" BASIS,
276 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
277 | * See the License for the specific language governing permissions and
278 | * limitations under the License.
279 | * ========================================================== */
280 |
281 |
282 | !function ($) {
283 |
284 | "use strict"; // jshint ;_;
285 |
286 |
287 | /* CAROUSEL CLASS DEFINITION
288 | * ========================= */
289 |
290 | var Carousel = function (element, options) {
291 | this.$element = $(element)
292 | this.$indicators = this.$element.find('.carousel-indicators')
293 | this.options = options
294 | this.options.pause == 'hover' && this.$element
295 | .on('mouseenter', $.proxy(this.pause, this))
296 | .on('mouseleave', $.proxy(this.cycle, this))
297 | }
298 |
299 | Carousel.prototype = {
300 |
301 | cycle: function (e) {
302 | if (!e) this.paused = false
303 | if (this.interval) clearInterval(this.interval);
304 | this.options.interval
305 | && !this.paused
306 | && (this.interval = setInterval($.proxy(this.next, this), this.options.interval))
307 | return this
308 | }
309 |
310 | , getActiveIndex: function () {
311 | this.$active = this.$element.find('.item.active')
312 | this.$items = this.$active.parent().children()
313 | return this.$items.index(this.$active)
314 | }
315 |
316 | , to: function (pos) {
317 | var activeIndex = this.getActiveIndex()
318 | , that = this
319 |
320 | if (pos > (this.$items.length - 1) || pos < 0) return
321 |
322 | if (this.sliding) {
323 | return this.$element.one('slid', function () {
324 | that.to(pos)
325 | })
326 | }
327 |
328 | if (activeIndex == pos) {
329 | return this.pause().cycle()
330 | }
331 |
332 | return this.slide(pos > activeIndex ? 'next' : 'prev', $(this.$items[pos]))
333 | }
334 |
335 | , pause: function (e) {
336 | if (!e) this.paused = true
337 | if (this.$element.find('.next, .prev').length && $.support.transition.end) {
338 | this.$element.trigger($.support.transition.end)
339 | this.cycle(true)
340 | }
341 | clearInterval(this.interval)
342 | this.interval = null
343 | return this
344 | }
345 |
346 | , next: function () {
347 | if (this.sliding) return
348 | return this.slide('next')
349 | }
350 |
351 | , prev: function () {
352 | if (this.sliding) return
353 | return this.slide('prev')
354 | }
355 |
356 | , slide: function (type, next) {
357 | var $active = this.$element.find('.item.active')
358 | , $next = next || $active[type]()
359 | , isCycling = this.interval
360 | , direction = type == 'next' ? 'left' : 'right'
361 | , fallback = type == 'next' ? 'first' : 'last'
362 | , that = this
363 | , e
364 |
365 | this.sliding = true
366 |
367 | isCycling && this.pause()
368 |
369 | $next = $next.length ? $next : this.$element.find('.item')[fallback]()
370 |
371 | e = $.Event('slide', {
372 | relatedTarget: $next[0]
373 | , direction: direction
374 | })
375 |
376 | if ($next.hasClass('active')) return
377 |
378 | if (this.$indicators.length) {
379 | this.$indicators.find('.active').removeClass('active')
380 | this.$element.one('slid', function () {
381 | var $nextIndicator = $(that.$indicators.children()[that.getActiveIndex()])
382 | $nextIndicator && $nextIndicator.addClass('active')
383 | })
384 | }
385 |
386 | if ($.support.transition && this.$element.hasClass('slide')) {
387 | this.$element.trigger(e)
388 | if (e.isDefaultPrevented()) return
389 | $next.addClass(type)
390 | $next[0].offsetWidth // force reflow
391 | $active.addClass(direction)
392 | $next.addClass(direction)
393 | this.$element.one($.support.transition.end, function () {
394 | $next.removeClass([type, direction].join(' ')).addClass('active')
395 | $active.removeClass(['active', direction].join(' '))
396 | that.sliding = false
397 | setTimeout(function () { that.$element.trigger('slid') }, 0)
398 | })
399 | } else {
400 | this.$element.trigger(e)
401 | if (e.isDefaultPrevented()) return
402 | $active.removeClass('active')
403 | $next.addClass('active')
404 | this.sliding = false
405 | this.$element.trigger('slid')
406 | }
407 |
408 | isCycling && this.cycle()
409 |
410 | return this
411 | }
412 |
413 | }
414 |
415 |
416 | /* CAROUSEL PLUGIN DEFINITION
417 | * ========================== */
418 |
419 | var old = $.fn.carousel
420 |
421 | $.fn.carousel = function (option) {
422 | return this.each(function () {
423 | var $this = $(this)
424 | , data = $this.data('carousel')
425 | , options = $.extend({}, $.fn.carousel.defaults, typeof option == 'object' && option)
426 | , action = typeof option == 'string' ? option : options.slide
427 | if (!data) $this.data('carousel', (data = new Carousel(this, options)))
428 | if (typeof option == 'number') data.to(option)
429 | else if (action) data[action]()
430 | else if (options.interval) data.pause().cycle()
431 | })
432 | }
433 |
434 | $.fn.carousel.defaults = {
435 | interval: 5000
436 | , pause: 'hover'
437 | }
438 |
439 | $.fn.carousel.Constructor = Carousel
440 |
441 |
442 | /* CAROUSEL NO CONFLICT
443 | * ==================== */
444 |
445 | $.fn.carousel.noConflict = function () {
446 | $.fn.carousel = old
447 | return this
448 | }
449 |
450 | /* CAROUSEL DATA-API
451 | * ================= */
452 |
453 | $(document).on('click.carousel.data-api', '[data-slide], [data-slide-to]', function (e) {
454 | var $this = $(this), href
455 | , $target = $($this.attr('data-target') || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) //strip for ie7
456 | , options = $.extend({}, $target.data(), $this.data())
457 | , slideIndex
458 |
459 | $target.carousel(options)
460 |
461 | if (slideIndex = $this.attr('data-slide-to')) {
462 | $target.data('carousel').pause().to(slideIndex).cycle()
463 | }
464 |
465 | e.preventDefault()
466 | })
467 |
468 | }(window.jQuery);/* =============================================================
469 | * bootstrap-collapse.js v2.3.2
470 | * http://twitter.github.com/bootstrap/javascript.html#collapse
471 | * =============================================================
472 | * Copyright 2012 Twitter, Inc.
473 | *
474 | * Licensed under the Apache License, Version 2.0 (the "License");
475 | * you may not use this file except in compliance with the License.
476 | * You may obtain a copy of the License at
477 | *
478 | * http://www.apache.org/licenses/LICENSE-2.0
479 | *
480 | * Unless required by applicable law or agreed to in writing, software
481 | * distributed under the License is distributed on an "AS IS" BASIS,
482 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
483 | * See the License for the specific language governing permissions and
484 | * limitations under the License.
485 | * ============================================================ */
486 |
487 |
488 | !function ($) {
489 |
490 | "use strict"; // jshint ;_;
491 |
492 |
493 | /* COLLAPSE PUBLIC CLASS DEFINITION
494 | * ================================ */
495 |
496 | var Collapse = function (element, options) {
497 | this.$element = $(element)
498 | this.options = $.extend({}, $.fn.collapse.defaults, options)
499 |
500 | if (this.options.parent) {
501 | this.$parent = $(this.options.parent)
502 | }
503 |
504 | this.options.toggle && this.toggle()
505 | }
506 |
507 | Collapse.prototype = {
508 |
509 | constructor: Collapse
510 |
511 | , dimension: function () {
512 | var hasWidth = this.$element.hasClass('width')
513 | return hasWidth ? 'width' : 'height'
514 | }
515 |
516 | , show: function () {
517 | var dimension
518 | , scroll
519 | , actives
520 | , hasData
521 |
522 | if (this.transitioning || this.$element.hasClass('in')) return
523 |
524 | dimension = this.dimension()
525 | scroll = $.camelCase(['scroll', dimension].join('-'))
526 | actives = this.$parent && this.$parent.find('> .accordion-group > .in')
527 |
528 | if (actives && actives.length) {
529 | hasData = actives.data('collapse')
530 | if (hasData && hasData.transitioning) return
531 | actives.collapse('hide')
532 | hasData || actives.data('collapse', null)
533 | }
534 |
535 | this.$element[dimension](0)
536 | this.transition('addClass', $.Event('show'), 'shown')
537 | $.support.transition && this.$element[dimension](this.$element[0][scroll])
538 | }
539 |
540 | , hide: function () {
541 | var dimension
542 | if (this.transitioning || !this.$element.hasClass('in')) return
543 | dimension = this.dimension()
544 | this.reset(this.$element[dimension]())
545 | this.transition('removeClass', $.Event('hide'), 'hidden')
546 | this.$element[dimension](0)
547 | }
548 |
549 | , reset: function (size) {
550 | var dimension = this.dimension()
551 |
552 | this.$element
553 | .removeClass('collapse')
554 | [dimension](size || 'auto')
555 | [0].offsetWidth
556 |
557 | this.$element[size !== null ? 'addClass' : 'removeClass']('collapse')
558 |
559 | return this
560 | }
561 |
562 | , transition: function (method, startEvent, completeEvent) {
563 | var that = this
564 | , complete = function () {
565 | if (startEvent.type == 'show') that.reset()
566 | that.transitioning = 0
567 | that.$element.trigger(completeEvent)
568 | }
569 |
570 | this.$element.trigger(startEvent)
571 |
572 | if (startEvent.isDefaultPrevented()) return
573 |
574 | this.transitioning = 1
575 |
576 | this.$element[method]('in')
577 |
578 | $.support.transition && this.$element.hasClass('collapse') ?
579 | this.$element.one($.support.transition.end, complete) :
580 | complete()
581 | }
582 |
583 | , toggle: function () {
584 | this[this.$element.hasClass('in') ? 'hide' : 'show']()
585 | }
586 |
587 | }
588 |
589 |
590 | /* COLLAPSE PLUGIN DEFINITION
591 | * ========================== */
592 |
593 | var old = $.fn.collapse
594 |
595 | $.fn.collapse = function (option) {
596 | return this.each(function () {
597 | var $this = $(this)
598 | , data = $this.data('collapse')
599 | , options = $.extend({}, $.fn.collapse.defaults, $this.data(), typeof option == 'object' && option)
600 | if (!data) $this.data('collapse', (data = new Collapse(this, options)))
601 | if (typeof option == 'string') data[option]()
602 | })
603 | }
604 |
605 | $.fn.collapse.defaults = {
606 | toggle: true
607 | }
608 |
609 | $.fn.collapse.Constructor = Collapse
610 |
611 |
612 | /* COLLAPSE NO CONFLICT
613 | * ==================== */
614 |
615 | $.fn.collapse.noConflict = function () {
616 | $.fn.collapse = old
617 | return this
618 | }
619 |
620 |
621 | /* COLLAPSE DATA-API
622 | * ================= */
623 |
624 | $(document).on('click.collapse.data-api', '[data-toggle=collapse]', function (e) {
625 | var $this = $(this), href
626 | , target = $this.attr('data-target')
627 | || e.preventDefault()
628 | || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '') //strip for ie7
629 | , option = $(target).data('collapse') ? 'toggle' : $this.data()
630 | $this[$(target).hasClass('in') ? 'addClass' : 'removeClass']('collapsed')
631 | $(target).collapse(option)
632 | })
633 |
634 | }(window.jQuery);/* ============================================================
635 | * bootstrap-dropdown.js v2.3.2
636 | * http://twitter.github.com/bootstrap/javascript.html#dropdowns
637 | * ============================================================
638 | * Copyright 2012 Twitter, Inc.
639 | *
640 | * Licensed under the Apache License, Version 2.0 (the "License");
641 | * you may not use this file except in compliance with the License.
642 | * You may obtain a copy of the License at
643 | *
644 | * http://www.apache.org/licenses/LICENSE-2.0
645 | *
646 | * Unless required by applicable law or agreed to in writing, software
647 | * distributed under the License is distributed on an "AS IS" BASIS,
648 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
649 | * See the License for the specific language governing permissions and
650 | * limitations under the License.
651 | * ============================================================ */
652 |
653 |
654 | !function ($) {
655 |
656 | "use strict"; // jshint ;_;
657 |
658 |
659 | /* DROPDOWN CLASS DEFINITION
660 | * ========================= */
661 |
662 | var toggle = '[data-toggle=dropdown]'
663 | , Dropdown = function (element) {
664 | var $el = $(element).on('click.dropdown.data-api', this.toggle)
665 | $('html').on('click.dropdown.data-api', function () {
666 | $el.parent().removeClass('open')
667 | })
668 | }
669 |
670 | Dropdown.prototype = {
671 |
672 | constructor: Dropdown
673 |
674 | , toggle: function (e) {
675 | var $this = $(this)
676 | , $parent
677 | , isActive
678 |
679 | if ($this.is('.disabled, :disabled')) return
680 |
681 | $parent = getParent($this)
682 |
683 | isActive = $parent.hasClass('open')
684 |
685 | clearMenus()
686 |
687 | if (!isActive) {
688 | if ('ontouchstart' in document.documentElement) {
689 | // if mobile we we use a backdrop because click events don't delegate
690 | $('
').insertBefore($(this)).on('click', clearMenus)
691 | }
692 | $parent.toggleClass('open')
693 | }
694 |
695 | $this.focus()
696 |
697 | return false
698 | }
699 |
700 | , keydown: function (e) {
701 | var $this
702 | , $items
703 | , $active
704 | , $parent
705 | , isActive
706 | , index
707 |
708 | if (!/(38|40|27)/.test(e.keyCode)) return
709 |
710 | $this = $(this)
711 |
712 | e.preventDefault()
713 | e.stopPropagation()
714 |
715 | if ($this.is('.disabled, :disabled')) return
716 |
717 | $parent = getParent($this)
718 |
719 | isActive = $parent.hasClass('open')
720 |
721 | if (!isActive || (isActive && e.keyCode == 27)) {
722 | if (e.which == 27) $parent.find(toggle).focus()
723 | return $this.click()
724 | }
725 |
726 | $items = $('[role=menu] li:not(.divider):visible a', $parent)
727 |
728 | if (!$items.length) return
729 |
730 | index = $items.index($items.filter(':focus'))
731 |
732 | if (e.keyCode == 38 && index > 0) index-- // up
733 | if (e.keyCode == 40 && index < $items.length - 1) index++ // down
734 | if (!~index) index = 0
735 |
736 | $items
737 | .eq(index)
738 | .focus()
739 | }
740 |
741 | }
742 |
743 | function clearMenus() {
744 | $('.dropdown-backdrop').remove()
745 | $(toggle).each(function () {
746 | getParent($(this)).removeClass('open')
747 | })
748 | }
749 |
750 | function getParent($this) {
751 | var selector = $this.attr('data-target')
752 | , $parent
753 |
754 | if (!selector) {
755 | selector = $this.attr('href')
756 | selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
757 | }
758 |
759 | $parent = selector && $(selector)
760 |
761 | if (!$parent || !$parent.length) $parent = $this.parent()
762 |
763 | return $parent
764 | }
765 |
766 |
767 | /* DROPDOWN PLUGIN DEFINITION
768 | * ========================== */
769 |
770 | var old = $.fn.dropdown
771 |
772 | $.fn.dropdown = function (option) {
773 | return this.each(function () {
774 | var $this = $(this)
775 | , data = $this.data('dropdown')
776 | if (!data) $this.data('dropdown', (data = new Dropdown(this)))
777 | if (typeof option == 'string') data[option].call($this)
778 | })
779 | }
780 |
781 | $.fn.dropdown.Constructor = Dropdown
782 |
783 |
784 | /* DROPDOWN NO CONFLICT
785 | * ==================== */
786 |
787 | $.fn.dropdown.noConflict = function () {
788 | $.fn.dropdown = old
789 | return this
790 | }
791 |
792 |
793 | /* APPLY TO STANDARD DROPDOWN ELEMENTS
794 | * =================================== */
795 |
796 | $(document)
797 | .on('click.dropdown.data-api', clearMenus)
798 | .on('click.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
799 | .on('click.dropdown.data-api' , toggle, Dropdown.prototype.toggle)
800 | .on('keydown.dropdown.data-api', toggle + ', [role=menu]' , Dropdown.prototype.keydown)
801 |
802 | }(window.jQuery);
803 | /* =========================================================
804 | * bootstrap-modal.js v2.3.2
805 | * http://twitter.github.com/bootstrap/javascript.html#modals
806 | * =========================================================
807 | * Copyright 2012 Twitter, Inc.
808 | *
809 | * Licensed under the Apache License, Version 2.0 (the "License");
810 | * you may not use this file except in compliance with the License.
811 | * You may obtain a copy of the License at
812 | *
813 | * http://www.apache.org/licenses/LICENSE-2.0
814 | *
815 | * Unless required by applicable law or agreed to in writing, software
816 | * distributed under the License is distributed on an "AS IS" BASIS,
817 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
818 | * See the License for the specific language governing permissions and
819 | * limitations under the License.
820 | * ========================================================= */
821 |
822 |
823 | !function ($) {
824 |
825 | "use strict"; // jshint ;_;
826 |
827 |
828 | /* MODAL CLASS DEFINITION
829 | * ====================== */
830 |
831 | var Modal = function (element, options) {
832 | this.options = options
833 | this.$element = $(element)
834 | .delegate('[data-dismiss="modal"]', 'click.dismiss.modal', $.proxy(this.hide, this))
835 | this.options.remote && this.$element.find('.modal-body').load(this.options.remote)
836 | }
837 |
838 | Modal.prototype = {
839 |
840 | constructor: Modal
841 |
842 | , toggle: function () {
843 | return this[!this.isShown ? 'show' : 'hide']()
844 | }
845 |
846 | , show: function () {
847 | var that = this
848 | , e = $.Event('show')
849 |
850 | this.$element.trigger(e)
851 |
852 | if (this.isShown || e.isDefaultPrevented()) return
853 |
854 | this.isShown = true
855 |
856 | this.escape()
857 |
858 | this.backdrop(function () {
859 | var transition = $.support.transition && that.$element.hasClass('fade')
860 |
861 | if (!that.$element.parent().length) {
862 | that.$element.appendTo(document.body) //don't move modals dom position
863 | }
864 |
865 | that.$element.show()
866 |
867 | if (transition) {
868 | that.$element[0].offsetWidth // force reflow
869 | }
870 |
871 | that.$element
872 | .addClass('in')
873 | .attr('aria-hidden', false)
874 |
875 | that.enforceFocus()
876 |
877 | transition ?
878 | that.$element.one($.support.transition.end, function () { that.$element.focus().trigger('shown') }) :
879 | that.$element.focus().trigger('shown')
880 |
881 | })
882 | }
883 |
884 | , hide: function (e) {
885 | e && e.preventDefault()
886 |
887 | var that = this
888 |
889 | e = $.Event('hide')
890 |
891 | this.$element.trigger(e)
892 |
893 | if (!this.isShown || e.isDefaultPrevented()) return
894 |
895 | this.isShown = false
896 |
897 | this.escape()
898 |
899 | $(document).off('focusin.modal')
900 |
901 | this.$element
902 | .removeClass('in')
903 | .attr('aria-hidden', true)
904 |
905 | $.support.transition && this.$element.hasClass('fade') ?
906 | this.hideWithTransition() :
907 | this.hideModal()
908 | }
909 |
910 | , enforceFocus: function () {
911 | var that = this
912 | $(document).on('focusin.modal', function (e) {
913 | if (that.$element[0] !== e.target && !that.$element.has(e.target).length) {
914 | that.$element.focus()
915 | }
916 | })
917 | }
918 |
919 | , escape: function () {
920 | var that = this
921 | if (this.isShown && this.options.keyboard) {
922 | this.$element.on('keyup.dismiss.modal', function ( e ) {
923 | e.which == 27 && that.hide()
924 | })
925 | } else if (!this.isShown) {
926 | this.$element.off('keyup.dismiss.modal')
927 | }
928 | }
929 |
930 | , hideWithTransition: function () {
931 | var that = this
932 | , timeout = setTimeout(function () {
933 | that.$element.off($.support.transition.end)
934 | that.hideModal()
935 | }, 500)
936 |
937 | this.$element.one($.support.transition.end, function () {
938 | clearTimeout(timeout)
939 | that.hideModal()
940 | })
941 | }
942 |
943 | , hideModal: function () {
944 | var that = this
945 | this.$element.hide()
946 | this.backdrop(function () {
947 | that.removeBackdrop()
948 | that.$element.trigger('hidden')
949 | })
950 | }
951 |
952 | , removeBackdrop: function () {
953 | this.$backdrop && this.$backdrop.remove()
954 | this.$backdrop = null
955 | }
956 |
957 | , backdrop: function (callback) {
958 | var that = this
959 | , animate = this.$element.hasClass('fade') ? 'fade' : ''
960 |
961 | if (this.isShown && this.options.backdrop) {
962 | var doAnimate = $.support.transition && animate
963 |
964 | this.$backdrop = $('
')
965 | .appendTo(document.body)
966 |
967 | this.$backdrop.click(
968 | this.options.backdrop == 'static' ?
969 | $.proxy(this.$element[0].focus, this.$element[0])
970 | : $.proxy(this.hide, this)
971 | )
972 |
973 | if (doAnimate) this.$backdrop[0].offsetWidth // force reflow
974 |
975 | this.$backdrop.addClass('in')
976 |
977 | if (!callback) return
978 |
979 | doAnimate ?
980 | this.$backdrop.one($.support.transition.end, callback) :
981 | callback()
982 |
983 | } else if (!this.isShown && this.$backdrop) {
984 | this.$backdrop.removeClass('in')
985 |
986 | $.support.transition && this.$element.hasClass('fade')?
987 | this.$backdrop.one($.support.transition.end, callback) :
988 | callback()
989 |
990 | } else if (callback) {
991 | callback()
992 | }
993 | }
994 | }
995 |
996 |
997 | /* MODAL PLUGIN DEFINITION
998 | * ======================= */
999 |
1000 | var old = $.fn.modal
1001 |
1002 | $.fn.modal = function (option) {
1003 | return this.each(function () {
1004 | var $this = $(this)
1005 | , data = $this.data('modal')
1006 | , options = $.extend({}, $.fn.modal.defaults, $this.data(), typeof option == 'object' && option)
1007 | if (!data) $this.data('modal', (data = new Modal(this, options)))
1008 | if (typeof option == 'string') data[option]()
1009 | else if (options.show) data.show()
1010 | })
1011 | }
1012 |
1013 | $.fn.modal.defaults = {
1014 | backdrop: true
1015 | , keyboard: true
1016 | , show: true
1017 | }
1018 |
1019 | $.fn.modal.Constructor = Modal
1020 |
1021 |
1022 | /* MODAL NO CONFLICT
1023 | * ================= */
1024 |
1025 | $.fn.modal.noConflict = function () {
1026 | $.fn.modal = old
1027 | return this
1028 | }
1029 |
1030 |
1031 | /* MODAL DATA-API
1032 | * ============== */
1033 |
1034 | $(document).on('click.modal.data-api', '[data-toggle="modal"]', function (e) {
1035 | var $this = $(this)
1036 | , href = $this.attr('href')
1037 | , $target = $($this.attr('data-target') || (href && href.replace(/.*(?=#[^\s]+$)/, ''))) //strip for ie7
1038 | , option = $target.data('modal') ? 'toggle' : $.extend({ remote:!/#/.test(href) && href }, $target.data(), $this.data())
1039 |
1040 | e.preventDefault()
1041 |
1042 | $target
1043 | .modal(option)
1044 | .one('hide', function () {
1045 | $this.focus()
1046 | })
1047 | })
1048 |
1049 | }(window.jQuery);
1050 | /* ===========================================================
1051 | * bootstrap-tooltip.js v2.3.2
1052 | * http://twitter.github.com/bootstrap/javascript.html#tooltips
1053 | * Inspired by the original jQuery.tipsy by Jason Frame
1054 | * ===========================================================
1055 | * Copyright 2012 Twitter, Inc.
1056 | *
1057 | * Licensed under the Apache License, Version 2.0 (the "License");
1058 | * you may not use this file except in compliance with the License.
1059 | * You may obtain a copy of the License at
1060 | *
1061 | * http://www.apache.org/licenses/LICENSE-2.0
1062 | *
1063 | * Unless required by applicable law or agreed to in writing, software
1064 | * distributed under the License is distributed on an "AS IS" BASIS,
1065 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1066 | * See the License for the specific language governing permissions and
1067 | * limitations under the License.
1068 | * ========================================================== */
1069 |
1070 |
1071 | !function ($) {
1072 |
1073 | "use strict"; // jshint ;_;
1074 |
1075 |
1076 | /* TOOLTIP PUBLIC CLASS DEFINITION
1077 | * =============================== */
1078 |
1079 | var Tooltip = function (element, options) {
1080 | this.init('tooltip', element, options)
1081 | }
1082 |
1083 | Tooltip.prototype = {
1084 |
1085 | constructor: Tooltip
1086 |
1087 | , init: function (type, element, options) {
1088 | var eventIn
1089 | , eventOut
1090 | , triggers
1091 | , trigger
1092 | , i
1093 |
1094 | this.type = type
1095 | this.$element = $(element)
1096 | this.options = this.getOptions(options)
1097 | this.enabled = true
1098 |
1099 | triggers = this.options.trigger.split(' ')
1100 |
1101 | for (i = triggers.length; i--;) {
1102 | trigger = triggers[i]
1103 | if (trigger == 'click') {
1104 | this.$element.on('click.' + this.type, this.options.selector, $.proxy(this.toggle, this))
1105 | } else if (trigger != 'manual') {
1106 | eventIn = trigger == 'hover' ? 'mouseenter' : 'focus'
1107 | eventOut = trigger == 'hover' ? 'mouseleave' : 'blur'
1108 | this.$element.on(eventIn + '.' + this.type, this.options.selector, $.proxy(this.enter, this))
1109 | this.$element.on(eventOut + '.' + this.type, this.options.selector, $.proxy(this.leave, this))
1110 | }
1111 | }
1112 |
1113 | this.options.selector ?
1114 | (this._options = $.extend({}, this.options, { trigger: 'manual', selector: '' })) :
1115 | this.fixTitle()
1116 | }
1117 |
1118 | , getOptions: function (options) {
1119 | options = $.extend({}, $.fn[this.type].defaults, this.$element.data(), options)
1120 |
1121 | if (options.delay && typeof options.delay == 'number') {
1122 | options.delay = {
1123 | show: options.delay
1124 | , hide: options.delay
1125 | }
1126 | }
1127 |
1128 | return options
1129 | }
1130 |
1131 | , enter: function (e) {
1132 | var defaults = $.fn[this.type].defaults
1133 | , options = {}
1134 | , self
1135 |
1136 | this._options && $.each(this._options, function (key, value) {
1137 | if (defaults[key] != value) options[key] = value
1138 | }, this)
1139 |
1140 | self = $(e.currentTarget)[this.type](options).data(this.type)
1141 |
1142 | if (!self.options.delay || !self.options.delay.show) return self.show()
1143 |
1144 | clearTimeout(this.timeout)
1145 | self.hoverState = 'in'
1146 | this.timeout = setTimeout(function() {
1147 | if (self.hoverState == 'in') self.show()
1148 | }, self.options.delay.show)
1149 | }
1150 |
1151 | , leave: function (e) {
1152 | var self = $(e.currentTarget)[this.type](this._options).data(this.type)
1153 |
1154 | if (this.timeout) clearTimeout(this.timeout)
1155 | if (!self.options.delay || !self.options.delay.hide) return self.hide()
1156 |
1157 | self.hoverState = 'out'
1158 | this.timeout = setTimeout(function() {
1159 | if (self.hoverState == 'out') self.hide()
1160 | }, self.options.delay.hide)
1161 | }
1162 |
1163 | , show: function () {
1164 | var $tip
1165 | , pos
1166 | , actualWidth
1167 | , actualHeight
1168 | , placement
1169 | , tp
1170 | , e = $.Event('show')
1171 |
1172 | if (this.hasContent() && this.enabled) {
1173 | this.$element.trigger(e)
1174 | if (e.isDefaultPrevented()) return
1175 | $tip = this.tip()
1176 | this.setContent()
1177 |
1178 | if (this.options.animation) {
1179 | $tip.addClass('fade')
1180 | }
1181 |
1182 | placement = typeof this.options.placement == 'function' ?
1183 | this.options.placement.call(this, $tip[0], this.$element[0]) :
1184 | this.options.placement
1185 |
1186 | $tip
1187 | .detach()
1188 | .css({ top: 0, left: 0, display: 'block' })
1189 |
1190 | this.options.container ? $tip.appendTo(this.options.container) : $tip.insertAfter(this.$element)
1191 |
1192 | pos = this.getPosition()
1193 |
1194 | actualWidth = $tip[0].offsetWidth
1195 | actualHeight = $tip[0].offsetHeight
1196 |
1197 | switch (placement) {
1198 | case 'bottom':
1199 | tp = {top: pos.top + pos.height, left: pos.left + pos.width / 2 - actualWidth / 2}
1200 | break
1201 | case 'top':
1202 | tp = {top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2}
1203 | break
1204 | case 'left':
1205 | tp = {top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth}
1206 | break
1207 | case 'right':
1208 | tp = {top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width}
1209 | break
1210 | }
1211 |
1212 | this.applyPlacement(tp, placement)
1213 | this.$element.trigger('shown')
1214 | }
1215 | }
1216 |
1217 | , applyPlacement: function(offset, placement){
1218 | var $tip = this.tip()
1219 | , width = $tip[0].offsetWidth
1220 | , height = $tip[0].offsetHeight
1221 | , actualWidth
1222 | , actualHeight
1223 | , delta
1224 | , replace
1225 |
1226 | $tip
1227 | .offset(offset)
1228 | .addClass(placement)
1229 | .addClass('in')
1230 |
1231 | actualWidth = $tip[0].offsetWidth
1232 | actualHeight = $tip[0].offsetHeight
1233 |
1234 | if (placement == 'top' && actualHeight != height) {
1235 | offset.top = offset.top + height - actualHeight
1236 | replace = true
1237 | }
1238 |
1239 | if (placement == 'bottom' || placement == 'top') {
1240 | delta = 0
1241 |
1242 | if (offset.left < 0){
1243 | delta = offset.left * -2
1244 | offset.left = 0
1245 | $tip.offset(offset)
1246 | actualWidth = $tip[0].offsetWidth
1247 | actualHeight = $tip[0].offsetHeight
1248 | }
1249 |
1250 | this.replaceArrow(delta - width + actualWidth, actualWidth, 'left')
1251 | } else {
1252 | this.replaceArrow(actualHeight - height, actualHeight, 'top')
1253 | }
1254 |
1255 | if (replace) $tip.offset(offset)
1256 | }
1257 |
1258 | , replaceArrow: function(delta, dimension, position){
1259 | this
1260 | .arrow()
1261 | .css(position, delta ? (50 * (1 - delta / dimension) + "%") : '')
1262 | }
1263 |
1264 | , setContent: function () {
1265 | var $tip = this.tip()
1266 | , title = this.getTitle()
1267 |
1268 | $tip.find('.tooltip-inner')[this.options.html ? 'html' : 'text'](title)
1269 | $tip.removeClass('fade in top bottom left right')
1270 | }
1271 |
1272 | , hide: function () {
1273 | var that = this
1274 | , $tip = this.tip()
1275 | , e = $.Event('hide')
1276 |
1277 | this.$element.trigger(e)
1278 | if (e.isDefaultPrevented()) return
1279 |
1280 | $tip.removeClass('in')
1281 |
1282 | function removeWithAnimation() {
1283 | var timeout = setTimeout(function () {
1284 | $tip.off($.support.transition.end).detach()
1285 | }, 500)
1286 |
1287 | $tip.one($.support.transition.end, function () {
1288 | clearTimeout(timeout)
1289 | $tip.detach()
1290 | })
1291 | }
1292 |
1293 | $.support.transition && this.$tip.hasClass('fade') ?
1294 | removeWithAnimation() :
1295 | $tip.detach()
1296 |
1297 | this.$element.trigger('hidden')
1298 |
1299 | return this
1300 | }
1301 |
1302 | , fixTitle: function () {
1303 | var $e = this.$element
1304 | if ($e.attr('title') || typeof($e.attr('data-original-title')) != 'string') {
1305 | $e.attr('data-original-title', $e.attr('title') || '').attr('title', '')
1306 | }
1307 | }
1308 |
1309 | , hasContent: function () {
1310 | return this.getTitle()
1311 | }
1312 |
1313 | , getPosition: function () {
1314 | var el = this.$element[0]
1315 | return $.extend({}, (typeof el.getBoundingClientRect == 'function') ? el.getBoundingClientRect() : {
1316 | width: el.offsetWidth
1317 | , height: el.offsetHeight
1318 | }, this.$element.offset())
1319 | }
1320 |
1321 | , getTitle: function () {
1322 | var title
1323 | , $e = this.$element
1324 | , o = this.options
1325 |
1326 | title = $e.attr('data-original-title')
1327 | || (typeof o.title == 'function' ? o.title.call($e[0]) : o.title)
1328 |
1329 | return title
1330 | }
1331 |
1332 | , tip: function () {
1333 | return this.$tip = this.$tip || $(this.options.template)
1334 | }
1335 |
1336 | , arrow: function(){
1337 | return this.$arrow = this.$arrow || this.tip().find(".tooltip-arrow")
1338 | }
1339 |
1340 | , validate: function () {
1341 | if (!this.$element[0].parentNode) {
1342 | this.hide()
1343 | this.$element = null
1344 | this.options = null
1345 | }
1346 | }
1347 |
1348 | , enable: function () {
1349 | this.enabled = true
1350 | }
1351 |
1352 | , disable: function () {
1353 | this.enabled = false
1354 | }
1355 |
1356 | , toggleEnabled: function () {
1357 | this.enabled = !this.enabled
1358 | }
1359 |
1360 | , toggle: function (e) {
1361 | var self = e ? $(e.currentTarget)[this.type](this._options).data(this.type) : this
1362 | self.tip().hasClass('in') ? self.hide() : self.show()
1363 | }
1364 |
1365 | , destroy: function () {
1366 | this.hide().$element.off('.' + this.type).removeData(this.type)
1367 | }
1368 |
1369 | }
1370 |
1371 |
1372 | /* TOOLTIP PLUGIN DEFINITION
1373 | * ========================= */
1374 |
1375 | var old = $.fn.tooltip
1376 |
1377 | $.fn.tooltip = function ( option ) {
1378 | return this.each(function () {
1379 | var $this = $(this)
1380 | , data = $this.data('tooltip')
1381 | , options = typeof option == 'object' && option
1382 | if (!data) $this.data('tooltip', (data = new Tooltip(this, options)))
1383 | if (typeof option == 'string') data[option]()
1384 | })
1385 | }
1386 |
1387 | $.fn.tooltip.Constructor = Tooltip
1388 |
1389 | $.fn.tooltip.defaults = {
1390 | animation: true
1391 | , placement: 'top'
1392 | , selector: false
1393 | , template: ''
1394 | , trigger: 'hover focus'
1395 | , title: ''
1396 | , delay: 0
1397 | , html: false
1398 | , container: false
1399 | }
1400 |
1401 |
1402 | /* TOOLTIP NO CONFLICT
1403 | * =================== */
1404 |
1405 | $.fn.tooltip.noConflict = function () {
1406 | $.fn.tooltip = old
1407 | return this
1408 | }
1409 |
1410 | }(window.jQuery);
1411 | /* ===========================================================
1412 | * bootstrap-popover.js v2.3.2
1413 | * http://twitter.github.com/bootstrap/javascript.html#popovers
1414 | * ===========================================================
1415 | * Copyright 2012 Twitter, Inc.
1416 | *
1417 | * Licensed under the Apache License, Version 2.0 (the "License");
1418 | * you may not use this file except in compliance with the License.
1419 | * You may obtain a copy of the License at
1420 | *
1421 | * http://www.apache.org/licenses/LICENSE-2.0
1422 | *
1423 | * Unless required by applicable law or agreed to in writing, software
1424 | * distributed under the License is distributed on an "AS IS" BASIS,
1425 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1426 | * See the License for the specific language governing permissions and
1427 | * limitations under the License.
1428 | * =========================================================== */
1429 |
1430 |
1431 | !function ($) {
1432 |
1433 | "use strict"; // jshint ;_;
1434 |
1435 |
1436 | /* POPOVER PUBLIC CLASS DEFINITION
1437 | * =============================== */
1438 |
1439 | var Popover = function (element, options) {
1440 | this.init('popover', element, options)
1441 | }
1442 |
1443 |
1444 | /* NOTE: POPOVER EXTENDS BOOTSTRAP-TOOLTIP.js
1445 | ========================================== */
1446 |
1447 | Popover.prototype = $.extend({}, $.fn.tooltip.Constructor.prototype, {
1448 |
1449 | constructor: Popover
1450 |
1451 | , setContent: function () {
1452 | var $tip = this.tip()
1453 | , title = this.getTitle()
1454 | , content = this.getContent()
1455 |
1456 | $tip.find('.popover-title')[this.options.html ? 'html' : 'text'](title)
1457 | $tip.find('.popover-content')[this.options.html ? 'html' : 'text'](content)
1458 |
1459 | $tip.removeClass('fade top bottom left right in')
1460 | }
1461 |
1462 | , hasContent: function () {
1463 | return this.getTitle() || this.getContent()
1464 | }
1465 |
1466 | , getContent: function () {
1467 | var content
1468 | , $e = this.$element
1469 | , o = this.options
1470 |
1471 | content = (typeof o.content == 'function' ? o.content.call($e[0]) : o.content)
1472 | || $e.attr('data-content')
1473 |
1474 | return content
1475 | }
1476 |
1477 | , tip: function () {
1478 | if (!this.$tip) {
1479 | this.$tip = $(this.options.template)
1480 | }
1481 | return this.$tip
1482 | }
1483 |
1484 | , destroy: function () {
1485 | this.hide().$element.off('.' + this.type).removeData(this.type)
1486 | }
1487 |
1488 | })
1489 |
1490 |
1491 | /* POPOVER PLUGIN DEFINITION
1492 | * ======================= */
1493 |
1494 | var old = $.fn.popover
1495 |
1496 | $.fn.popover = function (option) {
1497 | return this.each(function () {
1498 | var $this = $(this)
1499 | , data = $this.data('popover')
1500 | , options = typeof option == 'object' && option
1501 | if (!data) $this.data('popover', (data = new Popover(this, options)))
1502 | if (typeof option == 'string') data[option]()
1503 | })
1504 | }
1505 |
1506 | $.fn.popover.Constructor = Popover
1507 |
1508 | $.fn.popover.defaults = $.extend({} , $.fn.tooltip.defaults, {
1509 | placement: 'right'
1510 | , trigger: 'click'
1511 | , content: ''
1512 | , template: ''
1513 | })
1514 |
1515 |
1516 | /* POPOVER NO CONFLICT
1517 | * =================== */
1518 |
1519 | $.fn.popover.noConflict = function () {
1520 | $.fn.popover = old
1521 | return this
1522 | }
1523 |
1524 | }(window.jQuery);
1525 | /* =============================================================
1526 | * bootstrap-scrollspy.js v2.3.2
1527 | * http://twitter.github.com/bootstrap/javascript.html#scrollspy
1528 | * =============================================================
1529 | * Copyright 2012 Twitter, Inc.
1530 | *
1531 | * Licensed under the Apache License, Version 2.0 (the "License");
1532 | * you may not use this file except in compliance with the License.
1533 | * You may obtain a copy of the License at
1534 | *
1535 | * http://www.apache.org/licenses/LICENSE-2.0
1536 | *
1537 | * Unless required by applicable law or agreed to in writing, software
1538 | * distributed under the License is distributed on an "AS IS" BASIS,
1539 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1540 | * See the License for the specific language governing permissions and
1541 | * limitations under the License.
1542 | * ============================================================== */
1543 |
1544 |
1545 | !function ($) {
1546 |
1547 | "use strict"; // jshint ;_;
1548 |
1549 |
1550 | /* SCROLLSPY CLASS DEFINITION
1551 | * ========================== */
1552 |
1553 | function ScrollSpy(element, options) {
1554 | var process = $.proxy(this.process, this)
1555 | , $element = $(element).is('body') ? $(window) : $(element)
1556 | , href
1557 | this.options = $.extend({}, $.fn.scrollspy.defaults, options)
1558 | this.$scrollElement = $element.on('scroll.scroll-spy.data-api', process)
1559 | this.selector = (this.options.target
1560 | || ((href = $(element).attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) //strip for ie7
1561 | || '') + ' .nav li > a'
1562 | this.$body = $('body')
1563 | this.refresh()
1564 | this.process()
1565 | }
1566 |
1567 | ScrollSpy.prototype = {
1568 |
1569 | constructor: ScrollSpy
1570 |
1571 | , refresh: function () {
1572 | var self = this
1573 | , $targets
1574 |
1575 | this.offsets = $([])
1576 | this.targets = $([])
1577 |
1578 | $targets = this.$body
1579 | .find(this.selector)
1580 | .map(function () {
1581 | var $el = $(this)
1582 | , href = $el.data('target') || $el.attr('href')
1583 | , $href = /^#\w/.test(href) && $(href)
1584 | return ( $href
1585 | && $href.length
1586 | && [[ $href.position().top + (!$.isWindow(self.$scrollElement.get(0)) && self.$scrollElement.scrollTop()), href ]] ) || null
1587 | })
1588 | .sort(function (a, b) { return a[0] - b[0] })
1589 | .each(function () {
1590 | self.offsets.push(this[0])
1591 | self.targets.push(this[1])
1592 | })
1593 | }
1594 |
1595 | , process: function () {
1596 | var scrollTop = this.$scrollElement.scrollTop() + this.options.offset
1597 | , scrollHeight = this.$scrollElement[0].scrollHeight || this.$body[0].scrollHeight
1598 | , maxScroll = scrollHeight - this.$scrollElement.height()
1599 | , offsets = this.offsets
1600 | , targets = this.targets
1601 | , activeTarget = this.activeTarget
1602 | , i
1603 |
1604 | if (scrollTop >= maxScroll) {
1605 | return activeTarget != (i = targets.last()[0])
1606 | && this.activate ( i )
1607 | }
1608 |
1609 | for (i = offsets.length; i--;) {
1610 | activeTarget != targets[i]
1611 | && scrollTop >= offsets[i]
1612 | && (!offsets[i + 1] || scrollTop <= offsets[i + 1])
1613 | && this.activate( targets[i] )
1614 | }
1615 | }
1616 |
1617 | , activate: function (target) {
1618 | var active
1619 | , selector
1620 |
1621 | this.activeTarget = target
1622 |
1623 | $(this.selector)
1624 | .parent('.active')
1625 | .removeClass('active')
1626 |
1627 | selector = this.selector
1628 | + '[data-target="' + target + '"],'
1629 | + this.selector + '[href="' + target + '"]'
1630 |
1631 | active = $(selector)
1632 | .parent('li')
1633 | .addClass('active')
1634 |
1635 | if (active.parent('.dropdown-menu').length) {
1636 | active = active.closest('li.dropdown').addClass('active')
1637 | }
1638 |
1639 | active.trigger('activate')
1640 | }
1641 |
1642 | }
1643 |
1644 |
1645 | /* SCROLLSPY PLUGIN DEFINITION
1646 | * =========================== */
1647 |
1648 | var old = $.fn.scrollspy
1649 |
1650 | $.fn.scrollspy = function (option) {
1651 | return this.each(function () {
1652 | var $this = $(this)
1653 | , data = $this.data('scrollspy')
1654 | , options = typeof option == 'object' && option
1655 | if (!data) $this.data('scrollspy', (data = new ScrollSpy(this, options)))
1656 | if (typeof option == 'string') data[option]()
1657 | })
1658 | }
1659 |
1660 | $.fn.scrollspy.Constructor = ScrollSpy
1661 |
1662 | $.fn.scrollspy.defaults = {
1663 | offset: 10
1664 | }
1665 |
1666 |
1667 | /* SCROLLSPY NO CONFLICT
1668 | * ===================== */
1669 |
1670 | $.fn.scrollspy.noConflict = function () {
1671 | $.fn.scrollspy = old
1672 | return this
1673 | }
1674 |
1675 |
1676 | /* SCROLLSPY DATA-API
1677 | * ================== */
1678 |
1679 | $(window).on('load', function () {
1680 | $('[data-spy="scroll"]').each(function () {
1681 | var $spy = $(this)
1682 | $spy.scrollspy($spy.data())
1683 | })
1684 | })
1685 |
1686 | }(window.jQuery);/* ========================================================
1687 | * bootstrap-tab.js v2.3.2
1688 | * http://twitter.github.com/bootstrap/javascript.html#tabs
1689 | * ========================================================
1690 | * Copyright 2012 Twitter, Inc.
1691 | *
1692 | * Licensed under the Apache License, Version 2.0 (the "License");
1693 | * you may not use this file except in compliance with the License.
1694 | * You may obtain a copy of the License at
1695 | *
1696 | * http://www.apache.org/licenses/LICENSE-2.0
1697 | *
1698 | * Unless required by applicable law or agreed to in writing, software
1699 | * distributed under the License is distributed on an "AS IS" BASIS,
1700 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1701 | * See the License for the specific language governing permissions and
1702 | * limitations under the License.
1703 | * ======================================================== */
1704 |
1705 |
1706 | !function ($) {
1707 |
1708 | "use strict"; // jshint ;_;
1709 |
1710 |
1711 | /* TAB CLASS DEFINITION
1712 | * ==================== */
1713 |
1714 | var Tab = function (element) {
1715 | this.element = $(element)
1716 | }
1717 |
1718 | Tab.prototype = {
1719 |
1720 | constructor: Tab
1721 |
1722 | , show: function () {
1723 | var $this = this.element
1724 | , $ul = $this.closest('ul:not(.dropdown-menu)')
1725 | , selector = $this.attr('data-target')
1726 | , previous
1727 | , $target
1728 | , e
1729 |
1730 | if (!selector) {
1731 | selector = $this.attr('href')
1732 | selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7
1733 | }
1734 |
1735 | if ( $this.parent('li').hasClass('active') ) return
1736 |
1737 | previous = $ul.find('.active:last a')[0]
1738 |
1739 | e = $.Event('show', {
1740 | relatedTarget: previous
1741 | })
1742 |
1743 | $this.trigger(e)
1744 |
1745 | if (e.isDefaultPrevented()) return
1746 |
1747 | $target = $(selector)
1748 |
1749 | this.activate($this.parent('li'), $ul)
1750 | this.activate($target, $target.parent(), function () {
1751 | $this.trigger({
1752 | type: 'shown'
1753 | , relatedTarget: previous
1754 | })
1755 | })
1756 | }
1757 |
1758 | , activate: function ( element, container, callback) {
1759 | var $active = container.find('> .active')
1760 | , transition = callback
1761 | && $.support.transition
1762 | && $active.hasClass('fade')
1763 |
1764 | function next() {
1765 | $active
1766 | .removeClass('active')
1767 | .find('> .dropdown-menu > .active')
1768 | .removeClass('active')
1769 |
1770 | element.addClass('active')
1771 |
1772 | if (transition) {
1773 | element[0].offsetWidth // reflow for transition
1774 | element.addClass('in')
1775 | } else {
1776 | element.removeClass('fade')
1777 | }
1778 |
1779 | if ( element.parent('.dropdown-menu') ) {
1780 | element.closest('li.dropdown').addClass('active')
1781 | }
1782 |
1783 | callback && callback()
1784 | }
1785 |
1786 | transition ?
1787 | $active.one($.support.transition.end, next) :
1788 | next()
1789 |
1790 | $active.removeClass('in')
1791 | }
1792 | }
1793 |
1794 |
1795 | /* TAB PLUGIN DEFINITION
1796 | * ===================== */
1797 |
1798 | var old = $.fn.tab
1799 |
1800 | $.fn.tab = function ( option ) {
1801 | return this.each(function () {
1802 | var $this = $(this)
1803 | , data = $this.data('tab')
1804 | if (!data) $this.data('tab', (data = new Tab(this)))
1805 | if (typeof option == 'string') data[option]()
1806 | })
1807 | }
1808 |
1809 | $.fn.tab.Constructor = Tab
1810 |
1811 |
1812 | /* TAB NO CONFLICT
1813 | * =============== */
1814 |
1815 | $.fn.tab.noConflict = function () {
1816 | $.fn.tab = old
1817 | return this
1818 | }
1819 |
1820 |
1821 | /* TAB DATA-API
1822 | * ============ */
1823 |
1824 | $(document).on('click.tab.data-api', '[data-toggle="tab"], [data-toggle="pill"]', function (e) {
1825 | e.preventDefault()
1826 | $(this).tab('show')
1827 | })
1828 |
1829 | }(window.jQuery);/* =============================================================
1830 | * bootstrap-typeahead.js v2.3.2
1831 | * http://twitter.github.com/bootstrap/javascript.html#typeahead
1832 | * =============================================================
1833 | * Copyright 2012 Twitter, Inc.
1834 | *
1835 | * Licensed under the Apache License, Version 2.0 (the "License");
1836 | * you may not use this file except in compliance with the License.
1837 | * You may obtain a copy of the License at
1838 | *
1839 | * http://www.apache.org/licenses/LICENSE-2.0
1840 | *
1841 | * Unless required by applicable law or agreed to in writing, software
1842 | * distributed under the License is distributed on an "AS IS" BASIS,
1843 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1844 | * See the License for the specific language governing permissions and
1845 | * limitations under the License.
1846 | * ============================================================ */
1847 |
1848 |
1849 | !function($){
1850 |
1851 | "use strict"; // jshint ;_;
1852 |
1853 |
1854 | /* TYPEAHEAD PUBLIC CLASS DEFINITION
1855 | * ================================= */
1856 |
1857 | var Typeahead = function (element, options) {
1858 | this.$element = $(element)
1859 | this.options = $.extend({}, $.fn.typeahead.defaults, options)
1860 | this.matcher = this.options.matcher || this.matcher
1861 | this.sorter = this.options.sorter || this.sorter
1862 | this.highlighter = this.options.highlighter || this.highlighter
1863 | this.updater = this.options.updater || this.updater
1864 | this.source = this.options.source
1865 | this.$menu = $(this.options.menu)
1866 | this.shown = false
1867 | this.listen()
1868 | }
1869 |
1870 | Typeahead.prototype = {
1871 |
1872 | constructor: Typeahead
1873 |
1874 | , select: function () {
1875 | var val = this.$menu.find('.active').attr('data-value')
1876 | this.$element
1877 | .val(this.updater(val))
1878 | .change()
1879 | return this.hide()
1880 | }
1881 |
1882 | , updater: function (item) {
1883 | return item
1884 | }
1885 |
1886 | , show: function () {
1887 | var pos = $.extend({}, this.$element.position(), {
1888 | height: this.$element[0].offsetHeight
1889 | })
1890 |
1891 | this.$menu
1892 | .insertAfter(this.$element)
1893 | .css({
1894 | top: pos.top + pos.height
1895 | , left: pos.left
1896 | })
1897 | .show()
1898 |
1899 | this.shown = true
1900 | return this
1901 | }
1902 |
1903 | , hide: function () {
1904 | this.$menu.hide()
1905 | this.shown = false
1906 | return this
1907 | }
1908 |
1909 | , lookup: function (event) {
1910 | var items
1911 |
1912 | this.query = this.$element.val()
1913 |
1914 | if (!this.query || this.query.length < this.options.minLength) {
1915 | return this.shown ? this.hide() : this
1916 | }
1917 |
1918 | items = $.isFunction(this.source) ? this.source(this.query, $.proxy(this.process, this)) : this.source
1919 |
1920 | return items ? this.process(items) : this
1921 | }
1922 |
1923 | , process: function (items) {
1924 | var that = this
1925 |
1926 | items = $.grep(items, function (item) {
1927 | return that.matcher(item)
1928 | })
1929 |
1930 | items = this.sorter(items)
1931 |
1932 | if (!items.length) {
1933 | return this.shown ? this.hide() : this
1934 | }
1935 |
1936 | return this.render(items.slice(0, this.options.items)).show()
1937 | }
1938 |
1939 | , matcher: function (item) {
1940 | return ~item.toLowerCase().indexOf(this.query.toLowerCase())
1941 | }
1942 |
1943 | , sorter: function (items) {
1944 | var beginswith = []
1945 | , caseSensitive = []
1946 | , caseInsensitive = []
1947 | , item
1948 |
1949 | while (item = items.shift()) {
1950 | if (!item.toLowerCase().indexOf(this.query.toLowerCase())) beginswith.push(item)
1951 | else if (~item.indexOf(this.query)) caseSensitive.push(item)
1952 | else caseInsensitive.push(item)
1953 | }
1954 |
1955 | return beginswith.concat(caseSensitive, caseInsensitive)
1956 | }
1957 |
1958 | , highlighter: function (item) {
1959 | var query = this.query.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, '\\$&')
1960 | return item.replace(new RegExp('(' + query + ')', 'ig'), function ($1, match) {
1961 | return '' + match + ' '
1962 | })
1963 | }
1964 |
1965 | , render: function (items) {
1966 | var that = this
1967 |
1968 | items = $(items).map(function (i, item) {
1969 | i = $(that.options.item).attr('data-value', item)
1970 | i.find('a').html(that.highlighter(item))
1971 | return i[0]
1972 | })
1973 |
1974 | items.first().addClass('active')
1975 | this.$menu.html(items)
1976 | return this
1977 | }
1978 |
1979 | , next: function (event) {
1980 | var active = this.$menu.find('.active').removeClass('active')
1981 | , next = active.next()
1982 |
1983 | if (!next.length) {
1984 | next = $(this.$menu.find('li')[0])
1985 | }
1986 |
1987 | next.addClass('active')
1988 | }
1989 |
1990 | , prev: function (event) {
1991 | var active = this.$menu.find('.active').removeClass('active')
1992 | , prev = active.prev()
1993 |
1994 | if (!prev.length) {
1995 | prev = this.$menu.find('li').last()
1996 | }
1997 |
1998 | prev.addClass('active')
1999 | }
2000 |
2001 | , listen: function () {
2002 | this.$element
2003 | .on('focus', $.proxy(this.focus, this))
2004 | .on('blur', $.proxy(this.blur, this))
2005 | .on('keypress', $.proxy(this.keypress, this))
2006 | .on('keyup', $.proxy(this.keyup, this))
2007 |
2008 | if (this.eventSupported('keydown')) {
2009 | this.$element.on('keydown', $.proxy(this.keydown, this))
2010 | }
2011 |
2012 | this.$menu
2013 | .on('click', $.proxy(this.click, this))
2014 | .on('mouseenter', 'li', $.proxy(this.mouseenter, this))
2015 | .on('mouseleave', 'li', $.proxy(this.mouseleave, this))
2016 | }
2017 |
2018 | , eventSupported: function(eventName) {
2019 | var isSupported = eventName in this.$element
2020 | if (!isSupported) {
2021 | this.$element.setAttribute(eventName, 'return;')
2022 | isSupported = typeof this.$element[eventName] === 'function'
2023 | }
2024 | return isSupported
2025 | }
2026 |
2027 | , move: function (e) {
2028 | if (!this.shown) return
2029 |
2030 | switch(e.keyCode) {
2031 | case 9: // tab
2032 | case 13: // enter
2033 | case 27: // escape
2034 | e.preventDefault()
2035 | break
2036 |
2037 | case 38: // up arrow
2038 | e.preventDefault()
2039 | this.prev()
2040 | break
2041 |
2042 | case 40: // down arrow
2043 | e.preventDefault()
2044 | this.next()
2045 | break
2046 | }
2047 |
2048 | e.stopPropagation()
2049 | }
2050 |
2051 | , keydown: function (e) {
2052 | this.suppressKeyPressRepeat = ~$.inArray(e.keyCode, [40,38,9,13,27])
2053 | this.move(e)
2054 | }
2055 |
2056 | , keypress: function (e) {
2057 | if (this.suppressKeyPressRepeat) return
2058 | this.move(e)
2059 | }
2060 |
2061 | , keyup: function (e) {
2062 | switch(e.keyCode) {
2063 | case 40: // down arrow
2064 | case 38: // up arrow
2065 | case 16: // shift
2066 | case 17: // ctrl
2067 | case 18: // alt
2068 | break
2069 |
2070 | case 9: // tab
2071 | case 13: // enter
2072 | if (!this.shown) return
2073 | this.select()
2074 | break
2075 |
2076 | case 27: // escape
2077 | if (!this.shown) return
2078 | this.hide()
2079 | break
2080 |
2081 | default:
2082 | this.lookup()
2083 | }
2084 |
2085 | e.stopPropagation()
2086 | e.preventDefault()
2087 | }
2088 |
2089 | , focus: function (e) {
2090 | this.focused = true
2091 | }
2092 |
2093 | , blur: function (e) {
2094 | this.focused = false
2095 | if (!this.mousedover && this.shown) this.hide()
2096 | }
2097 |
2098 | , click: function (e) {
2099 | e.stopPropagation()
2100 | e.preventDefault()
2101 | this.select()
2102 | this.$element.focus()
2103 | }
2104 |
2105 | , mouseenter: function (e) {
2106 | this.mousedover = true
2107 | this.$menu.find('.active').removeClass('active')
2108 | $(e.currentTarget).addClass('active')
2109 | }
2110 |
2111 | , mouseleave: function (e) {
2112 | this.mousedover = false
2113 | if (!this.focused && this.shown) this.hide()
2114 | }
2115 |
2116 | }
2117 |
2118 |
2119 | /* TYPEAHEAD PLUGIN DEFINITION
2120 | * =========================== */
2121 |
2122 | var old = $.fn.typeahead
2123 |
2124 | $.fn.typeahead = function (option) {
2125 | return this.each(function () {
2126 | var $this = $(this)
2127 | , data = $this.data('typeahead')
2128 | , options = typeof option == 'object' && option
2129 | if (!data) $this.data('typeahead', (data = new Typeahead(this, options)))
2130 | if (typeof option == 'string') data[option]()
2131 | })
2132 | }
2133 |
2134 | $.fn.typeahead.defaults = {
2135 | source: []
2136 | , items: 8
2137 | , menu: ''
2138 | , item: ' '
2139 | , minLength: 1
2140 | }
2141 |
2142 | $.fn.typeahead.Constructor = Typeahead
2143 |
2144 |
2145 | /* TYPEAHEAD NO CONFLICT
2146 | * =================== */
2147 |
2148 | $.fn.typeahead.noConflict = function () {
2149 | $.fn.typeahead = old
2150 | return this
2151 | }
2152 |
2153 |
2154 | /* TYPEAHEAD DATA-API
2155 | * ================== */
2156 |
2157 | $(document).on('focus.typeahead.data-api', '[data-provide="typeahead"]', function (e) {
2158 | var $this = $(this)
2159 | if ($this.data('typeahead')) return
2160 | $this.typeahead($this.data())
2161 | })
2162 |
2163 | }(window.jQuery);
2164 | /* ==========================================================
2165 | * bootstrap-affix.js v2.3.2
2166 | * http://twitter.github.com/bootstrap/javascript.html#affix
2167 | * ==========================================================
2168 | * Copyright 2012 Twitter, Inc.
2169 | *
2170 | * Licensed under the Apache License, Version 2.0 (the "License");
2171 | * you may not use this file except in compliance with the License.
2172 | * You may obtain a copy of the License at
2173 | *
2174 | * http://www.apache.org/licenses/LICENSE-2.0
2175 | *
2176 | * Unless required by applicable law or agreed to in writing, software
2177 | * distributed under the License is distributed on an "AS IS" BASIS,
2178 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
2179 | * See the License for the specific language governing permissions and
2180 | * limitations under the License.
2181 | * ========================================================== */
2182 |
2183 |
2184 | !function ($) {
2185 |
2186 | "use strict"; // jshint ;_;
2187 |
2188 |
2189 | /* AFFIX CLASS DEFINITION
2190 | * ====================== */
2191 |
2192 | var Affix = function (element, options) {
2193 | this.options = $.extend({}, $.fn.affix.defaults, options)
2194 | this.$window = $(window)
2195 | .on('scroll.affix.data-api', $.proxy(this.checkPosition, this))
2196 | .on('click.affix.data-api', $.proxy(function () { setTimeout($.proxy(this.checkPosition, this), 1) }, this))
2197 | this.$element = $(element)
2198 | this.checkPosition()
2199 | }
2200 |
2201 | Affix.prototype.checkPosition = function () {
2202 | if (!this.$element.is(':visible')) return
2203 |
2204 | var scrollHeight = $(document).height()
2205 | , scrollTop = this.$window.scrollTop()
2206 | , position = this.$element.offset()
2207 | , offset = this.options.offset
2208 | , offsetBottom = offset.bottom
2209 | , offsetTop = offset.top
2210 | , reset = 'affix affix-top affix-bottom'
2211 | , affix
2212 |
2213 | if (typeof offset != 'object') offsetBottom = offsetTop = offset
2214 | if (typeof offsetTop == 'function') offsetTop = offset.top()
2215 | if (typeof offsetBottom == 'function') offsetBottom = offset.bottom()
2216 |
2217 | affix = this.unpin != null && (scrollTop + this.unpin <= position.top) ?
2218 | false : offsetBottom != null && (position.top + this.$element.height() >= scrollHeight - offsetBottom) ?
2219 | 'bottom' : offsetTop != null && scrollTop <= offsetTop ?
2220 | 'top' : false
2221 |
2222 | if (this.affixed === affix) return
2223 |
2224 | this.affixed = affix
2225 | this.unpin = affix == 'bottom' ? position.top - scrollTop : null
2226 |
2227 | this.$element.removeClass(reset).addClass('affix' + (affix ? '-' + affix : ''))
2228 | }
2229 |
2230 |
2231 | /* AFFIX PLUGIN DEFINITION
2232 | * ======================= */
2233 |
2234 | var old = $.fn.affix
2235 |
2236 | $.fn.affix = function (option) {
2237 | return this.each(function () {
2238 | var $this = $(this)
2239 | , data = $this.data('affix')
2240 | , options = typeof option == 'object' && option
2241 | if (!data) $this.data('affix', (data = new Affix(this, options)))
2242 | if (typeof option == 'string') data[option]()
2243 | })
2244 | }
2245 |
2246 | $.fn.affix.Constructor = Affix
2247 |
2248 | $.fn.affix.defaults = {
2249 | offset: 0
2250 | }
2251 |
2252 |
2253 | /* AFFIX NO CONFLICT
2254 | * ================= */
2255 |
2256 | $.fn.affix.noConflict = function () {
2257 | $.fn.affix = old
2258 | return this
2259 | }
2260 |
2261 |
2262 | /* AFFIX DATA-API
2263 | * ============== */
2264 |
2265 | $(window).on('load', function () {
2266 | $('[data-spy="affix"]').each(function () {
2267 | var $spy = $(this)
2268 | , data = $spy.data()
2269 |
2270 | data.offset = data.offset || {}
2271 |
2272 | data.offsetBottom && (data.offset.bottom = data.offsetBottom)
2273 | data.offsetTop && (data.offset.top = data.offsetTop)
2274 |
2275 | $spy.affix(data)
2276 | })
2277 | })
2278 |
2279 |
2280 | }(window.jQuery);
--------------------------------------------------------------------------------
/vendor/assets/stylesheets/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/vendor/assets/stylesheets/.gitkeep
--------------------------------------------------------------------------------
/vendor/assets/stylesheets/bootstrap-responsive.css:
--------------------------------------------------------------------------------
1 | /*!
2 | * Bootstrap Responsive v2.3.2
3 | *
4 | * Copyright 2012 Twitter, Inc
5 | * Licensed under the Apache License v2.0
6 | * http://www.apache.org/licenses/LICENSE-2.0
7 | *
8 | * Designed and built with all the love in the world @twitter by @mdo and @fat.
9 | */
10 |
11 | .clearfix {
12 | *zoom: 1;
13 | }
14 |
15 | .clearfix:before,
16 | .clearfix:after {
17 | display: table;
18 | line-height: 0;
19 | content: "";
20 | }
21 |
22 | .clearfix:after {
23 | clear: both;
24 | }
25 |
26 | .hide-text {
27 | font: 0/0 a;
28 | color: transparent;
29 | text-shadow: none;
30 | background-color: transparent;
31 | border: 0;
32 | }
33 |
34 | .input-block-level {
35 | display: block;
36 | width: 100%;
37 | min-height: 30px;
38 | -webkit-box-sizing: border-box;
39 | -moz-box-sizing: border-box;
40 | box-sizing: border-box;
41 | }
42 |
43 | @-ms-viewport {
44 | width: device-width;
45 | }
46 |
47 | .hidden {
48 | display: none;
49 | visibility: hidden;
50 | }
51 |
52 | .visible-phone {
53 | display: none !important;
54 | }
55 |
56 | .visible-tablet {
57 | display: none !important;
58 | }
59 |
60 | .hidden-desktop {
61 | display: none !important;
62 | }
63 |
64 | .visible-desktop {
65 | display: inherit !important;
66 | }
67 |
68 | @media (min-width: 768px) and (max-width: 979px) {
69 | .hidden-desktop {
70 | display: inherit !important;
71 | }
72 | .visible-desktop {
73 | display: none !important ;
74 | }
75 | .visible-tablet {
76 | display: inherit !important;
77 | }
78 | .hidden-tablet {
79 | display: none !important;
80 | }
81 | }
82 |
83 | @media (max-width: 767px) {
84 | .hidden-desktop {
85 | display: inherit !important;
86 | }
87 | .visible-desktop {
88 | display: none !important;
89 | }
90 | .visible-phone {
91 | display: inherit !important;
92 | }
93 | .hidden-phone {
94 | display: none !important;
95 | }
96 | }
97 |
98 | .visible-print {
99 | display: none !important;
100 | }
101 |
102 | @media print {
103 | .visible-print {
104 | display: inherit !important;
105 | }
106 | .hidden-print {
107 | display: none !important;
108 | }
109 | }
110 |
111 | @media (min-width: 1200px) {
112 | .row {
113 | margin-left: -30px;
114 | *zoom: 1;
115 | }
116 | .row:before,
117 | .row:after {
118 | display: table;
119 | line-height: 0;
120 | content: "";
121 | }
122 | .row:after {
123 | clear: both;
124 | }
125 | [class*="span"] {
126 | float: left;
127 | min-height: 1px;
128 | margin-left: 30px;
129 | }
130 | .container,
131 | .navbar-static-top .container,
132 | .navbar-fixed-top .container,
133 | .navbar-fixed-bottom .container {
134 | width: 1170px;
135 | }
136 | .span12 {
137 | width: 1170px;
138 | }
139 | .span11 {
140 | width: 1070px;
141 | }
142 | .span10 {
143 | width: 970px;
144 | }
145 | .span9 {
146 | width: 870px;
147 | }
148 | .span8 {
149 | width: 770px;
150 | }
151 | .span7 {
152 | width: 670px;
153 | }
154 | .span6 {
155 | width: 570px;
156 | }
157 | .span5 {
158 | width: 470px;
159 | }
160 | .span4 {
161 | width: 370px;
162 | }
163 | .span3 {
164 | width: 270px;
165 | }
166 | .span2 {
167 | width: 170px;
168 | }
169 | .span1 {
170 | width: 70px;
171 | }
172 | .offset12 {
173 | margin-left: 1230px;
174 | }
175 | .offset11 {
176 | margin-left: 1130px;
177 | }
178 | .offset10 {
179 | margin-left: 1030px;
180 | }
181 | .offset9 {
182 | margin-left: 930px;
183 | }
184 | .offset8 {
185 | margin-left: 830px;
186 | }
187 | .offset7 {
188 | margin-left: 730px;
189 | }
190 | .offset6 {
191 | margin-left: 630px;
192 | }
193 | .offset5 {
194 | margin-left: 530px;
195 | }
196 | .offset4 {
197 | margin-left: 430px;
198 | }
199 | .offset3 {
200 | margin-left: 330px;
201 | }
202 | .offset2 {
203 | margin-left: 230px;
204 | }
205 | .offset1 {
206 | margin-left: 130px;
207 | }
208 | .row-fluid {
209 | width: 100%;
210 | *zoom: 1;
211 | }
212 | .row-fluid:before,
213 | .row-fluid:after {
214 | display: table;
215 | line-height: 0;
216 | content: "";
217 | }
218 | .row-fluid:after {
219 | clear: both;
220 | }
221 | .row-fluid [class*="span"] {
222 | display: block;
223 | float: left;
224 | width: 100%;
225 | min-height: 30px;
226 | margin-left: 2.564102564102564%;
227 | *margin-left: 2.5109110747408616%;
228 | -webkit-box-sizing: border-box;
229 | -moz-box-sizing: border-box;
230 | box-sizing: border-box;
231 | }
232 | .row-fluid [class*="span"]:first-child {
233 | margin-left: 0;
234 | }
235 | .row-fluid .controls-row [class*="span"] + [class*="span"] {
236 | margin-left: 2.564102564102564%;
237 | }
238 | .row-fluid .span12 {
239 | width: 100%;
240 | *width: 99.94680851063829%;
241 | }
242 | .row-fluid .span11 {
243 | width: 91.45299145299145%;
244 | *width: 91.39979996362975%;
245 | }
246 | .row-fluid .span10 {
247 | width: 82.90598290598291%;
248 | *width: 82.8527914166212%;
249 | }
250 | .row-fluid .span9 {
251 | width: 74.35897435897436%;
252 | *width: 74.30578286961266%;
253 | }
254 | .row-fluid .span8 {
255 | width: 65.81196581196582%;
256 | *width: 65.75877432260411%;
257 | }
258 | .row-fluid .span7 {
259 | width: 57.26495726495726%;
260 | *width: 57.21176577559556%;
261 | }
262 | .row-fluid .span6 {
263 | width: 48.717948717948715%;
264 | *width: 48.664757228587014%;
265 | }
266 | .row-fluid .span5 {
267 | width: 40.17094017094017%;
268 | *width: 40.11774868157847%;
269 | }
270 | .row-fluid .span4 {
271 | width: 31.623931623931625%;
272 | *width: 31.570740134569924%;
273 | }
274 | .row-fluid .span3 {
275 | width: 23.076923076923077%;
276 | *width: 23.023731587561375%;
277 | }
278 | .row-fluid .span2 {
279 | width: 14.52991452991453%;
280 | *width: 14.476723040552828%;
281 | }
282 | .row-fluid .span1 {
283 | width: 5.982905982905983%;
284 | *width: 5.929714493544281%;
285 | }
286 | .row-fluid .offset12 {
287 | margin-left: 105.12820512820512%;
288 | *margin-left: 105.02182214948171%;
289 | }
290 | .row-fluid .offset12:first-child {
291 | margin-left: 102.56410256410257%;
292 | *margin-left: 102.45771958537915%;
293 | }
294 | .row-fluid .offset11 {
295 | margin-left: 96.58119658119658%;
296 | *margin-left: 96.47481360247316%;
297 | }
298 | .row-fluid .offset11:first-child {
299 | margin-left: 94.01709401709402%;
300 | *margin-left: 93.91071103837061%;
301 | }
302 | .row-fluid .offset10 {
303 | margin-left: 88.03418803418803%;
304 | *margin-left: 87.92780505546462%;
305 | }
306 | .row-fluid .offset10:first-child {
307 | margin-left: 85.47008547008548%;
308 | *margin-left: 85.36370249136206%;
309 | }
310 | .row-fluid .offset9 {
311 | margin-left: 79.48717948717949%;
312 | *margin-left: 79.38079650845607%;
313 | }
314 | .row-fluid .offset9:first-child {
315 | margin-left: 76.92307692307693%;
316 | *margin-left: 76.81669394435352%;
317 | }
318 | .row-fluid .offset8 {
319 | margin-left: 70.94017094017094%;
320 | *margin-left: 70.83378796144753%;
321 | }
322 | .row-fluid .offset8:first-child {
323 | margin-left: 68.37606837606839%;
324 | *margin-left: 68.26968539734497%;
325 | }
326 | .row-fluid .offset7 {
327 | margin-left: 62.393162393162385%;
328 | *margin-left: 62.28677941443899%;
329 | }
330 | .row-fluid .offset7:first-child {
331 | margin-left: 59.82905982905982%;
332 | *margin-left: 59.72267685033642%;
333 | }
334 | .row-fluid .offset6 {
335 | margin-left: 53.84615384615384%;
336 | *margin-left: 53.739770867430444%;
337 | }
338 | .row-fluid .offset6:first-child {
339 | margin-left: 51.28205128205128%;
340 | *margin-left: 51.175668303327875%;
341 | }
342 | .row-fluid .offset5 {
343 | margin-left: 45.299145299145295%;
344 | *margin-left: 45.1927623204219%;
345 | }
346 | .row-fluid .offset5:first-child {
347 | margin-left: 42.73504273504273%;
348 | *margin-left: 42.62865975631933%;
349 | }
350 | .row-fluid .offset4 {
351 | margin-left: 36.75213675213675%;
352 | *margin-left: 36.645753773413354%;
353 | }
354 | .row-fluid .offset4:first-child {
355 | margin-left: 34.18803418803419%;
356 | *margin-left: 34.081651209310785%;
357 | }
358 | .row-fluid .offset3 {
359 | margin-left: 28.205128205128204%;
360 | *margin-left: 28.0987452264048%;
361 | }
362 | .row-fluid .offset3:first-child {
363 | margin-left: 25.641025641025642%;
364 | *margin-left: 25.53464266230224%;
365 | }
366 | .row-fluid .offset2 {
367 | margin-left: 19.65811965811966%;
368 | *margin-left: 19.551736679396257%;
369 | }
370 | .row-fluid .offset2:first-child {
371 | margin-left: 17.094017094017094%;
372 | *margin-left: 16.98763411529369%;
373 | }
374 | .row-fluid .offset1 {
375 | margin-left: 11.11111111111111%;
376 | *margin-left: 11.004728132387708%;
377 | }
378 | .row-fluid .offset1:first-child {
379 | margin-left: 8.547008547008547%;
380 | *margin-left: 8.440625568285142%;
381 | }
382 | input,
383 | textarea,
384 | .uneditable-input {
385 | margin-left: 0;
386 | }
387 | .controls-row [class*="span"] + [class*="span"] {
388 | margin-left: 30px;
389 | }
390 | input.span12,
391 | textarea.span12,
392 | .uneditable-input.span12 {
393 | width: 1156px;
394 | }
395 | input.span11,
396 | textarea.span11,
397 | .uneditable-input.span11 {
398 | width: 1056px;
399 | }
400 | input.span10,
401 | textarea.span10,
402 | .uneditable-input.span10 {
403 | width: 956px;
404 | }
405 | input.span9,
406 | textarea.span9,
407 | .uneditable-input.span9 {
408 | width: 856px;
409 | }
410 | input.span8,
411 | textarea.span8,
412 | .uneditable-input.span8 {
413 | width: 756px;
414 | }
415 | input.span7,
416 | textarea.span7,
417 | .uneditable-input.span7 {
418 | width: 656px;
419 | }
420 | input.span6,
421 | textarea.span6,
422 | .uneditable-input.span6 {
423 | width: 556px;
424 | }
425 | input.span5,
426 | textarea.span5,
427 | .uneditable-input.span5 {
428 | width: 456px;
429 | }
430 | input.span4,
431 | textarea.span4,
432 | .uneditable-input.span4 {
433 | width: 356px;
434 | }
435 | input.span3,
436 | textarea.span3,
437 | .uneditable-input.span3 {
438 | width: 256px;
439 | }
440 | input.span2,
441 | textarea.span2,
442 | .uneditable-input.span2 {
443 | width: 156px;
444 | }
445 | input.span1,
446 | textarea.span1,
447 | .uneditable-input.span1 {
448 | width: 56px;
449 | }
450 | .thumbnails {
451 | margin-left: -30px;
452 | }
453 | .thumbnails > li {
454 | margin-left: 30px;
455 | }
456 | .row-fluid .thumbnails {
457 | margin-left: 0;
458 | }
459 | }
460 |
461 | @media (min-width: 768px) and (max-width: 979px) {
462 | .row {
463 | margin-left: -20px;
464 | *zoom: 1;
465 | }
466 | .row:before,
467 | .row:after {
468 | display: table;
469 | line-height: 0;
470 | content: "";
471 | }
472 | .row:after {
473 | clear: both;
474 | }
475 | [class*="span"] {
476 | float: left;
477 | min-height: 1px;
478 | margin-left: 20px;
479 | }
480 | .container,
481 | .navbar-static-top .container,
482 | .navbar-fixed-top .container,
483 | .navbar-fixed-bottom .container {
484 | width: 724px;
485 | }
486 | .span12 {
487 | width: 724px;
488 | }
489 | .span11 {
490 | width: 662px;
491 | }
492 | .span10 {
493 | width: 600px;
494 | }
495 | .span9 {
496 | width: 538px;
497 | }
498 | .span8 {
499 | width: 476px;
500 | }
501 | .span7 {
502 | width: 414px;
503 | }
504 | .span6 {
505 | width: 352px;
506 | }
507 | .span5 {
508 | width: 290px;
509 | }
510 | .span4 {
511 | width: 228px;
512 | }
513 | .span3 {
514 | width: 166px;
515 | }
516 | .span2 {
517 | width: 104px;
518 | }
519 | .span1 {
520 | width: 42px;
521 | }
522 | .offset12 {
523 | margin-left: 764px;
524 | }
525 | .offset11 {
526 | margin-left: 702px;
527 | }
528 | .offset10 {
529 | margin-left: 640px;
530 | }
531 | .offset9 {
532 | margin-left: 578px;
533 | }
534 | .offset8 {
535 | margin-left: 516px;
536 | }
537 | .offset7 {
538 | margin-left: 454px;
539 | }
540 | .offset6 {
541 | margin-left: 392px;
542 | }
543 | .offset5 {
544 | margin-left: 330px;
545 | }
546 | .offset4 {
547 | margin-left: 268px;
548 | }
549 | .offset3 {
550 | margin-left: 206px;
551 | }
552 | .offset2 {
553 | margin-left: 144px;
554 | }
555 | .offset1 {
556 | margin-left: 82px;
557 | }
558 | .row-fluid {
559 | width: 100%;
560 | *zoom: 1;
561 | }
562 | .row-fluid:before,
563 | .row-fluid:after {
564 | display: table;
565 | line-height: 0;
566 | content: "";
567 | }
568 | .row-fluid:after {
569 | clear: both;
570 | }
571 | .row-fluid [class*="span"] {
572 | display: block;
573 | float: left;
574 | width: 100%;
575 | min-height: 30px;
576 | margin-left: 2.7624309392265194%;
577 | *margin-left: 2.709239449864817%;
578 | -webkit-box-sizing: border-box;
579 | -moz-box-sizing: border-box;
580 | box-sizing: border-box;
581 | }
582 | .row-fluid [class*="span"]:first-child {
583 | margin-left: 0;
584 | }
585 | .row-fluid .controls-row [class*="span"] + [class*="span"] {
586 | margin-left: 2.7624309392265194%;
587 | }
588 | .row-fluid .span12 {
589 | width: 100%;
590 | *width: 99.94680851063829%;
591 | }
592 | .row-fluid .span11 {
593 | width: 91.43646408839778%;
594 | *width: 91.38327259903608%;
595 | }
596 | .row-fluid .span10 {
597 | width: 82.87292817679558%;
598 | *width: 82.81973668743387%;
599 | }
600 | .row-fluid .span9 {
601 | width: 74.30939226519337%;
602 | *width: 74.25620077583166%;
603 | }
604 | .row-fluid .span8 {
605 | width: 65.74585635359117%;
606 | *width: 65.69266486422946%;
607 | }
608 | .row-fluid .span7 {
609 | width: 57.18232044198895%;
610 | *width: 57.12912895262725%;
611 | }
612 | .row-fluid .span6 {
613 | width: 48.61878453038674%;
614 | *width: 48.56559304102504%;
615 | }
616 | .row-fluid .span5 {
617 | width: 40.05524861878453%;
618 | *width: 40.00205712942283%;
619 | }
620 | .row-fluid .span4 {
621 | width: 31.491712707182323%;
622 | *width: 31.43852121782062%;
623 | }
624 | .row-fluid .span3 {
625 | width: 22.92817679558011%;
626 | *width: 22.87498530621841%;
627 | }
628 | .row-fluid .span2 {
629 | width: 14.3646408839779%;
630 | *width: 14.311449394616199%;
631 | }
632 | .row-fluid .span1 {
633 | width: 5.801104972375691%;
634 | *width: 5.747913483013988%;
635 | }
636 | .row-fluid .offset12 {
637 | margin-left: 105.52486187845304%;
638 | *margin-left: 105.41847889972962%;
639 | }
640 | .row-fluid .offset12:first-child {
641 | margin-left: 102.76243093922652%;
642 | *margin-left: 102.6560479605031%;
643 | }
644 | .row-fluid .offset11 {
645 | margin-left: 96.96132596685082%;
646 | *margin-left: 96.8549429881274%;
647 | }
648 | .row-fluid .offset11:first-child {
649 | margin-left: 94.1988950276243%;
650 | *margin-left: 94.09251204890089%;
651 | }
652 | .row-fluid .offset10 {
653 | margin-left: 88.39779005524862%;
654 | *margin-left: 88.2914070765252%;
655 | }
656 | .row-fluid .offset10:first-child {
657 | margin-left: 85.6353591160221%;
658 | *margin-left: 85.52897613729868%;
659 | }
660 | .row-fluid .offset9 {
661 | margin-left: 79.8342541436464%;
662 | *margin-left: 79.72787116492299%;
663 | }
664 | .row-fluid .offset9:first-child {
665 | margin-left: 77.07182320441989%;
666 | *margin-left: 76.96544022569647%;
667 | }
668 | .row-fluid .offset8 {
669 | margin-left: 71.2707182320442%;
670 | *margin-left: 71.16433525332079%;
671 | }
672 | .row-fluid .offset8:first-child {
673 | margin-left: 68.50828729281768%;
674 | *margin-left: 68.40190431409427%;
675 | }
676 | .row-fluid .offset7 {
677 | margin-left: 62.70718232044199%;
678 | *margin-left: 62.600799341718584%;
679 | }
680 | .row-fluid .offset7:first-child {
681 | margin-left: 59.94475138121547%;
682 | *margin-left: 59.838368402492065%;
683 | }
684 | .row-fluid .offset6 {
685 | margin-left: 54.14364640883978%;
686 | *margin-left: 54.037263430116376%;
687 | }
688 | .row-fluid .offset6:first-child {
689 | margin-left: 51.38121546961326%;
690 | *margin-left: 51.27483249088986%;
691 | }
692 | .row-fluid .offset5 {
693 | margin-left: 45.58011049723757%;
694 | *margin-left: 45.47372751851417%;
695 | }
696 | .row-fluid .offset5:first-child {
697 | margin-left: 42.81767955801105%;
698 | *margin-left: 42.71129657928765%;
699 | }
700 | .row-fluid .offset4 {
701 | margin-left: 37.01657458563536%;
702 | *margin-left: 36.91019160691196%;
703 | }
704 | .row-fluid .offset4:first-child {
705 | margin-left: 34.25414364640884%;
706 | *margin-left: 34.14776066768544%;
707 | }
708 | .row-fluid .offset3 {
709 | margin-left: 28.45303867403315%;
710 | *margin-left: 28.346655695309746%;
711 | }
712 | .row-fluid .offset3:first-child {
713 | margin-left: 25.69060773480663%;
714 | *margin-left: 25.584224756083227%;
715 | }
716 | .row-fluid .offset2 {
717 | margin-left: 19.88950276243094%;
718 | *margin-left: 19.783119783707537%;
719 | }
720 | .row-fluid .offset2:first-child {
721 | margin-left: 17.12707182320442%;
722 | *margin-left: 17.02068884448102%;
723 | }
724 | .row-fluid .offset1 {
725 | margin-left: 11.32596685082873%;
726 | *margin-left: 11.219583872105325%;
727 | }
728 | .row-fluid .offset1:first-child {
729 | margin-left: 8.56353591160221%;
730 | *margin-left: 8.457152932878806%;
731 | }
732 | input,
733 | textarea,
734 | .uneditable-input {
735 | margin-left: 0;
736 | }
737 | .controls-row [class*="span"] + [class*="span"] {
738 | margin-left: 20px;
739 | }
740 | input.span12,
741 | textarea.span12,
742 | .uneditable-input.span12 {
743 | width: 710px;
744 | }
745 | input.span11,
746 | textarea.span11,
747 | .uneditable-input.span11 {
748 | width: 648px;
749 | }
750 | input.span10,
751 | textarea.span10,
752 | .uneditable-input.span10 {
753 | width: 586px;
754 | }
755 | input.span9,
756 | textarea.span9,
757 | .uneditable-input.span9 {
758 | width: 524px;
759 | }
760 | input.span8,
761 | textarea.span8,
762 | .uneditable-input.span8 {
763 | width: 462px;
764 | }
765 | input.span7,
766 | textarea.span7,
767 | .uneditable-input.span7 {
768 | width: 400px;
769 | }
770 | input.span6,
771 | textarea.span6,
772 | .uneditable-input.span6 {
773 | width: 338px;
774 | }
775 | input.span5,
776 | textarea.span5,
777 | .uneditable-input.span5 {
778 | width: 276px;
779 | }
780 | input.span4,
781 | textarea.span4,
782 | .uneditable-input.span4 {
783 | width: 214px;
784 | }
785 | input.span3,
786 | textarea.span3,
787 | .uneditable-input.span3 {
788 | width: 152px;
789 | }
790 | input.span2,
791 | textarea.span2,
792 | .uneditable-input.span2 {
793 | width: 90px;
794 | }
795 | input.span1,
796 | textarea.span1,
797 | .uneditable-input.span1 {
798 | width: 28px;
799 | }
800 | }
801 |
802 | @media (max-width: 767px) {
803 | body {
804 | padding-right: 20px;
805 | padding-left: 20px;
806 | }
807 | .navbar-fixed-top,
808 | .navbar-fixed-bottom,
809 | .navbar-static-top {
810 | margin-right: -20px;
811 | margin-left: -20px;
812 | }
813 | .container-fluid {
814 | padding: 0;
815 | }
816 | .dl-horizontal dt {
817 | float: none;
818 | width: auto;
819 | clear: none;
820 | text-align: left;
821 | }
822 | .dl-horizontal dd {
823 | margin-left: 0;
824 | }
825 | .container {
826 | width: auto;
827 | }
828 | .row-fluid {
829 | width: 100%;
830 | }
831 | .row,
832 | .thumbnails {
833 | margin-left: 0;
834 | }
835 | .thumbnails > li {
836 | float: none;
837 | margin-left: 0;
838 | }
839 | [class*="span"],
840 | .uneditable-input[class*="span"],
841 | .row-fluid [class*="span"] {
842 | display: block;
843 | float: none;
844 | width: 100%;
845 | margin-left: 0;
846 | -webkit-box-sizing: border-box;
847 | -moz-box-sizing: border-box;
848 | box-sizing: border-box;
849 | }
850 | .span12,
851 | .row-fluid .span12 {
852 | width: 100%;
853 | -webkit-box-sizing: border-box;
854 | -moz-box-sizing: border-box;
855 | box-sizing: border-box;
856 | }
857 | .row-fluid [class*="offset"]:first-child {
858 | margin-left: 0;
859 | }
860 | .input-large,
861 | .input-xlarge,
862 | .input-xxlarge,
863 | input[class*="span"],
864 | select[class*="span"],
865 | textarea[class*="span"],
866 | .uneditable-input {
867 | display: block;
868 | width: 100%;
869 | min-height: 30px;
870 | -webkit-box-sizing: border-box;
871 | -moz-box-sizing: border-box;
872 | box-sizing: border-box;
873 | }
874 | .input-prepend input,
875 | .input-append input,
876 | .input-prepend input[class*="span"],
877 | .input-append input[class*="span"] {
878 | display: inline-block;
879 | width: auto;
880 | }
881 | .controls-row [class*="span"] + [class*="span"] {
882 | margin-left: 0;
883 | }
884 | .modal {
885 | position: fixed;
886 | top: 20px;
887 | right: 20px;
888 | left: 20px;
889 | width: auto;
890 | margin: 0;
891 | }
892 | .modal.fade {
893 | top: -100px;
894 | }
895 | .modal.fade.in {
896 | top: 20px;
897 | }
898 | }
899 |
900 | @media (max-width: 480px) {
901 | .nav-collapse {
902 | -webkit-transform: translate3d(0, 0, 0);
903 | }
904 | .page-header h1 small {
905 | display: block;
906 | line-height: 20px;
907 | }
908 | input[type="checkbox"],
909 | input[type="radio"] {
910 | border: 1px solid #ccc;
911 | }
912 | .form-horizontal .control-label {
913 | float: none;
914 | width: auto;
915 | padding-top: 0;
916 | text-align: left;
917 | }
918 | .form-horizontal .controls {
919 | margin-left: 0;
920 | }
921 | .form-horizontal .control-list {
922 | padding-top: 0;
923 | }
924 | .form-horizontal .form-actions {
925 | padding-right: 10px;
926 | padding-left: 10px;
927 | }
928 | .media .pull-left,
929 | .media .pull-right {
930 | display: block;
931 | float: none;
932 | margin-bottom: 10px;
933 | }
934 | .media-object {
935 | margin-right: 0;
936 | margin-left: 0;
937 | }
938 | .modal {
939 | top: 10px;
940 | right: 10px;
941 | left: 10px;
942 | }
943 | .modal-header .close {
944 | padding: 10px;
945 | margin: -10px;
946 | }
947 | .carousel-caption {
948 | position: static;
949 | }
950 | }
951 |
952 | @media (max-width: 979px) {
953 | body {
954 | padding-top: 0;
955 | }
956 | .navbar-fixed-top,
957 | .navbar-fixed-bottom {
958 | position: static;
959 | }
960 | .navbar-fixed-top {
961 | margin-bottom: 20px;
962 | }
963 | .navbar-fixed-bottom {
964 | margin-top: 20px;
965 | }
966 | .navbar-fixed-top .navbar-inner,
967 | .navbar-fixed-bottom .navbar-inner {
968 | padding: 5px;
969 | }
970 | .navbar .container {
971 | width: auto;
972 | padding: 0;
973 | }
974 | .navbar .brand {
975 | padding-right: 10px;
976 | padding-left: 10px;
977 | margin: 0 0 0 -5px;
978 | }
979 | .nav-collapse {
980 | clear: both;
981 | }
982 | .nav-collapse .nav {
983 | float: none;
984 | margin: 0 0 10px;
985 | }
986 | .nav-collapse .nav > li {
987 | float: none;
988 | }
989 | .nav-collapse .nav > li > a {
990 | margin-bottom: 2px;
991 | }
992 | .nav-collapse .nav > .divider-vertical {
993 | display: none;
994 | }
995 | .nav-collapse .nav .nav-header {
996 | color: #777777;
997 | text-shadow: none;
998 | }
999 | .nav-collapse .nav > li > a,
1000 | .nav-collapse .dropdown-menu a {
1001 | padding: 9px 15px;
1002 | font-weight: bold;
1003 | color: #777777;
1004 | -webkit-border-radius: 3px;
1005 | -moz-border-radius: 3px;
1006 | border-radius: 3px;
1007 | }
1008 | .nav-collapse .btn {
1009 | padding: 4px 10px 4px;
1010 | font-weight: normal;
1011 | -webkit-border-radius: 4px;
1012 | -moz-border-radius: 4px;
1013 | border-radius: 4px;
1014 | }
1015 | .nav-collapse .dropdown-menu li + li a {
1016 | margin-bottom: 2px;
1017 | }
1018 | .nav-collapse .nav > li > a:hover,
1019 | .nav-collapse .nav > li > a:focus,
1020 | .nav-collapse .dropdown-menu a:hover,
1021 | .nav-collapse .dropdown-menu a:focus {
1022 | background-color: #f2f2f2;
1023 | }
1024 | .navbar-inverse .nav-collapse .nav > li > a,
1025 | .navbar-inverse .nav-collapse .dropdown-menu a {
1026 | color: #999999;
1027 | }
1028 | .navbar-inverse .nav-collapse .nav > li > a:hover,
1029 | .navbar-inverse .nav-collapse .nav > li > a:focus,
1030 | .navbar-inverse .nav-collapse .dropdown-menu a:hover,
1031 | .navbar-inverse .nav-collapse .dropdown-menu a:focus {
1032 | background-color: #111111;
1033 | }
1034 | .nav-collapse.in .btn-group {
1035 | padding: 0;
1036 | margin-top: 5px;
1037 | }
1038 | .nav-collapse .dropdown-menu {
1039 | position: static;
1040 | top: auto;
1041 | left: auto;
1042 | display: none;
1043 | float: none;
1044 | max-width: none;
1045 | padding: 0;
1046 | margin: 0 15px;
1047 | background-color: transparent;
1048 | border: none;
1049 | -webkit-border-radius: 0;
1050 | -moz-border-radius: 0;
1051 | border-radius: 0;
1052 | -webkit-box-shadow: none;
1053 | -moz-box-shadow: none;
1054 | box-shadow: none;
1055 | }
1056 | .nav-collapse .open > .dropdown-menu {
1057 | display: block;
1058 | }
1059 | .nav-collapse .dropdown-menu:before,
1060 | .nav-collapse .dropdown-menu:after {
1061 | display: none;
1062 | }
1063 | .nav-collapse .dropdown-menu .divider {
1064 | display: none;
1065 | }
1066 | .nav-collapse .nav > li > .dropdown-menu:before,
1067 | .nav-collapse .nav > li > .dropdown-menu:after {
1068 | display: none;
1069 | }
1070 | .nav-collapse .navbar-form,
1071 | .nav-collapse .navbar-search {
1072 | float: none;
1073 | padding: 10px 15px;
1074 | margin: 10px 0;
1075 | border-top: 1px solid #f2f2f2;
1076 | border-bottom: 1px solid #f2f2f2;
1077 | -webkit-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.1), 0 1px 0 rgba(255, 255, 255, 0.1);
1078 | -moz-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.1), 0 1px 0 rgba(255, 255, 255, 0.1);
1079 | box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.1), 0 1px 0 rgba(255, 255, 255, 0.1);
1080 | }
1081 | .navbar-inverse .nav-collapse .navbar-form,
1082 | .navbar-inverse .nav-collapse .navbar-search {
1083 | border-top-color: #111111;
1084 | border-bottom-color: #111111;
1085 | }
1086 | .navbar .nav-collapse .nav.pull-right {
1087 | float: none;
1088 | margin-left: 0;
1089 | }
1090 | .nav-collapse,
1091 | .nav-collapse.collapse {
1092 | height: 0;
1093 | overflow: hidden;
1094 | }
1095 | .navbar .btn-navbar {
1096 | display: block;
1097 | }
1098 | .navbar-static .navbar-inner {
1099 | padding-right: 10px;
1100 | padding-left: 10px;
1101 | }
1102 | }
1103 |
1104 | @media (min-width: 980px) {
1105 | .nav-collapse.collapse {
1106 | height: auto !important;
1107 | overflow: visible !important;
1108 | }
1109 | }
1110 |
--------------------------------------------------------------------------------
/vendor/plugins/.gitkeep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/vendor/plugins/.gitkeep
--------------------------------------------------------------------------------