BigW Consortium Gitlab

Commit 6f449c63 by Alfredo Sumaran

Apply styling and tweaks to autocomplete dropdown

parent 4b3d3446
...@@ -14,4 +14,36 @@ $.widget( "custom.catcomplete", $.ui.autocomplete, ...@@ -14,4 +14,36 @@ $.widget( "custom.catcomplete", $.ui.autocomplete,
if item.category? if item.category?
li.attr('aria-label', item.category + " : " + item.label) li.attr('aria-label', item.category + " : " + item.label)
_renderItem: (ul, item) ->
# Highlight occurrences
item.label = item.label.replace(new RegExp("(?![^&;]+;)(?!<[^<>]*)(" + $.ui.autocomplete.escapeRegex(this.term) + ")(?![^<>]*>)(?![^&;]+;)", "gi"), "<strong>$1</strong>");
return $( "<li></li>" )
.data( "item.autocomplete", item )
.append( "<a>#{item.label}</a>" )
.appendTo( ul );
_resizeMenu: ->
if (isNaN(this.options.maxShowItems))
return
ul = this.menu.element.css(overflowX: '', overflowY: '', width: '', maxHeight: '')
lis = ul.children('li').css('whiteSpace', 'nowrap');
if (lis.length > this.options.maxShowItems)
ulW = ul.prop('clientWidth')
ul.css(
overflowX: 'hidden'
overflowY: 'auto'
maxHeight: lis.eq(0).outerHeight() * this.options.maxShowItems + 1
)
barW = ulW - ul.prop('clientWidth');
ul.width('+=' + barW);
# Original code from jquery.ui.autocomplete.js _resizeMenu()
ul.outerWidth(Math.max(ul.outerWidth() + 1, this.element.outerWidth()));
) )
...@@ -24,7 +24,10 @@ class @SearchAutocomplete ...@@ -24,7 +24,10 @@ class @SearchAutocomplete
@scopeInputEl = @$('#scope') @scopeInputEl = @$('#scope')
@saveOriginalState() @saveOriginalState()
@createAutocomplete()
if @locationBadgeEl.is(':empty')
@createAutocomplete()
@bindEvents() @bindEvents()
$: (selector) -> $: (selector) ->
...@@ -66,6 +69,12 @@ class @SearchAutocomplete ...@@ -66,6 +69,12 @@ class @SearchAutocomplete
appendTo: 'form.navbar-form' appendTo: 'form.navbar-form'
source: @autocompletePath + @query source: @autocompletePath + @query
minLength: 1 minLength: 1
maxShowItems: 15
position:
# { my: "left top", at: "left bottom", collision: "none" }
my: "left-10 top+9"
at: "left bottom"
collision: "none"
close: (e) -> close: (e) ->
e.preventDefault() e.preventDefault()
...@@ -89,7 +98,9 @@ class @SearchAutocomplete ...@@ -89,7 +98,9 @@ class @SearchAutocomplete
bindEvents: -> bindEvents: ->
@searchInput.on 'keydown', @onSearchKeyDown @searchInput.on 'keydown', @onSearchInputKeyDown
@searchInput.on 'focus', @onSearchInputFocus
@searchInput.on 'blur', @onSearchInputBlur
@wrap.on 'click', '.remove-badge', @onRemoveLocationBadgeClick @wrap.on 'click', '.remove-badge', @onRemoveLocationBadgeClick
onRemoveLocationBadgeClick: (e) => onRemoveLocationBadgeClick: (e) =>
...@@ -97,7 +108,7 @@ class @SearchAutocomplete ...@@ -97,7 +108,7 @@ class @SearchAutocomplete
@removeLocationBadge() @removeLocationBadge()
@searchInput.focus() @searchInput.focus()
onSearchKeyDown: (e) => onSearchInputKeyDown: (e) =>
# Remove tag when pressing backspace and input search is empty # Remove tag when pressing backspace and input search is empty
if e.keyCode is @keyCode.BACKSPACE and e.currentTarget.value is '' if e.keyCode is @keyCode.BACKSPACE and e.currentTarget.value is ''
@removeLocationBadge() @removeLocationBadge()
...@@ -106,14 +117,24 @@ class @SearchAutocomplete ...@@ -106,14 +117,24 @@ class @SearchAutocomplete
else if e.keyCode is @keyCode.ESCAPE else if e.keyCode is @keyCode.ESCAPE
@restoreOriginalState() @restoreOriginalState()
else else
# Create new autocomplete instance if it's not created # Create new autocomplete if hasn't been created yet and there's no badge
@createAutocomplete() unless @catcomplete? if !@catComplete? and @locationBadgeEl.is(':empty')
@createAutocomplete()
onSearchInputFocus: =>
@wrap.addClass('search-active')
onSearchInputBlur: =>
@wrap.removeClass('search-active')
# If input is blank then restore state
@restoreOriginalState() if @searchInput.val() is ''
addLocationBadge: (item) -> addLocationBadge: (item) ->
category = if item.category? then "#{item.category}: " else '' category = if item.category? then "#{item.category}: " else ''
value = if item.value? then item.value else '' value = if item.value? then item.value else ''
html = "<span class='label label-primary'> html = "<span class='location-badge'>
<i class='location-text'>#{category}#{value}</i> <i class='location-text'>#{category}#{value}</i>
<a class='remove-badge' href='#'>x</a> <a class='remove-badge' href='#'>x</a>
</span>" </span>"
...@@ -160,5 +181,5 @@ class @SearchAutocomplete ...@@ -160,5 +181,5 @@ class @SearchAutocomplete
location.href = result.url location.href = result.url
destroyAutocomplete: -> destroyAutocomplete: ->
@catComplete.destroy() if @catcomplete? @catComplete.destroy() if @catComplete?
@catComplete = null @catComplete = null
...@@ -6,40 +6,6 @@ input { ...@@ -6,40 +6,6 @@ input {
border-radius: $border-radius-base; border-radius: $border-radius-base;
} }
input[type='search'] {
background-color: white;
padding-left: 10px;
}
input[type='search'].search-input {
background-repeat: no-repeat;
background-position: 10px;
background-size: 16px;
background-position-x: 30%;
padding-left: 10px;
background-color: $gray-light;
&.search-input[value=""] {
background-image: url('');
}
&.search-input::-webkit-input-placeholder {
text-align: center;
}
&.search-input:-moz-placeholder { /* Firefox 18- */
text-align: center;
}
&.search-input::-moz-placeholder { /* Firefox 19+ */
text-align: center;
}
&.search-input:-ms-input-placeholder {
text-align: center;
}
}
input[type='text'].danger { input[type='text'].danger {
background: #F2DEDE!important; background: #F2DEDE!important;
border-color: #D66; border-color: #D66;
......
...@@ -112,26 +112,6 @@ header { ...@@ -112,26 +112,6 @@ header {
} }
} }
.search {
margin-right: 10px;
margin-left: 10px;
margin-top: ($header-height - 36) / 2;
form {
margin: 0;
padding: 0;
}
.search-input {
width: 220px;
&:focus {
@include box-shadow(none);
outline: none;
}
}
}
.impersonation i { .impersonation i {
color: $red-normal; color: $red-normal;
} }
......
...@@ -19,13 +19,41 @@ ...@@ -19,13 +19,41 @@
} }
&.ui-autocomplete { &.ui-autocomplete {
border-color: #DDD;
padding: 0;
margin-top: 2px; margin-top: 2px;
z-index: 1001; z-index: 1001;
width: 240px;
margin-bottom: 0;
padding: 10px 10px;
font-size: 14px;
font-weight: normal;
background-color: $dropdown-bg;
border: 1px solid $dropdown-border-color;
border-radius: $border-radius-base;
box-shadow: 0 2px 4px $dropdown-shadow-color;
.ui-menu-item a { .ui-menu-item {
padding: 4px 10px; display: block;
position: relative;
padding: 0 10px;
color: $dropdown-link-color;
line-height: 34px;
text-overflow: ellipsis;
border-radius: 2px;
white-space: nowrap;
overflow: hidden;
border: none;
&.ui-state-focus {
background-color: $dropdown-link-hover-bg;
text-decoration: none;
margin: 0;
}
}
.ui-autocomplete-category {
text-transform: uppercase;
font-size: 11px;
color: #7f8fa4;
} }
} }
......
...@@ -21,3 +21,73 @@ ...@@ -21,3 +21,73 @@
} }
} }
.search {
margin-right: 10px;
margin-left: 10px;
margin-top: ($header-height - 35) / 2;
&.search-active {
form {
@extend .form-control:focus;
}
.location-badge {
@include transition(all .15s);
background-color: $input-border-focus;
color: $white-light;
}
}
form {
@extend .form-control;
margin: 0;
padding: 4px;
width: 350px;
line-height: 24px;
overflow: hidden;
}
.location-text {
font-style: normal;
}
.remove-badge {
display: none;
}
.search-input {
border: none;
font-size: 14px;
outline: none;
padding: 0;
margin-left: 2px;
line-height: 25px;
width: 100%;
}
.location-badge {
line-height: 25px;
padding: 0 5px;
border-radius: 2px;
font-size: 14px;
font-style: normal;
color: #AAAAAA;
display: inline-block;
background-color: #F5F5F5;
vertical-align: top;
}
.search-input-container {
display: flex;
}
.search-location-badge, .search-input-wrap {
// Fallback if flex is not supported
display: inline-block;
}
.search-input-wrap {
width: 100%;
}
}
...@@ -48,20 +48,19 @@ module SearchHelper ...@@ -48,20 +48,19 @@ module SearchHelper
# Autocomplete results for the current project, if it's defined # Autocomplete results for the current project, if it's defined
def project_autocomplete def project_autocomplete
if @project && @project.repository.exists? && @project.repository.root_ref if @project && @project.repository.exists? && @project.repository.root_ref
prefix = "Project - " + search_result_sanitize(@project.name_with_namespace)
ref = @ref || @project.repository.root_ref ref = @ref || @project.repository.root_ref
[ [
{ category: prefix, label: "Files", url: namespace_project_tree_path(@project.namespace, @project, ref) }, { category: "Current Project", label: "Files", url: namespace_project_tree_path(@project.namespace, @project, ref) },
{ category: prefix, label: "Commits", url: namespace_project_commits_path(@project.namespace, @project, ref) }, { category: "Current Project", label: "Commits", url: namespace_project_commits_path(@project.namespace, @project, ref) },
{ category: prefix, label: "Network", url: namespace_project_network_path(@project.namespace, @project, ref) }, { category: "Current Project", label: "Network", url: namespace_project_network_path(@project.namespace, @project, ref) },
{ category: prefix, label: "Graph", url: namespace_project_graph_path(@project.namespace, @project, ref) }, { category: "Current Project", label: "Graph", url: namespace_project_graph_path(@project.namespace, @project, ref) },
{ category: prefix, label: "Issues", url: namespace_project_issues_path(@project.namespace, @project) }, { category: "Current Project", label: "Issues", url: namespace_project_issues_path(@project.namespace, @project) },
{ category: prefix, label: "Merge Requests", url: namespace_project_merge_requests_path(@project.namespace, @project) }, { category: "Current Project", label: "Merge Requests", url: namespace_project_merge_requests_path(@project.namespace, @project) },
{ category: prefix, label: "Milestones", url: namespace_project_milestones_path(@project.namespace, @project) }, { category: "Current Project", label: "Milestones", url: namespace_project_milestones_path(@project.namespace, @project) },
{ category: prefix, label: "Snippets", url: namespace_project_snippets_path(@project.namespace, @project) }, { category: "Current Project", label: "Snippets", url: namespace_project_snippets_path(@project.namespace, @project) },
{ category: prefix, label: "Members", url: namespace_project_project_members_path(@project.namespace, @project) }, { category: "Current Project", label: "Members", url: namespace_project_project_members_path(@project.namespace, @project) },
{ category: prefix, label: "Wiki", url: namespace_project_wikis_path(@project.namespace, @project) }, { category: "Current Project", label: "Wiki", url: namespace_project_wikis_path(@project.namespace, @project) },
] ]
else else
[] []
......
.search .search.search-form
= form_tag search_path, method: :get, class: 'navbar-form pull-left' do |f| = form_tag search_path, method: :get, class: 'navbar-form' do |f|
= render 'shared/location_badge' .search-input-container
= search_field_tag "search", nil, placeholder: 'Search', class: "search-input form-control", spellcheck: false, tabindex: "1" .search-location-badge
= hidden_field_tag :group_id, @group.try(:id) = render 'shared/location_badge'
.search-input-wrap
= search_field_tag "search", nil, placeholder: 'Search', class: "search-input", spellcheck: false, tabindex: "1", autocomplete: 'off'
= hidden_field_tag :group_id, @group.try(:id)
= hidden_field_tag :project_id, @project && @project.persisted? ? @project.id : '' = hidden_field_tag :project_id, @project && @project.persisted? ? @project.id : ''
- if @project && @project.persisted? - if @project && @project.persisted?
......
...@@ -3,11 +3,10 @@ ...@@ -3,11 +3,10 @@
- if controller.controller_path =~ /^projects/ - if controller.controller_path =~ /^projects/
- label = 'This project' - label = 'This project'
.search-location-badge - if label.present?
- if label.present? %span.location-badge
%span.label.label-primary %i.location-text
%i.location-text = label
= label
%a.remove-badge{href: '#'} %a.remove-badge{href: '#'}
x x
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment