From fa5a53f5ed2914052e07ef8fd46879576c978563 Mon Sep 17 00:00:00 2001 From: Jakub Jirutka Date: Sun, 29 Jul 2012 16:15:52 +0200 Subject: [PATCH 01/46] Change identification of users with extern auth provider (LDAP) --- .../omniauth_callbacks_controller.rb | 3 +-- app/models/user.rb | 19 +++++++++++++------ ...31232_add_extern_auth_provider_to_users.rb | 8 ++++++++ db/schema.rb | 5 ++++- 4 files changed, 26 insertions(+), 9 deletions(-) create mode 100644 db/migrate/20120729131232_add_extern_auth_provider_to_users.rb diff --git a/app/controllers/omniauth_callbacks_controller.rb b/app/controllers/omniauth_callbacks_controller.rb index fb759c37..5bad6093 100644 --- a/app/controllers/omniauth_callbacks_controller.rb +++ b/app/controllers/omniauth_callbacks_controller.rb @@ -15,8 +15,7 @@ class OmniauthCallbacksController < Devise::OmniauthCallbacksController def ldap # We only find ourselves here if the authentication to LDAP was successful. - info = request.env["omniauth.auth"]["info"] - @user = User.find_for_ldap_auth(info) + @user = User.find_for_ldap_auth(request.env["omniauth.auth"], current_user) if @user.persisted? @user.remember_me = true end diff --git a/app/models/user.rb b/app/models/user.rb index ff27660a..f5ea70d5 100644 --- a/app/models/user.rb +++ b/app/models/user.rb @@ -7,7 +7,7 @@ class User < ActiveRecord::Base attr_accessible :email, :password, :password_confirmation, :remember_me, :bio, :name, :projects_limit, :skype, :linkedin, :twitter, :dark_scheme, - :theme_id, :force_random_password + :theme_id, :force_random_password, :extern_uid, :provider attr_accessor :force_random_password @@ -54,6 +54,8 @@ class User < ActiveRecord::Base validates :bio, :length => { :within => 0..255 } + validates :extern_uid, :allow_blank => true, :uniqueness => {:scope => :provider} + before_save :ensure_authentication_token alias_attribute :private_token, :authentication_token @@ -84,16 +86,21 @@ class User < ActiveRecord::Base where('id NOT IN (SELECT DISTINCT(user_id) FROM users_projects)') end - def self.find_for_ldap_auth(omniauth_info) - name = omniauth_info.name.force_encoding("utf-8") - email = omniauth_info.email.downcase unless omniauth_info.email.nil? - raise OmniAuth::Error, "LDAP accounts must provide an email address" if email.nil? + def self.find_for_ldap_auth(auth, signed_in_resource=nil) + uid = auth.info.uid + provider = auth.provider + name = auth.info.name.force_encoding("utf-8") + email = auth.info.email.downcase unless auth.info.email.nil? + raise OmniAuth::Error, "LDAP accounts must provide an uid and email address" if uid.nil? and email.nil? - if @user = User.find_by_email(email) + if @user = User.find_by_extern_uid_and_provider(uid, provider) @user else + logger.info "Creating user from LDAP login; uid = #{uid}, name = #{name}, email = #{email}" password = Devise.friendly_token[0, 8].downcase @user = User.create( + :extern_uid => uid, + :provider => provider, :name => name, :email => email, :password => password, diff --git a/db/migrate/20120729131232_add_extern_auth_provider_to_users.rb b/db/migrate/20120729131232_add_extern_auth_provider_to_users.rb new file mode 100644 index 00000000..d5e66ba4 --- /dev/null +++ b/db/migrate/20120729131232_add_extern_auth_provider_to_users.rb @@ -0,0 +1,8 @@ +class AddExternAuthProviderToUsers < ActiveRecord::Migration + def change + add_column :users, :extern_uid, :string + add_column :users, :provider, :string + + add_index :users, [:extern_uid, :provider], :unique => true + end +end diff --git a/db/schema.rb b/db/schema.rb index c4c54f56..46461e44 100644 --- a/db/schema.rb +++ b/db/schema.rb @@ -11,7 +11,7 @@ # # It's strongly recommended to check this file into your version control system. -ActiveRecord::Schema.define(:version => 20120712080407) do +ActiveRecord::Schema.define(:version => 20120729131232) do create_table "events", :force => true do |t| t.string "target_type" @@ -171,9 +171,12 @@ ActiveRecord::Schema.define(:version => 20120712080407) do t.boolean "blocked", :default => false, :null => false t.integer "failed_attempts", :default => 0 t.datetime "locked_at" + t.string "extern_uid" + t.string "provider" end add_index "users", ["email"], :name => "index_users_on_email", :unique => true + add_index "users", ["extern_uid", "provider"], :name => "index_users_on_extern_uid_and_provider", :unique => true add_index "users", ["reset_password_token"], :name => "index_users_on_reset_password_token", :unique => true create_table "users_projects", :force => true do |t| From ad265b9610b474132752da43cf845d82ddf5fd8c Mon Sep 17 00:00:00 2001 From: Jakub Jirutka Date: Sun, 5 Aug 2012 11:27:17 +0200 Subject: [PATCH 02/46] Adding workaround for backward compatibility with legacy LDAP users --- app/models/user.rb | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/app/models/user.rb b/app/models/user.rb index f5ea70d5..1b53bda2 100644 --- a/app/models/user.rb +++ b/app/models/user.rb @@ -95,8 +95,13 @@ class User < ActiveRecord::Base if @user = User.find_by_extern_uid_and_provider(uid, provider) @user + # workaround for backward compatibility + elsif @user = User.find_by_email(email) + logger.info "Updating legacy LDAP user #{email} with extern_uid => #{uid}" + @user.update_attributes(:extern_uid => uid, :provider => provider) + @user else - logger.info "Creating user from LDAP login; uid = #{uid}, name = #{name}, email = #{email}" + logger.info "Creating user from LDAP login {uid => #{uid}, name => #{name}, email => #{email}}" password = Devise.friendly_token[0, 8].downcase @user = User.create( :extern_uid => uid, From 0b7e67ad12dff04da6d7825a6125099ef0d21174 Mon Sep 17 00:00:00 2001 From: Nihad Abbasov Date: Tue, 14 Aug 2012 01:24:27 -0700 Subject: [PATCH 03/46] fix note preview link, and hide preview and errors after posting a comment --- app/assets/javascripts/application.js | 2 +- app/views/notes/_create_common.js.haml | 4 +++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/app/assets/javascripts/application.js b/app/assets/javascripts/application.js index 25732ae5..28e9d362 100644 --- a/app/assets/javascripts/application.js +++ b/app/assets/javascripts/application.js @@ -74,7 +74,7 @@ $(document).ready(function(){ * Note markdown preview * */ - $('#preview-link').on('click', function(e) { + $(document).on('click', '#preview-link', function(e) { $('#preview-note').text('Loading...'); var previewLinkText = ($(this).text() == 'Preview' ? 'Edit' : 'Preview'); diff --git a/app/views/notes/_create_common.js.haml b/app/views/notes/_create_common.js.haml index 847ff383..e9538902 100644 --- a/app/views/notes/_create_common.js.haml +++ b/app/views/notes/_create_common.js.haml @@ -1,7 +1,9 @@ - if note.valid? :plain - $("#new_note .errors").remove(); + $("#new_note .error").remove(); $('#new_note textarea').val(""); + $('#preview-link').text('Preview'); + $('#preview-note').hide(); $('#note_note').show(); NoteList.prepend(#{note.id}, "#{escape_javascript(render partial: "notes/show", locals: {note: note})}"); - else :plain From 65bcc41f3e0a8b678e201e7f3d6a63c5b463fbe3 Mon Sep 17 00:00:00 2001 From: Robert Speicher Date: Wed, 15 Aug 2012 21:06:08 -0400 Subject: [PATCH 04/46] Allow disabling Gravatars in gitlab.yml settings Closes #1237 --- app/helpers/application_helper.rb | 11 +++++++---- config/gitlab.yml.example | 13 ++++++------- config/initializers/1_settings.rb | 4 ++++ spec/helpers/application_helper_spec.rb | 26 +++++++++++++++++++++++++ 4 files changed, 43 insertions(+), 11 deletions(-) create mode 100644 spec/helpers/application_helper_spec.rb diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb index 7a9f0e9d..2d7e4fda 100644 --- a/app/helpers/application_helper.rb +++ b/app/helpers/application_helper.rb @@ -2,10 +2,13 @@ require 'digest/md5' module ApplicationHelper def gravatar_icon(user_email = '', size = 40) - return unless user_email - gravatar_host = request.ssl? ? "https://secure.gravatar.com" : "http://www.gravatar.com" - user_email.strip! - "#{gravatar_host}/avatar/#{Digest::MD5.hexdigest(user_email.downcase)}?s=#{size}&d=identicon" + if Gitlab.config.disable_gravatar? || user_email.blank? + 'no_avatar.png' + else + gravatar_prefix = request.ssl? ? "https://secure" : "http://www" + user_email.strip! + "#{gravatar_prefix}.gravatar.com/avatar/#{Digest::MD5.hexdigest(user_email.downcase)}?s=#{size}&d=identicon" + end end def request_protocol diff --git a/config/gitlab.yml.example b/config/gitlab.yml.example index 1818f2c0..be36ee6d 100644 --- a/config/gitlab.yml.example +++ b/config/gitlab.yml.example @@ -1,4 +1,4 @@ -# # # # # # # # # # # # # # # # # # +# # # # # # # # # # # # # # # # # # # Gitlab application config file # # # # # # # # # # # # # # # # # # # @@ -19,14 +19,14 @@ email: # Application specific settings # Like default project limit for user etc -app: - default_projects_limit: 10 +app: + default_projects_limit: 10 # backup_path: "/vol/backups" # default: Rails.root + backups/ # backup_keep_time: 604800 # default: 0 (forever) (in seconds) + # disable_gravatar: true # default: false - Disable user avatars from Gravatar.com - -# -# 2. Advanced settings: +# +# 2. Advanced settings: # ========================== # Git Hosting configuration @@ -39,7 +39,6 @@ git_host: receive_pack: true # port: 22 - # Git settings # Use default values unless you understand it git: diff --git a/config/initializers/1_settings.rb b/config/initializers/1_settings.rb index 5c5987a8..8165d6c2 100644 --- a/config/initializers/1_settings.rb +++ b/config/initializers/1_settings.rb @@ -111,5 +111,9 @@ class Settings < Settingslogic def backup_keep_time app['backup_keep_time'] || 0 end + + def disable_gravatar? + app['disable_gravatar'] || false + end end end diff --git a/spec/helpers/application_helper_spec.rb b/spec/helpers/application_helper_spec.rb new file mode 100644 index 00000000..9a2df314 --- /dev/null +++ b/spec/helpers/application_helper_spec.rb @@ -0,0 +1,26 @@ +require 'spec_helper' + +describe ApplicationHelper do + describe "gravatar_icon" do + let(:user_email) { 'user@email.com' } + + it "should return a generic avatar path when Gravatar is disabled" do + Gitlab.config.stub(:disable_gravatar?).and_return(true) + gravatar_icon(user_email).should == 'no_avatar.png' + end + + it "should return a generic avatar path when email is blank" do + gravatar_icon('').should == 'no_avatar.png' + end + + it "should use SSL when appropriate" do + stub!(:request).and_return(double(:ssl? => true)) + gravatar_icon(user_email).should match('https://secure.gravatar.com') + end + + it "should accept a custom size" do + stub!(:request).and_return(double(:ssl? => false)) + gravatar_icon(user_email, 64).should match(/\?s=64/) + end + end +end From 813814f02ed7bf226782861820570117fa196026 Mon Sep 17 00:00:00 2001 From: Dmitriy Zaporozhets Date: Fri, 17 Aug 2012 09:09:11 +0300 Subject: [PATCH 05/46] Application cleanup --- Gemfile | 6 +- Gemfile.lock | 8 +- app/assets/javascripts/application.js | 2 - app/assets/stylesheets/application.css | 3 +- app/assets/stylesheets/jquery_ui.scss | 33 - app/assets/stylesheets/main.scss | 30 +- app/models/issue.rb | 2 - lib/tasks/dev/tests.rake | 2 +- vendor/assets/images/bg_fallback.png | Bin 0 -> 3721 bytes vendor/assets/images/icon_sprite.png | Bin 0 -> 3217 bytes vendor/assets/images/progress_bar.gif | Bin 0 -> 502 bytes vendor/assets/images/slider_handles.png | Bin 0 -> 4453 bytes .../assets/images/ui-icons_222222_256x240.png | Bin 0 -> 4369 bytes .../assets/images/ui-icons_454545_256x240.png | Bin 0 -> 4369 bytes vendor/assets/javascripts/jquery.tagify.js | 143 --- .../javascripts/jquery.ui.selectmenu.js | 844 ------------------ .../stylesheets/jquery-ui/jquery.tagify.css | 34 - .../jquery-ui/jquery.ui.selectmenu.css | 33 - .../assets/stylesheets/jquery.ui.aristo.css | 738 +++++++++++++++ 19 files changed, 755 insertions(+), 1123 deletions(-) delete mode 100644 app/assets/stylesheets/jquery_ui.scss create mode 100644 vendor/assets/images/bg_fallback.png create mode 100644 vendor/assets/images/icon_sprite.png create mode 100644 vendor/assets/images/progress_bar.gif create mode 100644 vendor/assets/images/slider_handles.png create mode 100644 vendor/assets/images/ui-icons_222222_256x240.png create mode 100644 vendor/assets/images/ui-icons_454545_256x240.png delete mode 100644 vendor/assets/javascripts/jquery.tagify.js delete mode 100644 vendor/assets/javascripts/jquery.ui.selectmenu.js delete mode 100644 vendor/assets/stylesheets/jquery-ui/jquery.tagify.css delete mode 100644 vendor/assets/stylesheets/jquery-ui/jquery.ui.selectmenu.css create mode 100644 vendor/assets/stylesheets/jquery.ui.aristo.css diff --git a/Gemfile b/Gemfile index d2a5728f..26e4e195 100644 --- a/Gemfile +++ b/Gemfile @@ -76,10 +76,6 @@ gem 'settingslogic' gem "foreman" gem "git" -# Unused -gem 'tabs_on_rails' -gem "acts_as_list" - group :assets do gem "sass-rails", "3.2.5" gem "coffee-rails", "3.2.2" @@ -91,7 +87,7 @@ group :assets do gem "jquery-ui-rails", "0.5.0" gem "modernizr", "2.5.3" gem "raphael-rails", "1.5.2" - gem 'bootstrap-sass', "2.0.3.1" + gem 'bootstrap-sass', "2.0.4" end group :development do diff --git a/Gemfile.lock b/Gemfile.lock index 7356c35e..57ad1935 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -98,7 +98,6 @@ GEM multi_json (~> 1.0) acts-as-taggable-on (2.3.1) rails (~> 3.0) - acts_as_list (0.1.6) addressable (2.2.8) ansi (1.4.2) arel (3.0.2) @@ -109,7 +108,7 @@ GEM awesome_print (1.0.2) bcrypt-ruby (3.0.1) blankslate (2.1.2.4) - bootstrap-sass (2.0.3.1) + bootstrap-sass (2.0.4.0) builder (3.0.0) capybara (1.1.2) mime-types (>= 1.16) @@ -338,7 +337,6 @@ GEM tilt (~> 1.1, != 1.3.0) sqlite3 (1.3.6) stamp (0.1.6) - tabs_on_rails (2.1.1) therubyracer (0.10.1) libv8 (~> 3.3.10) thin (1.3.1) @@ -375,12 +373,11 @@ PLATFORMS DEPENDENCIES acts-as-taggable-on (= 2.3.1) - acts_as_list annotate! autotest autotest-rails awesome_print - bootstrap-sass (= 2.0.3.1) + bootstrap-sass (= 2.0.4) capybara capybara-webkit carrierwave @@ -431,7 +428,6 @@ DEPENDENCIES six sqlite3 stamp - tabs_on_rails therubyracer thin turn diff --git a/app/assets/javascripts/application.js b/app/assets/javascripts/application.js index 25732ae5..d9cbd5d6 100644 --- a/app/assets/javascripts/application.js +++ b/app/assets/javascripts/application.js @@ -7,8 +7,6 @@ //= require jquery //= require jquery.ui.all //= require jquery_ujs -//= require jquery.ui.selectmenu -//= require jquery.tagify //= require jquery.cookie //= require jquery.endless-scroll //= require jquery.highlight diff --git a/app/assets/stylesheets/application.css b/app/assets/stylesheets/application.css index 6ce23320..92d542a9 100644 --- a/app/assets/stylesheets/application.css +++ b/app/assets/stylesheets/application.css @@ -3,8 +3,7 @@ * and any sub-directories. You're free to add application-wide styles to this file and they'll appear at * the top of the compiled file, but it's generally better to create a new file per style scope. *= require jquery.ui.all - *= require jquery-ui/jquery.ui.selectmenu - *= require jquery-ui/jquery.tagify + *= require jquery.ui.aristo *= require chosen *= require_self *= require main diff --git a/app/assets/stylesheets/jquery_ui.scss b/app/assets/stylesheets/jquery_ui.scss deleted file mode 100644 index 1063f1d0..00000000 --- a/app/assets/stylesheets/jquery_ui.scss +++ /dev/null @@ -1,33 +0,0 @@ -/** - * JQUERY UI datepicker - * - */ -.ui-datepicker { - border-color:#eee; - padding:20px; - - .ui-state-default { - background:#f1f1f1; - padding:5px; - } - .ui-state-active { - background:#fff; - } -} - -/** - * JQUERY UI progressbar - * - */ -.ui-progressbar { - border:1px solid #ddd; - height:6px; - margin:0; - padding:0; - - .ui-progressbar-value { - background-color: #62C462;//$blue_link; - margin:0; - } -} - diff --git a/app/assets/stylesheets/main.scss b/app/assets/stylesheets/main.scss index 5613f1e8..71d09883 100644 --- a/app/assets/stylesheets/main.scss +++ b/app/assets/stylesheets/main.scss @@ -3,8 +3,8 @@ /** GITLAB colors **/ $text_color:#222; -$lite_text_color: #666; -$link_color:#2A79A3; +$lite_text_color: #666; +$link_color:#2A79A3; $active_link_color:#2FA0BB; $active_bg_color:#79C3E0; $active_bd_color: #2FA0BB; @@ -31,7 +31,7 @@ $hover: #FDF5D9; box-shadow: 0 0 3px #ddd; } -@mixin solid_shade { +@mixin solid_shade { -moz-box-shadow: 0 0 0 3px #eee; -webkit-box-shadow: 0 0 0 3px #eee; box-shadow: 0 0 0 3px #eee; @@ -73,21 +73,21 @@ $hover: #FDF5D9; /** - * Header of application. + * Header of application. * Contain application logo, search panel, profile icon */ @import "sections/header.scss"; /** - * Navigation menu of application. + * Navigation menu of application. * Panel with links to pages depends on project, profile or admin area */ @import "sections/nav.scss"; /** - * This file represent some UI that can be changed - * during web app restyle or theme select. - * + * This file represent some UI that can be changed + * during web app restyle or theme select. + * * Next items should be placed there * - link, button colors * - header restyles @@ -118,7 +118,7 @@ $hover: #FDF5D9; * Most of application styles placed here. * This file represent common UI that should not be changed between themes * or project restyling like form width or user avatar class or commit title - * + * * TODO: clean it */ @import "common.scss"; @@ -140,17 +140,17 @@ $hover: #FDF5D9; @import "ref_select.scss"; /** - * Code (files list) styles. Browsing project files there + * Code (files list) styles. Browsing project files there */ @import "sections/tree.scss"; /** - * This file represent notes(comments) styles + * This file represent notes(comments) styles */ @import "sections/notes.scss"; /** - * Devise styles + * Devise styles */ @import "sections/login.scss"; @@ -165,9 +165,3 @@ $hover: #FDF5D9; * */ @import "highlight/dark.scss"; - -/** - * JQUERY UI ext - * - */ -@import "jquery_ui.scss"; diff --git a/app/models/issue.rb b/app/models/issue.rb index 454b1358..6409eeba 100644 --- a/app/models/issue.rb +++ b/app/models/issue.rb @@ -9,8 +9,6 @@ class Issue < ActiveRecord::Base validates :description, length: { within: 0..2000 } - acts_as_list - def self.open_for(user) opened.assigned(user) end diff --git a/lib/tasks/dev/tests.rake b/lib/tasks/dev/tests.rake index f91320eb..d5c0e75e 100644 --- a/lib/tasks/dev/tests.rake +++ b/lib/tasks/dev/tests.rake @@ -3,7 +3,7 @@ namespace :dev do task :tests do ["cucumber", "rspec spec"].each do |cmd| puts "Starting to run #{cmd}..." - system("bundle exec #{cmd}") + system("export DISPLAY=:99.0 && bundle exec #{cmd}") raise "#{cmd} failed!" unless $?.exitstatus == 0 end end diff --git a/vendor/assets/images/bg_fallback.png b/vendor/assets/images/bg_fallback.png new file mode 100644 index 0000000000000000000000000000000000000000..4b2754b8040e4bb430bd910225bb9760d25e8794 GIT binary patch literal 3721 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 literal 0 HcmV?d00001 diff --git a/vendor/assets/images/icon_sprite.png b/vendor/assets/images/icon_sprite.png new file mode 100644 index 0000000000000000000000000000000000000000..636c80f221621c2461cb2f78764b21a1d5ce2f5a GIT binary patch literal 3217 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 literal 0 HcmV?d00001 diff --git a/vendor/assets/images/slider_handles.png b/vendor/assets/images/slider_handles.png new file mode 100644 index 0000000000000000000000000000000000000000..b95a46eca97b9001da25067948cb05a0021be2a1 GIT binary patch literal 4453 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 literal 0 HcmV?d00001 diff --git a/vendor/assets/images/ui-icons_222222_256x240.png b/vendor/assets/images/ui-icons_222222_256x240.png new file mode 100644 index 0000000000000000000000000000000000000000..b273ff111d219c9b9a8b96d57683d0075fb7871a GIT binary patch literal 4369 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 literal 0 HcmV?d00001 diff --git a/vendor/assets/images/ui-icons_454545_256x240.png b/vendor/assets/images/ui-icons_454545_256x240.png new file mode 100644 index 0000000000000000000000000000000000000000..59bd45b907c4fd965697774ce8c5fc6b2fd9c105 GIT binary patch literal 4369 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") - .attr( 'placeholder', opts.addTagPrompt ) - .keypress( function(e) { - var $this = $(this), - pressed = e.which; - - for ( i in opts.delimiters ) { - - if (pressed == opts.delimiters[i]) { - self.add( $this.val() ); - e.preventDefault(); - return false; - } - } - }) - // for some reason, in Safari, backspace is only recognized on keyup - .keyup( function(e) { - var $this = $(this), - pressed = e.which; - - // if backspace is hit with no input, remove the last tag - if (pressed == 8) { // backspace - if ( $this.val() == "" ) { - self.remove(); - return false; - } - return; - } - }); - - this.tagDiv = $("
") - .addClass( opts.cssClass ) - .click( function() { - $(this).children('input').focus(); - }) - .append( this.tagInput ) - .insertAfter( el.hide() ); - - // if the field isn't empty, parse the field for tags, and prepopulate existing tags - var initVal = $.trim( el.val() ); - - if ( initVal ) { - var initTags = initVal.split( opts.outputDelimiter ); - $.each( initTags, function(i, tag) { - self.add( tag ); - }); - } - }, - - _setOption: function( key, value ) { - options.key = value; - }, - - // add a tag, public function - add: function(text) { - var self = this; - text = text || self.tagInput.val(); - if (text) { - var tagIndex = self.tags.length; - - var removeButton = $("
x") - .click( function() { - self.remove( tagIndex ); - return false; - }); - var newTag = $("") - .text( text ) - .append( removeButton ); - - self.tagInput.before( newTag ); - self.tags.push( text ); - self.tagInput.val(''); - } - }, - - // remove a tag by index, public function - // if index is blank, remove the last tag - remove: function( tagIndex ) { - var self = this; - if ( tagIndex == null || tagIndex === (self.tags.length - 1) ) { - this.tagDiv.children("span").last().remove(); - self.tags.pop(); - } - if ( typeof(tagIndex) == 'number' ) { - // otherwise just hide this tag, and we don't mess up the index - this.tagDiv.children( "span:eq(" + tagIndex + ")" ).hide(); - // we rely on the serialize function to remove null values - delete( self.tags[tagIndex] ); - } - }, - - // serialize the tags with the given delimiter, and write it back into the tagified field - serialize: function() { - var self = this; - var delim = self.options.outputDelimiter; - var tagsStr = self.tags.join( delim ); - - // our tags might have deleted entries, remove them here - var dupes = new RegExp(delim + delim + '+', 'g'); // regex: /,,+/g - var ends = new RegExp('^' + delim + '|' + delim + '$', 'g'); // regex: /^,|,$/g - var outputStr = tagsStr.replace( dupes, delim ).replace(ends, ''); - - self.element.val(outputStr); - return outputStr; - }, - - inputField: function() { - return this.tagInput; - }, - - containerDiv: function() { - return this.tagDiv; - }, - - // remove the div, and show original input - destroy: function() { - $.Widget.prototype.destroy.apply(this); - this.tagDiv.remove(); - this.element.show(); - } - }); - -})(jQuery); \ No newline at end of file diff --git a/vendor/assets/javascripts/jquery.ui.selectmenu.js b/vendor/assets/javascripts/jquery.ui.selectmenu.js deleted file mode 100644 index 957fe4d8..00000000 --- a/vendor/assets/javascripts/jquery.ui.selectmenu.js +++ /dev/null @@ -1,844 +0,0 @@ - /* - * jQuery UI selectmenu dev version - * - * Copyright (c) 2009 AUTHORS.txt (http://jqueryui.com/about) - * Dual licensed under the MIT (MIT-LICENSE.txt) - * and GPL (GPL-LICENSE.txt) licenses. - * - * http://docs.jquery.com/UI - * https://github.com/fnagel/jquery-ui/wiki/Selectmenu - */ - -(function($) { - -$.widget("ui.selectmenu", { - getter: "value", - version: "1.8", - eventPrefix: "selectmenu", - options: { - transferClasses: true, - typeAhead: "sequential", - style: 'dropdown', - positionOptions: { - my: "left top", - at: "left bottom", - offset: null - }, - width: null, - menuWidth: null, - handleWidth: 26, - maxHeight: null, - icons: null, - format: null, - bgImage: function() {}, - wrapperElement: "
" - }, - - _create: function() { - var self = this, o = this.options; - - // set a default id value, generate a new random one if not set by developer - var selectmenuId = this.element.attr( 'id' ) || 'ui-selectmenu-' + Math.random().toString( 16 ).slice( 2, 10 ); - - // quick array of button and menu id's - this.ids = [ selectmenuId + '-button', selectmenuId + '-menu' ]; - - // define safe mouseup for future toggling - this._safemouseup = true; - - // create menu button wrapper - this.newelement = $( '', { - 'class': this.widgetBaseClass + ' ui-widget ui-state-default ui-corner-all', - 'id' : this.ids[ 0 ], - 'role': 'button', - 'href': '#nogo', - 'tabindex': this.element.attr( 'disabled' ) ? 1 : 0, - 'aria-haspopup': true, - 'aria-owns': this.ids[ 1 ] - }); - this.newelementWrap = $( o.wrapperElement ) - .append( this.newelement ) - .insertAfter( this.element ); - - // transfer tabindex - var tabindex = this.element.attr( 'tabindex' ); - if ( tabindex ) { - this.newelement.attr( 'tabindex', tabindex ); - } - - // save reference to select in data for ease in calling methods - this.newelement.data( 'selectelement', this.element ); - - // menu icon - this.selectmenuIcon = $( '' ) - .prependTo( this.newelement ); - - // append status span to button - this.newelement.prepend( '' ); - - // make associated form label trigger focus - $( 'label[for="' + selectmenuId + '"]' ) - .attr( 'for', this.ids[0] ) - .bind( 'click.selectmenu', function() { - self.newelement[0].focus(); - return false; - }); - - // click toggle for menu visibility - this.newelement - .bind('mousedown.selectmenu', function(event) { - self._toggle(event, true); - // make sure a click won't open/close instantly - if (o.style == "popup") { - self._safemouseup = false; - setTimeout(function() { self._safemouseup = true; }, 300); - } - return false; - }) - .bind('click.selectmenu', function() { - return false; - }) - .bind("keydown.selectmenu", function(event) { - var ret = false; - switch (event.keyCode) { - case $.ui.keyCode.ENTER: - ret = true; - break; - case $.ui.keyCode.SPACE: - self._toggle(event); - break; - case $.ui.keyCode.UP: - if (event.altKey) { - self.open(event); - } else { - self._moveSelection(-1); - } - break; - case $.ui.keyCode.DOWN: - if (event.altKey) { - self.open(event); - } else { - self._moveSelection(1); - } - break; - case $.ui.keyCode.LEFT: - self._moveSelection(-1); - break; - case $.ui.keyCode.RIGHT: - self._moveSelection(1); - break; - case $.ui.keyCode.TAB: - ret = true; - break; - default: - ret = true; - } - return ret; - }) - .bind('keypress.selectmenu', function(event) { - self._typeAhead(event.which, 'mouseup'); - return true; - }) - .bind('mouseover.selectmenu focus.selectmenu', function() { - if (!o.disabled) { - $(this).addClass(self.widgetBaseClass + '-focus ui-state-hover'); - } - }) - .bind('mouseout.selectmenu blur.selectmenu', function() { - if (!o.disabled) { - $(this).removeClass(self.widgetBaseClass + '-focus ui-state-hover'); - } - }); - - // document click closes menu - $(document).bind("mousedown.selectmenu", function(event) { - self.close(event); - }); - - // change event on original selectmenu - this.element - .bind("click.selectmenu", function() { - self._refreshValue(); - }) - // FIXME: newelement can be null under unclear circumstances in IE8 - // TODO not sure if this is still a problem (fnagel 20.03.11) - .bind("focus.selectmenu", function() { - if (self.newelement) { - self.newelement[0].focus(); - } - }); - - // set width when not set via options - if (!o.width) { - o.width = this.element.outerWidth(); - } - // set menu button width - this.newelement.width(o.width); - - // hide original selectmenu element - this.element.hide(); - - // create menu portion, append to body - this.list = $( '
    ', { - 'class': 'ui-widget ui-widget-content', - 'aria-hidden': true, - 'role': 'listbox', - 'aria-labelledby': this.ids[0], - 'id': this.ids[1] - }); - this.listWrap = $( o.wrapperElement ) - .addClass( self.widgetBaseClass + '-menu' ) - .append( this.list ) - .appendTo( 'body' ); - - // transfer menu click to menu button - this.list - .bind("keydown.selectmenu", function(event) { - var ret = false; - switch (event.keyCode) { - case $.ui.keyCode.UP: - if (event.altKey) { - self.close(event, true); - } else { - self._moveFocus(-1); - } - break; - case $.ui.keyCode.DOWN: - if (event.altKey) { - self.close(event, true); - } else { - self._moveFocus(1); - } - break; - case $.ui.keyCode.LEFT: - self._moveFocus(-1); - break; - case $.ui.keyCode.RIGHT: - self._moveFocus(1); - break; - case $.ui.keyCode.HOME: - self._moveFocus(':first'); - break; - case $.ui.keyCode.PAGE_UP: - self._scrollPage('up'); - break; - case $.ui.keyCode.PAGE_DOWN: - self._scrollPage('down'); - break; - case $.ui.keyCode.END: - self._moveFocus(':last'); - break; - case $.ui.keyCode.ENTER: - case $.ui.keyCode.SPACE: - self.close(event, true); - $(event.target).parents('li:eq(0)').trigger('mouseup'); - break; - case $.ui.keyCode.TAB: - ret = true; - self.close(event, true); - $(event.target).parents('li:eq(0)').trigger('mouseup'); - break; - case $.ui.keyCode.ESCAPE: - self.close(event, true); - break; - default: - ret = true; - } - return ret; - }) - .bind('keypress.selectmenu', function(event) { - self._typeAhead(event.which, 'focus'); - return true; - }) - // this allows for using the scrollbar in an overflowed list - .bind( 'mousedown.selectmenu mouseup.selectmenu', function() { return false; }); - - // needed when window is resized - // TODO seems to be useless, but causes errors (fnagel 01.08.11) - // see: https://github.com/fnagel/jquery-ui/issues/147 - // $(window).bind( "resize.selectmenu", $.proxy( self._refreshPosition, this ) ); - }, - - _init: function() { - var self = this, o = this.options; - - // serialize selectmenu element options - var selectOptionData = []; - this.element - .find('option') - .each(function() { - var opt = $(this); - selectOptionData.push({ - value: opt.attr('value'), - text: self._formatText(opt.text()), - selected: opt.attr('selected'), - disabled: opt.attr('disabled'), - classes: opt.attr('class'), - typeahead: opt.attr('typeahead'), - parentOptGroup: opt.parent('optgroup'), - bgImage: o.bgImage.call(opt) - }); - }); - - // active state class is only used in popup style - var activeClass = (self.options.style == "popup") ? " ui-state-active" : ""; - - // empty list so we can refresh the selectmenu via selectmenu() - this.list.html(""); - - // write li's - if (selectOptionData.length) { - for (var i = 0; i < selectOptionData.length; i++) { - var thisLiAttr = { role : 'presentation' }; - if ( selectOptionData[ i ].disabled ) { - thisLiAttr[ 'class' ] = this.namespace + '-state-disabled'; - } - var thisAAttr = { - html: selectOptionData[i].text, - href : '#nogo', - tabindex : -1, - role : 'option', - 'aria-selected' : false - }; - if ( selectOptionData[ i ].disabled ) { - thisAAttr[ 'aria-disabled' ] = selectOptionData[ i ].disabled; - } - if ( selectOptionData[ i ].typeahead ) { - thisAAttr[ 'typeahead' ] = selectOptionData[ i ].typeahead; - } - var thisA = $('', thisAAttr); - var thisLi = $('
  • ', thisLiAttr) - .append(thisA) - .data('index', i) - .addClass(selectOptionData[i].classes) - .data('optionClasses', selectOptionData[i].classes || '') - .bind("mouseup.selectmenu", function(event) { - if (self._safemouseup && !self._disabled(event.currentTarget) && !self._disabled($( event.currentTarget ).parents( "ul>li." + self.widgetBaseClass + "-group " )) ) { - var changed = $(this).data('index') != self._selectedIndex(); - self.index($(this).data('index')); - self.select(event); - if (changed) { - self.change(event); - } - self.close(event, true); - } - return false; - }) - .bind("click.selectmenu", function() { - return false; - }) - .bind('mouseover.selectmenu focus.selectmenu', function(e) { - // no hover if diabled - if (!$(e.currentTarget).hasClass(self.namespace + '-state-disabled') && !$(e.currentTarget).parent("ul").parent("li").hasClass(self.namespace + '-state-disabled')) { - self._selectedOptionLi().addClass(activeClass); - self._focusedOptionLi().removeClass(self.widgetBaseClass + '-item-focus ui-state-hover'); - $(this).removeClass('ui-state-active').addClass(self.widgetBaseClass + '-item-focus ui-state-hover'); - } - }) - .bind('mouseout.selectmenu blur.selectmenu', function() { - if ($(this).is(self._selectedOptionLi().selector)) { - $(this).addClass(activeClass); - } - $(this).removeClass(self.widgetBaseClass + '-item-focus ui-state-hover'); - }); - - // optgroup or not... - if ( selectOptionData[i].parentOptGroup.length ) { - var optGroupName = self.widgetBaseClass + '-group-' + this.element.find( 'optgroup' ).index( selectOptionData[i].parentOptGroup ); - if (this.list.find( 'li.' + optGroupName ).length ) { - this.list.find( 'li.' + optGroupName + ':last ul' ).append( thisLi ); - } else { - $('
  • ') - .appendTo( this.list ) - .find( 'ul' ) - .append( thisLi ); - } - } else { - thisLi.appendTo(this.list); - } - - // append icon if option is specified - if (o.icons) { - for (var j in o.icons) { - if (thisLi.is(o.icons[j].find)) { - thisLi - .data('optionClasses', selectOptionData[i].classes + ' ' + self.widgetBaseClass + '-hasIcon') - .addClass(self.widgetBaseClass + '-hasIcon'); - var iconClass = o.icons[j].icon || ""; - thisLi - .find('a:eq(0)') - .prepend(''); - if (selectOptionData[i].bgImage) { - thisLi.find('span').css('background-image', selectOptionData[i].bgImage); - } - } - } - } - } - } else { - $('
  • ').appendTo(this.list); - } - // we need to set and unset the CSS classes for dropdown and popup style - var isDropDown = ( o.style == 'dropdown' ); - this.newelement - .toggleClass( self.widgetBaseClass + '-dropdown', isDropDown ) - .toggleClass( self.widgetBaseClass + '-popup', !isDropDown ); - this.list - .toggleClass( self.widgetBaseClass + '-menu-dropdown ui-corner-bottom', isDropDown ) - .toggleClass( self.widgetBaseClass + '-menu-popup ui-corner-all', !isDropDown ) - // add corners to top and bottom menu items - .find( 'li:first' ) - .toggleClass( 'ui-corner-top', !isDropDown ) - .end().find( 'li:last' ) - .addClass( 'ui-corner-bottom' ); - this.selectmenuIcon - .toggleClass( 'ui-icon-triangle-1-s', isDropDown ) - .toggleClass( 'ui-icon-triangle-2-n-s', !isDropDown ); - - // transfer classes to selectmenu and list - if ( o.transferClasses ) { - var transferClasses = this.element.attr( 'class' ) || ''; - this.newelement.add( this.list ).addClass( transferClasses ); - } - - // set menu width to either menuWidth option value, width option value, or select width - if ( o.style == 'dropdown' ) { - this.list.width( o.menuWidth ? o.menuWidth : o.width ); - } else { - this.list.width( o.menuWidth ? o.menuWidth : o.width - o.handleWidth ); - } - - // reset height to auto - this.list.css( 'height', 'auto' ); - var listH = this.listWrap.height(); - // calculate default max height - if ( o.maxHeight && o.maxHeight < listH ) { - this.list.height( o.maxHeight ); - } else { - var winH = $( window ).height() / 3; - if ( winH < listH ) this.list.height( winH ); - } - - // save reference to actionable li's (not group label li's) - this._optionLis = this.list.find( 'li:not(.' + self.widgetBaseClass + '-group)' ); - - // transfer disabled state - if ( this.element.attr( 'disabled' ) ) { - this.disable(); - } else { - this.enable() - } - - // update value - this.index( this._selectedIndex() ); - - // needed when selectmenu is placed at the very bottom / top of the page - window.setTimeout( function() { - self._refreshPosition(); - }, 200 ); - }, - - destroy: function() { - this.element.removeData( this.widgetName ) - .removeClass( this.widgetBaseClass + '-disabled' + ' ' + this.namespace + '-state-disabled' ) - .removeAttr( 'aria-disabled' ) - .unbind( ".selectmenu" ); - - // TODO unneded as event binding has been disabled - // $( window ).unbind( ".selectmenu" ); - $( document ).unbind( ".selectmenu" ); - - // unbind click on label, reset its for attr - $( 'label[for=' + this.newelement.attr('id') + ']' ) - .attr( 'for', this.element.attr( 'id' ) ) - .unbind( '.selectmenu' ); - - this.newelementWrap.remove(); - this.listWrap.remove(); - - this.element.show(); - - // call widget destroy function - $.Widget.prototype.destroy.apply(this, arguments); - }, - - _typeAhead: function( code, eventType ){ - var self = this, focusFound = false, C = String.fromCharCode(code).toUpperCase(); - c = C.toLowerCase(); - - if ( self.options.typeAhead == 'sequential' ) { - // clear the timeout so we can use _prevChar - window.clearTimeout('ui.selectmenu-' + self.selectmenuId); - - // define our find var - var find = typeof( self._prevChar ) == 'undefined' ? '' : self._prevChar.join( '' ); - - function focusOptSeq( elem, ind, c ){ - focusFound = true; - $( elem ).trigger( eventType ); - typeof( self._prevChar ) == 'undefined' ? self._prevChar = [ c ] : self._prevChar[ self._prevChar.length ] = c; - } - this.list.find( 'li a' ).each( function( i ) { - if ( !focusFound ) { - // allow the typeahead attribute on the option tag for a more specific lookup - var thisText = $( this ).attr( 'typeahead' ) || $(this).text(); - if ( thisText.indexOf( find + C ) === 0 ) { - focusOptSeq( this, i, C ); - } else if (thisText.indexOf(find+c) === 0 ) { - focusOptSeq( this, i, c ); - } - } - }); - // set a 1 second timeout for sequenctial typeahead - // keep this set even if we have no matches so it doesnt typeahead somewhere else - window.setTimeout( function( el ) { - self._prevChar = undefined; - }, 1000, self ); - - } else { - // define self._prevChar if needed - if ( !self._prevChar ) { self._prevChar = [ '' , 0 ]; } - - focusFound = false; - function focusOpt( elem, ind ){ - focusFound = true; - $( elem ).trigger( eventType ); - self._prevChar[ 1 ] = ind; - } - this.list.find( 'li a' ).each(function( i ){ - if (!focusFound){ - var thisText = $(this).text(); - if ( thisText.indexOf( C ) === 0 || thisText.indexOf( c ) === 0 ) { - if (self._prevChar[0] == C){ - if ( self._prevChar[ 1 ] < i ){ focusOpt( this, i ); } - } else{ - focusOpt( this, i ); - } - } - } - }); - this._prevChar[ 0 ] = C; - } - }, - - // returns some usefull information, called by callbacks only - _uiHash: function() { - var index = this.index(); - return { - index: index, - option: $("option", this.element).get(index), - value: this.element[0].value - }; - }, - - open: function(event) { - var self = this, o = this.options; - if ( self.newelement.attr("aria-disabled") != 'true' ) { - self._closeOthers(event); - self.newelement.addClass('ui-state-active'); - - self.listWrap.appendTo( o.appendTo ); - self.list.attr('aria-hidden', false); - - if ( o.style == "dropdown" ) { - self.newelement.removeClass('ui-corner-all').addClass('ui-corner-top'); - } - - self.listWrap.addClass( self.widgetBaseClass + '-open' ); - // positioning needed for IE7 (tested 01.08.11 on MS VPC Image) - // see https://github.com/fnagel/jquery-ui/issues/147 - if ( $.browser.msie && $.browser.version.substr( 0,1 ) == 7 ) { - self._refreshPosition(); - } - var selected = self.list.attr('aria-hidden', false).find('li:not(.' + self.widgetBaseClass + '-group):eq(' + self._selectedIndex() + '):visible a'); - if (selected.length) selected[0].focus(); - // positioning needed for FF, Chrome, IE8, IE7, IE6 (tested 01.08.11 on MS VPC Image) - self._refreshPosition(); - - self._trigger("open", event, self._uiHash()); - } - }, - - close: function(event, retainFocus) { - if ( this.newelement.is('.ui-state-active') ) { - this.newelement - .removeClass('ui-state-active'); - this.listWrap.removeClass(this.widgetBaseClass + '-open'); - this.list.attr('aria-hidden', true); - if ( this.options.style == "dropdown" ) { - this.newelement.removeClass('ui-corner-top').addClass('ui-corner-all'); - } - if ( retainFocus ) { - this.newelement.focus(); - } - this._trigger("close", event, this._uiHash()); - } - }, - - change: function(event) { - this.element.trigger("change"); - this._trigger("change", event, this._uiHash()); - }, - - select: function(event) { - if (this._disabled(event.currentTarget)) { return false; } - this._trigger("select", event, this._uiHash()); - }, - - _closeOthers: function(event) { - $('.' + this.widgetBaseClass + '.ui-state-active').not(this.newelement).each(function() { - $(this).data('selectelement').selectmenu('close', event); - }); - $('.' + this.widgetBaseClass + '.ui-state-hover').trigger('mouseout'); - }, - - _toggle: function(event, retainFocus) { - if ( this.list.parent().is('.' + this.widgetBaseClass + '-open') ) { - this.close(event, retainFocus); - } else { - this.open(event); - } - }, - - _formatText: function(text) { - return (this.options.format ? this.options.format(text) : text); - }, - - _selectedIndex: function() { - return this.element[0].selectedIndex; - }, - - _selectedOptionLi: function() { - return this._optionLis.eq(this._selectedIndex()); - }, - - _focusedOptionLi: function() { - return this.list.find('.' + this.widgetBaseClass + '-item-focus'); - }, - - _moveSelection: function(amt, recIndex) { - // do nothing if disabled - if (!this.options.disabled) { - var currIndex = parseInt(this._selectedOptionLi().data('index') || 0, 10); - var newIndex = currIndex + amt; - // do not loop when using up key - - if (newIndex < 0) { - newIndex = 0; - } - if (newIndex > this._optionLis.size() - 1) { - newIndex = this._optionLis.size() - 1; - } - // Occurs when a full loop has been made - if (newIndex === recIndex) { return false; } - - if (this._optionLis.eq(newIndex).hasClass( this.namespace + '-state-disabled' )) { - // if option at newIndex is disabled, call _moveFocus, incrementing amt by one - (amt > 0) ? ++amt : --amt; - this._moveSelection(amt, newIndex); - } else { - return this._optionLis.eq(newIndex).trigger('mouseup'); - } - } - }, - - _moveFocus: function(amt, recIndex) { - if (!isNaN(amt)) { - var currIndex = parseInt(this._focusedOptionLi().data('index') || 0, 10); - var newIndex = currIndex + amt; - } else { - var newIndex = parseInt(this._optionLis.filter(amt).data('index'), 10); - } - - if (newIndex < 0) { - newIndex = 0; - } - if (newIndex > this._optionLis.size() - 1) { - newIndex = this._optionLis.size() - 1; - } - - //Occurs when a full loop has been made - if (newIndex === recIndex) { return false; } - - var activeID = this.widgetBaseClass + '-item-' + Math.round(Math.random() * 1000); - - this._focusedOptionLi().find('a:eq(0)').attr('id', ''); - - if (this._optionLis.eq(newIndex).hasClass( this.namespace + '-state-disabled' )) { - // if option at newIndex is disabled, call _moveFocus, incrementing amt by one - (amt > 0) ? ++amt : --amt; - this._moveFocus(amt, newIndex); - } else { - this._optionLis.eq(newIndex).find('a:eq(0)').attr('id',activeID).focus(); - } - - this.list.attr('aria-activedescendant', activeID); - }, - - _scrollPage: function(direction) { - var numPerPage = Math.floor(this.list.outerHeight() / this.list.find('li:first').outerHeight()); - numPerPage = (direction == 'up' ? -numPerPage : numPerPage); - this._moveFocus(numPerPage); - }, - - _setOption: function(key, value) { - this.options[key] = value; - // set - if (key == 'disabled') { - this.close(); - this.element - .add(this.newelement) - .add(this.list)[value ? 'addClass' : 'removeClass']( - this.widgetBaseClass + '-disabled' + ' ' + - this.namespace + '-state-disabled') - .attr("aria-disabled", value); - } - }, - - disable: function(index, type){ - // if options is not provided, call the parents disable function - if ( typeof( index ) == 'undefined' ) { - this._setOption( 'disabled', true ); - } else { - if ( type == "optgroup" ) { - this._disableOptgroup(index); - } else { - this._disableOption(index); - } - } - }, - - enable: function(index, type) { - // if options is not provided, call the parents enable function - if ( typeof( index ) == 'undefined' ) { - this._setOption('disabled', false); - } else { - if ( type == "optgroup" ) { - this._enableOptgroup(index); - } else { - this._enableOption(index); - } - } - }, - - _disabled: function(elem) { - return $(elem).hasClass( this.namespace + '-state-disabled' ); - }, - - _disableOption: function(index) { - var optionElem = this._optionLis.eq(index); - if (optionElem) { - optionElem.addClass(this.namespace + '-state-disabled') - .find("a").attr("aria-disabled", true); - this.element.find("option").eq(index).attr("disabled", "disabled"); - } - }, - - _enableOption: function(index) { - var optionElem = this._optionLis.eq(index); - if (optionElem) { - optionElem.removeClass( this.namespace + '-state-disabled' ) - .find("a").attr("aria-disabled", false); - this.element.find("option").eq(index).removeAttr("disabled"); - } - }, - - _disableOptgroup: function(index) { - var optGroupElem = this.list.find( 'li.' + this.widgetBaseClass + '-group-' + index ); - if (optGroupElem) { - optGroupElem.addClass(this.namespace + '-state-disabled') - .attr("aria-disabled", true); - this.element.find("optgroup").eq(index).attr("disabled", "disabled"); - } - }, - - _enableOptgroup: function(index) { - var optGroupElem = this.list.find( 'li.' + this.widgetBaseClass + '-group-' + index ); - if (optGroupElem) { - optGroupElem.removeClass(this.namespace + '-state-disabled') - .attr("aria-disabled", false); - this.element.find("optgroup").eq(index).removeAttr("disabled"); - } - }, - - index: function(newValue) { - if (arguments.length) { - if (!this._disabled($(this._optionLis[newValue]))) { - this.element[0].selectedIndex = newValue; - this._refreshValue(); - } else { - return false; - } - } else { - return this._selectedIndex(); - } - }, - - value: function(newValue) { - if (arguments.length) { - this.element[0].value = newValue; - this._refreshValue(); - } else { - return this.element[0].value; - } - }, - - _refreshValue: function() { - var activeClass = (this.options.style == "popup") ? " ui-state-active" : ""; - var activeID = this.widgetBaseClass + '-item-' + Math.round(Math.random() * 1000); - // deselect previous - this.list - .find('.' + this.widgetBaseClass + '-item-selected') - .removeClass(this.widgetBaseClass + "-item-selected" + activeClass) - .find('a') - .attr('aria-selected', 'false') - .attr('id', ''); - // select new - this._selectedOptionLi() - .addClass(this.widgetBaseClass + "-item-selected" + activeClass) - .find('a') - .attr('aria-selected', 'true') - .attr('id', activeID); - - // toggle any class brought in from option - var currentOptionClasses = (this.newelement.data('optionClasses') ? this.newelement.data('optionClasses') : ""); - var newOptionClasses = (this._selectedOptionLi().data('optionClasses') ? this._selectedOptionLi().data('optionClasses') : ""); - this.newelement - .removeClass(currentOptionClasses) - .data('optionClasses', newOptionClasses) - .addClass( newOptionClasses ) - .find('.' + this.widgetBaseClass + '-status') - .html( - this._selectedOptionLi() - .find('a:eq(0)') - .html() - ); - - this.list.attr('aria-activedescendant', activeID); - }, - - _refreshPosition: function() { - var o = this.options; - - // if its a native pop-up we need to calculate the position of the selected li - if ( o.style == "popup" && !o.positionOptions.offset ) { - var selected = this._selectedOptionLi(); - var _offset = "0 -" + ( selected.outerHeight() + selected.offset().top - this.list.offset().top ); - } - // update zIndex if jQuery UI is able to process - var zIndexElement = this.element.zIndex(); - if ( zIndexElement ) { - this.listWrap.css( 'zIndex', zIndexElement ); - } - this.listWrap.position({ - // set options for position plugin - of: o.positionOptions.of || this.newelement, - my: o.positionOptions.my, - at: o.positionOptions.at, - offset: o.positionOptions.offset || _offset, - collision: o.positionOptions.collision || 'flip' - }); - } -}); - -})(jQuery); diff --git a/vendor/assets/stylesheets/jquery-ui/jquery.tagify.css b/vendor/assets/stylesheets/jquery-ui/jquery.tagify.css deleted file mode 100644 index d6c178f7..00000000 --- a/vendor/assets/stylesheets/jquery-ui/jquery.tagify.css +++ /dev/null @@ -1,34 +0,0 @@ -/* Tagify styles -Author: Alicia Liu test -*/ - -.tagify-container { -} - -.tagify-container > span { - display: inline-block; - padding: 8px 11px 8px 11px; - margin: 1px 5px 0px 0px; - border-radius: 4px; - border: 1px solid #d0e1ff; - background-color: #d0e1ff; - color: #0f326d; - font-weight: bold; - font-size: 14px; -} - -.tagify-container > span > a { - padding-left: 5px !important; - color: #83a5e1; - text-decoration: none; - font-weight: bold; -} - -.tagify-container > input { - border: 0 none; - width: 100px !important; -} - -.tagify-container > input:focus { - outline: none; -} \ No newline at end of file diff --git a/vendor/assets/stylesheets/jquery-ui/jquery.ui.selectmenu.css b/vendor/assets/stylesheets/jquery-ui/jquery.ui.selectmenu.css deleted file mode 100644 index 481adf5a..00000000 --- a/vendor/assets/stylesheets/jquery-ui/jquery.ui.selectmenu.css +++ /dev/null @@ -1,33 +0,0 @@ -/* Selectmenu -----------------------------------*/ -.ui-selectmenu { background:none; font-size:12px;display: block; display: inline-block; position: relative; height: 2.2em; vertical-align: middle; text-decoration: none; overflow: hidden; zoom: 1; } -.ui-selectmenu-icon { position:absolute; right:6px; margin-top:-8px; top: 50%; } -.ui-selectmenu-menu { padding:0; margin:0; position:absolute; top: 0; display: none; z-index: 1005;} /* z-index: 1005 to make selectmenu work with dialog */ -.ui-selectmenu-menu ul { padding:0; margin:0; list-style:none; position: relative; overflow: auto; overflow-y: auto ; overflow-x: hidden; } -.ui-selectmenu-open { display: block; } -.ui-selectmenu.ui-widget { background:none; } -.ui-selectmenu-menu-popup { margin-top: -1px; } -.ui-selectmenu-menu-dropdown { } -.ui-selectmenu-menu li.ui-state-active { background:#F7FBFC; border:none; padding:1px 0;} -.ui-selectmenu-menu li { padding:0; margin:0; display: block; border-top: 1px dotted transparent; border-bottom: 1px dotted transparent; border-right-width: 0 !important; border-left-width: 0 !important; font-weight: normal !important; } -.ui-selectmenu-menu li a,.ui-selectmenu-status { line-height: 1.4em; display: block; padding: .405em 2.1em .405em 1em; outline:none; text-decoration:none; } -.ui-selectmenu-menu li.ui-state-disabled a, .ui-state-disabled { cursor: default; } -.ui-selectmenu-menu li.ui-selectmenu-hasIcon a, -.ui-selectmenu-hasIcon .ui-selectmenu-status { padding-left: 20px; position: relative; margin-left: 5px; } -.ui-selectmenu-menu li .ui-icon, .ui-selectmenu-status .ui-icon { position: absolute; top: 1em; margin-top: -8px; left: 0; } -.ui-selectmenu-status { line-height: 1.4em; } -.ui-selectmenu-open li.ui-selectmenu-item-focus { background: none repeat scroll 0 0 #FFF6BF; border:1px solid #eaeaea;} -.ui-selectmenu-open li.ui-selectmenu-item-selected { } -.ui-selectmenu-menu li span,.ui-selectmenu-status span { display:block; margin-bottom: .2em; } -.ui-selectmenu-menu li .ui-selectmenu-item-header { font-weight: bold; } -.ui-selectmenu-menu li .ui-selectmenu-item-content { } -.ui-selectmenu-menu li .ui-selectmenu-item-footer { opacity: .8; } -/* for optgroups */ -.ui-selectmenu-menu .ui-selectmenu-group { font-size: 1em; } -.ui-selectmenu-menu .ui-selectmenu-group .ui-selectmenu-group-label { line-height: 1.4em; display:block; padding: .6em .5em 0; font-weight: bold; } -.ui-selectmenu-menu .ui-selectmenu-group ul { margin: 0; padding: 0; } -/* IE6 workaround (dotted transparent borders) */ -* html .ui-selectmenu-menu li { border-color: pink; filter:chroma(color=pink); width:100%; } -* html .ui-selectmenu-menu li a { position: relative } -/* IE7 workaround (opacity disabled) */ -*+html .ui-state-disabled, *+html .ui-state-disabled a { color: silver; } diff --git a/vendor/assets/stylesheets/jquery.ui.aristo.css b/vendor/assets/stylesheets/jquery.ui.aristo.css new file mode 100644 index 00000000..8cc6e787 --- /dev/null +++ b/vendor/assets/stylesheets/jquery.ui.aristo.css @@ -0,0 +1,738 @@ +/* + * jQuery UI CSS Framework 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Theming/API + */ + +/* Layout helpers +----------------------------------*/ +.ui-helper-hidden { display: none; } +.ui-helper-hidden-accessible { position: absolute !important; clip: rect(1px 1px 1px 1px); clip: rect(1px,1px,1px,1px); } +.ui-helper-reset { margin: 0; padding: 0; border: 0; outline: 0; line-height: 1.3; text-decoration: none; font-size: 100%; list-style: none; } +.ui-helper-clearfix:after { content: "."; display: block; height: 0; clear: both; visibility: hidden; } +.ui-helper-clearfix { display: inline-block; } +/* required comment for clearfix to work in Opera \*/ +* html .ui-helper-clearfix { height:1%; } +.ui-helper-clearfix { display:block; } +/* end clearfix */ +.ui-helper-zfix { width: 100%; height: 100%; top: 0; left: 0; position: absolute; opacity: 0; filter:Alpha(Opacity=0); } + + +/* Interaction Cues +----------------------------------*/ +.ui-state-disabled { cursor: default !important; } + + +/* Icons +----------------------------------*/ + +/* states and images */ +.ui-icon { display: block; text-indent: -99999px; overflow: hidden; background-repeat: no-repeat; } + + +/* Misc visuals +----------------------------------*/ + +/* Overlays */ +.ui-widget-overlay { position: absolute; top: 0; left: 0; width: 100%; height: 100%; } + + +/* + * jQuery UI CSS Framework 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Theming/API + * + * To view and modify this theme, visit http://jqueryui.com/themeroller/?ctl=themeroller + */ + + +/* Component containers +----------------------------------*/ +.ui-widget { font-family: Arial,sans-serif; font-size: 1.1em; } +.ui-widget .ui-widget { font-size: 1em; } +.ui-widget input, .ui-widget select, .ui-widget textarea, .ui-widget button { font-family: Arial,sans-serif; font-size: 1em; } +.ui-widget-content { border: 1px solid #B6B6B6; background: #ffffff; color: #4F4F4F; } +.ui-widget-content a { color: #4F4F4F; } +.ui-widget-header { border: 1px solid #B6B6B6; color: #4F4F4F; font-weight: bold; } +.ui-widget-header { + background: #ededed url(bg_fallback.png) 0 0 repeat-x; /* Old browsers */ + background: -moz-linear-gradient(top, #ededed 0%, #c4c4c4 100%); /* FF3.6+ */ + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%,#ededed), color-stop(100%,#c4c4c4)); /* Chrome,Safari4+ */ + background: -webkit-linear-gradient(top, #ededed 0%,#c4c4c4 100%); /* Chrome10+,Safari5.1+ */ + background: -o-linear-gradient(top, #ededed 0%,#c4c4c4 100%); /* Opera11.10+ */ + background: -ms-linear-gradient(top, #ededed 0%,#c4c4c4 100%); /* IE10+ */ + background: linear-gradient(top, #ededed 0%,#c4c4c4 100%); /* W3C */ +} +.ui-widget-header a { color: #4F4F4F; } + +/* Interaction states +----------------------------------*/ +.ui-state-default, .ui-widget-content .ui-state-default, .ui-widget-header .ui-state-default { border: 1px solid #B6B6B6; font-weight: normal; color: #4F4F4F; } +.ui-state-default, .ui-widget-content .ui-state-default, .ui-widget-header .ui-state-default { + background: #ededed url(bg_fallback.png) 0 0 repeat-x; /* Old browsers */ + background: -moz-linear-gradient(top, #ededed 0%, #c4c4c4 100%); /* FF3.6+ */ + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%,#ededed), color-stop(100%,#c4c4c4)); /* Chrome,Safari4+ */ + background: -webkit-linear-gradient(top, #ededed 0%,#c4c4c4 100%); /* Chrome10+,Safari5.1+ */ + background: -o-linear-gradient(top, #ededed 0%,#c4c4c4 100%); /* Opera11.10+ */ + background: -ms-linear-gradient(top, #ededed 0%,#c4c4c4 100%); /* IE10+ */ + background: linear-gradient(top, #ededed 0%,#c4c4c4 100%); /* W3C */ + -webkit-box-shadow: 0 1px 0 rgba(255,255,255,0.6) inset; + -moz-box-shadow: 0 1px 0 rgba(255,255,255,0.6) inset; + box-shadow: 0 1px 0 rgba(255,255,255,0.6) inset; +} +.ui-state-default a, .ui-state-default a:link, .ui-state-default a:visited { color: #4F4F4F; text-decoration: none; } +.ui-state-hover, .ui-widget-content .ui-state-hover, .ui-widget-header .ui-state-hover, .ui-state-focus, .ui-widget-content .ui-state-focus, .ui-widget-header .ui-state-focus { border: 1px solid #9D9D9D; font-weight: normal; color: #313131; } +.ui-state-hover a, .ui-state-hover a:hover { color: #313131; text-decoration: none; } +.ui-state-active, .ui-widget-content .ui-state-active, .ui-widget-header .ui-state-active { + outline: none; + color: #1c4257; border: 1px solid #7096ab; + background: #ededed url(bg_fallback.png) 0 -50px repeat-x; /* Old browsers */ + background: -moz-linear-gradient(top, #b9e0f5 0%, #92bdd6 100%); /* FF3.6+ */ + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%,#b9e0f5), color-stop(100%,#92bdd6)); /* Chrome,Safari4+ */ + background: -webkit-linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* Chrome10+,Safari5.1+ */ + background: -o-linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* Opera11.10+ */ + background: -ms-linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* IE10+ */ + background: linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* W3C */ + -webkit-box-shadow: none; + -moz-box-shadow: none; + box-shadow: none; +} +.ui-state-active a, .ui-state-active a:link, .ui-state-active a:visited { color: #313131; text-decoration: none; } +.ui-widget :active { outline: none; } + +/* Interaction Cues +----------------------------------*/ +.ui-state-highlight, .ui-widget-content .ui-state-highlight, .ui-widget-header .ui-state-highlight { border: 1px solid #d2dbf4; background: #f4f8fd; color: #0d2054; -moz-border-radius: 0 !important; -webkit-border-radius: 0 !important; border-radius: 0 !important; } +.ui-state-highlight a, .ui-widget-content .ui-state-highlight a,.ui-widget-header .ui-state-highlight a { color: #363636; } +.ui-state-error, .ui-widget-content .ui-state-error, .ui-widget-header .ui-state-error { border: 1px solid #e2d0d0; background: #fcf0f0; color: #280b0b; -moz-border-radius: 0 !important; -webkit-border-radius: 0 !important; border-radius: 0 !important; } +.ui-state-error a, .ui-widget-content .ui-state-error a, .ui-widget-header .ui-state-error a { color: #cd0a0a; } +.ui-state-error-text, .ui-widget-content .ui-state-error-text, .ui-widget-header .ui-state-error-text { color: #cd0a0a; } +.ui-priority-primary, .ui-widget-content .ui-priority-primary, .ui-widget-header .ui-priority-primary { font-weight: bold; } +.ui-priority-secondary, .ui-widget-content .ui-priority-secondary, .ui-widget-header .ui-priority-secondary { opacity: .7; filter:Alpha(Opacity=70); font-weight: normal; } +.ui-state-disabled, .ui-widget-content .ui-state-disabled, .ui-widget-header .ui-state-disabled { opacity: .35; filter:Alpha(Opacity=35); background-image: none; } + +/* Icons +----------------------------------*/ + +/* states and images */ +.ui-icon { width: 16px; height: 16px; background-image: url(ui-icons_222222_256x240.png); } +.ui-widget-content .ui-icon {background-image: url(ui-icons_222222_256x240.png); } +.ui-widget-header .ui-icon {background-image: url(ui-icons_222222_256x240.png); } +.ui-state-default .ui-icon { background-image: url(ui-icons_454545_256x240.png); } +.ui-state-hover .ui-icon, .ui-state-focus .ui-icon {background-image: url(ui-icons_454545_256x240.png); } +.ui-state-active .ui-icon {background-image: url(ui-icons_454545_256x240.png); } +.ui-state-highlight .ui-icon {background-image: url(ui-icons_454545_256x240.png); } +.ui-state-error .ui-icon, .ui-state-error-text .ui-icon { background: url(icon_sprite.png) -16px 0 no-repeat !important; } +.ui-state-highlight .ui-icon, .ui-state-error .ui-icon { margin-top: -1px; } + +/* positioning */ +.ui-icon-carat-1-n { background-position: 0 0; } +.ui-icon-carat-1-ne { background-position: -16px 0; } +.ui-icon-carat-1-e { background-position: -32px 0; } +.ui-icon-carat-1-se { background-position: -48px 0; } +.ui-icon-carat-1-s { background-position: -64px 0; } +.ui-icon-carat-1-sw { background-position: -80px 0; } +.ui-icon-carat-1-w { background-position: -96px 0; } +.ui-icon-carat-1-nw { background-position: -112px 0; } +.ui-icon-carat-2-n-s { background-position: -128px 0; } +.ui-icon-carat-2-e-w { background-position: -144px 0; } +.ui-icon-triangle-1-n { background-position: 0 -16px; } +.ui-icon-triangle-1-ne { background-position: -16px -16px; } +.ui-icon-triangle-1-e { background-position: -32px -16px; } +.ui-icon-triangle-1-se { background-position: -48px -16px; } +.ui-icon-triangle-1-s { background-position: -64px -16px; } +.ui-icon-triangle-1-sw { background-position: -80px -16px; } +.ui-icon-triangle-1-w { background-position: -96px -16px; } +.ui-icon-triangle-1-nw { background-position: -112px -16px; } +.ui-icon-triangle-2-n-s { background-position: -128px -16px; } +.ui-icon-triangle-2-e-w { background-position: -144px -16px; } +.ui-icon-arrow-1-n { background-position: 0 -32px; } +.ui-icon-arrow-1-ne { background-position: -16px -32px; } +.ui-icon-arrow-1-e { background-position: -32px -32px; } +.ui-icon-arrow-1-se { background-position: -48px -32px; } +.ui-icon-arrow-1-s { background-position: -64px -32px; } +.ui-icon-arrow-1-sw { background-position: -80px -32px; } +.ui-icon-arrow-1-w { background-position: -96px -32px; } +.ui-icon-arrow-1-nw { background-position: -112px -32px; } +.ui-icon-arrow-2-n-s { background-position: -128px -32px; } +.ui-icon-arrow-2-ne-sw { background-position: -144px -32px; } +.ui-icon-arrow-2-e-w { background-position: -160px -32px; } +.ui-icon-arrow-2-se-nw { background-position: -176px -32px; } +.ui-icon-arrowstop-1-n { background-position: -192px -32px; } +.ui-icon-arrowstop-1-e { background-position: -208px -32px; } +.ui-icon-arrowstop-1-s { background-position: -224px -32px; } +.ui-icon-arrowstop-1-w { background-position: -240px -32px; } +.ui-icon-arrowthick-1-n { background-position: 0 -48px; } +.ui-icon-arrowthick-1-ne { background-position: -16px -48px; } +.ui-icon-arrowthick-1-e { background-position: -32px -48px; } +.ui-icon-arrowthick-1-se { background-position: -48px -48px; } +.ui-icon-arrowthick-1-s { background-position: -64px -48px; } +.ui-icon-arrowthick-1-sw { background-position: -80px -48px; } +.ui-icon-arrowthick-1-w { background-position: -96px -48px; } +.ui-icon-arrowthick-1-nw { background-position: -112px -48px; } +.ui-icon-arrowthick-2-n-s { background-position: -128px -48px; } +.ui-icon-arrowthick-2-ne-sw { background-position: -144px -48px; } +.ui-icon-arrowthick-2-e-w { background-position: -160px -48px; } +.ui-icon-arrowthick-2-se-nw { background-position: -176px -48px; } +.ui-icon-arrowthickstop-1-n { background-position: -192px -48px; } +.ui-icon-arrowthickstop-1-e { background-position: -208px -48px; } +.ui-icon-arrowthickstop-1-s { background-position: -224px -48px; } +.ui-icon-arrowthickstop-1-w { background-position: -240px -48px; } +.ui-icon-arrowreturnthick-1-w { background-position: 0 -64px; } +.ui-icon-arrowreturnthick-1-n { background-position: -16px -64px; } +.ui-icon-arrowreturnthick-1-e { background-position: -32px -64px; } +.ui-icon-arrowreturnthick-1-s { background-position: -48px -64px; } +.ui-icon-arrowreturn-1-w { background-position: -64px -64px; } +.ui-icon-arrowreturn-1-n { background-position: -80px -64px; } +.ui-icon-arrowreturn-1-e { background-position: -96px -64px; } +.ui-icon-arrowreturn-1-s { background-position: -112px -64px; } +.ui-icon-arrowrefresh-1-w { background-position: -128px -64px; } +.ui-icon-arrowrefresh-1-n { background-position: -144px -64px; } +.ui-icon-arrowrefresh-1-e { background-position: -160px -64px; } +.ui-icon-arrowrefresh-1-s { background-position: -176px -64px; } +.ui-icon-arrow-4 { background-position: 0 -80px; } +.ui-icon-arrow-4-diag { background-position: -16px -80px; } +.ui-icon-extlink { background-position: -32px -80px; } +.ui-icon-newwin { background-position: -48px -80px; } +.ui-icon-refresh { background-position: -64px -80px; } +.ui-icon-shuffle { background-position: -80px -80px; } +.ui-icon-transfer-e-w { background-position: -96px -80px; } +.ui-icon-transferthick-e-w { background-position: -112px -80px; } +.ui-icon-folder-collapsed { background-position: 0 -96px; } +.ui-icon-folder-open { background-position: -16px -96px; } +.ui-icon-document { background-position: -32px -96px; } +.ui-icon-document-b { background-position: -48px -96px; } +.ui-icon-note { background-position: -64px -96px; } +.ui-icon-mail-closed { background-position: -80px -96px; } +.ui-icon-mail-open { background-position: -96px -96px; } +.ui-icon-suitcase { background-position: -112px -96px; } +.ui-icon-comment { background-position: -128px -96px; } +.ui-icon-person { background-position: -144px -96px; } +.ui-icon-print { background-position: -160px -96px; } +.ui-icon-trash { background-position: -176px -96px; } +.ui-icon-locked { background-position: -192px -96px; } +.ui-icon-unlocked { background-position: -208px -96px; } +.ui-icon-bookmark { background-position: -224px -96px; } +.ui-icon-tag { background-position: -240px -96px; } +.ui-icon-home { background-position: 0 -112px; } +.ui-icon-flag { background-position: -16px -112px; } +.ui-icon-calendar { background-position: -32px -112px; } +.ui-icon-cart { background-position: -48px -112px; } +.ui-icon-pencil { background-position: -64px -112px; } +.ui-icon-clock { background-position: -80px -112px; } +.ui-icon-disk { background-position: -96px -112px; } +.ui-icon-calculator { background-position: -112px -112px; } +.ui-icon-zoomin { background-position: -128px -112px; } +.ui-icon-zoomout { background-position: -144px -112px; } +.ui-icon-search { background-position: -160px -112px; } +.ui-icon-wrench { background-position: -176px -112px; } +.ui-icon-gear { background-position: -192px -112px; } +.ui-icon-heart { background-position: -208px -112px; } +.ui-icon-star { background-position: -224px -112px; } +.ui-icon-link { background-position: -240px -112px; } +.ui-icon-cancel { background-position: 0 -128px; } +.ui-icon-plus { background-position: -16px -128px; } +.ui-icon-plusthick { background-position: -32px -128px; } +.ui-icon-minus { background-position: -48px -128px; } +.ui-icon-minusthick { background-position: -64px -128px; } +.ui-icon-close { background-position: -80px -128px; } +.ui-icon-closethick { background-position: -96px -128px; } +.ui-icon-key { background-position: -112px -128px; } +.ui-icon-lightbulb { background-position: -128px -128px; } +.ui-icon-scissors { background-position: -144px -128px; } +.ui-icon-clipboard { background-position: -160px -128px; } +.ui-icon-copy { background-position: -176px -128px; } +.ui-icon-contact { background-position: -192px -128px; } +.ui-icon-image { background-position: -208px -128px; } +.ui-icon-video { background-position: -224px -128px; } +.ui-icon-script { background-position: -240px -128px; } +.ui-icon-alert { background-position: 0 -144px; } +.ui-icon-info { background: url(icon_sprite.png) 0 0 no-repeat !important; } +.ui-icon-notice { background-position: -32px -144px; } +.ui-icon-help { background-position: -48px -144px; } +.ui-icon-check { background-position: -64px -144px; } +.ui-icon-bullet { background-position: -80px -144px; } +.ui-icon-radio-off { background-position: -96px -144px; } +.ui-icon-radio-on { background-position: -112px -144px; } +.ui-icon-pin-w { background-position: -128px -144px; } +.ui-icon-pin-s { background-position: -144px -144px; } +.ui-icon-play { background-position: 0 -160px; } +.ui-icon-pause { background-position: -16px -160px; } +.ui-icon-seek-next { background-position: -32px -160px; } +.ui-icon-seek-prev { background-position: -48px -160px; } +.ui-icon-seek-end { background-position: -64px -160px; } +.ui-icon-seek-start { background-position: -80px -160px; } +/* ui-icon-seek-first is deprecated, use ui-icon-seek-start instead */ +.ui-icon-seek-first { background-position: -80px -160px; } +.ui-icon-stop { background-position: -96px -160px; } +.ui-icon-eject { background-position: -112px -160px; } +.ui-icon-volume-off { background-position: -128px -160px; } +.ui-icon-volume-on { background-position: -144px -160px; } +.ui-icon-power { background-position: 0 -176px; } +.ui-icon-signal-diag { background-position: -16px -176px; } +.ui-icon-signal { background-position: -32px -176px; } +.ui-icon-battery-0 { background-position: -48px -176px; } +.ui-icon-battery-1 { background-position: -64px -176px; } +.ui-icon-battery-2 { background-position: -80px -176px; } +.ui-icon-battery-3 { background-position: -96px -176px; } +.ui-icon-circle-plus { background-position: 0 -192px; } +.ui-icon-circle-minus { background-position: -16px -192px; } +.ui-icon-circle-close { background-position: -32px -192px; } +.ui-icon-circle-triangle-e { background-position: -48px -192px; } +.ui-icon-circle-triangle-s { background-position: -64px -192px; } +.ui-icon-circle-triangle-w { background-position: -80px -192px; } +.ui-icon-circle-triangle-n { background-position: -96px -192px; } +.ui-icon-circle-arrow-e { background-position: -112px -192px; } +.ui-icon-circle-arrow-s { background-position: -128px -192px; } +.ui-icon-circle-arrow-w { background-position: -144px -192px; } +.ui-icon-circle-arrow-n { background-position: -160px -192px; } +.ui-icon-circle-zoomin { background-position: -176px -192px; } +.ui-icon-circle-zoomout { background-position: -192px -192px; } +.ui-icon-circle-check { background-position: -208px -192px; } +.ui-icon-circlesmall-plus { background-position: 0 -208px; } +.ui-icon-circlesmall-minus { background-position: -16px -208px; } +.ui-icon-circlesmall-close { background-position: -32px -208px; } +.ui-icon-squaresmall-plus { background-position: -48px -208px; } +.ui-icon-squaresmall-minus { background-position: -64px -208px; } +.ui-icon-squaresmall-close { background-position: -80px -208px; } +.ui-icon-grip-dotted-vertical { background-position: 0 -224px; } +.ui-icon-grip-dotted-horizontal { background-position: -16px -224px; } +.ui-icon-grip-solid-vertical { background-position: -32px -224px; } +.ui-icon-grip-solid-horizontal { background-position: -48px -224px; } +.ui-icon-gripsmall-diagonal-se { background-position: -64px -224px; } +.ui-icon-grip-diagonal-se { background-position: -80px -224px; } + + +/* Misc visuals +----------------------------------*/ + +/* Corner radius */ +.ui-corner-tl { -moz-border-radius-topleft: 3px; -webkit-border-top-left-radius: 3px; border-top-left-radius: 3px; } +.ui-corner-tr { -moz-border-radius-topright: 3px; -webkit-border-top-right-radius: 3px; border-top-right-radius: 3px; } +.ui-corner-bl { -moz-border-radius-bottomleft: 3px; -webkit-border-bottom-left-radius: 3px; border-bottom-left-radius: 3px; } +.ui-corner-br { -moz-border-radius-bottomright: 3px; -webkit-border-bottom-right-radius: 3px; border-bottom-right-radius: 3px; } +.ui-corner-top { -moz-border-radius-topleft: 3px; -webkit-border-top-left-radius: 3px; border-top-left-radius: 3px; -moz-border-radius-topright: 3px; -webkit-border-top-right-radius: 3px; border-top-right-radius: 3px; } +.ui-corner-bottom { -moz-border-radius-bottomleft: 3px; -webkit-border-bottom-left-radius: 3px; border-bottom-left-radius: 3px; -moz-border-radius-bottomright: 3px; -webkit-border-bottom-right-radius: 3px; border-bottom-right-radius: 3px; } +.ui-corner-right { -moz-border-radius-topright: 3px; -webkit-border-top-right-radius: 3px; border-top-right-radius: 3px; -moz-border-radius-bottomright: 3px; -webkit-border-bottom-right-radius: 3px; border-bottom-right-radius: 3px; } +.ui-corner-left { -moz-border-radius-topleft: 3px; -webkit-border-top-left-radius: 3px; border-top-left-radius: 3px; -moz-border-radius-bottomleft: 3px; -webkit-border-bottom-left-radius: 3px; border-bottom-left-radius: 3px; } +.ui-corner-all { -moz-border-radius: 3px; -webkit-border-radius: 3px; border-radius: 3px; } + +/* Overlays */ +.ui-widget-overlay { background: #262b33; opacity: .70;filter:Alpha(Opacity=70); } +.ui-widget-shadow { margin: -8px 0 0 -8px; padding: 8px; background: #000000; opacity: .30;filter:Alpha(Opacity=30); -moz-border-radius: 8px; -webkit-border-radius: 8px; border-radius: 8px; }/* + * jQuery UI Resizable 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Resizable#theming + */ +.ui-resizable { position: relative;} +.ui-resizable-handle { position: absolute; font-size: 0.1px; z-index: 999; display: block;} +.ui-resizable-disabled .ui-resizable-handle, .ui-resizable-autohide .ui-resizable-handle { display: none; } +.ui-resizable-n { cursor: n-resize; height: 7px; width: 100%; top: -5px; left: 0; } +.ui-resizable-s { cursor: s-resize; height: 7px; width: 100%; bottom: -5px; left: 0; } +.ui-resizable-e { cursor: e-resize; width: 7px; right: -5px; top: 0; height: 100%; } +.ui-resizable-w { cursor: w-resize; width: 7px; left: -5px; top: 0; height: 100%; } +.ui-resizable-se { cursor: se-resize; width: 12px; height: 12px; right: 1px; bottom: 1px; } +.ui-resizable-sw { cursor: sw-resize; width: 9px; height: 9px; left: -5px; bottom: -5px; } +.ui-resizable-nw { cursor: nw-resize; width: 9px; height: 9px; left: -5px; top: -5px; } +.ui-resizable-ne { cursor: ne-resize; width: 9px; height: 9px; right: -5px; top: -5px;}/* + * jQuery UI Selectable 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Selectable#theming + */ +.ui-selectable-helper { position: absolute; z-index: 100; border:1px dotted black; } +/* + * jQuery UI Accordion 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Accordion#theming + */ +/* IE/Win - Fix animation bug - #4615 */ +.ui-accordion { width: 100%; } +.ui-accordion .ui-accordion-header { cursor: pointer; position: relative; margin-top: 1px; zoom: 1; } +.ui-accordion .ui-accordion-header, .ui-accordion .ui-accordion-content { -moz-border-radius: 0; -webkit-border-radius: 0; border-radius: 0; } +.ui-accordion .ui-accordion-li-fix { display: inline; } +.ui-accordion .ui-accordion-header-active { border-bottom: 0 !important; } +.ui-accordion .ui-accordion-header a { display: block; font-size: 12px; font-weight: bold; padding: .5em .5em .5em .7em; } +.ui-accordion-icons .ui-accordion-header a { padding-left: 2.2em; } +.ui-accordion .ui-accordion-header .ui-icon { position: absolute; left: .5em; top: 50%; margin-top: -8px; } +.ui-accordion .ui-accordion-content { padding: 1em 2.2em; border-top: 0; margin-top: -2px; position: relative; top: 1px; margin-bottom: 2px; overflow: auto; display: none; zoom: 1; } +.ui-accordion .ui-accordion-content-active { display: block; }/* + * jQuery UI Autocomplete 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Autocomplete#theming + */ +.ui-autocomplete { + position: absolute; cursor: default; z-index: 3; + -moz-border-radius: 0; + -webkit-border-radius: 0; + border-radius: 0; + -moz-box-shadow: 0 1px 5px rgba(0,0,0,0.3); + -webkit-box-shadow: 0 1px 5px rgba(0,0,0,0.3); + box-shadow: 0 1px 5px rgba(0,0,0,0.3); +} + +/* workarounds */ +* html .ui-autocomplete { width:1px; } /* without this, the menu expands to 100% in IE6 */ + +/* + * jQuery UI Menu 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Menu#theming + */ +.ui-menu { + list-style:none; + padding: 2px; + margin: 0; + display:block; + float: left; +} +.ui-menu .ui-menu { + margin-top: -3px; +} +.ui-menu .ui-menu-item { + margin:0; + padding: 0; + zoom: 1; + float: left; + clear: left; + width: 100%; +} +.ui-menu .ui-menu-item a { + text-decoration:none; + display:block; + padding:.2em .4em; + line-height:1.5; + zoom:1; +} +.ui-menu .ui-menu-item a.ui-state-hover, +.ui-menu .ui-menu-item a.ui-state-active { + font-weight: normal; + margin: -1px; + background: #5f83b9; + color: #FFFFFF; + text-shadow: 0px 1px 1px #234386; + border-color: #466086; + -moz-border-radius: 0; + -webkit-border-radius: 0; + border-radius: 0; +} +/* + * jQuery UI Button 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Button#theming + */ +.ui-button { display: inline-block; position: relative; padding: 0; margin-right: .1em; text-decoration: none !important; cursor: pointer; text-align: center; zoom: 1; overflow: visible; -webkit-user-select: none; -moz-user-select: none; user-select: none; } /* the overflow property removes extra width in IE */ +.ui-button-icon-only { width: 2.2em; } /* to make room for the icon, a width needs to be set here */ +button.ui-button-icon-only { width: 2.4em; } /* button elements seem to need a little more width */ +.ui-button-icons-only { width: 3.4em; } +button.ui-button-icons-only { width: 3.7em; } + +/* button animation properties */ +.ui-button { + -webkit-transition: all 250ms ease-in-out; + -moz-transition: all 250ms ease-in-out; + -o-transition: all 250ms ease-in-out; + transition: all 250ms ease-in-out; +} + +/*states*/ +.ui-button.ui-state-hover { + -moz-box-shadow: 0 0 8px rgba(0, 0, 0, 0.15), 0 1px 0 rgba(255,255,255,0.8) inset; + -webkit-box-shadow: 0 0 8px rgba(0, 0, 0, 0.15), 0 1px 0 rgba(255,255,255,0.8) inset; + box-shadow: 0 0 8px rgba(0, 0, 0, 0.15), 0 1px 0 rgba(255,255,255,0.8) inset; +} +.ui-button.ui-state-focus { + outline: none; + color: #1c4257; + border-color: #7096ab; + background: #ededed url(bg_fallback.png) 0 -50px repeat-x; /* Old browsers */ + background: -moz-linear-gradient(top, #b9e0f5 0%, #92bdd6 100%); /* FF3.6+ */ + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%,#b9e0f5), color-stop(100%,#92bdd6)); /* Chrome,Safari4+ */ + background: -webkit-linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* Chrome10+,Safari5.1+ */ + background: -o-linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* Opera11.10+ */ + background: -ms-linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* IE10+ */ + background: linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* W3C */ + -moz-box-shadow: 0 0 8px rgba(0, 0, 0, 0.15), 0 1px 0 rgba(255,255,255,0.8) inset; + -webkit-box-shadow: 0 0 8px rgba(0, 0, 0, 0.15), 0 1px 0 rgba(255,255,255,0.8) inset; + box-shadow: 0 0 8px rgba(0, 0, 0, 0.15), 0 1px 0 rgba(255,255,255,0.8) inset; +} + +/*button text element */ +.ui-button .ui-button-text { display: block; line-height: 1.4; font-size: 14px; font-weight: bold; text-shadow: 0 1px 0 rgba(255, 255, 255, 0.6); } +.ui-button-text-only .ui-button-text { padding: .4em 1em; } +.ui-button-icon-only .ui-button-text, .ui-button-icons-only .ui-button-text { padding: .4em; text-indent: -9999999px; } +.ui-button-text-icon-primary .ui-button-text, .ui-button-text-icons .ui-button-text { padding: .4em 1em .4em 2.1em; } +.ui-button-text-icon-secondary .ui-button-text, .ui-button-text-icons .ui-button-text { padding: .4em 2.1em .4em 1em; } +.ui-button-text-icons .ui-button-text { padding-left: 2.1em; padding-right: 2.1em; } +/* no icon support for input elements, provide padding by default */ +input.ui-button, .ui-widget-content input.ui-button { font-size: 14px; font-weight: bold; text-shadow: 0 1px 0 rgba(255, 255, 255, 0.6); padding: 0 1em !important; height: 33px; } +/*remove submit button internal padding in Firefox*/ +input.ui-button::-moz-focus-inner { + border: 0; + padding: 0; +} +/* fix webkits handling of the box model */ +@media screen and (-webkit-min-device-pixel-ratio:0) { + input.ui-button { + height: 31px !important; + } +} + + +/*button icon element(s) */ +.ui-button-icon-only .ui-icon, .ui-button-text-icon-primary .ui-icon, .ui-button-text-icon-secondary .ui-icon, .ui-button-text-icons .ui-icon, .ui-button-icons-only .ui-icon { position: absolute; top: 50%; margin-top: -8px; } +.ui-button-icon-only .ui-icon { left: 50%; margin-left: -8px; } +.ui-button-text-icon-primary .ui-button-icon-primary, .ui-button-text-icons .ui-button-icon-primary, .ui-button-icons-only .ui-button-icon-primary { left: .5em; } +.ui-button-text-icon-secondary .ui-button-icon-secondary, .ui-button-text-icons .ui-button-icon-secondary, .ui-button-icons-only .ui-button-icon-secondary { right: .5em; } +.ui-button-text-icons .ui-button-icon-secondary, .ui-button-icons-only .ui-button-icon-secondary { right: .5em; } + +/*button sets*/ +.ui-buttonset { margin-right: 7px; } +.ui-buttonset .ui-button { margin-left: 0; margin-right: -.3em; } +.ui-buttonset .ui-button.ui-state-active { color: #1c4257; border-color: #7096ab; } +.ui-buttonset .ui-button.ui-state-active { + background: #ededed url(bg_fallback.png) 0 -50px repeat-x; /* Old browsers */ + background: -moz-linear-gradient(top, #b9e0f5 0%, #92bdd6 100%); /* FF3.6+ */ + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%,#b9e0f5), color-stop(100%,#92bdd6)); /* Chrome,Safari4+ */ + background: -webkit-linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* Chrome10+,Safari5.1+ */ + background: -o-linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* Opera11.10+ */ + background: -ms-linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* IE10+ */ + background: linear-gradient(top, #b9e0f5 0%,#92bdd6 100%); /* W3C */ + -webkit-box-shadow: none; + -moz-box-shadow: none; + box-shadow: none; +} + +/* workarounds */ +button.ui-button::-moz-focus-inner { border: 0; padding: 0; } /* reset extra padding in Firefox */ +/* + * jQuery UI Dialog 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Dialog#theming + */ +.ui-dialog { position: absolute; padding: 0; width: 300px; overflow: hidden; } +.ui-dialog { + -webkit-box-shadow: 0 2px 12px rgba(0,0,0,0.6); + -moz-box-shadow: 0 2px 12px rgba(0,0,0,0.6); + box-shadow: 0 2px 12px rgba(0,0,0,0.6); +} +.ui-dialog .ui-dialog-titlebar { padding: 0.7em 1em 0.6em 1em; position: relative; border: none; border-bottom: 1px solid #979797; -moz-border-radius: 0; -webkit-border-radius: 0; border-radius: 0; } +.ui-dialog .ui-dialog-title { float: left; margin: .1em 16px .2em 0; font-size: 14px; text-shadow: 0 1px 0 rgba(255,255,255,0.5); } +.ui-dialog .ui-dialog-titlebar-close { position: absolute; right: .8em; top: 55%; width: 16px; margin: -10px 0 0 0; padding: 0; height: 16px; } +.ui-dialog .ui-dialog-titlebar-close span { display: block; margin: 1px; background: url(icon_sprite.png) 0 -16px no-repeat; } +.ui-dialog .ui-dialog-titlebar-close:hover span { background-position: -16px -16px; } +.ui-dialog .ui-dialog-titlebar-close:hover, .ui-dialog .ui-dialog-titlebar-close:focus { padding: 0; border: 0; } +.ui-dialog .ui-dialog-content { position: relative; border: 0; padding: .5em 1em; background: none; overflow: auto; zoom: 1; } +.ui-dialog .ui-dialog-buttonpane { text-align: left; border-width: 1px 0 0 0; background-image: none; margin: .5em 0 0 0; padding: .3em 1em .5em .4em; } +.ui-dialog .ui-dialog-buttonpane .ui-dialog-buttonset { float: right; } +.ui-dialog .ui-dialog-buttonpane button { margin: .5em .4em .5em 0; cursor: pointer; } +.ui-dialog .ui-resizable-se { width: 14px; height: 14px; right: 3px; bottom: 3px; } +.ui-draggable .ui-dialog-titlebar { cursor: move; } +/* + * jQuery UI Slider 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Slider#theming + */ +.ui-slider { position: relative; text-align: left; background: #d7d7d7; z-index: 1; } +.ui-slider { -moz-box-shadow: 0 1px 2px rgba(0,0,0,0.5) inset; -webkit-box-shadow: 0 1px 2px rgba(0,0,0,0.5) inset; box-shadow: 0 1px 2px rgba(0,0,0,0.5) inset; } +.ui-slider .ui-slider-handle { background: url(slider_handles.png) 0px -23px no-repeat; position: absolute; z-index: 2; width: 23px; height: 23px; cursor: default; border: none; outline: none; -moz-box-shadow: none; -webkit-box-shadow: none; box-shadow: none; } +.ui-slider .ui-state-hover, .ui-slider .ui-state-active { background-position: 0 0; } +.ui-slider .ui-slider-range { background: #a3cae0; position: absolute; z-index: 1; font-size: .7em; display: block; border: 0; background-position: 0 0; } +.ui-slider .ui-slider-range { -moz-box-shadow: 0 1px 2px rgba(17,35,45,0.6) inset; -webkit-box-shadow: 0 1px 2px rgba(17,35,45,0.6) inset; box-shadow: 0 1px 2px rgba(17,35,45,0.6) inset; } + + +.ui-slider-horizontal { height: 5px; } +.ui-slider-horizontal .ui-slider-handle { top: -8px; margin-left: -13px; } +.ui-slider-horizontal .ui-slider-range { top: 0; height: 100%; } +.ui-slider-horizontal .ui-slider-range-min { left: 0; } +.ui-slider-horizontal .ui-slider-range-max { right: 0; } + +.ui-slider-vertical { width: 5px; height: 100px; } +.ui-slider-vertical .ui-slider-handle { left: -8px; margin-left: 0; margin-bottom: -13px; } +.ui-slider-vertical .ui-slider-range { left: 0; width: 100%; } +.ui-slider-vertical .ui-slider-range-min { bottom: 0; } +.ui-slider-vertical .ui-slider-range-max { top: 0; }/* + * jQuery UI Tabs 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Tabs#theming + */ +.ui-tabs { position: relative; zoom: 1; border: 0; background: transparent; } /* position: relative prevents IE scroll bug (element with position: relative inside container with overflow: auto appear as "fixed") */ +.ui-tabs .ui-tabs-nav { margin: 0; padding: 0; background: transparent; border-width: 0 0 1px 0; } +.ui-tabs .ui-tabs-nav { + -moz-border-radius: 0; + -webkit-border-radius: 0; + border-radius: 0; +} +.ui-tabs .ui-tabs-nav li { list-style: none; float: left; position: relative; top: 1px; margin: 0 .2em 1px 0; border-bottom: 0 !important; padding: 0; white-space: nowrap; } +.ui-tabs .ui-tabs-nav li a { float: left; padding: .5em 1em; text-decoration: none; font-size: 12px; font-weight: bold; text-shadow: 0 1px 0 rgba(255,255,255,0.5); } +.ui-tabs .ui-tabs-nav li.ui-tabs-selected { margin-bottom: 0; padding-bottom: 1px; background: #fff; border-color: #B6B6B6; } +.ui-tabs .ui-tabs-nav li.ui-tabs-selected a, .ui-tabs .ui-tabs-nav li.ui-state-disabled a, .ui-tabs .ui-tabs-nav li.ui-state-processing a { cursor: text; outline: none; } +.ui-tabs .ui-tabs-nav li a, .ui-tabs.ui-tabs-collapsible .ui-tabs-nav li.ui-tabs-selected a { cursor: pointer; } /* first selector in group seems obsolete, but required to overcome bug in Opera applying cursor: text overall if defined elsewhere... */ +.ui-tabs .ui-tabs-panel { display: block; border-width: 0 1px 1px 1px; padding: 1em 1.4em; background: none; } +.ui-tabs .ui-tabs-panel { background: #FFF; + -moz-border-radius: 0; + -webkit-border-radius: 0; + border-radius: 0; +} +.ui-tabs .ui-tabs-hide { display: none !important; } +/* + * jQuery UI Datepicker 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Datepicker#theming + */ +.ui-datepicker { width: 17em; padding: 0; display: none; border-color: #DDDDDD; } +.ui-datepicker { + -moz-box-shadow: 0 4px 8px rgba(0,0,0,0.5); + -webkit-box-shadow: 0 4px 8px rgba(0,0,0,0.5); + box-shadow: 0 4px 8px rgba(0,0,0,0.5); +} +.ui-datepicker .ui-datepicker-header { position:relative; padding:.35em 0; border: none; border-bottom: 1px solid #B6B6B6; -moz-border-radius: 0; -webkit-border-radius: 0; border-radius: 0; } +.ui-datepicker .ui-datepicker-prev, .ui-datepicker .ui-datepicker-next { position:absolute; top: 6px; width: 1.8em; height: 1.8em; } +.ui-datepicker .ui-datepicker-prev-hover, .ui-datepicker .ui-datepicker-next-hover { border: 1px none; } +.ui-datepicker .ui-datepicker-prev { left:2px; } +.ui-datepicker .ui-datepicker-next { right:2px; } +.ui-datepicker .ui-datepicker-prev span { background-position: 0px -32px !important; } +.ui-datepicker .ui-datepicker-next span { background-position: -16px -32px !important; } +.ui-datepicker .ui-datepicker-prev-hover span { background-position: 0px -48px !important; } +.ui-datepicker .ui-datepicker-next-hover span { background-position: -16px -48px !important; } +.ui-datepicker .ui-datepicker-prev span, .ui-datepicker .ui-datepicker-next span { display: block; position: absolute; left: 50%; margin-left: -8px; top: 50%; margin-top: -8px; background: url(icon_sprite.png) no-repeat; } +.ui-datepicker .ui-datepicker-title { margin: 0 2.3em; line-height: 1.8em; text-align: center; font-size: 12px; text-shadow: 0 1px 0 rgba(255,255,255,0.6); } +.ui-datepicker .ui-datepicker-title select { font-size:1em; margin:1px 0; } +.ui-datepicker select.ui-datepicker-month-year {width: 100%;} +.ui-datepicker select.ui-datepicker-month, +.ui-datepicker select.ui-datepicker-year { width: 49%;} +.ui-datepicker table {width: 100%; font-size: .9em; border-collapse: collapse; margin:0 0 .4em; } +.ui-datepicker th { padding: .7em .3em; text-align: center; font-weight: bold; border: 0; } +.ui-datepicker td { border: 0; padding: 1px; } +.ui-datepicker td span, .ui-datepicker td a { display: block; padding: .2em; text-align: right; text-decoration: none; } +.ui-datepicker .ui-datepicker-buttonpane { background-image: none; margin: .7em 0 0 0; padding:0 .2em; border-left: 0; border-right: 0; border-bottom: 0; } +.ui-datepicker .ui-datepicker-buttonpane button { float: right; margin: .5em .2em .4em; cursor: pointer; padding: .2em .6em .3em .6em; width:auto; overflow:visible; } +.ui-datepicker .ui-datepicker-buttonpane button.ui-datepicker-current { float:left; } +.ui-datepicker table .ui-state-highlight { border-color: #5F83B9; } +.ui-datepicker table .ui-state-hover { background: #5F83B9; color: #FFF; font-weight: bold; text-shadow: 0 1px 1px #234386; -webkit-box-shadow: 0 0px 0 rgba(255,255,255,0.6) inset; -moz-box-shadow: 0 0px 0 rgba(255,255,255,0.6) inset; box-shadow: 0 0px 0 rgba(255,255,255,0.6) inset; border-color: #5F83B9; } +.ui-datepicker-calendar .ui-state-default { background: transparent; border-color: #FFF; } +.ui-datepicker-calendar .ui-state-active { background: #5F83B9; border-color: #5F83B9; color: #FFF; font-weight: bold; text-shadow: 0 1px 1px #234386; } + +/* with multiple calendars */ +.ui-datepicker.ui-datepicker-multi { width:auto; } +.ui-datepicker-multi .ui-datepicker-group { float:left; } +.ui-datepicker-multi .ui-datepicker-group table { width:95%; margin:0 auto .4em; } +.ui-datepicker-multi-2 .ui-datepicker-group { width:50%; } +.ui-datepicker-multi-3 .ui-datepicker-group { width:33.3%; } +.ui-datepicker-multi-4 .ui-datepicker-group { width:25%; } +.ui-datepicker-multi .ui-datepicker-group-last .ui-datepicker-header { border-left-width:0; } +.ui-datepicker-multi .ui-datepicker-group-middle .ui-datepicker-header { border-left-width:0; } +.ui-datepicker-multi .ui-datepicker-buttonpane { clear:left; } +.ui-datepicker-row-break { clear:both; width:100%; } + +/* RTL support */ +.ui-datepicker-rtl { direction: rtl; } +.ui-datepicker-rtl .ui-datepicker-prev { right: 2px; left: auto; } +.ui-datepicker-rtl .ui-datepicker-next { left: 2px; right: auto; } +.ui-datepicker-rtl .ui-datepicker-prev:hover { right: 1px; left: auto; } +.ui-datepicker-rtl .ui-datepicker-next:hover { left: 1px; right: auto; } +.ui-datepicker-rtl .ui-datepicker-buttonpane { clear:right; } +.ui-datepicker-rtl .ui-datepicker-buttonpane button { float: left; } +.ui-datepicker-rtl .ui-datepicker-buttonpane button.ui-datepicker-current { float:right; } +.ui-datepicker-rtl .ui-datepicker-group { float:right; } +.ui-datepicker-rtl .ui-datepicker-group-last .ui-datepicker-header { border-right-width:0; border-left-width:1px; } +.ui-datepicker-rtl .ui-datepicker-group-middle .ui-datepicker-header { border-right-width:0; border-left-width:1px; } + +/* IE6 IFRAME FIX (taken from datepicker 1.5.3 */ +.ui-datepicker-cover { + display: none; /*sorry for IE5*/ + display/**/: block; /*sorry for IE5*/ + position: absolute; /*must have*/ + z-index: -1; /*must have*/ + filter: mask(); /*must have*/ + top: -4px; /*must have*/ + left: -4px; /*must have*/ + width: 200px; /*must have*/ + height: 200px; /*must have*/ +}/* + * jQuery UI Progressbar 1.8.7 + * + * Copyright 2010, AUTHORS.txt (http://jqueryui.com/about) + * Dual licensed under the MIT or GPL Version 2 licenses. + * http://jquery.org/license + * + * http://docs.jquery.com/UI/Progressbar#theming + */ +.ui-progressbar { height: 12px; text-align: left; background: #FFF url(progress_bar.gif) 0 -14px repeat-x; } +.ui-progressbar .ui-progressbar-value {margin: -1px; height:100%; background: url(progress_bar.gif) 0 0 repeat-x; } + +/* Extra Input Field Styling */ +.ui-form textarea, .ui-form input:not([type="submit"]):not([type="button"]):not([type="checkbox"]):not([type="radio"]):not([type="file"]):not([type="range"]) { + padding: 3px; + -webkit-border-radius: 2px; + -moz-border-radius: 2px; + border-radius: 2px; + border: 1px solid #cecece; + outline: none; + -webkit-box-shadow: 0 1px 3px rgba(0,0,0,0.1) inset, 0 1px 0 rgba(255,255,255,0.2); + -moz-box-shadow: 0 1px 3px rgba(0,0,0,0.1) inset, 0 1px 0 rgba(255,255,255,0.2); + box-shadow: 0 1px 3px rgba(0,0,0,0.1) inset, 0 1px 0 rgba(255,255,255,0.2); + -webkit-transition: all 250ms ease-in-out; + -moz-transition: all 250ms ease-in-out; + -o-transition: all 250ms ease-in-out; + transition: all 250ms ease-in-out; +} +.ui-form textarea:hover, .ui-form input:not([type="submit"]):not([type="button"]):not([type="checkbox"]):not([type="radio"]):not([type="file"]):not([type="range"]):hover { + border: 1px solid #bdbdbd; + -webkit-box-shadow: 0 1px 3px rgba(0,0,0,0.2) inset, 0 1px 0 rgba(255,255,255,0.2); + -moz-box-shadow: 0 1px 3px rgba(0,0,0,0.2) inset, 0 1px 0 rgba(255,255,255,0.2); + box-shadow: 0 1px 3px rgba(0,0,0,0.2) inset, 0 1px 0 rgba(255,255,255,0.2); +} +.ui-form textarea:focus, .ui-form input:not([type="submit"]):not([type="button"]):not([type="checkbox"]):not([type="radio"]):not([type="file"]):not([type="range"]):focus { + border: 1px solid #95bdd4; + -webkit-box-shadow: 0 2px 3px rgba(161,202,226,0.5) inset, 0 1px 0 rgba(255,255,255,0.2); + -moz-box-shadow: 0 2px 3px rgba(161,202,226,0.5) inset, 0 1px 0 rgba(255,255,255,0.2); + box-shadow: 0 2px 3px rgba(161,202,226,0.5) inset, 0 1px 0 rgba(255,255,255,0.2); +} From 93401ef9887b297343d5bf33b4fbf39f13090894 Mon Sep 17 00:00:00 2001 From: Dmitriy Zaporozhets Date: Fri, 17 Aug 2012 09:26:59 +0300 Subject: [PATCH 06/46] Few css improvements --- app/views/commits/_commit_box.html.haml | 6 +++--- app/views/issues/_form.html.haml | 12 ++++++------ app/views/issues/index.html.haml | 5 +++-- app/views/milestones/_form.html.haml | 6 +++--- 4 files changed, 15 insertions(+), 14 deletions(-) diff --git a/app/views/commits/_commit_box.html.haml b/app/views/commits/_commit_box.html.haml index 52f03ba7..506f4e09 100644 --- a/app/views/commits/_commit_box.html.haml +++ b/app/views/commits/_commit_box.html.haml @@ -5,10 +5,10 @@ %span.btn.disabled.grouped %i.icon-comment = @notes_count - = link_to patch_project_commit_path(@project, @commit.id), class: "btn small grouped" do + = link_to patch_project_commit_path(@project, @commit.id), class: "btn small grouped" do %i.icon-download-alt - Get Patch - = link_to tree_project_ref_path(@project, @commit.id), class: "browse-button primary grouped" do + Get Patch + = link_to tree_project_ref_path(@project, @commit.id), class: "browse-button primary grouped" do %strong Browse Code » %h3.commit-title.page_title = gfm @commit.title diff --git a/app/views/issues/_form.html.haml b/app/views/issues/_form.html.haml index 6139f3d4..1b67eabd 100644 --- a/app/views/issues/_form.html.haml +++ b/app/views/issues/_form.html.haml @@ -1,5 +1,5 @@ %div.issue-form-holder - %h3= @issue.new_record? ? "New Issue" : "Edit Issue ##{@issue.id}" + %h3.page_title= @issue.new_record? ? "New Issue" : "Edit Issue ##{@issue.id}" = form_for [@project, @issue], remote: request.xhr? do |f| -if @issue.errors.any? .alert-message.block-message.error @@ -9,26 +9,26 @@ .issue_form_box .issue_title .clearfix - = f.label :title do + = f.label :title do %strong= "Subject *" .input = f.text_field :title, maxlength: 255, class: "xxlarge" .issue_middle_block .issue_assignee - = f.label :assignee_id do + = f.label :assignee_id do %i.icon-user Assign to .input= f.select(:assignee_id, @project.users.all.collect {|p| [ p.name, p.id ] }, { include_blank: "Select a user" }) .issue_milestone - = f.label :milestone_id do + = f.label :milestone_id do %i.icon-time Milestone .input= f.select(:milestone_id, @project.milestones.active.all.collect {|p| [ p.title, p.id ] }, { include_blank: "Select milestone" }) .issue_description .clearfix - = f.label :label_list do - %i.icon-tag + = f.label :label_list do + %i.icon-tag Labels .input = f.text_field :label_list, maxlength: 2000, class: "xxlarge" diff --git a/app/views/issues/index.html.haml b/app/views/issues/index.html.haml index 8876d24d..a6836fd4 100644 --- a/app/views/issues/index.html.haml +++ b/app/views/issues/index.html.haml @@ -7,6 +7,7 @@ .span5 - if can? current_user, :write_issue, @project = link_to new_project_issue_path(@project), class: "right btn small", title: "New Issue", remote: true do + %i.icon-plus New Issue = form_tag search_project_issues_path(@project), method: :get, remote: true, id: "issue_search_form", class: :right do = hidden_field_tag :project_id, @project.id, { id: 'project_id' } @@ -21,7 +22,7 @@ .issues_bulk_update.hide - = form_tag bulk_update_project_issues_path(@project), method: :post do + = form_tag bulk_update_project_issues_path(@project), method: :post do %span.update_issues_text Update selected issues with   .left = select_tag('update[status]', options_for_select(['open', 'closed']), prompt: "Status") @@ -53,7 +54,7 @@ = select_tag(:milestone_id, options_from_collection_for_select([unassigned_filter] + @project.milestones.order("id desc").all, "id", "title", params[:milestone_id]), prompt: "Milestone") = hidden_field_tag :f, params[:f] .clearfix - + %ul#issues-table.unstyled.issues_table = render "issues" diff --git a/app/views/milestones/_form.html.haml b/app/views/milestones/_form.html.haml index daae58fe..1cd08ac3 100644 --- a/app/views/milestones/_form.html.haml +++ b/app/views/milestones/_form.html.haml @@ -1,4 +1,4 @@ -%h3= @milestone.new_record? ? "New Milestone" : "Edit Milestone ##{@milestone.id}" +%h3.page_title= @milestone.new_record? ? "New Milestone" : "Edit Milestone ##{@milestone.id}" .back_link = link_to project_milestones_path(@project) do ← To milestones @@ -17,12 +17,12 @@ = f.label :title, "Title", class: "control-label" .controls = f.text_field :title, maxlength: 255, class: "input-xlarge" - %p.help-block Required + %p.hint Required .control-group = f.label :description, "Description", class: "control-label" .controls = f.text_area :description, maxlength: 2000, class: "input-xlarge", rows: 10 - %p.help-block Markdown is enabled. + %p.hint Markdown is enabled. .span6 .control-group = f.label :due_date, "Due Date", class: "control-label" From 3f9749dd86874c4b6dcbdad7d3cd0dc7406a20b8 Mon Sep 17 00:00:00 2001 From: randx Date: Sat, 18 Aug 2012 23:53:58 +0300 Subject: [PATCH 07/46] Fix branch selector. Few css fixed also --- app/assets/stylesheets/sections/login.scss | 1 + app/assets/stylesheets/sections/notes.scss | 2 +- app/assets/stylesheets/themes/ui_modern.scss | 7 +++++-- app/views/notes/_show.html.haml | 2 +- app/views/refs/_tree.html.haml | 2 -- app/views/refs/_tree_commit.html.haml | 2 +- 6 files changed, 9 insertions(+), 7 deletions(-) diff --git a/app/assets/stylesheets/sections/login.scss b/app/assets/stylesheets/sections/login.scss index 3726d9f0..5b8763cf 100644 --- a/app/assets/stylesheets/sections/login.scss +++ b/app/assets/stylesheets/sections/login.scss @@ -27,6 +27,7 @@ body.login-page{ -moz-border-radius-topright: 5px; border-top-left-radius: 5px; border-top-right-radius: 5px; + margin-bottom:0px; } .login-box input.text.bottom{ diff --git a/app/assets/stylesheets/sections/notes.scss b/app/assets/stylesheets/sections/notes.scss index b498277d..c846e6a7 100644 --- a/app/assets/stylesheets/sections/notes.scss +++ b/app/assets/stylesheets/sections/notes.scss @@ -48,7 +48,7 @@ p { color:$style_color; } .note-author { color: $style_color;} - .note-title { margin-left:50px; padding-top: 5px;} + .note-title { padding-top: 5px;} .avatar { margin-top:3px; } diff --git a/app/assets/stylesheets/themes/ui_modern.scss b/app/assets/stylesheets/themes/ui_modern.scss index 29c857e5..3fd9bb5e 100644 --- a/app/assets/stylesheets/themes/ui_modern.scss +++ b/app/assets/stylesheets/themes/ui_modern.scss @@ -37,6 +37,7 @@ * */ .app_logo { + width:160px; a { h1 { opacity: 0.7; @@ -56,7 +57,7 @@ .separator { width: 1px; height: 40px; - margin: 0 9px; + margin: 0 10px; overflow: hidden; background: #222; border-left: 1px solid #333; @@ -73,6 +74,7 @@ background:none; margin-left:8px; font-size: 13px; + font-weight:bold; line-height: 19px; color:#ccc; &:hover { @@ -81,6 +83,7 @@ border: none; box-shadow:none; text-shadow: 0 -1px 0 #000000; + border-left: 1px solid #333; } } @@ -115,7 +118,7 @@ .project_name { line-height:34px; font-size:22px; - color:#fff; + color:#ccc; text-shadow: 0 1px 1px #111; } diff --git a/app/views/notes/_show.html.haml b/app/views/notes/_show.html.haml index bdb00546..3412e4eb 100644 --- a/app/views/notes/_show.html.haml +++ b/app/views/notes/_show.html.haml @@ -1,5 +1,5 @@ %li{id: dom_id(note), class: "note"} - = image_tag gravatar_icon(note.author.email), class: "avatar" + = image_tag gravatar_icon(note.author.email), class: "avatar s32" %div.note-author %strong= note.author_name = link_to "##{dom_id(note)}", name: dom_id(note) do diff --git a/app/views/refs/_tree.html.haml b/app/views/refs/_tree.html.haml index 6f8175da..c231c407 100644 --- a/app/views/refs/_tree.html.haml +++ b/app/views/refs/_tree.html.haml @@ -51,8 +51,6 @@ :javascript $(function(){ - $('select#branch').selectmenu({style:'popup', width:200}); - $('select#tag').selectmenu({style:'popup', width:200}); $('.project-refs-select').chosen(); history.pushState({ path: this.path }, '', "#{@history_path}"); diff --git a/app/views/refs/_tree_commit.html.haml b/app/views/refs/_tree_commit.html.haml index a5681aa2..1bcf1a7e 100644 --- a/app/views/refs/_tree_commit.html.haml +++ b/app/views/refs/_tree_commit.html.haml @@ -1,3 +1,3 @@ - if tm - %strong= link_to "[#{tm.user_name}]", project_team_member_path(@project, tm) + = link_to "[#{tm.user_name}]", project_team_member_path(@project, tm) = link_to_gfm truncate(content_commit.title, length: tm ? 30 : 50), project_commit_path(@project, content_commit.id), class: "tree-commit-link" From 7530fa9decae2bb3d8622e1a40a080deb831af9f Mon Sep 17 00:00:00 2001 From: randx Date: Sun, 19 Aug 2012 00:08:20 +0300 Subject: [PATCH 08/46] Restored margin for text in notes --- app/assets/stylesheets/sections/notes.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/assets/stylesheets/sections/notes.scss b/app/assets/stylesheets/sections/notes.scss index c846e6a7..30587ef5 100644 --- a/app/assets/stylesheets/sections/notes.scss +++ b/app/assets/stylesheets/sections/notes.scss @@ -48,7 +48,7 @@ p { color:$style_color; } .note-author { color: $style_color;} - .note-title { padding-top: 5px;} + .note-title { margin-left:45px; padding-top: 5px;} .avatar { margin-top:3px; } From 8eaead6a01739fee37d7b29b78decc2e8a4b89a3 Mon Sep 17 00:00:00 2001 From: randx Date: Sun, 19 Aug 2012 00:45:46 +0300 Subject: [PATCH 09/46] Text message for mergeing MR --- app/assets/javascripts/merge_requests.js | 1 + app/assets/stylesheets/sections/merge_requests.scss | 5 +++++ app/views/merge_requests/_show.html.haml | 3 ++- app/views/merge_requests/show/_mr_accept.html.haml | 3 +++ 4 files changed, 11 insertions(+), 1 deletion(-) diff --git a/app/assets/javascripts/merge_requests.js b/app/assets/javascripts/merge_requests.js index 4b155192..0ab6f6e2 100644 --- a/app/assets/javascripts/merge_requests.js +++ b/app/assets/javascripts/merge_requests.js @@ -112,6 +112,7 @@ var MergeRequest = { already_cannot_be_merged: function(){ $(".automerge_widget").hide(); + $(".merge_in_progress").hide(); $(".automerge_widget.already_cannot_be_merged").show(); } } diff --git a/app/assets/stylesheets/sections/merge_requests.scss b/app/assets/stylesheets/sections/merge_requests.scss index 34f43acf..ec84a64e 100644 --- a/app/assets/stylesheets/sections/merge_requests.scss +++ b/app/assets/stylesheets/sections/merge_requests.scss @@ -94,3 +94,8 @@ li.merge_request { padding-bottom: 2px; } } + +.merge_in_progress { + @extend .padded; + @extend .append-bottom-10; +} diff --git a/app/views/merge_requests/_show.html.haml b/app/views/merge_requests/_show.html.haml index 072bf259..f1b3fa9f 100644 --- a/app/views/merge_requests/_show.html.haml +++ b/app/views/merge_requests/_show.html.haml @@ -33,7 +33,8 @@ }); $(".edit_merge_request").live("ajax:beforeSend", function() { - $(this).replaceWith('#{image_tag "ajax_loader.gif"}'); + $('.can_be_merged').hide(); + $('.merge_in_progress').show(); }) }) diff --git a/app/views/merge_requests/show/_mr_accept.html.haml b/app/views/merge_requests/show/_mr_accept.html.haml index efd47af0..f2422885 100644 --- a/app/views/merge_requests/show/_mr_accept.html.haml +++ b/app/views/merge_requests/show/_mr_accept.html.haml @@ -40,3 +40,6 @@ .alert.alert-info %strong This merge request already can not be merged. Try to reload page. + .merge_in_progress.hide + %span.cgray Merge is in progress. Please wait. Page will be automatically reloaded.   + = image_tag "ajax_loader.gif" From 05c86fb0fc5b9e90889a96f9de301756ce3d5552 Mon Sep 17 00:00:00 2001 From: randx Date: Sun, 19 Aug 2012 10:20:42 +0300 Subject: [PATCH 10/46] Specify charlock holmes version in installation docs --- doc/installation.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/doc/installation.md b/doc/installation.md index 524c8e86..5611c3b3 100644 --- a/doc/installation.md +++ b/doc/installation.md @@ -134,7 +134,7 @@ Permissions: # 4. Install gitlab and configuration. Check status configuration. - sudo gem install charlock_holmes + sudo gem install charlock_holmes --version '0.6.8' sudo pip install pygments sudo gem install bundler cd /home/gitlab From d4059ac966e26c4f384c53ca14319c56619fef78 Mon Sep 17 00:00:00 2001 From: randx Date: Sun, 19 Aug 2012 10:58:10 +0300 Subject: [PATCH 11/46] Move graph_commit under gitlab module --- app/controllers/projects_controller.rb | 4 +- lib/gitlab/graph_commit.rb | 183 +++++++++++++++++++++++++ lib/graph_commit.rb | 181 ------------------------ 3 files changed, 185 insertions(+), 183 deletions(-) create mode 100644 lib/gitlab/graph_commit.rb delete mode 100644 lib/graph_commit.rb diff --git a/app/controllers/projects_controller.rb b/app/controllers/projects_controller.rb index b596a5a6..bd7f811e 100644 --- a/app/controllers/projects_controller.rb +++ b/app/controllers/projects_controller.rb @@ -1,4 +1,4 @@ -require File.join(Rails.root, 'lib', 'graph_commit') +require Rails.root.join('lib', 'gitlab', 'graph_commit') class ProjectsController < ApplicationController before_filter :project, except: [:index, :new, :create] @@ -78,7 +78,7 @@ class ProjectsController < ApplicationController end def graph - @days_json, @commits_json = GraphCommit.to_graph(project) + @days_json, @commits_json = Gitlab::GraphCommit.to_graph(project) end def destroy diff --git a/lib/gitlab/graph_commit.rb b/lib/gitlab/graph_commit.rb new file mode 100644 index 00000000..b9859d79 --- /dev/null +++ b/lib/gitlab/graph_commit.rb @@ -0,0 +1,183 @@ +require "grit" + +module Gitlab + class GraphCommit + attr_accessor :time, :space + attr_accessor :refs + + def self.to_graph(project) + @repo = project.repo + commits = Grit::Commit.find_all(@repo, nil, {max_count: 650}) + + ref_cache = {} + + commits.map! {|c| GraphCommit.new(Commit.new(c))} + commits.each { |commit| commit.add_refs(ref_cache, @repo) } + + days = GraphCommit.index_commits(commits) + @days_json = days.compact.collect{|d| [d.day, d.strftime("%b")] }.to_json + @commits_json = commits.map(&:to_graph_hash).to_json + + return @days_json, @commits_json + end + + # Method is adding time and space on the + # list of commits. As well as returns date list + # corelated with time set on commits. + # + # @param [Array] comits to index + # + # @return [Array] list of commit dates corelated with time on commits + def self.index_commits(commits) + days, heads = [], [] + map = {} + + commits.reverse.each_with_index do |c,i| + c.time = i + days[i] = c.committed_date + map[c.id] = c + heads += c.refs unless c.refs.nil? + end + + heads.select!{|h| h.is_a? Grit::Head or h.is_a? Grit::Remote} + # sort heads so the master is top and current branches are closer + heads.sort! do |a,b| + if a.name == "master" + -1 + elsif b.name == "master" + 1 + else + b.commit.committed_date <=> a.commit.committed_date + end + end + + @_reserved = {} + days.each_index do |i| + @_reserved[i] = [] + end + + heads.each do |h| + if map.include? h.commit.id then + place_chain(map[h.commit.id], map) + end + end + days + end + + # Add space mark on commit and its parents + # + # @param [GraphCommit] the commit object. + # @param [Hash] map of commits + def self.place_chain(commit, map, parent_time = nil) + leaves = take_left_leaves(commit, map) + if leaves.empty? then + return + end + space = find_free_space(leaves.last.time..leaves.first.time) + leaves.each{|l| l.space = space} + # and mark it as reserved + min_time = leaves.last.time + parents = leaves.last.parents.collect + parents.each do |p| + if map.include? p.id then + parent = map[p.id] + if parent.time < min_time then + min_time = parent.time + end + end + end + if parent_time.nil? then + max_time = leaves.first.time + else + max_time = parent_time - 1 + end + mark_reserved(min_time..max_time, space) + # Visit branching chains + leaves.each do |l| + parents = l.parents.collect + .select{|p| map.include? p.id and map[p.id].space == 0} + for p in parents + place_chain(map[p.id], map, l.time) + end + end + end + + def self.mark_reserved(time_range, space) + for day in time_range + @_reserved[day].push(space) + end + end + + def self.find_free_space(time_range) + reserved = [] + for day in time_range + reserved += @_reserved[day] + end + space = 1 + while reserved.include? space do + space += 1 + end + space + end + + # Takes most left subtree branch of commits + # which don't have space mark yet. + # + # @param [GraphCommit] the commit object. + # @param [Hash] map of commits + # + # @return [Array] list of branch commits + def self.take_left_leaves(commit, map) + leaves = [] + leaves.push(commit) if commit.space == 0 + while true + parent = commit.parents.collect + .select{|p| map.include? p.id and map[p.id].space == 0} + if parent.count == 0 then + return leaves + else + commit = map[parent.first.id] + leaves.push(commit) + end + end + end + + + def initialize(commit) + @_commit = commit + @time = -1 + @space = 0 + end + + def method_missing(m, *args, &block) + @_commit.send(m, *args, &block) + end + + def to_graph_hash + h = {} + h[:parents] = self.parents.collect do |p| + [p.id,0,0] + end + h[:author] = Gitlab::Encode.utf8(author.name) + h[:time] = time + h[:space] = space + h[:refs] = refs.collect{|r|r.name}.join(" ") unless refs.nil? + h[:id] = sha + h[:date] = date + h[:message] = Gitlab::Encode.utf8(message) + h[:login] = author.email + h + end + + def add_refs(ref_cache, repo) + if ref_cache.empty? + repo.refs.each do |ref| + ref_cache[ref.commit.id] ||= [] + ref_cache[ref.commit.id] << ref + end + end + @refs = ref_cache[@_commit.id] if ref_cache.include?(@_commit.id) + @refs ||= [] + end + end +end diff --git a/lib/graph_commit.rb b/lib/graph_commit.rb deleted file mode 100644 index e08a8fad..00000000 --- a/lib/graph_commit.rb +++ /dev/null @@ -1,181 +0,0 @@ -require "grit" - -class GraphCommit - attr_accessor :time, :space - attr_accessor :refs - - def self.to_graph(project) - @repo = project.repo - commits = Grit::Commit.find_all(@repo, nil, {max_count: 650}) - - ref_cache = {} - - commits.map! {|c| GraphCommit.new(Commit.new(c))} - commits.each { |commit| commit.add_refs(ref_cache, @repo) } - - days = GraphCommit.index_commits(commits) - @days_json = days.compact.collect{|d| [d.day, d.strftime("%b")] }.to_json - @commits_json = commits.map(&:to_graph_hash).to_json - - return @days_json, @commits_json - end - - # Method is adding time and space on the - # list of commits. As well as returns date list - # corelated with time set on commits. - # - # @param [Array] comits to index - # - # @return [Array] list of commit dates corelated with time on commits - def self.index_commits(commits) - days, heads = [], [] - map = {} - - commits.reverse.each_with_index do |c,i| - c.time = i - days[i] = c.committed_date - map[c.id] = c - heads += c.refs unless c.refs.nil? - end - - heads.select!{|h| h.is_a? Grit::Head or h.is_a? Grit::Remote} - # sort heads so the master is top and current branches are closer - heads.sort! do |a,b| - if a.name == "master" - -1 - elsif b.name == "master" - 1 - else - b.commit.committed_date <=> a.commit.committed_date - end - end - - @_reserved = {} - days.each_index do |i| - @_reserved[i] = [] - end - - heads.each do |h| - if map.include? h.commit.id then - place_chain(map[h.commit.id], map) - end - end - days - end - - # Add space mark on commit and its parents - # - # @param [GraphCommit] the commit object. - # @param [Hash] map of commits - def self.place_chain(commit, map, parent_time = nil) - leaves = take_left_leaves(commit, map) - if leaves.empty? then - return - end - space = find_free_space(leaves.last.time..leaves.first.time) - leaves.each{|l| l.space = space} - # and mark it as reserved - min_time = leaves.last.time - parents = leaves.last.parents.collect - parents.each do |p| - if map.include? p.id then - parent = map[p.id] - if parent.time < min_time then - min_time = parent.time - end - end - end - if parent_time.nil? then - max_time = leaves.first.time - else - max_time = parent_time - 1 - end - mark_reserved(min_time..max_time, space) - # Visit branching chains - leaves.each do |l| - parents = l.parents.collect - .select{|p| map.include? p.id and map[p.id].space == 0} - for p in parents - place_chain(map[p.id], map, l.time) - end - end - end - - def self.mark_reserved(time_range, space) - for day in time_range - @_reserved[day].push(space) - end - end - - def self.find_free_space(time_range) - reserved = [] - for day in time_range - reserved += @_reserved[day] - end - space = 1 - while reserved.include? space do - space += 1 - end - space - end - - # Takes most left subtree branch of commits - # which don't have space mark yet. - # - # @param [GraphCommit] the commit object. - # @param [Hash] map of commits - # - # @return [Array] list of branch commits - def self.take_left_leaves(commit, map) - leaves = [] - leaves.push(commit) if commit.space == 0 - while true - parent = commit.parents.collect - .select{|p| map.include? p.id and map[p.id].space == 0} - if parent.count == 0 then - return leaves - else - commit = map[parent.first.id] - leaves.push(commit) - end - end - end - - - def initialize(commit) - @_commit = commit - @time = -1 - @space = 0 - end - - def method_missing(m, *args, &block) - @_commit.send(m, *args, &block) - end - - def to_graph_hash - h = {} - h[:parents] = self.parents.collect do |p| - [p.id,0,0] - end - h[:author] = Gitlab::Encode.utf8(author.name) - h[:time] = time - h[:space] = space - h[:refs] = refs.collect{|r|r.name}.join(" ") unless refs.nil? - h[:id] = sha - h[:date] = date - h[:message] = Gitlab::Encode.utf8(message) - h[:login] = author.email - h - end - - def add_refs(ref_cache, repo) - if ref_cache.empty? - repo.refs.each do |ref| - ref_cache[ref.commit.id] ||= [] - ref_cache[ref.commit.id] << ref - end - end - @refs = ref_cache[@_commit.id] if ref_cache.include?(@_commit.id) - @refs ||= [] - end -end From d656cb74f7e1becdb6ed686a7d36da48ab521030 Mon Sep 17 00:00:00 2001 From: randx Date: Sun, 19 Aug 2012 11:36:37 +0300 Subject: [PATCH 12/46] Headless gem added --- Gemfile | 1 + Gemfile.lock | 2 ++ features/support/env.rb | 5 +++++ spec/spec_helper.rb | 6 ++++++ 4 files changed, 14 insertions(+) diff --git a/Gemfile b/Gemfile index 26e4e195..e8b0b244 100644 --- a/Gemfile +++ b/Gemfile @@ -101,6 +101,7 @@ group :development, :test do gem "rspec-rails" gem "capybara" gem "capybara-webkit" + gem "headless" gem "autotest" gem "autotest-rails" gem "pry" diff --git a/Gemfile.lock b/Gemfile.lock index 57ad1935..b23bc47c 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -186,6 +186,7 @@ GEM railties (~> 3.0) hashery (1.4.0) hashie (1.2.0) + headless (0.3.1) hike (1.2.1) httparty (0.8.3) multi_json (~> 1.0) @@ -398,6 +399,7 @@ DEPENDENCIES grape (~> 0.2.1) grit! haml-rails + headless httparty jquery-rails (= 2.0.2) jquery-ui-rails (= 0.5.0) diff --git a/features/support/env.rb b/features/support/env.rb index b47349c8..496f23f9 100644 --- a/features/support/env.rb +++ b/features/support/env.rb @@ -39,3 +39,8 @@ rescue NameError end Cucumber::Rails::Database.javascript_strategy = :truncation + +require 'headless' + +headless = Headless.new +headless.start diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index f87c9a50..5c0bb618 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -12,6 +12,7 @@ require 'webmock/rspec' require 'factories' require 'monkeypatch' require 'email_spec' +require 'headless' # Requires supporting ruby files with custom matchers and macros, etc, # in spec/support/ and its subdirectories. @@ -30,6 +31,11 @@ RSpec.configure do |config| # instead of true. config.use_transactional_fixtures = false + config.before :all do + headless = Headless.new + headless.start + end + config.before :each, type: :integration do DeviseSessionMock.disable end From 8e3a2def8af612689ddc830a3eab9e463b8bc205 Mon Sep 17 00:00:00 2001 From: randx Date: Sun, 19 Aug 2012 11:44:48 +0300 Subject: [PATCH 13/46] Project /lib cleanup --- lib/color.rb | 31 ------------------------------- lib/tasks/dev/repo.rake | 26 -------------------------- lib/tasks/dev/tests.rake | 10 ---------- lib/tasks/dev/user.sh | 7 ------- 4 files changed, 74 deletions(-) delete mode 100644 lib/color.rb delete mode 100644 lib/tasks/dev/repo.rake delete mode 100644 lib/tasks/dev/tests.rake delete mode 100755 lib/tasks/dev/user.sh diff --git a/lib/color.rb b/lib/color.rb deleted file mode 100644 index 4723804e..00000000 --- a/lib/color.rb +++ /dev/null @@ -1,31 +0,0 @@ -module Color - extend self - - def colorize(text, color_code) - "\033[#{color_code}#{text}\033[0m" - end - - def red(text) - colorize(text, "31m") - end - - def green(text) - colorize(text, "32m") - end - - def yellow(text) - colorize(text, "93m") - end - - def command(string) - `#{string}` - if $?.to_i > 0 - puts red " == #{string} - FAIL" - puts red " == Error during configure" - exit - else - puts green " == #{string} - OK" - end - end -end - diff --git a/lib/tasks/dev/repo.rake b/lib/tasks/dev/repo.rake deleted file mode 100644 index 7b389a55..00000000 --- a/lib/tasks/dev/repo.rake +++ /dev/null @@ -1,26 +0,0 @@ -namespace :dev do - desc "Prepare for development (run dev_user.sh first)" - task :repos => :environment do - key = `sudo -u gitlabdev -H cat /home/gitlabdev/.ssh/id_rsa.pub` - raise "\n *** Run ./lib/tasks/dev/user.sh first *** \n" if key.empty? - Key.create(:user_id => User.first, :key => key, :title => "gitlabdev") - - puts "\n *** Clone diaspora from github" - `sudo -u gitlabdev -H sh -c "cd /home/gitlabdev; git clone git://github.com/diaspora/diaspora.git /home/gitlabdev/diaspora"` - - puts "\n *** Push diaspora source to gitlab" - `sudo -u gitlabdev -H sh -c "cd /home/gitlabdev/diaspora; git remote add local git@localhost:diaspora.git; git push local master; git push local --tags; git checkout -b api origin/api; git push local api; git checkout -b heroku origin/heroku; git push local heroku"` - - puts "\n *** Clone rails from github" - `sudo -u gitlabdev -H sh -c "cd /home/gitlabdev; git clone git://github.com/rails/rails.git /home/gitlabdev/rails"` - - puts "\n *** Push rails source to gitlab" - `sudo -u gitlabdev -H sh -c "cd /home/gitlabdev/rails; git remote add local git@localhost:ruby_on_rails.git; git push local master; git push local --tags"` - - puts "\n *** Clone rubinius from github" - `sudo -u gitlabdev -H sh -c "cd /home/gitlabdev; git clone git://github.com/rubinius/rubinius.git /home/gitlabdev/rubinius"` - - puts "\n *** Push rubinius source to gitlab" - `sudo -u gitlabdev -H sh -c "cd /home/gitlabdev/rubinius; git remote add local git@localhost:rubinius.git; git push local master; git push local --tags"` - end -end diff --git a/lib/tasks/dev/tests.rake b/lib/tasks/dev/tests.rake deleted file mode 100644 index d5c0e75e..00000000 --- a/lib/tasks/dev/tests.rake +++ /dev/null @@ -1,10 +0,0 @@ -namespace :dev do - desc "DEV | Run cucumber and rspec" - task :tests do - ["cucumber", "rspec spec"].each do |cmd| - puts "Starting to run #{cmd}..." - system("export DISPLAY=:99.0 && bundle exec #{cmd}") - raise "#{cmd} failed!" unless $?.exitstatus == 0 - end - end -end diff --git a/lib/tasks/dev/user.sh b/lib/tasks/dev/user.sh deleted file mode 100755 index d6b20df2..00000000 --- a/lib/tasks/dev/user.sh +++ /dev/null @@ -1,7 +0,0 @@ -sudo adduser \ - --gecos 'gitlab dev user' \ - --disabled-password \ - --home /home/gitlabdev \ - gitlabdev - -sudo -i -u gitlabdev -H sh -c "ssh-keygen -t rsa" From ccdea8b80dc3b7642c5d99e8f5dc8e80ed1e19c2 Mon Sep 17 00:00:00 2001 From: randx Date: Sun, 19 Aug 2012 12:14:36 +0300 Subject: [PATCH 14/46] Fix dashboard random test fail --- features/step_definitions/dashboard_steps.rb | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/features/step_definitions/dashboard_steps.rb b/features/step_definitions/dashboard_steps.rb index bd9d69b8..90c3a69c 100644 --- a/features/step_definitions/dashboard_steps.rb +++ b/features/step_definitions/dashboard_steps.rb @@ -93,11 +93,11 @@ end Given /^I have assigned issues$/ do project1 = Factory :project, :path => "project1", - :code => "TEST1" + :code => "gitlabhq_1" project2 = Factory :project, :path => "project2", - :code => "TEST2" + :code => "gitlabhq_2" project1.add_access(@user, :read, :write) project2.add_access(@user, :read, :write) @@ -116,11 +116,11 @@ end Given /^I have authored merge requests$/ do project1 = Factory :project, :path => "project1", - :code => "TEST1" + :code => "gitlabhq_1" project2 = Factory :project, :path => "project2", - :code => "TEST2" + :code => "gitlabhq_2" project1.add_access(@user, :read, :write) project2.add_access(@user, :read, :write) From 335b3ed19791d2bf5aea9c95f46af925ebc80412 Mon Sep 17 00:00:00 2001 From: Jakub Jirutka Date: Mon, 20 Aug 2012 12:58:03 +0200 Subject: [PATCH 15/46] fix condition in find_for_ldap_auth --- app/models/user.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/models/user.rb b/app/models/user.rb index 1b53bda2..b0538cb7 100644 --- a/app/models/user.rb +++ b/app/models/user.rb @@ -91,7 +91,7 @@ class User < ActiveRecord::Base provider = auth.provider name = auth.info.name.force_encoding("utf-8") email = auth.info.email.downcase unless auth.info.email.nil? - raise OmniAuth::Error, "LDAP accounts must provide an uid and email address" if uid.nil? and email.nil? + raise OmniAuth::Error, "LDAP accounts must provide an uid and email address" if uid.nil? or email.nil? if @user = User.find_by_extern_uid_and_provider(uid, provider) @user From b2b88b2ff2a3044efa259d06c630dc903d0851bd Mon Sep 17 00:00:00 2001 From: randx Date: Mon, 20 Aug 2012 22:51:37 +0300 Subject: [PATCH 16/46] Added font for head panel. Major restyle for header --- .../fonts/korolev-medium-compressed.otf | Bin 0 -> 28796 bytes app/assets/images/logo_dark.png | Bin 0 -> 2858 bytes app/assets/images/logo_white.png | Bin 0 -> 1681 bytes app/assets/stylesheets/main.scss | 2 ++ app/assets/stylesheets/sections/header.scss | 20 ++++++----- app/assets/stylesheets/themes/ui_basic.scss | 32 ++++++++++++++++++ app/assets/stylesheets/themes/ui_mars.scss | 12 ++----- app/assets/stylesheets/themes/ui_modern.scss | 28 ++++++--------- app/views/layouts/_head.html.haml | 1 - config/application.rb | 3 ++ 10 files changed, 63 insertions(+), 35 deletions(-) create mode 100644 app/assets/fonts/korolev-medium-compressed.otf create mode 100644 app/assets/images/logo_dark.png create mode 100644 app/assets/images/logo_white.png diff --git a/app/assets/fonts/korolev-medium-compressed.otf b/app/assets/fonts/korolev-medium-compressed.otf new file mode 100644 index 0000000000000000000000000000000000000000..e3817cec85765374e2853696aca7c889a849fcd0 GIT binary patch literal 28796 zcmdqJ2V4}#|2IB+$L=0;dZKtJN3w?{_JY_e78DCAidYe)S!n`_*n7d=3sz7>V(-0R z?=^-5EKy?;LV_VCb7V31f6sy?zi+Xez2kXFwKX}HeEuc!Z@2SqCham^#7@;D#=fc9Nn{rK}RLdc|^{`EcQ zRY_<>h~Rgs81Y2@d3f$l1fG=;{SL1)L&~)Jjp+5LmyoORFXz0A?a%FA`YHMuxYiRLzl%SV z(fw?fa4QAR@93v!H5ETA&M<-yKO!I`I-#fVhKR&LcS@Ip>xIR!@EGm*1CQUA+Cs?Q zwuE$jcOxb-HV$cbRaf$^8RR^gyn;X1HPD^s&$QxhLf7~2Z;QKVofeiOZA;QRB9Y!D zX+7TCza;HITuD+%S|p`OYDrom70JGmv`p%gt0idzDNk5Qnv!OMu_SFGW}!n#+KGI4 zZ`qIDTaLI2{lP=Q!4a*SCX6Xb3&d0Sr6jE*CS8@1w4Q|N8kD3Rh^KBt?~%b#?w!XD9}ynsUf@?*EvzZgJig+I#=$r+u)n9}?+E<&toVBtBqSu%3)5IM zEIKMau3l(#y|JU;wdX@a-HVAO9Y{2ZA&DfGj3mR!2oev4u0cXcO~jtWgERo#NeFVd zLw!e*U=oE?XEIjH9ZuqKRUhys;ba`1ha#?r>vkjpKlhL77oL>z&|GLCv=mwiy#;@vj}QQH%!Dw;p{^h@Sm;iMk|9vZ*<>EZC>zKZ z?|C==eQFj`!(hhfiiEZ>*Y?_zNm|$sePbu|sw>GFa*sR`+Cj*^qz?%oy`b7NP$m=- zB2bV4WT4;=N{C21T8%a&UQu36?K<&S*Tg3P(@MNDt(oq$|z-;QsD0FUeWy% zcXzGk=I?73w3;yvvS$LP@RpK4`@MH5LjKGxP8VJLaPM8}S+i%&pEV=onftSD&)lEh zcybvJ55i1AaBT*(BmT5!dqe|f5H+R zU<*#L25t*gU=7?BOpsY~*h350LQ673XbXGj%`HM0Dzt}|t`h{ZUeJ+bK~FXb64@-s z;2R}d1OrLM$Zfk&n(PwFlih-u>=i1IJ%S7QTyQ4)1vhd)_=Fr3J|%|)g&Y>FX1yKF1aYwCl`czUmP?1~_JV}<&fjq^i`GwGv{2~k{uZ2MJt1yK8CIrKugu#;#odf&; z#_QlM5k?bHh$Rh$7*buBP8tclk!}Vg;yMXFZ4c5Ti3Rx@3XOnlxwNokZTg|Toccp_)eQVpmJnt5CebXFtTsF$b1;760;UpLx+1E1=W z&B(_9+y^hT=a700@)NGR!YBVg%7H&Mph+g-Dc;i`@AX5v5&ZXc;)eHELVI0>$*4b; zScN$98MsnONI+dYjW&~w@aC)VtPd#x4?a}AMuaG9oka~?Wm+Z=kYw; z6X0=nqYueI|AG9_pfb=XXF$e#+yN^rz#Xs(>quo`CusTs{OSPg#GHWR1?n9LdfX6m z1R(EVJof`1UZRiU03S5mLwoO`zNe7KQ^e2FQ$5FfJV6gf6yDJr@i_2gF=*yIPbal? zi%C^|3h2zm9HuAgZU?j?)pZQ{ypX>ouFC_wJ?&vjgOlK(D6>lfeIfg~`x`4rmY8v3R`e9k{-6%z!R&y;}qNIk^7)GxRKZ z)*dm}&yN9J73QOt-vu7+f@l2Po~y+f_B^u7UPLnNHtpFS;-TqtJo0mW?o|@ELEMx$ zzSnQA;~#_T`M<&(@U^x^6St*e#F54&>B@*dhN4|}Bo)BZ58$@*AAsA|$I#@>W&RGG zp^NWOYL~}$TK(UG`zh|D zJ_h$yAEOTB^$xvY-}>U7-F_HTrN93ZuqM=&2tq>)Pz>+^dLr8YKA&HfugS1q*I z3y^lx(i`wpLY^Fi745$x|ccZ9rcCXFH!>e-ES2Q7b3tsXzL$3%MDuOSS;h|)wJ z*`Hzt(g>U^hgLm9SjwJ?&;@TV2TuNk@DnZUh&Gf1U7HbZKv`#$Qb~EOh2_zbKQM#w zMEhM(PYy!S{v*OtXr~KmQxSe@zmBjnsB=MUGZA_~?k;HUC4`>P5Esndu}tcdsKA>9$RRm6LDBHRF~T~R|WLSBO_%05E)Ddg^o zvTqP}L`z(eKNH~w`)TxwJf{UY|3LUD>a-x|YlI$G@*{>q>wVy$_!M*|WN3QLk zpw4`R{OzA;b$+7BNP!gcF{ku|ms7xVo^FD)l{7?~E2Gp;2s?t$mGQ=h2z$aWRVIGW z&dSi|M+gI;%~es)ZwPrk)luqK%tJqgUDrd|OoTr6%Lx5ZPd&)!I{fUXXtO86e1uke zA;OOKCkTB&X+zvEKxjp64RN2}@j=c;+WkhL^eNKaz@*0+ri8q^|U5!(f&4CJzfZZz-*J(;|(jg zhmd)lxRZ&nk^M`^LvMeH zko$+8koYsmMvt6+xU-4$Li#qsQufsdKgIoCqz-JY7pVtYdJ}il+?&+LTY5t)RLs8h z;8h>wk3{$>?6(i{|BSGn7B+xflA%$5Aap^^$)Ms1Lhjp>(W(=_er9!-}KD zA@_dhuV6u0sNp$67qlu%>rJxIswYUhX?Hr>_aMzJJqu;GVa`+8o`=v=3me(9F#CZO zBZTil=wrWxuqXQHijYhNg#Pv%gaM=?dRp#((1XIV6s+M05yA0Muqg-L)neAqp9uW> z3+^GM!9n{Pp(mvbnQjk>pfuBRPZvrF3TCXHn4ty<5kiu%LD(al6K?AqFnh16>!kD5 z_0@0=dWgr_09Bc^xpcedOv+%eV{%>AE6(o zpQc}~-=;sJzpDRQpQr!R!O5YrgNH*K2VaMQ4&xo>IIMI?aoFc@+aXV+Vhyp0=q>gZ zhl)|+Br!=`D`tr2#M@$ym@C>ONh&XOm8MBYq&(>l*&w^fRpn0d2sugKB_EKF$!Fw? z@^$&H{7BA`pUdCNKg)0AKMcem8Qcxk40Q~ihGvE~25&=GgP);oUTUGXsm*F8R-5H#)KqY(IF=r129@GE#=?J5s_KJT}Af6aSq-dFRUn)lPZ zm;P%>870%TzLfW*ydUMgDDOjg56a(O{1&cD{7nvc%TK5~19fl4oy@=FQQx=bL)%KD zHRrhApu}OVHN3aueI4)Vct6K`IrMS7hvWU5*1I`qJsR(^cz?xvD?RGCj;l;uordld z*LYj&GkA}|`wQM%@VYfLxi# zbs6our+N6|`+Ci{bI;EGI=^Db`5agIxZ-}CdvWf=CCz(t-_1QY_uJfSbDyo(yfydL z+*1n(^T9*zskxu#UYh%8e*Xzt&wVua(6Z*Cxqs%~nfqq$nYmx)UYY0qN(+A|DJej5 zymsz)d6=z*Un1l_Ri}AP?lX0o&*UDH-}_nfi~p!||L5;K`HKzwRp0;ft9O<$6_S`! zVj1%^pUFMuUoGQ5zxvlU^#9(||Gb6&Gdcg!8}Qi+_d8FZ-Fv`OZkyZwxB2+5TKT`{ zMgIR^>aSkyKfn6-{_p>r9Nf2LX};y1hb{J$|NW4AYTjpD;^PcH%HU%RKEn8Kz3m@8 z`+suhKcdE^E&MpaI)*pCajnH#^CxqQe}IT^FWWp*9!Pb!E?kffS16xz#H8A4fq{k0L8fO53&MS39JHE z18ab_z&c<(kPMiS^BKl}si+&XCG_4aaJ>>(1*`_v0BeDDzZQu@Y7q|!92Oa9yaoOQ z?Dp>j0tkQ(&;t&D1jv8^a0Dn|1WEy=0TWOLa01E#5VWd<{vyh9qA@lCL4jJV-JRlFWl7^B~DQNHPzS%!4HJAjv#PG7pl> zgCz4H$vj9h50cCidf0y#dIEkxFCfsK3p>q)o#w($b77~su+v=FX)f$E7gEf_H-EnR z;12N7YZZWxVCw?)0Tn0$-U5FDc6&ZJln)N&gG2e?P(C=64-VynL;2uPJ~)&Q4&{SG z`QT7KIFt_#<%2`{;7~p|ln)N&gG2e?P(C=64-VynL;2uPJ~)&Q4&{SG`QS!AxRDQT zUIy(z0jMhgbp@cV0Mr$Lx&lyF0O|@rT>+>o0CfeRt^m{(fVu)u zR{-h?KwSZ-D*$x`psoPa6@a<|P*(uz3P4=}s4D<<1)#0~)D?ib0#H`~>Iy(z0jMhg zbp@cV0Mr$Lx&lyF0O|@rT>+>o0CfeRu7EUy>j{{055@Wf!DbI3s3<33cNv{-+>oA{!;LQ6d{9 zvQZ)%C9+W>8zr()ViQJGeBF<)`tdbCE8q@P0;&MDfx19_U&d? zNCwWK9u;|ufVaS(fZd)AOV5U-XT#F7Vd>eh^lVsqHY`0GmYxku&xWOE!_u>1>DjRK zY*>0WEIk{Ro()UShNWl2(z9Xd*|79%Sb8=rJsXyu4NK34rDwy^vtjAku=H$LdbZFK zXa%$e+5lcaJHQ9%40Hjy0^I;#U^v<<60C2+zpL=?D*U?&|E^-B%-1{lYA0Xov;yux zC7=pW8>kD^2l&e83glf0tO8a8Yk;-DI$%AJ44gw`ukVt?_Z(6e}(@375e*E=lzkh}P z{+0Hvk^7V9sO33od5&71a6X=YsdS(B52(9dp5d?5hCy%D5G92Py$o0KQIM3#g5B9jzb8CG`;3 z$2DI)ZwNF7HsN|Rum#u(qyYPY1HeJx5O5B-1>6Si0C$0VzcvL0R%t?;5!>EBo`Kv3k%7Gh2+9Qa$zC4u#j9>NG>cS7Z#EW z3(19rcJqxG^8&#(bd( z&w#oo6YAa!Yyq|cDZqZ<0B{gE1e`;@TZnH1 zcYwRVJ>Wj@0C)&|0Xzb7fXBcS;3@D7_y$nXrXt`i@F!r$x-|AGLZX?FXeK0@35jMx zqM49rCM22(iDp8gnUH8EB$^3{WW)@cjsumNGJ;u z%7TQlAfYTsC<_wGf`qalp)5!!3lhqLgt8!^EJ!E|63T*vvLK->NGJ;u%7TQlAfYTs zC<_wGf`qalp)5!v3)0AfG_oL#EJz~@(#V1|vLKBtNFxi<$bvMoAdM_YBMZ{Vf;6(g z{Va{be_Z?EYaV>PgRgb`pI%q^@2ny4^#Gkdg0F}+4-)&8IS*?&fFk3ZkOXyP!9%{whqGGf%e6n$_XDp1-)~SB*k-?NFRuif$-uP9;?gd|*Oz4SU4eT`NjnfHz@T-ko+Q}%#ciK0` zH}^x>b&-i*8*-6c#z~1=_yXS^-_x_O*C7Xc+I$ICZn4W##7^oU?8vT2!mtP1i$r11 zwJ-LB3s^aJKi!-}9?zw9yOTzHInY-XyPc}zX9i_;ke46&v0!fr zc9($04d7HNsH;KFgSr+Pb*(_%BhdI2=;{Kx{wSdCxl_Y^+L3`p4MXQU&D`|Gn7ETjGu%y^PeMrI<&boVlGPxNtMMNXZ&<%b9t;n zoAJ}3&DfWvQCKY9(x8&lSsA}F8ij&JpKOzI&k6bM6Qlcl3fccCkBYd?~B(r5s8(()hwp)`%$2xll^`71Lizqra3!c`1!@ z3Hn+AbaDyHns$pC{f>~|r)a$uk~V0(5U~5X3ex<1jS(7L8|gau8L>mNF4CMYMp!|8 zq&au6;|@RU+=DJ+XCr=kjbA#AUk+pwNkPjvcS@0S_;HRrLYi}-l*R=KJ38~xHqHxK z^pdJNBas8mCM+ ztuP3CJ_E69U9WLXCkz*cqs_(pNhooLp8nT;Cpzt^E@I^HU}B03j);f;6zAum0KQA0 zmw&+#pc}>eQYs^M2MpTNQi3bu&(KyKtUDBhnq{$U1VD+zUR z2&g3(>Vv}`eL3rd7-51i4aZ@Y3u}c`VYhHtI4@ig?g@{DZ-noJJfT1+)JeM1I%gak zs;aB2Yp83Z^Vapy1?UFpLUa+jSlwjZOx+^g3f)Fss&1!lA2#2d#sQ;yx*XkeY{9YV z^?F%f8k=w|`bzpb*oafS4JS}PR6j}|gKao7^t1H~uoY*sey4t~{;>YM{)+y-K1csj z|62dM192#YjX5q3ZVr_kYB)4@Xywq}p|gX(L!d*5L!?8T!(@k<4i_9$vAkF}DkLr@ zI5fQD*x2Z(==kt4V}m1NMutX5Mg|9mhK5JQw~Gu8#iMp1vEk#w+eHtLjtU>$J}f%E z_-TjW(6RC19YROOhK`LK77;$aLufE^cX*${Tf6rD=sGq!IKHEHw_|bD9kr@E7FXTz zea=qrQ$8Q6_tEnDeDvr;OFMta);T0NwoClTh_LW3#qa2%mFx1pT$kdvb^TDJ+eb9} zYE=1Zjr9HaKFW80Uq=ruR}U>$kB@S7j2hlkdtJ}sclR7IHfnfq?AXYN;IZ+3?~C`= ziucxv_x`B3zjojML(BZ%7x53p+2)9dVE>P@_t7fo^FB|XkDq}J0U8?uVn;>|571f_ zP~573k6RT`%#MEAtNRU)4IUTX??VIng^diy+3JyT1GFatK0Fzy6&U#b<-x^u2Wu}1 zF0M4>{nJn_cj$-Wp<3HR-)9KZuERdM=E@ST-Nkuou7}}T)#1fehriD`?0stZhw6uG zd53@W=!3G1_>gS`xHMAZ(#Ya>jMU1FEPltR4;e>)#M;rrW5dIvB7&pBMutXcbVX>* zjQA)|1l}C^{_Rm(t|%>6)JM6(qlQOoFN`jJd-TVGiFsc6_5R>c${`W?n5i% z-WQ20QJ1)nvd3!`#J|rI|M4?0Vywo9v0PooYONYu+^Vr3w`y!LYZA0qCumBM@Sy<- z@6;txdy@F!$po#yg!eBGuE#r*e}C@%U;h^X|Gxa+pPT>3Bl7>{EPsdh|EIvav-4VS zg@Z7pvMXutO3E292ttpBJ_UUpnMlxE5mFd`=SZz(XwY{y5~ zILL^Tyg1H^ZitK`IMYZ*3+Us~SCKdY2e&XrCF=;r6C_4J$AyEdIN`{rfvpLSJK`)R zjss!M6$56RB<2U_J|!5cV9Y~;1z{e+VLIL$V>my7kZ}CR9R$NNj74z_2`3_Pa0_Qu zgd~EquIL5H6oQ@*=aO-NjLaf96iKEL95BXdUL5HY@ah>jvqUgt#>qtt>CjmU3w8W| zuvntU_(fQu=O;uD5Okq9z9_6Q;&3ZY;*z6+kV-LF#lBM5Mr>Pwh1FKnQ0j7(N2$!DDdFdmNNu62`@wAj*O+B%GO*zfSCD2bKC>ey}TBl?R&#L;5BI8|IG9u$v>XT*!*L-DCtD9KWN zsk!7QZIWH(#RK91>8|oX{8U`2=4QCz8IM#4%<=EMAu;UEJG{+;3SEwWHLWj{Q z^fzM@<6`4+51uEQ=aKf8EcuoWd@fCFB4TJq0E{xTgvP#bD+$j zGDpgsE_1QW%`y+m@;0-R+0|U%+{`?|{K)*+{L=iRxxifLBDfe_>bNv?Y3UN~GS6kH%UYMsE{9!C zx%}wztIMAiXa%SYaI3=03V9WNt)N!aRg^1M zsaUIGgNn^6dR6RHv3tcn6$e$k;M&@ClqU!VX=?GZ z_*f=eW?JT3mRiR_Giz5FHhI=u+?ULE&1eI3q zMnfpxnr==VzcD_3d}7?#P2*EiHf`Ffm=>)<-j*wOoB#S+&$xz=>s>&q_yk$Mv>WOQAz{QDs#Jx%Q^CA-Jkz*^8Oozb(X*FcdAYQ=rNP0D=V)jvvLb8%qWLb=jspr`#@X9W}_R_L03u#|<7bV%s&vbnLQis5xQ4{9s>;>M7U1*!|V3OV_?r zSU>r3@Qudf;^xd4r~JSsLXQ@7RNF;&P*}N<=0p3&^{}cvwQ2$*Q|~LRr+hJE+i}G- zkJWWywV<48N+;G+Qmt>DMAIqObEf(E^&79O+47Cp<9#+|Qs3*18usw5uXK~Ur37Rq zD*g#$gSuH-UgeClzLwJ|c&M;!2Y&etE*s<0bd23(^Ua%2Ejn?@!Wzoo^}goe(W`rX zMeQdyInv{sEh$S@Zc*y06QJn}?y`324-~cBZga@6Emy2);Ju7(Cr+lvc2v-C|A<(~ zIhNXL2hgJ0SLsl^#^xbk+RQlzeMB5C&VLb$7i-L&10nnkwr-(Em6sd+qNZdOy3=8OIn}=yfm+;u$Z_wSd3y? z$NKFzZ>4dY5>rz*ZA?{SYz5{6d&Ug1R#zQ8)%uo*C8JgkS5l<8Taq?yWSv><>#U|F zyWa&*kKTQ1LHRE3N;}4Dsaoj4n~<8ac~fd?!p3;5_Ozk`bNKL-)7GDv<11F*vUBdP zX?vA;X>r1$@rh~|wN5v+hNb2CJ}0ZvDn#^k3Y~D-PQ>As$&pW6G8_dp_Gu768WK$rUucb#I zy^}rC(ua{o9qeoVHr%XbzMvlV$GzT%Y}3u>q$T$k-&=g&0DJDl8spA++*x#Q$$dk6 zN!6<(aJ?8gv}==E_ayAs`iIO(CpT8tviobSFj9ySe!Z z3lZB(-P8~fH(AIHiWGEpnr)ytnX29{$<#D20~a^d@?BN4cwTzS%uQK&@wUqiwR|_+ z$&?;5b93A=sb!h06+?;_DamkNdcw@jc!nn|*>=KwPx`LsT~80cVjbV!_xyOpe|+3v zZ%c2v#wqXDUtK)@)hcE28S%t*)zDb#JtAf(cNsrTqsrOq1Eio4+b(K)Z~B>CDAbt; z#Y(GYC2kz$77#ZG-m*yUnh|wk*TxO2mn*UIxqEY-TWzIfMmpc9RlQR;l_*6OP!G;w-NlA^lWT9`A|ZP@Yo%7idQ{YDO) z5kGqPZ1`bQJ?6K7Hj;*qnH*s4EB8xH+`D=$)W7J^O&Ppv!X}!(Ja#h2x zAp=?~bLAFC2Ys!Wem>2bh0zA;IdgwGaQ2vCF^TJUDK;waO5TvZXGQ!VWzgAc@{Lgk zy8Dg{>!zr0ZL`e#H%u8~Rolvg$4?1Y;@BZcEnDOfhR&=3>#=~Ub%&l;6OXaP^Aa@O|yY<+yp(CnctgSQ8N zb!>hk@r*R@`By<{hg%*--^mu+GR!BMU#3OZ&)crcxUtCA)qJwW$($1{F4I9R2IUlu zI&T{#o7S_zwja&iC6%c0;%(_ZYxq!YDBhArJR0`Jh(`uf3aiDcnNRQ7v0)Rno$eMc zC(npV8{pP4B&2~=Jtun{3cjaoxnd5DOF3y}7v&SFsR#FMj}KDREApV&_;6*Y+Ufu` ztzs=%RrAvmna{0kzx*sPvuTsSK1~#Lzuf3VpJ&Q7hp^bx6IOO!D}P{HT%e+!mjh$t z!jxdO6)Zk6g}b8E@tfk}$B&1Ps>w#5q8%l*;@d;wUFj&>bd+sA#x@)9;Im+;br9P$ zgl)z@LnjH{J`+z-pSOqZ@boeE$XrD3*of@&wfPP!^ubL>=Dpe6W9F7KR@+Yb^vW$) zlxwmoLf7{Hv2eV41>dd83G@o9-tPOCGdgve z5<7WXaCeJZTkbz`+#s~&mz}4W!OB|8+m^52pbmf@QmcWRp`Lxx6ewP!Kei=vQWh;_axGi0} zVBu2hkc`vvwd4b5?pjVwKRD!uqE=-7V)Vh;X*(@PKHqlWz}~o?=sCU$J=J@oax5ZD zo-=#l95>UNj`UW<$-Wzvb9m?MSqo>+nv*o!?RZ2#**`IIP-lyJRPMNQ(1i(j+i`i> z(xm0fmM>i9HaO#~d~RdL$-5TQ$=}&?bB~Az(xTZb$1I#r?zf0X-)NUP4;*uWuFxJiXCOsPi)>P+C}QGUK5-8K9pwk z6}xpmD-BR@!5~kwTjB*g;^{L;KfdNItvn}QyWSs{*TmvdeWYo==1g9&k7Qyww&8OI zQrqjo6?4pAAET{WaGQ&j8=g2!4{nL=Wt|}R8aF0%Xu|eON~(M=CFP)^o>S)!r1jCd zOXfZFvb>!Z!49Xz4Z|4dr;V4G#cG>(@2YHC$xg8x^YxlPcdt>)H?R7&eAzUI?O{3B z=yaCOQq9-Z>Ao_X&J5;j>h$ii$=2~V+c)Op>bagQCpw9VMMXQ7MzGeb3g!Y8Ss3e} zdb%#y#)9s#rHXnSm!;H`Y9MQDVAs^m=1GaeKA$~%=A4;xT}Mm|SsJx9dUVCa_369U ztXnpF+1%xGUAIizmXxx3`Ql}Zmo5$3S21ejh@DsNpPw-@^l13{t&7vAZg(AjAaFxi z;`}83Te@T!a-Q6JAt7bZ`q8t(5~2<2=WMSzY3ba>bCPDvuQ-3*=DBlH=Wa<%p1H}e zXz`-OODrp9Et@qHZJ51y&a(N+xM|~-Bq&SbHmq7QdHVc{bCTxGTR3Oo>?QM;F5I=O z;(>*9p!L@ zK7V+P4glZho3HIVbn`~|!5-a4hWK{g6L#IonmVL!oG>gXabi@|g!KmxC2!hkd?#b~Z7b^~-yOZrTWMajU&^x0Fdy7J3B8V=JapW&FeQea=eOhHoE<{d`V(j)UP%&FOq6sd3((>&~7p`2O6b5*|4UiYin>f*e6^Lr5 z!2YG+R5Zmpgxaofx))tNo3_U&nqRv|)0v2zSgv{8h6!7?Y}l}6^MnoK#!Z-zpqM7H zBbX95V~3>^$zp~aI8kJGz18pK!IQ)Y`S^NKea@!JSlalSJ*+{8P}P+aHG~f4O~++& z;SVi&+$frcQ`V#?PHZptpD-~Ledo*Bd+uaeupW|D=;KAJ(Tu`QA&B}H`qW0$uy9xv zM86d-_eWH#aEm9R?+Sh9B8n>XiKI7}*x$V7qBv{QmN_YID>03$C0~p?9+swTcqm3l zQ~bmgi3v-_x;5<4t-iIWmfUA+(7qUDVtes7CYci_E?GIniUoyKaoN-*lM*dcXzQnD zTm5M?IeG5PHHs=1Z5HROp1(HPVq%H5wHQ?{ijp=aM64a+*2ceQV{4J8?0Loa$>;l5 zZCkHIvPE~*P9F@TldNVhFPv2e7fmOZBB|sk)HYhqyi0dBn|l#Yr2fn$hVObz{Hyv+C;y?Dd(H{%qr~5>}!DPI*xe zI^<*sLbU=^3sH;gDu}i-Cx1k(9{1+=yxFIoh*zJQi}15kk+cJCdBX4h6eVeP(u`>d zbcOtI_30~*Ew*#sGM2H}9@L9yA1X!^h>8(4N5w1O|FfozimSm zx+sRK5?48%n9P-pCz^5tai`mDXT1;|x1A0_G{|7l4Q z>^CkhXz;ky%L?keU_Oo~hqlG`QdE=bDq~@9w(V&Ry1o@v9c+sq zu5ZoD6rgEB`}a{y%_wt40+*-+6(@>7S3+EQUx zUUkM|nx4Fg7S=qL5p=6_mktpRRCjsP)R^r9-8v2#>d9$oaE@L!hsE&u$4&Wo8b+f# z;)W>d9XT)>^AFP&=68yAl6G_vLnF>f3wF0y9jh#vGI`Ntw-!Ntn_ATaa>H{0j~|~q z|3YD(%U34sowQGxxG&<11(u^3y``b5zxm+y*v?jUmFyi89~K^+f`zD6@*T(w^9A}X zyW~Zy(%0-_2%=_rts!M18G5wS+>;#|a$wog3i26lxxnj2EDef@7v<=Q!Y zN&;OigDa=6S=7Po{$Pq)KW9JGptY9JTI`b=G=*I#Ss+zu;bL>$n4mh=9%pElymxup zu|umcp}r&CjymV5V6Jzeu)O(pOr}TOn89_Fo@eOS^4=Ax$Br&d2wJGTErWS)Q!c6d zwJBnc!p}SrSF`;JDGV%3F#D>m=pOelS5f^$JzJkI&n_+8Q5z!*sj#N`eunogpDq|G zzpZ)4<|`dbPuaiw^D#k6QJ1?mANhx&`#TMePK=p|iB==#XuGr;9ow(A-#XJfgOK#Z ztvAEzg+Qd6@WfKFCx3&FhKBIQ&FSBxk zdBG`#z6~gIQ9sjaDYKild0XAQVBVBO%fPhA%j=RCF4=@p#k}d$=g)MDo3bKf$--3& zthP%&GM1J17fz~8SFWOJt7y7%9cA-oC+OfX ztid#2L3>8i4l5vzLUR|oaZ>X5@slPdg2KAY#PZBgNt<)0BIm&o$JM;M5>}?csPZ*v zADV0X%?lB0S=iPe@sPrPEfJlJraM`0DAW%-kLO!$@nfm2`ypC1OYS>4F<8NpcDn7f z7yXRxwZScY2A^}XHljv0xTVjEeaL6DPNB0Wu3p(r&P5b!I}=H_?4ui~l^vDOCmb4{ zs%**;^ot< zKiR5_OQtWLHpMcXcD`?BSGg4(T)TSS>{UvwqR+&6E0b2OLEjW->&~cocuY)KcvR{| z1%tK0te*UO^rhPUL*hawDhDTprSx~J9@(XqRoyH%JQncm%z>2s$x8V81Mz3wo*v7* zYhC6et6Oa4%|}zlU~M-;_KlAlGBkcW+}(Eh%GUIw3UliWCzEpEaB6f<1&ivgOysM3 zfmr4r5EDIdx|L1y`O{}PeKquuw=%Am`CwWc)|zh~r0kA-JUwNfg3kiOnQIN&U?lZn zdM}FgeVEm-mi&h5@wQ1{T9c-o@In}feyk>S=iOLMI`UBs+KhHMI0&w4KpnWMv|RDw z%DopeEk6AFK>XW@eQ7rP*7m3QbNO!A(T2)wspiPRwXA+aeUB8)xoi7VzPfhvab=CP zcT>V}YqUIaazdcuD?eTt+|H_wkb6$;-D)|mne&ugU%j%hIr5#cXM1c^_OD6adBkFB z&W74vo6j3-(TW8;zS{ho z0aDwklZPn1_NxKm5&KnVx#ou6Z`Q3_yg*61DlV8lf9jxVTTd#c#&pn0FWQ)nIuU|U zH6E|cqdH$#L3Egv_D9s^>)!lc69ymEm~Ml1HKw&sM^bzf@ueLO47@!}b5>Im=oO-gQTWlR+S3M|Oj;ax@MBRwmpQz0v1$BqgYqp=g5Z$)rg&>+}`>8gfeuc&=bQY~) zd*g*jP2jeHxE8mZ)hvOT%ppmMkwI>l)*iDSmtS30 zw>XpC{L?c3Klxi}d)5N+IXNC*;#cCM{Q-PHzli_Zw=6b7wh&qi1BFr8i|Y2k zg#w)(d(NusJh0!atu9CxuA8V^qT8Xnrpwj6(Y@8z)i>04!tSwk`p@}QnQ7R)CW|7X1t`0QZU{aM)}%eSWJpHjSo|d=IZo~OLc0T zZGLv@{9S89+VE7sH%ez)_rcUyiJCUBHkGOIojVJpd|5FX%aXn5bZW$Blv?ena@U~^ zHFlw!Cas4ro-`hHCaj;hd2@2|=FJn=CnQXoglR-G+TzTBZ(bg~e#M6x+tF&*{a?Je zeCeA`)M!4Fd8UteQS!&lOt{DBe7Ec*;Pxu0w}58ck29r6;da zj2)=QwQet-@27m+X*BoQGwjOMJ-aR`s{%up<0h<0Q`QB@#s>Ie>oZ#pKTk)D zPCKkH7r55?^6Rimbs7)!kEcdokkt2S&bf0r3b#AsRyL~emeDCkFw#S`lGe6g%Q`1DnE z@w)V7zSvtD(5GLfXlky>bOTk4s#4e&A4gU9w|&J2(oQ~SuO+3co}!U|2lK3z9c>vD5jme4yHSs;-Ch`Pvh7&V z?mc`=@UY;mAo+rK(K*?O!QU;J3++G}U%b962gVpN)Uuj=rb>gb^^z-+(u+>MvrIOg znsk^iVLCA%F?QeNj9r%FpKm`@vV{4Q@A(#r@zBgYgD%{p%q)&eote7XvTxhA{kzh! zNPAoQX3){b(7GGH;8V}w&qU++^^-PjTEBji(S?Oykau)S^G@qz_*$yR62xsnNy`?rqE8E*viruC5MJwAN(Lpk@y3E1lmUP8d;c8I@EGzzuZ`Pfp zE=86eHjC&dbuY4XMWT~rG&kxVnOmo?p_lyqmF?BOziKer>i46@lAiuR8opB+&C_SD zShZ>e$IR)|XU?2%9qzC*HFaltTxw)wTwGLSY8;lWdh*X|qOH66XnN|teLFFAEb4m4 z<};ScM)|w$HyZL8FlP^Qok>SxJZYLnk3t)bN#s+j?_nsk*wMx|&== zO*Y(;hT#Lr*Qd@uw{Dl84a{t%RWU|xgenfC#`U$v%YLzO;UVL;T~boybE&EO71Lo> zrYkj$(p1)HKADlaw|H(-3UeEyd4G|E)M)E~pVlQW($+j?B+UvOzb;d;8MUrg*1BFJ zRvMZXO#$j!|_GsTeTNHbg;&S zGoVQ=ikh~tHHAzWEjl9_%__OwBQNjvRs46MHYfhg?P{%BoqOP$8#`0vWcCag(6VK~ z>6iT3%k!tRv(FFkRE%GkPzM*xq&KRHM%Ewe z3sqrpRk0>;kqK6OG5fv8#fA%&uhr(}A}V(X9TTW@wcWfaoj5qAqt(cUW9+qH&ca## zagmry9_)64+Um)dckaBfbU8jg8|4!B057G@ERP1?Gj6oa3&J}ye})!@))1MC%lV(h zUOZ_!dd3AGQeIp>@V%9lmT!(t3tzA7UAH;os^yLRo!{Yxis~lae_Phfk{`&&w@>I| zttWSgnbcqD!XBF0!mILyl=0qHZ@FFEltB0x@4l2B$C^=hiS;RT64i~JSO-b<$9@2o zCK)emOkTWrEv5uXbEeLl>=r+5-BzpdEc2Uf#uARv_7GoUZK7I1G8Vfw`Q^|f-UA|| zCQQ?)G+r~S4J6jwHdj>V_F(lSwMEf9ku8vnmcXqMW2kYb_h`BI@R*_SXNs|8jEed8 zH0!}`Z)Ys0-wr*DA*vrAq6S*k8yKSADjuR5XTUofSx4AWfca4RrquM*@k5m&|2wt- z`AEiwtvj}rT>8mR!w+;H92FBcMcFkmVq>UV(Ab!%GpxJ1o`Aivz}e>eM-G2$#axXb47Xg)K(cZjOD;S}RS6(e~3>DvH_nG2n;3S=y* zuTh~GbH1>ZHe>Nn{Ta&f@=A$vFrnuKPerW?z4kGm*%OPcHZSG+!(#_PBpHt~+OoBh ziLW&sHDWH#U9DndCMe?D0HfGWGLB)J#=+AZRn{x50RA0U- zHJQeDv8n;@mZM}iEp6&&;#&|>H*HGY3MLO3GA#9m;w>4~dK0LzFu)A9i#A^~CZqc} zn=u)MGtwrbA*1cG!>^l9OW)dD#G(M~=IJP_hN5+NuTt{s^Af&Uf5N#HWO;vMw zmlg?CicGtx9Pl0{dtB)D;>E=quN2$)4ze+conM3o4q}J+68ZL-^fPt77;WUNIZlnu z)bdZX-2+{j`6f087@wX#|H8=5FQCR%*03AKnBI!<+KpGnl<}LfMI<5Khz&t8Loo_F zoq@)d!G|EM66b&zZ}m2okpG{Z19_T_ySu)xz%?(^Sk5-pZzDHwCj8z?po4+y8k)Jh zKnoj=tlJKBQtD|g;DQM5?>m4Y+c;DB-|Xn=VkaDZ8k zd)fBwYk`%}?t_ozm#{yGKEw@5P~3lxp67no&1DYsz*V+0z!e?bKf6S^frDhgB2%9G zIM=xo9>DOKww&$b?`gobUEIKuO>O~qL2(MV+`)NE51f`d&HkMGZ|Eg%c7qe#JGnr? x%l%y-^tXUj%pGoEpn=mC_bE`Ll^Zyt*6zd3cZ3Vrf@$XR=9c@-%+0{S002HoAfW&N literal 0 HcmV?d00001 diff --git a/app/assets/images/logo_dark.png b/app/assets/images/logo_dark.png new file mode 100644 index 0000000000000000000000000000000000000000..fab64c2d5a9d7b46ad90daa132623a6d2046dddd GIT binary patch literal 2858 zcmV+_3)S?AP)Px#24YJ`L;(K){{a7>y{D4^000SaNLh0L01FcU01FcV0GgZ_00007bV*G`2iyo0 z5;qr0Rj&{L01B!}L_t(o!|hpna8%V9|IXdrdv`aRY#y70knCn3+1I_<`+d7vOypUN zf;d_n3L`d@hqe_e0>!EcI(F)4Ne9L84`({ou~u7JA2TRZE2X0COh;f4xTJX?lo&B! zv|&gfhOn>GKitJ7$?`HfcIx?O_nz-@zjMCd`5xy2d^^4!|3?E61tPNe{eEL+ZsW#{ zU%MTg^Yj1RZnqH;EjF8tW%CHx4~((t@p!yVlB8FO=n+ILXlZG=BIC1p5%C5@Y%VJ+ zvtEfjV+_n@bD65Dj~k7~R{-FU&1UoF6soJM`>w;`*a`r8L~KFCqeL_#LvL+u9V0W| z+`M`7bUvT&Eh2gY5%(dYGn?*lxyHz>TDdZN!M7682M&khuJrS6BASxZ@MttT*X44( zW-^)F002bf3WY*jWLf@0C={v$0GK;x&bSbYMx(V3hht$R5?QAx$^luH*8%`|z1}87 z+*(ytwb<+RMgTxJdeIqOZnrxi%kpc8_>A3dF9!f{I-LbX)b4h>Ll>YU`=PkFxYlmB zztq>)cl#%we9{I0;Bq)XFG-DNv-!D`Cr_^H?d{!~;g@B(gmeCoBuU>pckW!j!C=_= z*=L`#O7O_zqDNAJsIiJEMz;?&6#` z1Hi%+D^~1Vw{9Jn&1N<{JggAWo0_H(0G!P5>qoh`qobpvu)n|mQzH7lrfFM#v~neE z*sx(I5lt;ADS5o4q+|-`{BOx*@`r*T6a&CE&Ut-TSJ%hm!D!dJ@51Zbw(UA`;>0ch zc$kQa7-P4@VzJGEKp-gy!V97(Hl96uHm|U-5cllabGp0x^Ys)J6*#{B&JP$UuwX1f#-A0nbth`8PB^|~u6D)g7adp2zpT~}08RH3Tsi$pYt zh<5@2tXZ=rLpLH~6A>*WqD=q*Wkp4~4X>`Qo~o+q(*OWfRaG<6a1vBi-AqKstE;P* zO?3E^WqBSVejv;8hVuv5{@7xqePWZHl5;E(M3H9T^xQq9Gg(@9}s%Pc$|*=E<_GMIw=DSENi)lm`*< zU$QKJ6pcp9zv?WS(T#|CMAShHaWaRlwzl>OBDx0vpvXG5Vqm#m5C{Z1&1Un` z{rmU7mr>r_+zg25W{j-@fFc0+`^d=1pAQ{6GynjQZ?V9cp`n}xx3shXBI+4qP5`() z9*=wUdi}eMvF9~Si)Hf@Q90*)-@$_i-2f1~EW^EC@6CF>{%2iXUEj%`_xXHIqtW;_ zV=SLB#sI*ON~J!I$K${6>gsy*VqR=+ZU!Q91HeYk`7A*Y3OMI607#`$shvGNJx!-i zpN;_l?C$D7I><_w1!SNZ*ZQ53~>CX?y*(aecNqFxY$e8yOOXlSTw zU|?XoAP5#w6j!OL`jRZm5&*yh_uY4aT-sAU&iQuE`BcUj*EFqZaB%P`&iOE7Ot#zY zr5XN)U@%kOKE^q}&Fl3FqoFEfE007XB_@;Uc~KOt?d|QtICXO0=RZ6q2*TY~t93a5 z6b}y%caDsV>@*k*FXrXt3BTI32}ZenDgbQl?(V*2aB%Rwcs!mL4u`iJjm8}SzyV++ z9*>WV^S<6xDs@H_#UEsB94`T}p`oED=e+*ZsZ+1TVzJ|+6~*ImEtN_gVT>7*$>eN< z!H`U)QvdGm?*8rJ!-t2zc;7+OQ2(CTraa;&$P8w`dp0Bq~&>3NKZ#<-`qw|B6= zzklV($jC?e`S~0A`uZYmZEZItlgTYUpYImVd5I{B*9QWD^I_oJxpSz~=@bAc@%elY zmXwry+R@Rm;`s67YeZ387K_DR(KPLW@gke1X{R~oHv_=O{r&yDKVG_Y?BOJ$LICIh zfS+ob_A(;s7-OQQX=gJ$PrdL0tX;Ykyk2jfD2l(->-CEV2M60lQ7kr_&65CtCzHv< z$jC^;kt0V=q!UUDV{APDBsu2{0Ah?WHat9>uvjeX+S}V-kHumdYg#O^Q)@rLc}L8CmGG0J{{8a zk=)eO!~mdFQIs|!@}_kgF4gTzMD)Jj@2^j{PjWOG)wQ;^<}^67b2ghTkUdvdSEnm4 zFR#2*Ml~6Yz&Qt-&E}subEdJPqM{TLyJT6Oep$s`#;~%ovLF%dKtx?bLjyy^MgV~8 zuDdR$?ELxjbJD`9nv*7cam(PgTCFAN&HH006q;SX*0LjflO7_$ea(hlq~IvRr&I?@E$XX0cdm002Z$ ztS6#F;c&Rj@ArSHYSZHL`S#oGb{PPmzP{cb2n4RZ819@-r-z8%Afi{(#aPZ^v)S?_ zNxCT=rR=Jz?ngxLm$KMkFk~)@WJOU9R#jC+0RRdL3hb(?%HtWo_KRls{IKTdj}4`M zKUl3+o6TlxNW-HN5%+pLo>>!G|9ib&zpAS5*VNRg000(?WqK$SntZVc0{~zeMQeWG z^CUTA#tgw?vCK&~<<+XHcDh`yITPBM+<*W5nJ^h3qFso%Fn!TehluWrz(yEv_rl?@ zSy7a^=>#)PRn^_WU~oDBz)!NS;wE}nh)6@kWt?-c+wHR}D=Q5E0Cy~34xDqy%gehd z5{W!XL{A~&joAxZy?Qm2mzU3Uxm*Gw-iU}th{*Ys_Jm>@^N~oz==1rugu~%qFIu!n zlq6}E-|sKIoOaFY_0BFWEiECUM^sgP0|0Da)$Kt}{5l+tyNPHYBIdhXu4qL?g)St^ z%_iTpgnCVCOUU~rNs zicbT8l`%HRIWK06-I;wC{!RZ7wuFe*{?B~58?R;;zk>Jt7aI_+8J-YZZ2$lO07*qo IM6N<$g8p)8-~a#s literal 0 HcmV?d00001 diff --git a/app/assets/images/logo_white.png b/app/assets/images/logo_white.png new file mode 100644 index 0000000000000000000000000000000000000000..3f74025449c2e1a3aa4af6052098366e89ae90c2 GIT binary patch literal 1681 zcmV;C25$L@P)Px#24YJ`L;wH)0002_L%V+f000SaNLh0L01FcU01FcV0GgZ_00007bV*G`2iyo0 z6B-YA(FdLY00t3BL_t(o!|j<_Y}8d0#_x2xOrZ;9DTON178oeB2|GxE07{V{1w+Kx zrX`TDXaxgd6O4g}GKSrVF)nBTQBb2%iHah|B^Db{P&SQ-Pz7u%wr0(mt09N`KOmWW(I$)nP64k0tAVUU zg?GmSHv`Ln>_q1SxxhN$RiF{*6;1HTz)L_fupIab7-{=Mq0nV>yxIUi@HVgqI0$ri z`-=(+E}JuN?p$~N6yR%Mj_UQm;An!E11|%Gsuut|foFhxV~i;uKi<=XfPTO<;9lSp zV1e2zrHng(0f0aJ(2kaSYyZc9&T7j7J_kyo3EmGl3Y6Nmf^OXmFb?M>|1F#V| z0~`X@0@HyZu1x~NWqKd#+P7NeIAi8W%L6xU+GN{&lIbkaA)4Sm;9DRMxYHP8>gwuj zUq0|GPzQ|BExZoQ1G;#v7Yc<;#pKE1d*lGs((H>s1@I8?r1h!!^XJ>KLxB;%$#^_^ zw=`S=^l`5gSPA?Bya5yf$+7I+?Q{F(2VH>eKoc-4Oz42jY#OlE7-MpCa-s@;Bd|sP zN9q=mfDO8hnpS|nTVy8pIc<~lyAvpgXAwWJ7sv+Mt8G2-8Iar3`0hAn2JjxR%JH{C zIkvAyQBhnPm;&t6{|A60Kt?QcXl}7)1f&25q^P#Qr@)9ruWyC4eAYr!YerE=^KFG} z#}YQnMGF_&MfL-Ou2krEKp-F3pU}R}UQ-M{Qf7EiJusKCKR8mw#6ZhB&G(=cvG;jep4U_`u z(%f6R^4)e5kDzuG2nv^f)7%j{$U!eI0;EanHL-X~xv1=l2xGEEl+FVSbUQ~ZuMyFz z3!7sD%VE_^#Ib$ABf$5-gAw-2(wgUhT2H~xG&UNM{*CrEg`VGPj|;`-nPvR5}PIbKJo8! zvi&naS7+Vj@{98te^!S$tqARlpde!rON{aW(8d#bMQ8T{3mwx4M8aP&xIw^Rpg-`q z1n=yOnJ1yA%P)oi!AOj9MorB{ICLHt^@8645g(`$X14`@VQyNyK?iup9j{l^#(09! zsIoFUr&vDFNj}t5?J2QvJPkMr40RUrheDx<0y}Wp+ijD{<;RL0>4AVXkD!#-2{;av z#YOI%2-inu>a|so5ZOuf`|Uy-?8`*; zae%LDUS71@$`X34*Zf6po+iAc{oSPcI%$pPnnmuas;Z2F#6cT&flOd{3>&{esBuAP z@}+%{#K@uaaNe0GfZSj(_>Z1d8N$(G)vve0N3uJG>OH^wHcV;9;~*u)+SezfC+Nm?ydSJOuc0D0ny(_#SljWo#M>k zjuCxcEeGz{TG@kW%b52uJll6;a;7wqi?QatZT)E`?ZHkiO@D0FB zk*KsN5Lu(KRmK=oSy_2iJ{*Lz#2vsZKwrQYW*$9(r$iyL{xzRqoCr8y2vHqI6FvY| zw`BUx*9zLtGsbK&#$+2~nv5~+jWM%)KHsVT%7?Hj`&FVfN^)zU4q$vf-;RqgYK-}_ bh4=Xj|A5=F>y1FP00000NkvXXu0mjfpXw6O literal 0 HcmV?d00001 diff --git a/app/assets/stylesheets/main.scss b/app/assets/stylesheets/main.scss index 71d09883..ad8be0bc 100644 --- a/app/assets/stylesheets/main.scss +++ b/app/assets/stylesheets/main.scss @@ -23,6 +23,8 @@ $blue_link: #2fa0bb; $style_color: #474D57; $hover: #FDF5D9; +/** GITLAB Fonts **/ +@font-face { font-family: Korolev; src: url('korolev-medium-compressed.otf'); } /** MIXINS **/ @mixin shade { diff --git a/app/assets/stylesheets/sections/header.scss b/app/assets/stylesheets/sections/header.scss index c08404c3..d0fb662e 100644 --- a/app/assets/stylesheets/sections/header.scss +++ b/app/assets/stylesheets/sections/header.scss @@ -26,23 +26,25 @@ header { float:left; position:relative; top:-5px; - a { float:left; h1 { - text-indent:-9999px; + padding-top: 5px; width:102px; - background: url('logo_text.png') no-repeat 0px -3px; + background: url('logo_dark.png') no-repeat 0px -3px; float:left; margin-left:5px; - font-size:20px; + font-size:36px; line-height:36px; - font-weight:bold; - color:#aaa; + font-weight:normal; + color:$style_color; text-shadow: 0 1px 1px #FFF; padding-left:50px; + height:40px; + font-family: 'Korolev', sans-serif; } + } .separator { margin-left:20px; @@ -68,14 +70,16 @@ header { * */ .project_name { + position:relative; float:left; margin:0; margin-right:30px; - font-size:24px; + font-size:36px; line-height:36px; - font-weight:500; + font-weight:normal; color:$style_color; text-shadow: 0 1px 1px #FFF; + font-family: 'Korolev', sans-serif; } .fbtn { diff --git a/app/assets/stylesheets/themes/ui_basic.scss b/app/assets/stylesheets/themes/ui_basic.scss index 09ff0747..cf5eda1c 100644 --- a/app/assets/stylesheets/themes/ui_basic.scss +++ b/app/assets/stylesheets/themes/ui_basic.scss @@ -15,4 +15,36 @@ color: $blue_link; } } + + header { + .fbtn { + .btn { + background-color: #F8F8F8; + background-image: -webkit-gradient(linear,left top,left bottom,from(#F8F8F8),to(#ECECEC)); + background-image: -webkit-linear-gradient(top,#F8F8F8,#ECECEC); + background-image: -moz-linear-gradient(top,#F8F8F8,#ECECEC); + background-image: -ms-linear-gradient(top,#F8F8F8,#ECECEC); + background-image: -o-linear-gradient(top,#F8F8F8,#ECECEC); + background-image: linear-gradient(top,#F8F8F8,#ECECEC); + filter: progid:DXImageTransform.Microsoft.gradient(startColorStr='#f8f8f8',EndColorStr='#ececec'); + border-color: #C6C6C6; + margin-left:7px; + @include border-radius(3px); + box-shadow:none; + color:#666; + } + } + .search { + .search-input { + @include border-radius(3px); + border-color: #C6C6C6; + box-shadow:none; + } + } + .pic { + img { + @include border-radius(3px); + } + } + } } diff --git a/app/assets/stylesheets/themes/ui_mars.scss b/app/assets/stylesheets/themes/ui_mars.scss index 488c2af5..2808ad32 100644 --- a/app/assets/stylesheets/themes/ui_mars.scss +++ b/app/assets/stylesheets/themes/ui_mars.scss @@ -59,14 +59,9 @@ .app_logo { a { h1 { - background: url('images.png') no-repeat -3px -6px; - width: 65px; - height: 26px; - margin: 6px 0; - padding: 0; - float: left; - text-indent: -1000em; - float:left; + background: url('logo_white.png') no-repeat 0px -3px; + color:#fff; + text-shadow: 0 1px 1px #111; } } .separator { @@ -75,7 +70,6 @@ } .project_name { - line-height:38px; color:#fff; text-shadow: 0 1px 1px #111; } diff --git a/app/assets/stylesheets/themes/ui_modern.scss b/app/assets/stylesheets/themes/ui_modern.scss index 3fd9bb5e..1f0d7955 100644 --- a/app/assets/stylesheets/themes/ui_modern.scss +++ b/app/assets/stylesheets/themes/ui_modern.scss @@ -40,18 +40,11 @@ width:160px; a { h1 { - opacity: 0.7; - background: url('images.png') no-repeat -3px -6px; - width: 65px; - height: 26px; - margin: 6px 0; - padding: 0; - float: left; - text-indent: -1000em; - float:left; - &:hover { - opacity: 1.0; - } + background: none; + color:#DDD; + font-size:30px; + text-shadow: 0 1px 1px #111; + padding-left: 0; } } .separator { @@ -67,18 +60,19 @@ .fbtn { .btn { i { - @extend .icon-white; position: relative; top: 2px; } background:none; margin-left:8px; font-size: 13px; - font-weight:bold; line-height: 19px; color:#ccc; &:hover { color:#fff; + i { + @extend .icon-white; + } } border: none; box-shadow:none; @@ -116,9 +110,9 @@ * */ .project_name { - line-height:34px; - font-size:22px; - color:#ccc; + line-height:36px; + font-size:30px; + color:#DDD; text-shadow: 0 1px 1px #111; } diff --git a/app/views/layouts/_head.html.haml b/app/views/layouts/_head.html.haml index 3a2586a3..c076a3a1 100644 --- a/app/views/layouts/_head.html.haml +++ b/app/views/layouts/_head.html.haml @@ -6,7 +6,6 @@ = favicon_link_tag 'favicon.ico' = stylesheet_link_tag "application" = javascript_include_tag "application" - -# Atom feed - if controller_name == 'projects' && action_name == 'index' = auto_discovery_link_tag :atom, projects_url(:atom, private_token: current_user.private_token), title: "Dashboard feed" diff --git a/config/application.rb b/config/application.rb index 93726223..ecd88b15 100644 --- a/config/application.rb +++ b/config/application.rb @@ -44,5 +44,8 @@ module Gitlab # Version of your assets, change this if you want to expire all your assets config.assets.version = '1.0' + + # Add fonts + config.assets.paths << "#{Rails.root}/app/assets/fonts" end end From 652d955c92937d74bd81b9af4adf3b1549d89876 Mon Sep 17 00:00:00 2001 From: Robert Speicher Date: Mon, 20 Aug 2012 23:04:53 -0400 Subject: [PATCH 17/46] Remove some duplication in the Notify mailer --- app/mailers/notify.rb | 44 ++++++++++++++++++++++++++++--------- spec/mailers/notify_spec.rb | 6 ++--- 2 files changed, 37 insertions(+), 13 deletions(-) diff --git a/app/mailers/notify.rb b/app/mailers/notify.rb index 9563fdbc..c3b2e490 100644 --- a/app/mailers/notify.rb +++ b/app/mailers/notify.rb @@ -12,20 +12,20 @@ class Notify < ActionMailer::Base def new_user_email(user_id, password) @user = User.find(user_id) @password = password - mail(to: @user.email, subject: "gitlab | Account was created for you") + mail(to: @user.email, subject: subject("Account was created for you")) end def new_issue_email(issue_id) @issue = Issue.find(issue_id) @project = @issue.project - mail(to: @issue.assignee_email, subject: "gitlab | new issue ##{@issue.id} | #{@issue.title} | #{@project.name}") + mail(to: @issue.assignee_email, subject: subject("new issue ##{@issue.id}", @issue.title)) end def note_wall_email(recipient_id, note_id) recipient = User.find(recipient_id) @note = Note.find(note_id) @project = @note.project - mail(to: recipient.email, subject: "gitlab | #{@project.name}") + mail(to: recipient.email, subject: subject) end def note_commit_email(recipient_id, note_id) @@ -34,7 +34,7 @@ class Notify < ActionMailer::Base @commit = @note.target @commit = CommitDecorator.decorate(@commit) @project = @note.project - mail(to: recipient.email, subject: "gitlab | note for commit #{@commit.short_id} | #{@commit.title} | #{@project.name}") + mail(to: recipient.email, subject: subject("note for commit #{@commit.short_id}", @commit.title)) end def note_merge_request_email(recipient_id, note_id) @@ -42,7 +42,7 @@ class Notify < ActionMailer::Base @note = Note.find(note_id) @merge_request = @note.noteable @project = @note.project - mail(to: recipient.email, subject: "gitlab | note for merge request !#{@merge_request.id} | #{@project.name}") + mail(to: recipient.email, subject: subject("note for merge request !#{@merge_request.id}")) end def note_issue_email(recipient_id, note_id) @@ -50,7 +50,7 @@ class Notify < ActionMailer::Base @note = Note.find(note_id) @issue = @note.noteable @project = @note.project - mail(to: recipient.email, subject: "gitlab | note for issue ##{@issue.id} | #{@project.name}") + mail(to: recipient.email, subject: subject("note for issue ##{@issue.id}")) end def note_wiki_email(recipient_id, note_id) @@ -58,13 +58,13 @@ class Notify < ActionMailer::Base @note = Note.find(note_id) @wiki = @note.noteable @project = @note.project - mail(to: recipient.email, subject: "gitlab | note for wiki | #{@project.name}") + mail(to: recipient.email, subject: subject("note for wiki")) end def new_merge_request_email(merge_request_id) @merge_request = MergeRequest.find(merge_request_id) @project = @merge_request.project - mail(to: @merge_request.assignee_email, subject: "gitlab | new merge request !#{@merge_request.id} | #{@merge_request.title} | #{@project.name}") + mail(to: @merge_request.assignee_email, subject: subject("new merge request !#{@merge_request.id}", @merge_request.title)) end def reassigned_merge_request_email(recipient_id, merge_request_id, previous_assignee_id) @@ -72,7 +72,7 @@ class Notify < ActionMailer::Base @merge_request = MergeRequest.find(merge_request_id) @previous_assignee ||= User.find(previous_assignee_id) @project = @merge_request.project - mail(to: recipient.email, subject: "gitlab | changed merge request !#{@merge_request.id} | #{@merge_request.title} | #{@project.name}") + mail(to: recipient.email, subject: subject("changed merge request !#{@merge_request.id}", @merge_request.title)) end def reassigned_issue_email(recipient_id, issue_id, previous_assignee_id) @@ -80,6 +80,30 @@ class Notify < ActionMailer::Base @issue = Issue.find(issue_id) @previous_assignee ||= User.find(previous_assignee_id) @project = @issue.project - mail(to: recipient.email, subject: "gitlab | changed issue ##{@issue.id} | #{@issue.title} | #{@project.name}") + mail(to: recipient.email, subject: subject("changed issue ##{@issue.id}", @issue.title)) + end + + private + + # Formats arguments into a String suitable for use as an email subject + # + # extra - Extra Strings to be inserted into the subject + # + # Examples + # + # >> subject('Lorem ipsum') + # => "gitlab | Lorem ipsum" + # + # # Automatically inserts Project name when @project is set + # >> @project = Project.last + # => # + # >> subject('Lorem ipsum') + # => "gitlab | Lorem ipsum | Ruby on Rails" + # + # # Accepts multiple arguments + # >> subject('Lorem ipsum', 'Dolor sit amet') + # => "gitlab | Lorem ipsum | Dolor sit amet" + def subject(*extra) + "gitlab | " << extra.join(' | ') << (@project ? " | #{@project.name}" : "") end end diff --git a/spec/mailers/notify_spec.rb b/spec/mailers/notify_spec.rb index 27af1e38..93427ebf 100644 --- a/spec/mailers/notify_spec.rb +++ b/spec/mailers/notify_spec.rb @@ -24,7 +24,7 @@ describe Notify do end it 'has the correct subject' do - should have_subject /Account was created for you/ + should have_subject /^gitlab \| Account was created for you$/ end it 'contains the new user\'s login name' do @@ -60,7 +60,7 @@ describe Notify do it_behaves_like 'an assignee email' it 'has the correct subject' do - should have_subject /new issue ##{issue.id}/ + should have_subject /new issue ##{issue.id} \| #{issue.title} \| #{project.name}/ end it 'contains a link to the new issue' do @@ -76,7 +76,7 @@ describe Notify do it_behaves_like 'a multiple recipients email' it 'has the correct subject' do - should have_subject /changed issue/ + should have_subject /changed issue ##{issue.id} \| #{issue.title}/ end it 'contains the name of the previous assignee' do From c5b13cc9897d9b7175722589e6f0c3b66ded143d Mon Sep 17 00:00:00 2001 From: Robert Speicher Date: Mon, 20 Aug 2012 23:18:57 -0400 Subject: [PATCH 18/46] Remove more duplication in the Notify mailer --- app/mailers/notify.rb | 32 ++++++++++++++++++-------------- 1 file changed, 18 insertions(+), 14 deletions(-) diff --git a/app/mailers/notify.rb b/app/mailers/notify.rb index c3b2e490..d0571b7b 100644 --- a/app/mailers/notify.rb +++ b/app/mailers/notify.rb @@ -22,43 +22,38 @@ class Notify < ActionMailer::Base end def note_wall_email(recipient_id, note_id) - recipient = User.find(recipient_id) @note = Note.find(note_id) @project = @note.project - mail(to: recipient.email, subject: subject) + mail(to: recipient(recipient_id), subject: subject) end def note_commit_email(recipient_id, note_id) - recipient = User.find(recipient_id) @note = Note.find(note_id) @commit = @note.target @commit = CommitDecorator.decorate(@commit) @project = @note.project - mail(to: recipient.email, subject: subject("note for commit #{@commit.short_id}", @commit.title)) + mail(to: recipient(recipient_id), subject: subject("note for commit #{@commit.short_id}", @commit.title)) end def note_merge_request_email(recipient_id, note_id) - recipient = User.find(recipient_id) @note = Note.find(note_id) @merge_request = @note.noteable @project = @note.project - mail(to: recipient.email, subject: subject("note for merge request !#{@merge_request.id}")) + mail(to: recipient(recipient_id), subject: subject("note for merge request !#{@merge_request.id}")) end def note_issue_email(recipient_id, note_id) - recipient = User.find(recipient_id) @note = Note.find(note_id) @issue = @note.noteable @project = @note.project - mail(to: recipient.email, subject: subject("note for issue ##{@issue.id}")) + mail(to: recipient(recipient_id), subject: subject("note for issue ##{@issue.id}")) end def note_wiki_email(recipient_id, note_id) - recipient = User.find(recipient_id) @note = Note.find(note_id) @wiki = @note.noteable @project = @note.project - mail(to: recipient.email, subject: subject("note for wiki")) + mail(to: recipient(recipient_id), subject: subject("note for wiki")) end def new_merge_request_email(merge_request_id) @@ -68,23 +63,32 @@ class Notify < ActionMailer::Base end def reassigned_merge_request_email(recipient_id, merge_request_id, previous_assignee_id) - recipient = User.find(recipient_id) @merge_request = MergeRequest.find(merge_request_id) @previous_assignee ||= User.find(previous_assignee_id) @project = @merge_request.project - mail(to: recipient.email, subject: subject("changed merge request !#{@merge_request.id}", @merge_request.title)) + mail(to: recipient(recipient_id), subject: subject("changed merge request !#{@merge_request.id}", @merge_request.title)) end def reassigned_issue_email(recipient_id, issue_id, previous_assignee_id) - recipient = User.find(recipient_id) @issue = Issue.find(issue_id) @previous_assignee ||= User.find(previous_assignee_id) @project = @issue.project - mail(to: recipient.email, subject: subject("changed issue ##{@issue.id}", @issue.title)) + mail(to: recipient(recipient_id), subject: subject("changed issue ##{@issue.id}", @issue.title)) end private + # Look up a User by their ID and return their email address + # + # recipient_id - User ID + # + # Returns a String containing the User's email address. + def recipient(recipient_id) + if recipient = User.find(recipient_id) + recipient.email + end + end + # Formats arguments into a String suitable for use as an email subject # # extra - Extra Strings to be inserted into the subject From 2e7ca8c866179e78866cba1659dee45185911f5a Mon Sep 17 00:00:00 2001 From: Dmitriy Zaporozhets Date: Tue, 21 Aug 2012 08:20:11 +0300 Subject: [PATCH 19/46] Show only tm events related to this project --- app/assets/stylesheets/sections/projects.scss | 21 ++++++++++--------- app/controllers/team_members_controller.rb | 1 + app/views/team_members/show.html.haml | 2 +- 3 files changed, 13 insertions(+), 11 deletions(-) diff --git a/app/assets/stylesheets/sections/projects.scss b/app/assets/stylesheets/sections/projects.scss index 8c79e45e..1269ac8a 100644 --- a/app/assets/stylesheets/sections/projects.scss +++ b/app/assets/stylesheets/sections/projects.scss @@ -1,17 +1,18 @@ -.projects { +.projects { @extend .row; .activities { } - .side { + .side { @extend .span4; @extend .right; - .projects_box { - h5 { + .projects_box { + h5 { color:$style_color; font-size:16px; text-shadow: 0 1px 1px #fff; + padding: 2px 10px; } @extend .leftbar; @extend .ui-box; @@ -19,19 +20,19 @@ } } -.new_project, -.edit_project { - .project_name_holder { +.new_project, +.edit_project { + .project_name_holder { input, - label { + label { font-size:16px; line-height:20px; padding:8px; } - label { + label { color:#888; } - .btn { + .btn { padding:6px; margin-left:10px; } diff --git a/app/controllers/team_members_controller.rb b/app/controllers/team_members_controller.rb index 0cc58c3e..0846f096 100644 --- a/app/controllers/team_members_controller.rb +++ b/app/controllers/team_members_controller.rb @@ -9,6 +9,7 @@ class TeamMembersController < ApplicationController def show @team_member = project.users_projects.find(params[:id]) + @events = @team_member.user.recent_events.where(:project_id => @project.id).limit(7) end def new diff --git a/app/views/team_members/show.html.haml b/app/views/team_members/show.html.haml index d7e09bce..6cb357cd 100644 --- a/app/views/team_members/show.html.haml +++ b/app/views/team_members/show.html.haml @@ -51,7 +51,7 @@ = form_for(@team_member, as: :team_member, url: project_team_member_path(@project, @team_member)) do |f| = f.select :project_access, options_for_select(Project.access_options, @team_member.project_access), {}, class: "project-access-select", disabled: !allow_admin %hr - = render user.recent_events.limit(5) + = render @events :javascript $(function(){ $('.repo-access-select, .project-access-select').live("change", function() { From c625293b995cf21a1b551cecaac3cb742fcb2e66 Mon Sep 17 00:00:00 2001 From: randx Date: Tue, 21 Aug 2012 20:14:06 +0300 Subject: [PATCH 20/46] Handle post-receive files via gitolite, not gitlab --- app/roles/repository.rb | 18 --------------- app/views/errors/gitolite.html.haml | 2 -- lib/{post-receive-hook => hooks/post-receive} | 0 lib/tasks/gitlab/setup.rake | 7 +++++- lib/tasks/gitlab/status.rake | 6 ----- lib/tasks/gitlab/update_hooks.rake | 19 --------------- lib/tasks/gitlab/write_hook.rake | 23 +++++++++++++++++++ 7 files changed, 29 insertions(+), 46 deletions(-) rename lib/{post-receive-hook => hooks/post-receive} (100%) delete mode 100644 lib/tasks/gitlab/update_hooks.rake create mode 100644 lib/tasks/gitlab/write_hook.rake diff --git a/app/roles/repository.rb b/app/roles/repository.rb index 8d5b018d..7f1d6f84 100644 --- a/app/roles/repository.rb +++ b/app/roles/repository.rb @@ -30,26 +30,10 @@ module Repository Commit.commits_between(repo, from, to) end - def write_hooks - %w(post-receive).each do |hook| - write_hook(hook, File.read(File.join(Rails.root, 'lib', "#{hook}-hook"))) - end - end - def satellite @satellite ||= Gitlab::Satellite.new(self) end - def write_hook(name, content) - hook_file = File.join(path_to_repo, 'hooks', name) - - File.open(hook_file, 'w') do |f| - f.write(content) - end - - File.chmod(0775, hook_file) - end - def has_post_receive_file? hook_file = File.join(path_to_repo, 'hooks', 'post-receive') File.exists?(hook_file) @@ -73,8 +57,6 @@ module Repository def update_repository Gitlab::GitHost.system.update_project(path, self) - - write_hooks if File.exists?(path_to_repo) end def destroy_repository diff --git a/app/views/errors/gitolite.html.haml b/app/views/errors/gitolite.html.haml index 4788c2e5..50268b1a 100644 --- a/app/views/errors/gitolite.html.haml +++ b/app/views/errors/gitolite.html.haml @@ -23,5 +23,3 @@ = preserve do sudo chmod -R 770 /home/git/repositories/ sudo chown -R git:git /home/git/repositories/ - sudo chown gitlab:gitlab /home/git/repositories/**/hooks/post-receive - diff --git a/lib/post-receive-hook b/lib/hooks/post-receive similarity index 100% rename from lib/post-receive-hook rename to lib/hooks/post-receive diff --git a/lib/tasks/gitlab/setup.rake b/lib/tasks/gitlab/setup.rake index d60e73e9..21ce5d70 100644 --- a/lib/tasks/gitlab/setup.rake +++ b/lib/tasks/gitlab/setup.rake @@ -1,7 +1,12 @@ namespace :gitlab do namespace :app do desc "GITLAB | Setup production application" - task :setup => ['db:setup', 'db:seed_fu', 'gitlab:app:enable_automerge'] + task :setup => [ + 'db:setup', + 'db:seed_fu', + 'gitlab:gitolite:write_hooks', + 'gitlab:app:enable_automerge' + ] end end diff --git a/lib/tasks/gitlab/status.rake b/lib/tasks/gitlab/status.rake index bc4e86ea..a16b1512 100644 --- a/lib/tasks/gitlab/status.rake +++ b/lib/tasks/gitlab/status.rake @@ -67,12 +67,6 @@ namespace :gitlab do next end - - unless File.owned?(hook_file) - puts "post-receive file is not owner by gitlab".red - next - end - puts "post-reveice file ok".green end end diff --git a/lib/tasks/gitlab/update_hooks.rake b/lib/tasks/gitlab/update_hooks.rake deleted file mode 100644 index 44e1617e..00000000 --- a/lib/tasks/gitlab/update_hooks.rake +++ /dev/null @@ -1,19 +0,0 @@ -namespace :gitlab do - namespace :gitolite do - desc "GITLAB | Rewrite hooks for repos" - task :update_hooks => :environment do - puts "Starting Projects" - Project.find_each(:batch_size => 100) do |project| - begin - if project.commit - project.write_hooks - print ".".green - end - rescue Exception => e - print e.message.red - end - end - puts "\nDone with projects" - end - end -end diff --git a/lib/tasks/gitlab/write_hook.rake b/lib/tasks/gitlab/write_hook.rake new file mode 100644 index 00000000..098331b8 --- /dev/null +++ b/lib/tasks/gitlab/write_hook.rake @@ -0,0 +1,23 @@ +namespace :gitlab do + namespace :gitolite do + desc "GITLAB | Write GITLAB hook for gitolite" + task :write_hooks => :environment do + gitolite_hooks_path = File.join("/home", Gitlab.config.ssh_user, "share", "gitolite", "hooks", "common") + gitlab_hooks_path = Rails.root.join("lib", "hooks") + + gitlab_hook_files = ['post-receive'] + + gitlab_hook_files.each do |file_name| + source = File.join(gitlab_hooks_path, file_name) + dest = File.join(gitolite_hooks_path, file_name) + + puts "sudo -u root cp #{source} #{dest}".yellow + `sudo -u root cp #{source} #{dest}` + + puts "sudo -u root chown git:git #{dest}".yellow + `sudo -u root chown git:git #{dest}` + end + end + end +end + From 64f3682feb90c3eb8dbd4f0cd16ad6057b302a6c Mon Sep 17 00:00:00 2001 From: randx Date: Tue, 21 Aug 2012 20:24:04 +0300 Subject: [PATCH 21/46] project should not respond to write_hooks any more --- spec/models/project_spec.rb | 1 - 1 file changed, 1 deletion(-) diff --git a/spec/models/project_spec.rb b/spec/models/project_spec.rb index 5bba4ff6..af193295 100644 --- a/spec/models/project_spec.rb +++ b/spec/models/project_spec.rb @@ -40,7 +40,6 @@ describe Project do it { should respond_to(:commits_with_refs) } it { should respond_to(:commits_since) } it { should respond_to(:commits_between) } - it { should respond_to(:write_hooks) } it { should respond_to(:satellite) } it { should respond_to(:update_repository) } it { should respond_to(:destroy_repository) } From 7fa6a23416e08fe6ac832e60e78bd6fe61f21955 Mon Sep 17 00:00:00 2001 From: randx Date: Tue, 21 Aug 2012 20:53:33 +0300 Subject: [PATCH 22/46] Better fonts for all code like wiki etc --- app/assets/stylesheets/gitlab_bootstrap.scss | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/app/assets/stylesheets/gitlab_bootstrap.scss b/app/assets/stylesheets/gitlab_bootstrap.scss index 550046d0..e3f3e7af 100644 --- a/app/assets/stylesheets/gitlab_bootstrap.scss +++ b/app/assets/stylesheets/gitlab_bootstrap.scss @@ -1,6 +1,11 @@ body { margin-bottom:20px; } + +pre { + font-family:'Menlo', 'Liberation Mono', 'Consolas', 'Courier New', 'andale mono','lucida console',monospace; +} + a { outline: none; color: $link_color; From f088eaa972ae3bc3beddb85adb98d185ea8a2ddc Mon Sep 17 00:00:00 2001 From: randx Date: Tue, 21 Aug 2012 21:26:56 +0300 Subject: [PATCH 23/46] Refactoring & minor css changes --- CHANGELOG | 5 +++++ app/assets/stylesheets/gitlab_bootstrap.scss | 6 +++++- app/assets/stylesheets/themes/ui_mars.scss | 8 ++++++++ app/models/project.rb | 2 +- app/roles/{project_push.rb => push_observer.rb} | 2 +- app/views/team_members/_show.html.haml | 2 +- 6 files changed, 21 insertions(+), 4 deletions(-) rename app/roles/{project_push.rb => push_observer.rb} (99%) diff --git a/CHANGELOG b/CHANGELOG index 80b68b0a..3a916766 100644 --- a/CHANGELOG +++ b/CHANGELOG @@ -3,6 +3,11 @@ v 2.8.0 - Bulk issues update - Issues API - Cucumber coverage increased + - Post-receive files fixed + - UI improved + - Application cleanup + - more cucumber + - capybara-webkit + headless v 2.7.0 - Issue Labels diff --git a/app/assets/stylesheets/gitlab_bootstrap.scss b/app/assets/stylesheets/gitlab_bootstrap.scss index e3f3e7af..8bc63e25 100644 --- a/app/assets/stylesheets/gitlab_bootstrap.scss +++ b/app/assets/stylesheets/gitlab_bootstrap.scss @@ -330,16 +330,20 @@ img.avatar { float:left; margin-right:15px; width:40px; - border:2px solid #ddd; + border:1px solid #ddd; + padding:1px; &.s16 { width:16px; + height:16px; } &.s24 { width:24px; + height:24px; } &.s32 { width:32px; + height:32px; } } diff --git a/app/assets/stylesheets/themes/ui_mars.scss b/app/assets/stylesheets/themes/ui_mars.scss index 2808ad32..c630f388 100644 --- a/app/assets/stylesheets/themes/ui_mars.scss +++ b/app/assets/stylesheets/themes/ui_mars.scss @@ -20,6 +20,10 @@ .fbtn { .btn { + i { + position: relative; + top: 1px; + } margin-left:8px; background-image: -webkit-gradient(linear, 0 0, 0 30, color-stop(0.066, #595D63), to(#31363E)); background-image: -webkit-linear-gradient(#595D63 6.6%, #31363E); @@ -32,6 +36,10 @@ background-image: -moz-linear-gradient(#595D63 6.6%, #202227); background-image: -o-linear-gradient(#595D63 6.6%, #202227); background-position:0 0; + color:#fff; + i { + @extend .icon-white; + } } border: 1px solid #31363E; diff --git a/app/models/project.rb b/app/models/project.rb index 714953c6..3fe11916 100644 --- a/app/models/project.rb +++ b/app/models/project.rb @@ -2,7 +2,7 @@ require "grit" class Project < ActiveRecord::Base include Repository - include ProjectPush + include PushObserver include Authority include Team diff --git a/app/roles/project_push.rb b/app/roles/push_observer.rb similarity index 99% rename from app/roles/project_push.rb rename to app/roles/push_observer.rb index 02025384..1067404d 100644 --- a/app/roles/project_push.rb +++ b/app/roles/push_observer.rb @@ -1,4 +1,4 @@ -module ProjectPush +module PushObserver def observe_push(oldrev, newrev, ref, user) data = post_receive_data(oldrev, newrev, ref, user) diff --git a/app/views/team_members/_show.html.haml b/app/views/team_members/_show.html.haml index f47554c1..2dc4fb65 100644 --- a/app/views/team_members/_show.html.haml +++ b/app/views/team_members/_show.html.haml @@ -9,7 +9,7 @@ %span.label Blocked = link_to project_team_member_path(@project, member), title: user.name, class: "dark" do - = image_tag gravatar_icon(user.email, 40), class: "avatar" + = image_tag gravatar_icon(user.email, 40), class: "avatar s32" = link_to project_team_member_path(@project, member), title: user.name, class: "dark" do %strong= truncate(user.name, lenght: 40) %br From 3095483e650ae4fc79ff5012117877e5734b75ee Mon Sep 17 00:00:00 2001 From: Robert Speicher Date: Tue, 21 Aug 2012 22:20:09 -0400 Subject: [PATCH 24/46] Speed up the "Project Network Graph" cucumber feature This change involves stubbing out the call to `Grit::Commit.find_all` and limiting the number of commits to 10 vs. the standard of 650 used by `Gitlab::GraphCommit.to_graph`. Prior to this change, this single feature required almost 3 minutes of running time and over 2 GB of memory on my machine. Now it takes less than 3 seconds. --- features/projects/network.feature | 4 +--- features/step_definitions/project/projects_steps.rb | 10 ++++++++-- features/support/env.rb | 2 ++ 3 files changed, 11 insertions(+), 5 deletions(-) diff --git a/features/projects/network.feature b/features/projects/network.feature index 9655184c..61c05eb3 100644 --- a/features/projects/network.feature +++ b/features/projects/network.feature @@ -4,9 +4,7 @@ Feature: Project Network Graph Background: Given I signin as a user And I own project "Shop" - And I visit project "Shop" network page + And I visit project "Shop" network page Scenario: I should see project network Then page should have network graph - - diff --git a/features/step_definitions/project/projects_steps.rb b/features/step_definitions/project/projects_steps.rb index c9af346e..1a336ed2 100644 --- a/features/step_definitions/project/projects_steps.rb +++ b/features/step_definitions/project/projects_steps.rb @@ -57,6 +57,11 @@ end Given /^I visit project "(.*?)" network page$/ do |arg1| project = Project.find_by_name(arg1) + + # Stub out find_all to speed this up (10 commits vs. 650) + commits = Grit::Commit.find_all(project.repo, nil, {max_count: 10}) + Grit::Commit.stub(:find_all).and_return(commits) + visit graph_project_path(project) end @@ -67,8 +72,9 @@ end Given /^page should have network graph$/ do page.should have_content "Project Network Graph" within ".graph" do - page.should have_content "stable" - page.should have_content "notes_refacto..." + page.should have_content "master" + page.should have_content "github" + page.should have_content "scss_refactor..." end end diff --git a/features/support/env.rb b/features/support/env.rb index 496f23f9..78d829c1 100644 --- a/features/support/env.rb +++ b/features/support/env.rb @@ -44,3 +44,5 @@ require 'headless' headless = Headless.new headless.start + +require 'cucumber/rspec/doubles' From 6baf9c441d7051276b5bfdbe43499d74d3acc65d Mon Sep 17 00:00:00 2001 From: Robert Speicher Date: Tue, 21 Aug 2012 22:48:15 -0400 Subject: [PATCH 25/46] Fix cucumber failure that only happened on Travis --- features/step_definitions/project/projects_steps.rb | 1 - 1 file changed, 1 deletion(-) diff --git a/features/step_definitions/project/projects_steps.rb b/features/step_definitions/project/projects_steps.rb index 1a336ed2..3ff08d58 100644 --- a/features/step_definitions/project/projects_steps.rb +++ b/features/step_definitions/project/projects_steps.rb @@ -73,7 +73,6 @@ Given /^page should have network graph$/ do page.should have_content "Project Network Graph" within ".graph" do page.should have_content "master" - page.should have_content "github" page.should have_content "scss_refactor..." end end From d1daeba1736ba145fe525ce08a91f29495a3abf1 Mon Sep 17 00:00:00 2001 From: randx Date: Tue, 21 Aug 2012 22:44:49 +0300 Subject: [PATCH 26/46] Updated app:status & docs with hooks. Removed write_hooks from app:setup --- doc/installation.md | 7 ++++++- lib/tasks/gitlab/setup.rake | 1 - lib/tasks/gitlab/status.rake | 14 ++++++++++++++ 3 files changed, 20 insertions(+), 2 deletions(-) diff --git a/doc/installation.md b/doc/installation.md index 5611c3b3..1d32e1b7 100644 --- a/doc/installation.md +++ b/doc/installation.md @@ -119,7 +119,6 @@ Permissions: sudo chmod -R g+rwX /home/git/repositories/ sudo chown -R git:git /home/git/repositories/ - sudo chown gitlab:gitlab /home/git/repositories/**/hooks/post-receive #### CHECK: Logout & login again to apply git group to your user @@ -177,6 +176,11 @@ Permissions: #### Setup DB sudo -u gitlab bundle exec rake gitlab:app:setup RAILS_ENV=production + +#### Setup gitlab hooks + + sudo cp ./lib/hooks/post-receive /home/git/share/gitolite/hooks/common/post-receive + sudo chown git:git /home/git/share/gitolite/hooks/common/post-receive Checking status: @@ -196,6 +200,7 @@ Checking status: Resolving deltas: 100% (174/174), done. Can clone gitolite-admin?............YES UMASK for .gitolite.rc is 0007? ............YES + /home/git/share/gitolite/hooks/common/post-receive exists? ............YES If you got all YES - congrats! You can go to next step. diff --git a/lib/tasks/gitlab/setup.rake b/lib/tasks/gitlab/setup.rake index 21ce5d70..49c86461 100644 --- a/lib/tasks/gitlab/setup.rake +++ b/lib/tasks/gitlab/setup.rake @@ -4,7 +4,6 @@ namespace :gitlab do task :setup => [ 'db:setup', 'db:seed_fu', - 'gitlab:gitolite:write_hooks', 'gitlab:app:enable_automerge' ] end diff --git a/lib/tasks/gitlab/status.rake b/lib/tasks/gitlab/status.rake index a16b1512..02d27d4b 100644 --- a/lib/tasks/gitlab/status.rake +++ b/lib/tasks/gitlab/status.rake @@ -56,6 +56,20 @@ namespace :gitlab do return end + gitolite_hooks_path = File.join("/home", Gitlab.config.ssh_user, "share", "gitolite", "hooks", "common") + gitlab_hook_files = ['post-receive'] + gitlab_hook_files.each do |file_name| + dest = File.join(gitolite_hooks_path, file_name) + print "#{dest} exists? ............" + if File.exists?(dest) + puts "YES".green + else + puts "NO".red + return + end + end + + if Project.count > 0 puts "Validating projects repositories:".yellow Project.find_each(:batch_size => 100) do |project| From 9ec4c2d214b045f3fb207edd67055c4ccaebd381 Mon Sep 17 00:00:00 2001 From: Sytse Sijbrandij Date: Tue, 21 Aug 2012 23:07:11 +0200 Subject: [PATCH 27/46] Show only the commits that are newer in the merge request. --- app/models/merge_request.rb | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/app/models/merge_request.rb b/app/models/merge_request.rb index 47966d66..542817b0 100644 --- a/app/models/merge_request.rb +++ b/app/models/merge_request.rb @@ -88,8 +88,11 @@ class MergeRequest < ActiveRecord::Base end def unmerged_diffs - commits = project.repo.commits_between(target_branch, source_branch).map {|c| Commit.new(c)} - diffs = project.repo.diff(commits.first.prev_commit.id, commits.last.id) rescue [] + # Only show what is new in the source branch compared to the target branch, not the other way around. + # The linex below with merge_base is equivalent to diff with three dots (git diff branch1...branch2) + # From the git documentation: "git diff A...B" is equivalent to "git diff $(git-merge-base A B) B" + common_commit = project.repo.git.native(:merge_base, {}, [target_branch, source_branch]).strip + diffs = project.repo.diff(common_commit, source_branch) end def last_commit From 5c7ed6fa26b47ac71ff6ba04720d85df6d74b200 Mon Sep 17 00:00:00 2001 From: Dmitriy Zaporozhets Date: Wed, 22 Aug 2012 01:15:26 +0300 Subject: [PATCH 28/46] Up to 2.8 --- VERSION | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/VERSION b/VERSION index 3f80e494..834f2629 100644 --- a/VERSION +++ b/VERSION @@ -1 +1 @@ -2.8.0pre +2.8.0 From aa50408ecb5c7355d7cac86f7a59f02ae087714c Mon Sep 17 00:00:00 2001 From: Dmitriy Zaporozhets Date: Wed, 22 Aug 2012 03:56:53 +0300 Subject: [PATCH 29/46] 1. Better message if no ssh key 2. SSH Keys Help page --- app/assets/stylesheets/common.scss | 8 +++ app/assets/stylesheets/gitlab_bootstrap.scss | 5 ++ app/views/dashboard/index.html.haml | 11 +-- app/views/help/index.html.haml | 3 + app/views/help/permissions.html.haml | 4 +- app/views/help/ssh.html.haml | 25 +++++++ app/views/help/system_hooks.html.haml | 4 +- app/views/help/web_hooks.html.haml | 8 +-- app/views/help/workflow.html.haml | 13 ++-- app/views/keys/_form.html.haml | 8 ++- app/views/keys/new.html.haml | 2 +- .../show/_how_to_merge.html.haml | 11 ++- app/views/projects/empty.html.haml | 67 +++++++------------ app/views/shared/_no_ssh.html.haml | 8 +++ config/routes.rb | 1 + 15 files changed, 104 insertions(+), 74 deletions(-) create mode 100644 app/views/help/ssh.html.haml create mode 100644 app/views/shared/_no_ssh.html.haml diff --git a/app/assets/stylesheets/common.scss b/app/assets/stylesheets/common.scss index 6103d05c..3c160358 100644 --- a/app/assets/stylesheets/common.scss +++ b/app/assets/stylesheets/common.scss @@ -735,3 +735,11 @@ li.note { font-size: 12px; } } + +.error_message { + @extend .cred; + border-bottom: 1px solid #D21; + padding-bottom:20px; + text-align:center; + margin-bottom:10px; +} diff --git a/app/assets/stylesheets/gitlab_bootstrap.scss b/app/assets/stylesheets/gitlab_bootstrap.scss index 8bc63e25..03beeaef 100644 --- a/app/assets/stylesheets/gitlab_bootstrap.scss +++ b/app/assets/stylesheets/gitlab_bootstrap.scss @@ -4,6 +4,11 @@ body { pre { font-family:'Menlo', 'Liberation Mono', 'Consolas', 'Courier New', 'andale mono','lucida console',monospace; + + &.dark { + background: #333; + color:#f5f5f5; + } } a { diff --git a/app/views/dashboard/index.html.haml b/app/views/dashboard/index.html.haml index 9fea5acb..ba7d019c 100644 --- a/app/views/dashboard/index.html.haml +++ b/app/views/dashboard/index.html.haml @@ -1,14 +1,7 @@ - if @projects.any? .projects .activities.span8 - - if current_user.require_ssh_key? - .alert.alert-error.padded - %span - You wont be able to pull/push project code unless you - %strong - = link_to new_key_path, class: "vlink" do - add new key - to your profile + = render 'shared/no_ssh' - if @events.any? .content_list= render @events - else @@ -57,5 +50,5 @@ If you will be added to project - it will be displayed here -:javascript +:javascript $(function(){ Pager.init(20); }); diff --git a/app/views/help/index.html.haml b/app/views/help/index.html.haml index 66f7c722..02549577 100644 --- a/app/views/help/index.html.haml +++ b/app/views/help/index.html.haml @@ -31,3 +31,6 @@ %li %h5= link_to "Gitlab Markdown", help_markdown_path + + %li + %h5= link_to "SSH keys", help_ssh_path diff --git a/app/views/help/permissions.html.haml b/app/views/help/permissions.html.haml index 7511d15d..f9287fa0 100644 --- a/app/views/help/permissions.html.haml +++ b/app/views/help/permissions.html.haml @@ -1,6 +1,6 @@ -%h3 Permissions +%h3.page_title Permissions .back_link - = link_to help_path do + = link_to help_path do ← to index %hr diff --git a/app/views/help/ssh.html.haml b/app/views/help/ssh.html.haml new file mode 100644 index 00000000..6a581204 --- /dev/null +++ b/app/views/help/ssh.html.haml @@ -0,0 +1,25 @@ +%h3.page_title SSH Keys +.back_link + = link_to help_path do + ← to index +%hr + +%p.slead + SSH key allows you to establish a secure connection between your computer and Gitlab + +%p.slead + To generate a new SSH key just open your terminal and use code below. + +%pre.dark + ssh-keygen -t rsa -C "#{current_user.email}" + + \# Creates a new ssh key using the provided email + \# Generating public/private rsa key pair... + +%p.slead + Next just use code below to dump your public key and add to GITLAB SSH Keys + +%pre.dark + cat ~/.ssh/id_rsa.pub + + \# ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQC6eNtGpNGwstc.... diff --git a/app/views/help/system_hooks.html.haml b/app/views/help/system_hooks.html.haml index 2088208a..9fc8cbab 100644 --- a/app/views/help/system_hooks.html.haml +++ b/app/views/help/system_hooks.html.haml @@ -1,10 +1,10 @@ %h3 System hooks .back_link - = link_to :back do + = link_to :back do ← back %hr -%p.slead +%p.slead Your Gitlab instance can perform HTTP POST request on next event: create_project, delete_project, create_user, delete_user, change_team_member. %br System Hooks can be used for logging or change information in LDAP server. diff --git a/app/views/help/web_hooks.html.haml b/app/views/help/web_hooks.html.haml index 3acea62c..263eadf6 100644 --- a/app/views/help/web_hooks.html.haml +++ b/app/views/help/web_hooks.html.haml @@ -1,11 +1,11 @@ -%h3 Web hooks +%h3.page_title Web hooks .back_link - = link_to help_path do + = link_to help_path do ← to index %hr -%p.slead - Every Gitlab project can trigger a web server whenever the repo is pushed to. +%p.slead + Every Gitlab project can trigger a web server whenever the repo is pushed to. %br Web Hooks can be used to update an external issue tracker, trigger CI builds, update a backup mirror, or even deploy to your production server. %br diff --git a/app/views/help/workflow.html.haml b/app/views/help/workflow.html.haml index 7db8133b..a3fe3b01 100644 --- a/app/views/help/workflow.html.haml +++ b/app/views/help/workflow.html.haml @@ -1,7 +1,6 @@ -- bash_lexer = Pygments::Lexer[:bash] -%h3 Workflow +%h3.page_title Workflow .back_link - = link_to help_path do + = link_to help_path do ← to index %hr @@ -9,25 +8,25 @@ %li %p Clone project .bash - %pre + %pre.dark git clone git@example.com:project-name.git %li %p Create branch with your feature .bash - %pre + %pre.dark git checkout -b $feature_name %li %p Write code. Commit changes .bash - %pre + %pre.dark git commit -am "My feature is ready" %li %p Push your branch to gitlabhq .bash - %pre + %pre.dark git push origin $feature_name %li diff --git a/app/views/keys/_form.html.haml b/app/views/keys/_form.html.haml index ee2eafdd..9c6e229b 100644 --- a/app/views/keys/_form.html.haml +++ b/app/views/keys/_form.html.haml @@ -11,7 +11,13 @@ .input= f.text_field :title .clearfix = f.label :key - .input= f.text_area :key, class: [:xxlarge, :thin_area] + .input + = f.text_area :key, class: [:xxlarge, :thin_area] + %p.hint + Paste your public key here. Read more about how generate it + = link_to "here", help_ssh_path + + .actions = f.submit 'Save', class: "primary btn" = link_to "Cancel", keys_path, class: "btn" diff --git a/app/views/keys/new.html.haml b/app/views/keys/new.html.haml index 02e782b9..fff38058 100644 --- a/app/views/keys/new.html.haml +++ b/app/views/keys/new.html.haml @@ -1,4 +1,4 @@ -%h3.page_title New key +%h3.page_title Add an SSH Key %hr = render 'form' diff --git a/app/views/merge_requests/show/_how_to_merge.html.haml b/app/views/merge_requests/show/_how_to_merge.html.haml index c21f2727..69881d43 100644 --- a/app/views/merge_requests/show/_how_to_merge.html.haml +++ b/app/views/merge_requests/show/_how_to_merge.html.haml @@ -3,13 +3,12 @@ %a.close{href: "#"} × %h3 How To Merge .modal-body - %pre + %pre.dark = preserve do - :erb - git checkout <%= @merge_request.target_branch %> - git fetch origin - git merge origin/<%= @merge_request.source_branch %> - git push origin <%= @merge_request.target_branch %> + git checkout #{@merge_request.target_branch} + git fetch origin + git merge origin/#{@merge_request.source_branch} + git push origin #{@merge_request.target_branch} :javascript diff --git a/app/views/projects/empty.html.haml b/app/views/projects/empty.html.haml index b8d0dad9..907d5ef4 100644 --- a/app/views/projects/empty.html.haml +++ b/app/views/projects/empty.html.haml @@ -1,47 +1,30 @@ -- if current_user.require_ssh_key? - .alert-message.block-message.error - %ul - %li You have no ssh keys added to your profile. - %li You wont be able to pull/push repository. - %li Visit profile → keys and add public key of every machine you want to use for work with gitlabhq. - -.alert-message.block-message.error - %ul.unstyled.alert_holder - %li You should push repository to proceed. - %li After push you will be able to browse code, commits etc. - -- bash_lexer = Pygments::Lexer[:bash] += render 'shared/no_ssh' %div.git-empty - %h3 Git global setup: - - setup_str = ["git config --global user.name \"#{current_user.name}\"", - "git config --global user.email \"#{current_user.email}\""].join("\n") - = preserve do - = raw bash_lexer.highlight(setup_str, lexer: 'bash', options: {encoding: 'utf-8'}) + %h4 Git global setup: + %pre.dark + = preserve do + git config --global user.name "#{current_user.name}" + git config --global user.email "#{current_user.email}" - %br - %br - %h3 Create Repository - - repo_setup_str = ["mkdir #{@project.path}", - "cd #{@project.path}", - "git init", - "touch README", - "git add README", - "git commit -m 'first commit'", - "git remote add origin #{@project.url_to_repo}", - "git push -u origin master"].join("\n") + %h4.prepend-top-20 Create Repository + %pre.dark + = preserve do + mkdir #{@project.path} + cd #{@project.path} + git init + touch README + git add README + git commit -m 'first commit' + git remote add origin #{@project.url_to_repo} + git push -u origin master - = preserve do - = raw bash_lexer.highlight(repo_setup_str) - - %br - %br - %h3 Existing Git Repo? - - exist_repo_setup_str = ["cd existing_git_repo", - "git remote add origin #{@project.url_to_repo}", - "git push -u origin master"].join("\n") - = preserve do - = raw bash_lexer.highlight(exist_repo_setup_str) + %h4.prepend-top-20 Existing Git Repo? + %pre.dark + = preserve do + cd existing_git_repo + git remote add origin #{@project.url_to_repo} + git push -u origin master - if can? current_user, :admin_project, @project - .alert-message.block-message.error.prepend-top-20 - = link_to 'Remove project', @project, confirm: 'Are you sure?', method: :delete, class: "btn danger" + .prepend-top-20 + = link_to 'Remove project', @project, confirm: 'Are you sure?', method: :delete, class: "btn danger right" diff --git a/app/views/shared/_no_ssh.html.haml b/app/views/shared/_no_ssh.html.haml new file mode 100644 index 00000000..b6ab666b --- /dev/null +++ b/app/views/shared/_no_ssh.html.haml @@ -0,0 +1,8 @@ +- if current_user.require_ssh_key? + %h6.error_message + %span + You wont be able to pull/push project code unless you + %strong + = link_to new_key_path, class: "vlink" do + add SSH key + to your profile diff --git a/config/routes.rb b/config/routes.rb index 04e13bc4..97594d57 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -30,6 +30,7 @@ Gitlab::Application.routes.draw do get 'help/web_hooks' => 'help#web_hooks' get 'help/system_hooks' => 'help#system_hooks' get 'help/markdown' => 'help#markdown' + get 'help/ssh' => 'help#ssh' # # Admin Area From 6428d4dd63d112e3d1b2c3fa2f751651318a96d4 Mon Sep 17 00:00:00 2001 From: Dmitriy Zaporozhets Date: Thu, 23 Aug 2012 08:15:30 +0300 Subject: [PATCH 30/46] gitlab_meta gem added --- Gemfile | 1 + Gemfile.lock | 2 ++ 2 files changed, 3 insertions(+) diff --git a/Gemfile b/Gemfile index e8b0b244..c3a01cb5 100644 --- a/Gemfile +++ b/Gemfile @@ -75,6 +75,7 @@ gem 'settingslogic' # Misc gem "foreman" gem "git" +gem "gitlab_meta" group :assets do gem "sass-rails", "3.2.5" diff --git a/Gemfile.lock b/Gemfile.lock index b23bc47c..b8bc95c6 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -172,6 +172,7 @@ GEM gherkin (2.11.0) json (>= 1.4.6) git (1.2.5) + gitlab_meta (2.8.1) grape (0.2.1) hashie (~> 1.2) multi_json @@ -394,6 +395,7 @@ DEPENDENCIES ffaker foreman git + gitlab_meta gitolite! grack! grape (~> 0.2.1) From a94b2d1ccfa17f92d2d17f20a23ab86e4456291e Mon Sep 17 00:00:00 2001 From: Dmitriy Zaporozhets Date: Thu, 23 Aug 2012 08:28:00 +0300 Subject: [PATCH 31/46] Fix meta gem version --- Gemfile | 2 +- Gemfile.lock | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/Gemfile b/Gemfile index c3a01cb5..f256eccb 100644 --- a/Gemfile +++ b/Gemfile @@ -75,7 +75,7 @@ gem 'settingslogic' # Misc gem "foreman" gem "git" -gem "gitlab_meta" +gem "gitlab_meta", '2.8' group :assets do gem "sass-rails", "3.2.5" diff --git a/Gemfile.lock b/Gemfile.lock index b8bc95c6..b2a00118 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -172,7 +172,7 @@ GEM gherkin (2.11.0) json (>= 1.4.6) git (1.2.5) - gitlab_meta (2.8.1) + gitlab_meta (2.8) grape (0.2.1) hashie (~> 1.2) multi_json @@ -395,7 +395,7 @@ DEPENDENCIES ffaker foreman git - gitlab_meta + gitlab_meta (= 2.8) gitolite! grack! grape (~> 0.2.1) From ed2b53cd1c34c421b23208eeb502a141a6829f9d Mon Sep 17 00:00:00 2001 From: Dmitriy Zaporozhets Date: Thu, 23 Aug 2012 09:17:18 +0300 Subject: [PATCH 32/46] Up to 2.8.1 --- CHANGELOG | 5 +++++ VERSION | 2 +- 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/CHANGELOG b/CHANGELOG index 3a916766..6868c07b 100644 --- a/CHANGELOG +++ b/CHANGELOG @@ -1,3 +1,8 @@ +v 2.8.1 + - ability to disable gravatars + - improved MR diff logic + - ssh key help page + v 2.8.0 - Gitlab Flavored Markdown - Bulk issues update diff --git a/VERSION b/VERSION index 834f2629..dbe59006 100644 --- a/VERSION +++ b/VERSION @@ -1 +1 @@ -2.8.0 +2.8.1 From db4c3e58bca67bf773d9fe3e73acd6b687498334 Mon Sep 17 00:00:00 2001 From: Robert Speicher Date: Thu, 23 Aug 2012 07:24:25 -0400 Subject: [PATCH 33/46] Don't run SimpleCov on Travis --- features/support/env.rb | 6 ++++-- spec/spec_helper.rb | 6 ++++-- 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/features/support/env.rb b/features/support/env.rb index 78d829c1..b69a5fe7 100644 --- a/features/support/env.rb +++ b/features/support/env.rb @@ -1,5 +1,7 @@ -require 'simplecov' -SimpleCov.start 'rails' +unless ENV['CI'] + require 'simplecov' + SimpleCov.start 'rails' +end require 'cucumber/rails' require 'webmock/cucumber' diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index 5c0bb618..cd931475 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -1,5 +1,7 @@ -require 'simplecov' -SimpleCov.start 'rails' +unless ENV['CI'] + require 'simplecov' + SimpleCov.start 'rails' +end # This file is copied to spec/ when you run 'rails generate rspec:install' ENV["RAILS_ENV"] ||= 'test' From 32ae7fb616673a003ba24dcde1ded3bad8ff37d1 Mon Sep 17 00:00:00 2001 From: Riyad Preukschas Date: Thu, 23 Aug 2012 20:10:06 +0200 Subject: [PATCH 34/46] Improve GFM code documentation --- app/helpers/gitlab_markdown_helper.rb | 21 +++++++++++++++++++-- lib/gitlab/markdown.rb | 19 ++++++++++++++----- 2 files changed, 33 insertions(+), 7 deletions(-) diff --git a/app/helpers/gitlab_markdown_helper.rb b/app/helpers/gitlab_markdown_helper.rb index 24bc3e85..88e3473b 100644 --- a/app/helpers/gitlab_markdown_helper.rb +++ b/app/helpers/gitlab_markdown_helper.rb @@ -1,9 +1,18 @@ module GitlabMarkdownHelper + # Replaces references (i.e. @abc, #123, !456, ...) in the text with links to + # the appropriate items in Gitlab. + # + # text - the source text + # html_options - extra options for the reference links as given to link_to + # + # note: reference links will only be generated if @project is set + # + # see Gitlab::Markdown for details on the supported syntax def gfm(text, html_options = {}) return text if text.nil? return text if @project.nil? - # Extract pre blocks + # Extract pre blocks so they are not altered # from http://github.github.com/github-flavored-markdown/ extractions = {} text.gsub!(%r{
    .*?
    |.*?}m) do |match| @@ -25,7 +34,15 @@ module GitlabMarkdownHelper text.html_safe end - # circumvents nesting links, which will behave bad in browsers + # Use this in places where you would normally use link_to(gfm(...), ...). + # + # It solves a problem occurring with nested links (i.e. + # "outer text gfm ref more outer text"). This will not be + # interpreted as intended. Browsers will parse something like + # "outer text gfm ref more outer text" (notice the last part is + # not linked any more). link_to_gfm corrects that. It wraps all parts to + # explicitly produce the correct linking behavior (i.e. + # "outer text gfm ref more outer text"). def link_to_gfm(body, url, html_options = {}) gfm_body = gfm(body, html_options) diff --git a/lib/gitlab/markdown.rb b/lib/gitlab/markdown.rb index d3daed91..75fa835d 100644 --- a/lib/gitlab/markdown.rb +++ b/lib/gitlab/markdown.rb @@ -1,5 +1,14 @@ module Gitlab - # Custom parsing for Gitlab-flavored Markdown + # Custom parser for Gitlab-flavored Markdown + # + # It replaces references in the text with links to the appropriate items in Gitlab. + # + # Supported reference formats are: + # * @foo for team members + # * #123 for issues + # * !123 for merge requests + # * $123 for snippets + # * 123456 for commits # # Examples # @@ -67,25 +76,25 @@ module Gitlab def reference_user(identifier) if user = @project.users.where(name: identifier).first member = @project.users_projects.where(user_id: user).first - link_to("@#{user.name}", project_team_member_path(@project, member), html_options.merge(class: "gfm gfm-team_member #{html_options[:class]}")) if member + link_to("@#{identifier}", project_team_member_path(@project, member), html_options.merge(class: "gfm gfm-team_member #{html_options[:class]}")) if member end end def reference_issue(identifier) if issue = @project.issues.where(id: identifier).first - link_to("##{issue.id}", project_issue_path(@project, issue), html_options.merge(title: "Issue: #{issue.title}", class: "gfm gfm-issue #{html_options[:class]}")) + link_to("##{identifier}", project_issue_path(@project, issue), html_options.merge(title: "Issue: #{issue.title}", class: "gfm gfm-issue #{html_options[:class]}")) end end def reference_merge_request(identifier) if merge_request = @project.merge_requests.where(id: identifier).first - link_to("!#{merge_request.id}", project_merge_request_path(@project, merge_request), html_options.merge(title: "Merge Request: #{merge_request.title}", class: "gfm gfm-merge_request #{html_options[:class]}")) + link_to("!#{identifier}", project_merge_request_path(@project, merge_request), html_options.merge(title: "Merge Request: #{merge_request.title}", class: "gfm gfm-merge_request #{html_options[:class]}")) end end def reference_snippet(identifier) if snippet = @project.snippets.where(id: identifier).first - link_to("$#{snippet.id}", project_snippet_path(@project, snippet), html_options.merge(title: "Snippet: #{snippet.title}", class: "gfm gfm-snippet #{html_options[:class]}")) + link_to("$#{identifier}", project_snippet_path(@project, snippet), html_options.merge(title: "Snippet: #{snippet.title}", class: "gfm gfm-snippet #{html_options[:class]}")) end end From e2f19befbcc2c0b347507171ef94047426aba5b0 Mon Sep 17 00:00:00 2001 From: Cyril Date: Thu, 23 Aug 2012 23:04:43 +0200 Subject: [PATCH 35/46] fix inline forms --- app/assets/stylesheets/sections/projects.scss | 1 + app/views/admin/hooks/index.html.haml | 2 +- app/views/admin/projects/index.html.haml | 2 +- app/views/admin/users/index.html.haml | 2 +- app/views/hooks/index.html.haml | 2 +- app/views/search/show.html.haml | 2 +- 6 files changed, 6 insertions(+), 5 deletions(-) diff --git a/app/assets/stylesheets/sections/projects.scss b/app/assets/stylesheets/sections/projects.scss index 1269ac8a..0866b43f 100644 --- a/app/assets/stylesheets/sections/projects.scss +++ b/app/assets/stylesheets/sections/projects.scss @@ -35,6 +35,7 @@ .btn { padding:6px; margin-left:10px; + margin-bottom:8px; } } } diff --git a/app/views/admin/hooks/index.html.haml b/app/views/admin/hooks/index.html.haml index d34acffe..43288424 100644 --- a/app/views/admin/hooks/index.html.haml +++ b/app/views/admin/hooks/index.html.haml @@ -5,7 +5,7 @@ Read more about system hooks %strong #{link_to "here", help_system_hooks_path, class: "vlink"} -= form_for @hook, as: :hook, url: admin_hooks_path do |f| += form_for @hook, as: :hook, url: admin_hooks_path, html: { class: 'form-inline' } do |f| -if @hook.errors.any? .alert-message.block-message.error - @hook.errors.full_messages.each do |msg| diff --git a/app/views/admin/projects/index.html.haml b/app/views/admin/projects/index.html.haml index 4512bb7e..882b2ab5 100644 --- a/app/views/admin/projects/index.html.haml +++ b/app/views/admin/projects/index.html.haml @@ -2,7 +2,7 @@ Projects = link_to 'New Project', new_admin_project_path, class: "btn small right" %br -= form_tag admin_projects_path, method: :get do += form_tag admin_projects_path, method: :get, class: 'form-inline' do = text_field_tag :name, params[:name], class: "xlarge" = submit_tag "Search", class: "btn submit primary" diff --git a/app/views/admin/users/index.html.haml b/app/views/admin/users/index.html.haml index 5edca312..3d027217 100644 --- a/app/views/admin/users/index.html.haml +++ b/app/views/admin/users/index.html.haml @@ -3,7 +3,7 @@ = link_to 'New User', new_admin_user_path, class: "btn small right" %br -= form_tag admin_users_path, method: :get do += form_tag admin_users_path, method: :get, class: 'form-inline' do = text_field_tag :name, params[:name], class: "xlarge" = submit_tag "Search", class: "btn submit primary" %ul.nav.nav-pills diff --git a/app/views/hooks/index.html.haml b/app/views/hooks/index.html.haml index 4e15dc50..3d2a381e 100644 --- a/app/views/hooks/index.html.haml +++ b/app/views/hooks/index.html.haml @@ -8,7 +8,7 @@ Read more about web hooks %strong #{link_to "here", help_web_hooks_path, class: "vlink"} -= form_for [@project, @hook], as: :hook, url: project_hooks_path(@project) do |f| += form_for [@project, @hook], as: :hook, url: project_hooks_path(@project), html: { class: 'form-inline' } do |f| -if @hook.errors.any? .alert-message.block-message.error - @hook.errors.full_messages.each do |msg| diff --git a/app/views/search/show.html.haml b/app/views/search/show.html.haml index ed9da1f0..9a0b4789 100644 --- a/app/views/search/show.html.haml +++ b/app/views/search/show.html.haml @@ -1,4 +1,4 @@ -= form_tag search_path, method: :get do |f| += form_tag search_path, method: :get, class: 'form-inline' do |f| .padded = label_tag :search do %strong Looking for From 2c128f3874eabf642ba78775e169177129f10a97 Mon Sep 17 00:00:00 2001 From: Robert Speicher Date: Thu, 23 Aug 2012 19:20:28 -0400 Subject: [PATCH 36/46] Remove rails-footnotes gem and its related files --- .rails_footnotes | 3 --- Gemfile | 1 - Gemfile.lock | 3 --- config/initializers/rails_footnotes.rb | 3 --- 4 files changed, 10 deletions(-) delete mode 100644 .rails_footnotes delete mode 100644 config/initializers/rails_footnotes.rb diff --git a/.rails_footnotes b/.rails_footnotes deleted file mode 100644 index 1019a70a..00000000 --- a/.rails_footnotes +++ /dev/null @@ -1,3 +0,0 @@ -#this code temporarily disables notes for all controllers -# Footnotes::Filter.notes = [] - diff --git a/Gemfile b/Gemfile index f256eccb..214c06a9 100644 --- a/Gemfile +++ b/Gemfile @@ -93,7 +93,6 @@ end group :development do gem "letter_opener" - gem "rails-footnotes" gem "annotate", :git => "https://github.com/ctran/annotate_models.git" gem 'rack-mini-profiler' end diff --git a/Gemfile.lock b/Gemfile.lock index b2a00118..c2d60d79 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -259,8 +259,6 @@ GEM activesupport (= 3.2.8) bundler (~> 1.0) railties (= 3.2.8) - rails-footnotes (3.7.8) - rails (>= 3.0.0) railties (3.2.8) actionpack (= 3.2.8) activesupport (= 3.2.8) @@ -417,7 +415,6 @@ DEPENDENCIES pygments.rb! rack-mini-profiler rails (= 3.2.8) - rails-footnotes raphael-rails (= 1.5.2) redcarpet (~> 2.1.1) resque (~> 1.20.0) diff --git a/config/initializers/rails_footnotes.rb b/config/initializers/rails_footnotes.rb deleted file mode 100644 index afe6f3ad..00000000 --- a/config/initializers/rails_footnotes.rb +++ /dev/null @@ -1,3 +0,0 @@ -#if defined?(Footnotes) && Rails.env.development? - #Footnotes.run! # first of all -#end From 4adac4deba315c182820a1fe820fe0d297ae21ab Mon Sep 17 00:00:00 2001 From: Robert Speicher Date: Thu, 23 Aug 2012 19:21:20 -0400 Subject: [PATCH 37/46] Move gitlab_meta gem to :production group --- Gemfile | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/Gemfile b/Gemfile index 214c06a9..73daebec 100644 --- a/Gemfile +++ b/Gemfile @@ -75,7 +75,6 @@ gem 'settingslogic' # Misc gem "foreman" gem "git" -gem "gitlab_meta", '2.8' group :assets do gem "sass-rails", "3.2.5" @@ -120,3 +119,7 @@ group :test do gem 'resque_spec' gem "webmock" end + +group :production do + gem "gitlab_meta", '2.8' +end From af2e2e29b9b537f6bae752b74280ca7a3819d468 Mon Sep 17 00:00:00 2001 From: Robert Speicher Date: Thu, 23 Aug 2012 19:21:55 -0400 Subject: [PATCH 38/46] Remove unused minitest and turn gems --- Gemfile | 2 -- Gemfile.lock | 6 ------ 2 files changed, 8 deletions(-) diff --git a/Gemfile b/Gemfile index 73daebec..4629383e 100644 --- a/Gemfile +++ b/Gemfile @@ -111,8 +111,6 @@ end group :test do gem 'cucumber-rails', :require => false - gem 'minitest', ">= 2.10" - gem "turn", :require => false gem "simplecov", :require => false gem "shoulda-matchers" gem 'email_spec' diff --git a/Gemfile.lock b/Gemfile.lock index c2d60d79..88da1cd9 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -99,7 +99,6 @@ GEM acts-as-taggable-on (2.3.1) rails (~> 3.0) addressable (2.2.8) - ansi (1.4.2) arel (3.0.2) autotest (4.4.6) ZenTest (>= 4.4.1) @@ -219,7 +218,6 @@ GEM treetop (~> 1.4.8) method_source (0.7.1) mime-types (1.19) - minitest (3.1.0) modernizr (2.5.3) sprockets (~> 2.0) multi_json (1.3.6) @@ -348,8 +346,6 @@ GEM treetop (1.4.10) polyglot polyglot (>= 0.3.1) - turn (0.9.5) - ansi tzinfo (0.3.33) uglifier (1.0.3) execjs (>= 0.3.0) @@ -407,7 +403,6 @@ DEPENDENCIES launchy letter_opener linguist (~> 1.0.0)! - minitest (>= 2.10) modernizr (= 2.5.3) mysql2 omniauth-ldap! @@ -431,7 +426,6 @@ DEPENDENCIES stamp therubyracer thin - turn uglifier (= 1.0.3) unicorn webmock From f9ab136abdedb1ea1085e4e0a881371d6ad0dffb Mon Sep 17 00:00:00 2001 From: Steve Klabnik Date: Thu, 23 Aug 2012 17:01:30 -0700 Subject: [PATCH 39/46] Misspelled Draper :) --- app/decorators/application_decorator.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/decorators/application_decorator.rb b/app/decorators/application_decorator.rb index 7bc88648..3023699e 100644 --- a/app/decorators/application_decorator.rb +++ b/app/decorators/application_decorator.rb @@ -1,4 +1,4 @@ -class ApplicationDecorator < Drapper::Base +class ApplicationDecorator < Draper::Base # Lazy Helpers # PRO: Call Rails helpers without the h. proxy # ex: number_to_currency(model.price) From 4d9197efa819730de6ed4b927dc2c34270be1ec6 Mon Sep 17 00:00:00 2001 From: Steve Klabnik Date: Thu, 23 Aug 2012 17:11:37 -0700 Subject: [PATCH 40/46] Fix Draper spelling in Gemfile --- Gemfile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Gemfile b/Gemfile index f256eccb..2d448ced 100644 --- a/Gemfile +++ b/Gemfile @@ -54,7 +54,7 @@ gem "unicorn" gem "acts-as-taggable-on", "2.3.1" # Decorators -gem "drapper" +gem "draper" # Background jobs gem "resque", "~> 1.20.0" From b962bcd4a7fbe017430f4f0ea84074088fc8c8b1 Mon Sep 17 00:00:00 2001 From: randx Date: Wed, 22 Aug 2012 01:25:11 +0300 Subject: [PATCH 41/46] draper Gemfile.lock --- Gemfile.lock | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/Gemfile.lock b/Gemfile.lock index 88da1cd9..656bede4 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -155,7 +155,9 @@ GEM railties (~> 3.1) warden (~> 1.2.1) diff-lcs (1.1.3) - drapper (0.8.4) + draper (0.17.0) + actionpack (~> 3.2) + activesupport (~> 3.2) email_spec (1.2.1) mail (~> 2.2) rspec (~> 2.0) @@ -384,7 +386,7 @@ DEPENDENCIES cucumber-rails database_cleaner devise (~> 2.1.0) - drapper + draper email_spec ffaker foreman From 4381084af341684240b1a671d368511afcf5774a Mon Sep 17 00:00:00 2001 From: randx Date: Fri, 24 Aug 2012 11:48:30 +0300 Subject: [PATCH 42/46] Trying to get rid of random dashboard/issues cucumber fail --- features/step_definitions/dashboard_steps.rb | 16 ++++------------ 1 file changed, 4 insertions(+), 12 deletions(-) diff --git a/features/step_definitions/dashboard_steps.rb b/features/step_definitions/dashboard_steps.rb index 90c3a69c..1eec7619 100644 --- a/features/step_definitions/dashboard_steps.rb +++ b/features/step_definitions/dashboard_steps.rb @@ -91,26 +91,18 @@ Then /^I should see my merge requests$/ do end Given /^I have assigned issues$/ do - project1 = Factory :project, - :path => "project1", - :code => "gitlabhq_1" - - project2 = Factory :project, - :path => "project2", - :code => "gitlabhq_2" - - project1.add_access(@user, :read, :write) - project2.add_access(@user, :read, :write) + project = Factory :project + project.add_access(@user, :read, :write) issue1 = Factory :issue, :author => @user, :assignee => @user, - :project => project1 + :project => project issue2 = Factory :issue, :author => @user, :assignee => @user, - :project => project2 + :project => project end Given /^I have authored merge requests$/ do From a502f67c0b358cc6b391df0c5dca48375c21fcad Mon Sep 17 00:00:00 2001 From: randx Date: Fri, 24 Aug 2012 12:03:48 +0300 Subject: [PATCH 43/46] Up to 2.8.2 --- VERSION | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/VERSION b/VERSION index dbe59006..1817afea 100644 --- a/VERSION +++ b/VERSION @@ -1 +1 @@ -2.8.1 +2.8.2 From 14bd9c9228c50d2a9c5914394603d7ff2b2aa145 Mon Sep 17 00:00:00 2001 From: randx Date: Fri, 24 Aug 2012 12:17:05 +0300 Subject: [PATCH 44/46] Fix dashboard issues atom feed rspec --- spec/requests/atom/dashboard_issues_spec.rb | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/spec/requests/atom/dashboard_issues_spec.rb b/spec/requests/atom/dashboard_issues_spec.rb index 9b4ffc0e..1d208c70 100644 --- a/spec/requests/atom/dashboard_issues_spec.rb +++ b/spec/requests/atom/dashboard_issues_spec.rb @@ -7,11 +7,11 @@ describe "User Issues Dashboard" do login_as :user @project1 = Factory :project, - path: "project1", + path: "gitlabhq_0", code: "TEST1" @project2 = Factory :project, - path: "project2", + path: "gitlabhq_1", code: "TEST2" @project1.add_access(@user, :read, :write) From 3dd7703b8083d44c64a03cef6b72d161aed04239 Mon Sep 17 00:00:00 2001 From: randx Date: Fri, 24 Aug 2012 13:05:40 +0300 Subject: [PATCH 45/46] Feature: Labels page. Index page --- app/controllers/labels_controller.rb | 25 +++++++++++++++++++ app/views/issues/_head.html.haml | 3 +++ app/views/labels/_label.html.haml | 4 +++ app/views/labels/index.html.haml | 14 +++++++++++ config/routes.rb | 2 ++ features/projects/issues/labels.feature | 13 ++++++++++ .../project/project_issues_steps.rb | 21 +++++++++++++++- 7 files changed, 81 insertions(+), 1 deletion(-) create mode 100644 app/controllers/labels_controller.rb create mode 100644 app/views/labels/_label.html.haml create mode 100644 app/views/labels/index.html.haml create mode 100644 features/projects/issues/labels.feature diff --git a/app/controllers/labels_controller.rb b/app/controllers/labels_controller.rb new file mode 100644 index 00000000..f52fc2d8 --- /dev/null +++ b/app/controllers/labels_controller.rb @@ -0,0 +1,25 @@ +class LabelsController < ApplicationController + before_filter :authenticate_user! + before_filter :project + before_filter :module_enabled + + layout "project" + + # Authorize + before_filter :add_project_abilities + + # Allow read any issue + before_filter :authorize_read_issue! + + respond_to :js, :html + + def index + @labels = Issue.tag_counts_on(:labels) + end + + protected + + def module_enabled + return render_404 unless @project.issues_enabled + end +end diff --git a/app/views/issues/_head.html.haml b/app/views/issues/_head.html.haml index 1f6e7d7f..8ebe3e05 100644 --- a/app/views/issues/_head.html.haml +++ b/app/views/issues/_head.html.haml @@ -5,6 +5,9 @@ %li{class: "#{'active' if current_page?(project_milestones_path(@project))}"} = link_to project_milestones_path(@project), class: "tab" do Milestones + %li{class: "#{'active' if current_page?(project_labels_path(@project))}"} + = link_to project_labels_path(@project), class: "tab" do + Labels %li.right %span.rss-icon = link_to project_issues_path(@project, :atom, { private_token: current_user.private_token }) do diff --git a/app/views/labels/_label.html.haml b/app/views/labels/_label.html.haml new file mode 100644 index 00000000..32158c20 --- /dev/null +++ b/app/views/labels/_label.html.haml @@ -0,0 +1,4 @@ +%li.wll + %strong= label.name + .right + %span= pluralize label.count, 'issue' diff --git a/app/views/labels/index.html.haml b/app/views/labels/index.html.haml new file mode 100644 index 00000000..4e41d375 --- /dev/null +++ b/app/views/labels/index.html.haml @@ -0,0 +1,14 @@ += render "issues/head" + +%h3.page_title + Labels +%br +%div.ui-box + %ul.unstyled.labels-table + - @labels.each do |label| + = render 'label', label: label + + - unless @labels.present? + %li + %h3.nothing_here_message Nothing to show here + diff --git a/config/routes.rb b/config/routes.rb index 97594d57..f895478f 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -197,7 +197,9 @@ Gitlab::Application.routes.draw do end resources :team_members resources :milestones + resources :labels, :only => [:index] resources :issues do + collection do post :sort post :bulk_update diff --git a/features/projects/issues/labels.feature b/features/projects/issues/labels.feature new file mode 100644 index 00000000..5a20bfd6 --- /dev/null +++ b/features/projects/issues/labels.feature @@ -0,0 +1,13 @@ +Feature: Labels + Background: + Given I signin as a user + And I own project "Shop" + And project "Shop" have issues tags: + | name | + | bug | + | feature | + Given I visit project "Shop" labels page + + Scenario: I should see active milestones + Then I should see label "bug" + And I should see label "feature" diff --git a/features/step_definitions/project/project_issues_steps.rb b/features/step_definitions/project/project_issues_steps.rb index 00a1721f..27de03d5 100644 --- a/features/step_definitions/project/project_issues_steps.rb +++ b/features/step_definitions/project/project_issues_steps.rb @@ -33,6 +33,25 @@ Given /^I visit issue page "(.*?)"$/ do |arg1| end Given /^I submit new issue "(.*?)"$/ do |arg1| - fill_in "issue_title", :with => arg1 + fill_in "issue_title", with: arg1 click_button "Submit new issue" end + +Given /^project "(.*?)" have issues tags:$/ do |arg1, table| + project = Project.find_by_name(arg1) + table.hashes.each do |hash| + Factory :issue, + project: project, + label_list: [hash[:name]] + end +end + +Given /^I visit project "(.*?)" labels page$/ do |arg1| + visit project_labels_path(Project.find_by_name(arg1)) +end + +Then /^I should see label "(.*?)"$/ do |arg1| + within ".labels-table" do + page.should have_content arg1 + end +end From 6d4ae75f544d9819954865c105414a722344336a Mon Sep 17 00:00:00 2001 From: randx Date: Fri, 24 Aug 2012 13:17:19 +0300 Subject: [PATCH 46/46] Final fix for dashboard cucumber feature --- features/step_definitions/dashboard_steps.rb | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/features/step_definitions/dashboard_steps.rb b/features/step_definitions/dashboard_steps.rb index 1eec7619..d910ec90 100644 --- a/features/step_definitions/dashboard_steps.rb +++ b/features/step_definitions/dashboard_steps.rb @@ -107,11 +107,11 @@ end Given /^I have authored merge requests$/ do project1 = Factory :project, - :path => "project1", + :path => "gitlabhq_1", :code => "gitlabhq_1" project2 = Factory :project, - :path => "project2", + :path => "gitlabhq_2", :code => "gitlabhq_2" project1.add_access(@user, :read, :write)