Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
499bb9f3
Commit
499bb9f3
authored
May 13, 2016
by
Rémy Coutable
Committed by
Alfredo Sumaran
Jun 06, 2016
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Improve Issuable.order_labels_priority
Signed-off-by:
Rémy Coutable
<
remy@rymai.me
>
parent
d8263b28
Changes
11
Hide whitespace changes
Inline
Side-by-side
Showing
11 changed files
with
85 additions
and
61 deletions
+85
-61
app/assets/javascripts/LabelManager.js.coffee
app/assets/javascripts/LabelManager.js.coffee
+12
-9
app/controllers/projects/labels_controller.rb
app/controllers/projects/labels_controller.rb
+10
-5
app/finders/issuable_finder.rb
app/finders/issuable_finder.rb
+1
-1
app/models/concerns/issuable.rb
app/models/concerns/issuable.rb
+22
-8
app/models/issue.rb
app/models/issue.rb
+2
-2
app/models/label.rb
app/models/label.rb
+6
-19
app/views/projects/labels/_label.html.haml
app/views/projects/labels/_label.html.haml
+1
-1
config/initializers/nulls_last.rb
config/initializers/nulls_last.rb
+0
-15
config/routes.rb
config/routes.rb
+1
-1
lib/gitlab/database.rb
lib/gitlab/database.rb
+14
-0
spec/lib/gitlab/database_spec.rb
spec/lib/gitlab/database_spec.rb
+16
-0
No files found.
app/assets/javascripts/LabelManager.js.coffee
View file @
499bb9f3
...
@@ -25,7 +25,7 @@ class @LabelManager
...
@@ -25,7 +25,7 @@ class @LabelManager
action
=
if
$btn
.
parents
(
'.js-prioritized-labels'
).
length
then
'remove'
else
'add'
action
=
if
$btn
.
parents
(
'.js-prioritized-labels'
).
length
then
'remove'
else
'add'
_this
.
toggleLabelPriority
(
$label
,
action
)
_this
.
toggleLabelPriority
(
$label
,
action
)
toggleLabelPriority
:
(
$label
,
action
,
p
asiv
e
=
false
)
->
toggleLabelPriority
:
(
$label
,
action
,
p
ersistStat
e
=
false
)
->
_this
=
@
_this
=
@
url
=
$label
.
find
(
'.js-toggle-priority'
).
data
'url'
url
=
$label
.
find
(
'.js-toggle-priority'
).
data
'url'
...
@@ -46,16 +46,19 @@ class @LabelManager
...
@@ -46,16 +46,19 @@ class @LabelManager
$label
.
detach
().
appendTo
(
$target
)
$label
.
detach
().
appendTo
(
$target
)
# Return if we are not persisting state
# Return if we are not persisting state
return
if
p
asiv
e
return
if
p
ersistStat
e
xhr
=
$
.
post
url
if
action
is
'remove'
xhr
=
$
.
ajax
url
:
url
,
type
:
'DELETE'
# If request fails, put label back to Other labels group
# If request fails, put label back to Other labels group
xhr
.
fail
->
xhr
.
fail
->
_this
.
toggleLabelPriority
(
$label
,
'remove'
,
true
)
_this
.
toggleLabelPriority
(
$label
,
'remove'
,
true
)
# Show a message
# Show a message
new
Flash
(
'Unable to update label prioritization at this time'
,
'alert'
)
new
Flash
(
'Unable to update label prioritization at this time'
,
'alert'
)
else
@
savePrioritySort
()
onPrioritySortUpdate
:
->
onPrioritySortUpdate
:
->
@
savePrioritySort
()
@
savePrioritySort
()
...
@@ -76,4 +79,4 @@ class @LabelManager
...
@@ -76,4 +79,4 @@ class @LabelManager
sortedIds
=
[]
sortedIds
=
[]
@
prioritizedLabels
.
find
(
'li'
).
each
->
@
prioritizedLabels
.
find
(
'li'
).
each
->
sortedIds
.
push
$
(
@
).
data
'id'
sortedIds
.
push
$
(
@
).
data
'id'
sortedIds
sortedIds
\ No newline at end of file
app/controllers/projects/labels_controller.rb
View file @
499bb9f3
...
@@ -72,11 +72,9 @@ class Projects::LabelsController < Projects::ApplicationController
...
@@ -72,11 +72,9 @@ class Projects::LabelsController < Projects::ApplicationController
end
end
end
end
def
toggle_priority
def
remove_priority
priority
=
label
.
priority
respond_to
do
|
format
|
respond_to
do
|
format
|
if
label
.
update_attribute
s
(
priority:
!
priority
)
if
label
.
update_attribute
(
:priority
,
nil
)
format
.
json
{
render
json:
label
}
format
.
json
{
render
json:
label
}
else
else
message
=
label
.
errors
.
full_messages
.
uniq
.
join
(
'. '
)
message
=
label
.
errors
.
full_messages
.
uniq
.
join
(
'. '
)
...
@@ -86,8 +84,15 @@ class Projects::LabelsController < Projects::ApplicationController
...
@@ -86,8 +84,15 @@ class Projects::LabelsController < Projects::ApplicationController
end
end
def
set_sorting
def
set_sorting
Label
.
transaction
do
params
[
:label_ids
].
each_with_index
do
|
label_id
,
index
|
label
=
@project
.
labels
.
find_by_id
(
label_id
)
label
.
update_attribute
(
:priority
,
index
)
if
label
end
end
respond_to
do
|
format
|
respond_to
do
|
format
|
format
.
json
{
render
json:
{
message:
'success'
}
}
format
.
json
{
render
json:
{
message:
'success'
}
}
end
end
end
end
...
...
app/finders/issuable_finder.rb
View file @
499bb9f3
...
@@ -224,7 +224,7 @@ class IssuableFinder
...
@@ -224,7 +224,7 @@ class IssuableFinder
def
sort
(
items
)
def
sort
(
items
)
# Ensure we always have an explicit sort order (instead of inheriting
# Ensure we always have an explicit sort order (instead of inheriting
# multiple orders when combining ActiveRecord::Relation objects).
# multiple orders when combining ActiveRecord::Relation objects).
params
[
:sort
]
?
items
.
sort
(
params
[
:sort
],
label_names
)
:
items
.
reorder
(
id: :desc
)
params
[
:sort
]
?
items
.
sort
(
params
[
:sort
],
excluded_labels:
label_names
)
:
items
.
reorder
(
id: :desc
)
end
end
def
by_assignee
(
items
)
def
by_assignee
(
items
)
...
...
app/models/concerns/issuable.rb
View file @
499bb9f3
...
@@ -48,12 +48,6 @@ module Issuable
...
@@ -48,12 +48,6 @@ module Issuable
scope
:non_archived
,
->
{
join_project
.
where
(
projects:
{
archived:
false
})
}
scope
:non_archived
,
->
{
join_project
.
where
(
projects:
{
archived:
false
})
}
def
self
.
order_priority
(
labels
)
select
(
"
#{
table_name
}
.*, (
#{
Label
.
high_priority
(
name
,
table_name
,
labels
).
to_sql
}
) AS highest_priority"
)
.
group
(
"
#{
table_name
}
.id"
)
.
reorder
(
nulls_last
(
'highest_priority'
,
'ASC'
))
end
delegate
:name
,
delegate
:name
,
:email
,
:email
,
to: :author
,
to: :author
,
...
@@ -111,18 +105,24 @@ module Issuable
...
@@ -111,18 +105,24 @@ module Issuable
where
(
t
[
:title
].
matches
(
pattern
).
or
(
t
[
:description
].
matches
(
pattern
)))
where
(
t
[
:title
].
matches
(
pattern
).
or
(
t
[
:description
].
matches
(
pattern
)))
end
end
def
sort
(
method
,
labels
=
[])
def
sort
(
method
,
excluded_labels:
[])
case
method
.
to_s
case
method
.
to_s
when
'milestone_due_asc'
then
order_milestone_due_asc
when
'milestone_due_asc'
then
order_milestone_due_asc
when
'milestone_due_desc'
then
order_milestone_due_desc
when
'milestone_due_desc'
then
order_milestone_due_desc
when
'downvotes_desc'
then
order_downvotes_desc
when
'downvotes_desc'
then
order_downvotes_desc
when
'upvotes_desc'
then
order_upvotes_desc
when
'upvotes_desc'
then
order_upvotes_desc
when
'priority'
then
order_
priority
(
labels
)
when
'priority'
then
order_
labels_priority
(
excluded_labels:
excluded_
labels
)
else
else
order_by
(
method
)
order_by
(
method
)
end
end
end
end
def
order_labels_priority
(
excluded_labels:
[])
select
(
"
#{
table_name
}
.*, (
#{
highest_label_priority
(
excluded_labels
).
to_sql
}
) AS highest_priority"
).
group
(
arel_table
[
:id
]).
reorder
(
Gitlab
::
Database
.
nulls_last_order
(
'highest_priority'
,
'ASC'
))
end
def
with_label
(
title
,
sort
=
nil
)
def
with_label
(
title
,
sort
=
nil
)
if
title
.
is_a?
(
Array
)
&&
title
.
size
>
1
if
title
.
is_a?
(
Array
)
&&
title
.
size
>
1
joins
(
:labels
).
where
(
labels:
{
title:
title
}).
group
(
*
grouping_columns
(
sort
)).
having
(
"COUNT(DISTINCT labels.title) =
#{
title
.
size
}
"
)
joins
(
:labels
).
where
(
labels:
{
title:
title
}).
group
(
*
grouping_columns
(
sort
)).
having
(
"COUNT(DISTINCT labels.title) =
#{
title
.
size
}
"
)
...
@@ -146,6 +146,20 @@ module Issuable
...
@@ -146,6 +146,20 @@ module Issuable
grouping_columns
grouping_columns
end
end
private
def
highest_label_priority
(
excluded_labels
)
query
=
Label
.
select
(
Label
.
arel_table
[
:priority
].
minimum
).
joins
(
:label_links
).
where
(
label_links:
{
target_type:
name
}).
where
(
"label_links.target_id =
#{
table_name
}
.id"
).
reorder
(
nil
)
query
.
where
.
not
(
title:
excluded_labels
)
if
excluded_labels
.
present?
query
end
end
end
def
today?
def
today?
...
...
app/models/issue.rb
View file @
499bb9f3
...
@@ -75,10 +75,10 @@ class Issue < ActiveRecord::Base
...
@@ -75,10 +75,10 @@ class Issue < ActiveRecord::Base
@link_reference_pattern
||=
super
(
"issues"
,
/(?<issue>\d+)/
)
@link_reference_pattern
||=
super
(
"issues"
,
/(?<issue>\d+)/
)
end
end
def
self
.
sort
(
method
)
def
self
.
sort
(
method
,
excluded_labels:
[]
)
case
method
.
to_s
case
method
.
to_s
when
'due_date_asc'
then
order_due_date_asc
when
'due_date_asc'
then
order_due_date_asc
when
'due_date_desc'
then
order_due_date_desc
when
'due_date_desc'
then
order_due_date_desc
else
else
super
super
end
end
...
...
app/models/label.rb
View file @
499bb9f3
...
@@ -19,7 +19,6 @@ class Label < ActiveRecord::Base
...
@@ -19,7 +19,6 @@ class Label < ActiveRecord::Base
validates
:color
,
color:
true
,
allow_blank:
false
validates
:color
,
color:
true
,
allow_blank:
false
validates
:project
,
presence:
true
,
unless:
Proc
.
new
{
|
service
|
service
.
template?
}
validates
:project
,
presence:
true
,
unless:
Proc
.
new
{
|
service
|
service
.
template?
}
validates
:priority
,
presence:
false
,
default:
false
# Don't allow '?', '&', and ',' for label titles
# Don't allow '?', '&', and ',' for label titles
validates
:title
,
validates
:title
,
...
@@ -32,21 +31,11 @@ class Label < ActiveRecord::Base
...
@@ -32,21 +31,11 @@ class Label < ActiveRecord::Base
default_scope
{
order
(
title: :asc
)
}
default_scope
{
order
(
title: :asc
)
}
scope
:templates
,
->
{
where
(
template:
true
)
}
scope
:templates
,
->
{
where
(
template:
true
)
}
scope
:prioritized
,
->
(
value
=
true
)
{
where
(
priority:
value
)
}
def
self
.
prioritized
(
bool
=
true
)
def
self
.
high_priority
(
name
,
table_name
,
labels
)
query
=
bool
?
where
.
not
(
priority:
nil
)
:
where
(
priority:
nil
)
unfiltered
=
unscoped
.
select
(
"MIN(labels.priority)"
)
query
.
reorder
(
Gitlab
::
Database
.
nulls_last_order
(
:priority
),
:title
)
.
joins
(
"INNER JOIN label_links ON label_links.label_id = labels.id"
)
.
where
(
"label_links.target_type = '
#{
name
}
'"
)
.
where
(
"label_links.target_id =
#{
table_name
}
.id"
)
.
where
(
"labels.project_id =
#{
table_name
}
.project_id"
)
if
labels
.
empty?
unfiltered
else
unfiltered
.
where
(
"labels.title NOT IN (?)"
,
labels
)
end
end
end
alias_attribute
:name
,
:title
alias_attribute
:name
,
:title
...
@@ -139,8 +128,6 @@ class Label < ActiveRecord::Base
...
@@ -139,8 +128,6 @@ class Label < ActiveRecord::Base
end
end
def
nillify_priority
def
nillify_priority
unless
self
.
priority
.
present?
self
.
priority
=
nil
if
priority
.
blank?
self
.
priority
=
nil
end
end
end
end
end
app/views/projects/labels/_label.html.haml
View file @
499bb9f3
-
label_css_id
=
dom_id
(
label
)
-
label_css_id
=
dom_id
(
label
)
%li
{
id:
label_css_id
,
:"data-id"
=>
label
.
id
}
%li
{
id:
label_css_id
,
:"data-id"
=>
label
.
id
}
%a
.js-toggle-priority
{
:href
=>
"#"
,
%a
.js-toggle-priority
{
:href
=>
"#"
,
:"data-url"
=>
toggl
e_priority_namespace_project_label_path
(
@project
.
namespace
,
@project
,
label
),
:"data-url"
=>
remov
e_priority_namespace_project_label_path
(
@project
.
namespace
,
@project
,
label
),
:"data-dom-id"
=>
"#{label_css_id}"
}
:"data-dom-id"
=>
"#{label_css_id}"
}
%span
.add-priority
%span
.add-priority
(+)
(+)
...
...
config/initializers/nulls_last.rb
deleted
100644 → 0
View file @
d8263b28
module
ActiveRecord
class
Base
def
self
.
nulls_last
(
field
,
direction
=
'ASC'
)
if
Gitlab
::
Database
.
postgresql?
"
#{
field
}
#{
direction
}
NULLS LAST"
else
if
direction
.
upcase
==
'ASC'
"-
#{
field
}
DESC"
else
"
#{
field
}
DESC"
end
end
end
end
end
config/routes.rb
View file @
499bb9f3
...
@@ -724,7 +724,7 @@ Rails.application.routes.draw do
...
@@ -724,7 +724,7 @@ Rails.application.routes.draw do
member
do
member
do
post
:toggle_subscription
post
:toggle_subscription
post
:toggl
e_priority
delete
:remov
e_priority
end
end
end
end
...
...
lib/gitlab/database.rb
View file @
499bb9f3
...
@@ -16,6 +16,20 @@ module Gitlab
...
@@ -16,6 +16,20 @@ module Gitlab
database_version
.
match
(
/\A(?:PostgreSQL |)([^\s]+).*\z/
)[
1
]
database_version
.
match
(
/\A(?:PostgreSQL |)([^\s]+).*\z/
)[
1
]
end
end
def
self
.
nulls_last_order
(
field
,
direction
=
'ASC'
)
order
=
"
#{
field
}
#{
direction
}
"
if
Gitlab
::
Database
.
postgresql?
order
<<
' NULLS LAST'
else
# `field IS NULL` will be `0` for non-NULL columns and `1` for NULL
# columns. In the (default) ascending order, `0` comes first.
order
.
prepend
(
"
#{
field
}
IS NULL, "
)
if
direction
==
'ASC'
end
order
end
def
true_value
def
true_value
if
Gitlab
::
Database
.
postgresql?
if
Gitlab
::
Database
.
postgresql?
"'t'"
"'t'"
...
...
spec/lib/gitlab/database_spec.rb
View file @
499bb9f3
...
@@ -39,6 +39,22 @@ describe Gitlab::Database, lib: true do
...
@@ -39,6 +39,22 @@ describe Gitlab::Database, lib: true do
end
end
end
end
describe
'.nulls_last_order'
do
context
'when using PostgreSQL'
do
before
{
expect
(
described_class
).
to
receive
(
:postgresql?
).
and_return
(
true
)
}
it
{
expect
(
described_class
.
nulls_last_order
(
'column'
,
'ASC'
)).
to
eq
'column ASC NULLS LAST'
}
it
{
expect
(
described_class
.
nulls_last_order
(
'column'
,
'DESC'
)).
to
eq
'column DESC NULLS LAST'
}
end
context
'when using MySQL'
do
before
{
expect
(
described_class
).
to
receive
(
:postgresql?
).
and_return
(
false
)
}
it
{
expect
(
described_class
.
nulls_last_order
(
'column'
,
'ASC'
)).
to
eq
'column IS NULL, column ASC'
}
it
{
expect
(
described_class
.
nulls_last_order
(
'column'
,
'DESC'
)).
to
eq
'column DESC'
}
end
end
describe
'#true_value'
do
describe
'#true_value'
do
it
'returns correct value for PostgreSQL'
do
it
'returns correct value for PostgreSQL'
do
expect
(
described_class
).
to
receive
(
:postgresql?
).
and_return
(
true
)
expect
(
described_class
).
to
receive
(
:postgresql?
).
and_return
(
true
)
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment