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
bca169a3
Commit
bca169a3
authored
Aug 26, 2020
by
Alex Kalderimis
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add mover abstraction
parent
d1a6ff1d
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
688 additions
and
0 deletions
+688
-0
app/models/concerns/relative_positioning.rb
app/models/concerns/relative_positioning.rb
+345
-0
spec/models/concerns/relative_positioning_spec.rb
spec/models/concerns/relative_positioning_spec.rb
+343
-0
No files found.
app/models/concerns/relative_positioning.rb
View file @
bca169a3
...
@@ -26,6 +26,351 @@
...
@@ -26,6 +26,351 @@
# end
# end
#
#
module
RelativePositioning
module
RelativePositioning
class
Gap
attr_reader
:start_pos
,
:end_pos
def
initialize
(
start_pos
,
end_pos
)
@start_pos
,
@end_pos
=
start_pos
,
end_pos
end
def
delta
((
start_pos
-
end_pos
)
/
2.0
).
abs
.
ceil
.
clamp
(
0
,
RelativePositioning
::
IDEAL_DISTANCE
)
end
end
class
ItemContext
include
Gitlab
::
Utils
::
StrongMemoize
attr_reader
:object
,
:model_class
,
:range
attr_accessor
:ignoring
def
initialize
(
object
,
range
)
@object
=
object
@range
=
range
@model_class
=
object
.
class
end
def
min_relative_position
strong_memoize
(
:min_relative_position
)
{
calculate_relative_position
(
'MIN'
)
}
end
def
max_relative_position
strong_memoize
(
:max_relative_position
)
{
calculate_relative_position
(
'MAX'
)
}
end
def
prev_relative_position
calculate_relative_position
(
'MAX'
)
{
|
r
|
nextify
(
r
,
false
)
}
if
object
.
relative_position
end
def
next_relative_position
calculate_relative_position
(
'MIN'
)
{
|
r
|
nextify
(
r
)
}
if
object
.
relative_position
end
def
nextify
(
relation
,
gt
=
true
)
op
=
gt
?
'>'
:
'<'
relation
.
where
(
"relative_position
#{
op
}
?"
,
object
.
relative_position
)
end
def
relative_siblings
(
relation
=
scoped_items
)
relation
.
id_not_in
(
object
.
id
)
end
# Handles the possibility that the position is already occupied by a sibling
def
place_at_position
(
position
,
lhs
)
current_occupant
=
relative_siblings
.
find_by
(
relative_position:
position
)
if
current_occupant
.
present?
Mover
.
new
(
position
,
range
).
move
(
object
,
lhs
.
object
,
current_occupant
)
else
object
.
relative_position
=
position
end
end
def
lhs_neighbour
scoped_items
.
where
(
'relative_position < ?'
,
relative_position
)
.
reorder
(
relative_position: :desc
)
.
first
.
then
{
|
x
|
neighbour
(
x
)
}
end
def
rhs_neighbour
scoped_items
.
where
(
'relative_position > ?'
,
relative_position
)
.
reorder
(
relative_position: :asc
)
.
first
.
then
{
|
x
|
neighbour
(
x
)
}
end
def
neighbour
(
item
)
return
unless
item
.
present?
n
=
self
.
class
.
new
(
item
,
range
)
n
.
ignoring
=
ignoring
n
end
def
scoped_items
r
=
model_class
.
relative_positioning_query_base
(
object
)
r
=
r
.
id_not_in
(
ignoring
.
id
)
if
ignoring
.
present?
r
end
def
calculate_relative_position
(
calculation
)
# When calculating across projects, this is much more efficient than
# MAX(relative_position) without the GROUP BY, due to index usage:
# https://gitlab.com/gitlab-org/gitlab-foss/issues/54276#note_119340977
relation
=
scoped_items
.
order
(
Gitlab
::
Database
.
nulls_last_order
(
'position'
,
'DESC'
))
.
group
(
grouping_column
)
.
limit
(
1
)
relation
=
yield
relation
if
block_given?
relation
.
pluck
(
grouping_column
,
Arel
.
sql
(
"
#{
calculation
}
(relative_position) AS position"
))
.
first
&
.
last
end
def
grouping_column
model_class
.
relative_positioning_parent_column
end
def
max_sibling
sib
=
relative_siblings
.
order
(
Gitlab
::
Database
.
nulls_last_order
(
'relative_position'
,
'DESC'
))
.
first
self
.
class
.
new
(
sib
,
range
)
end
def
min_sibling
sib
=
relative_siblings
.
order
(
Gitlab
::
Database
.
nulls_last_order
(
'relative_position'
,
'ASC'
))
.
first
self
.
class
.
new
(
sib
,
range
)
end
def
shift_left
move_sequence_before
(
true
)
object
.
reset
end
def
shift_right
move_sequence_after
(
true
)
object
.
reset
end
def
create_space_left
(
gap:
nil
)
move_sequence_before
(
false
,
next_gap:
gap
)
end
def
create_space_right
(
gap:
nil
)
move_sequence_after
(
false
,
next_gap:
gap
)
end
def
move_sequence_before
(
include_self
=
false
,
next_gap:
find_next_gap_before
)
raise
NoSpaceLeft
unless
next_gap
.
present?
delta
=
next_gap
.
delta
move_sequence
(
next_gap
.
start_pos
,
relative_position
,
-
delta
,
include_self
)
end
def
move_sequence_after
(
include_self
=
false
,
next_gap:
find_next_gap_after
)
raise
NoSpaceLeft
unless
next_gap
.
present?
delta
=
next_gap
.
delta
move_sequence
(
relative_position
,
next_gap
.
start_pos
,
delta
,
include_self
)
end
def
move_sequence
(
start_pos
,
end_pos
,
delta
,
include_self
=
false
)
relation
=
include_self
?
scoped_items
:
relative_siblings
relation
.
where
(
'relative_position BETWEEN ? AND ?'
,
start_pos
,
end_pos
)
.
update_all
(
"relative_position = relative_position +
#{
delta
}
"
)
end
def
find_next_gap_before
items_with_next_pos
=
scoped_items
.
select
(
'relative_position AS pos, LEAD(relative_position) OVER (ORDER BY relative_position DESC) AS next_pos'
)
.
where
(
'relative_position <= ?'
,
relative_position
)
.
order
(
relative_position: :desc
)
find_next_gap
(
items_with_next_pos
,
range
.
first
)
end
def
find_next_gap_after
items_with_next_pos
=
scoped_items
.
select
(
'relative_position AS pos, LEAD(relative_position) OVER (ORDER BY relative_position ASC) AS next_pos'
)
.
where
(
'relative_position >= ?'
,
relative_position
)
.
order
(
:relative_position
)
find_next_gap
(
items_with_next_pos
,
range
.
last
)
end
def
find_next_gap
(
items_with_next_pos
,
default_end
)
gap
=
model_class
.
from
(
items_with_next_pos
,
:items
)
.
where
(
'next_pos IS NULL OR ABS(pos::bigint - next_pos::bigint) >= ?'
,
RelativePositioning
::
MIN_GAP
)
.
limit
(
1
)
.
pluck
(
:pos
,
:next_pos
)
.
first
return
if
gap
.
nil?
||
gap
.
first
==
default_end
Gap
.
new
(
gap
.
first
,
gap
.
second
||
default_end
)
end
def
relative_position
object
.
relative_position
end
end
class
Mover
attr_reader
:range
,
:start_position
def
initialize
(
start
,
range
)
@range
=
range
@start_position
=
start
end
def
move_to_end
(
object
)
focus
=
context
(
object
,
ignoring:
object
)
max_pos
=
focus
.
max_relative_position
move_to_range_end
(
focus
,
max_pos
)
end
def
move_to_start
(
object
)
focus
=
context
(
object
,
ignoring:
object
)
min_pos
=
focus
.
min_relative_position
move_to_range_start
(
focus
,
min_pos
)
end
def
move
(
object
,
first
,
last
)
raise
ArgumentError
unless
object
&&
(
first
||
last
)
&&
(
first
!=
last
)
# Moving a object next to itself is a no-op
return
if
object
==
first
||
object
==
last
lhs
=
context
(
first
,
ignoring:
object
)
rhs
=
context
(
last
,
ignoring:
object
)
focus
=
context
(
object
)
lhs
||=
rhs
.
lhs_neighbour
rhs
||=
lhs
.
rhs_neighbour
if
lhs
.
nil?
move_to_range_start
(
focus
,
rhs
.
relative_position
)
elsif
rhs
.
nil?
move_to_range_end
(
focus
,
lhs
.
relative_position
)
else
pos_left
,
pos_right
=
create_space_between
(
lhs
,
rhs
)
desired_position
=
position_between
(
pos_left
,
pos_right
)
focus
.
place_at_position
(
desired_position
,
lhs
)
end
end
private
def
gap_too_small?
(
pos_a
,
pos_b
)
return
false
unless
pos_a
&&
pos_b
(
pos_a
-
pos_b
).
abs
<
MIN_GAP
end
def
move_to_range_end
(
context
,
max_pos
)
range_end
=
range
.
last
+
1
new_pos
=
if
max_pos
.
nil?
start_position
elsif
gap_too_small?
(
max_pos
,
range_end
)
max
=
context
.
max_sibling
max
.
ignoring
=
context
.
object
max
.
shift_left
position_between
(
max
.
relative_position
,
range_end
)
else
position_between
(
max_pos
,
range_end
)
end
context
.
object
.
relative_position
=
new_pos
end
def
move_to_range_start
(
context
,
min_pos
)
range_end
=
range
.
first
-
1
new_pos
=
if
min_pos
.
nil?
start_position
elsif
gap_too_small?
(
min_pos
,
range_end
)
sib
=
context
.
min_sibling
sib
.
ignoring
=
context
.
object
sib
.
shift_right
position_between
(
sib
.
relative_position
,
range_end
)
else
position_between
(
min_pos
,
range_end
)
end
context
.
object
.
relative_position
=
new_pos
end
def
create_space_between
(
lhs
,
rhs
)
pos_left
=
lhs
&
.
relative_position
pos_right
=
rhs
&
.
relative_position
return
[
pos_left
,
pos_right
]
unless
gap_too_small?
(
pos_left
,
pos_right
)
gap
=
rhs
.
find_next_gap_before
if
gap
.
present?
rhs
.
create_space_left
(
gap:
gap
)
[
pos_left
-
gap
.
delta
,
pos_right
]
else
gap
=
lhs
.
find_next_gap_after
lhs
.
create_space_right
(
gap:
gap
)
[
pos_left
,
pos_right
+
gap
.
delta
]
end
end
def
context
(
object
,
ignoring:
nil
)
return
unless
object
c
=
ItemContext
.
new
(
object
,
range
)
c
.
ignoring
=
ignoring
c
end
def
position_between
(
pos_before
,
pos_after
)
pos_before
||=
range
.
first
pos_after
||=
range
.
last
pos_before
,
pos_after
=
[
pos_before
,
pos_after
].
sort
gap_width
=
pos_after
-
pos_before
if
gap_too_small?
(
pos_before
,
pos_after
)
raise
RelativePositioning
::
NoSpaceLeft
elsif
gap_width
>
RelativePositioning
::
MAX_GAP
if
pos_before
<=
range
.
first
pos_after
-
RelativePositioning
::
IDEAL_DISTANCE
elsif
pos_after
>=
range
.
last
pos_before
+
RelativePositioning
::
IDEAL_DISTANCE
else
midpoint
(
pos_before
,
pos_after
)
end
else
midpoint
(
pos_before
,
pos_after
)
end
end
def
midpoint
(
lower_bound
,
upper_bound
)
((
lower_bound
+
upper_bound
)
/
2.0
).
ceil
.
clamp
(
lower_bound
,
upper_bound
-
1
)
end
end
extend
ActiveSupport
::
Concern
extend
ActiveSupport
::
Concern
STEPS
=
10
STEPS
=
10
...
...
spec/models/concerns/relative_positioning_spec.rb
0 → 100644
View file @
bca169a3
# frozen_string_literal: true
require
'spec_helper'
RSpec
.
describe
RelativePositioning
do
let_it_be
(
:default_user
)
{
create_default
(
:user
)
}
let_it_be
(
:project
)
{
create
(
:project
)
}
def
create_issue
(
pos
)
create
(
:issue
,
project:
project
,
relative_position:
pos
)
end
# Increase the range size to convice yourself that this covers ALL arrangements
range
=
(
101
..
104
)
indices
=
range
.
each_with_index
.
to_a
.
map
(
&
:second
)
describe
'Mover'
do
let
(
:start
)
{
((
range
.
first
+
range
.
last
)
/
2.0
).
floor
}
subject
{
RelativePositioning
::
Mover
.
new
(
start
,
range
)
}
describe
'#move_to_end'
do
shared_examples
'able to place a new item at the end'
do
it
'can place any new item'
do
new_item
=
create_issue
(
nil
)
subject
.
move_to_end
(
new_item
)
new_item
.
save!
expect
(
new_item
.
relative_position
).
to
eq
(
project
.
issues
.
maximum
(
:relative_position
))
end
end
shared_examples
'able to move existing items to the end'
do
it
'can move any existing item'
do
issue
=
issues
[
index
]
subject
.
move_to_end
(
issue
)
issue
.
save!
project
.
reset
expect
(
project
.
issues
.
pluck
(
:relative_position
)).
to
all
(
be_between
(
range
.
first
,
range
.
last
))
expect
(
issue
.
relative_position
).
to
eq
(
project
.
issues
.
maximum
(
:relative_position
))
end
end
context
'all positions are taken'
do
let_it_be
(
:issues
)
do
range
.
map
{
|
pos
|
create_issue
(
pos
)
}
end
it
'raises an error when placing a new item'
do
new_item
=
create
(
:issue
,
project:
project
,
relative_position:
nil
)
expect
{
subject
.
move_to_end
(
new_item
)
}.
to
raise_error
(
RelativePositioning
::
NoSpaceLeft
)
end
where
(
:index
)
{
indices
}
with_them
do
it_behaves_like
'able to move existing items to the end'
end
end
context
'there are no siblings'
do
it_behaves_like
'able to place a new item at the end'
end
context
'there is only one sibling'
do
where
(
:pos
)
{
range
.
to_a
}
with_them
do
let!
(
:issues
)
{
[
create_issue
(
pos
)]
}
let
(
:index
)
{
0
}
it_behaves_like
'able to place a new item at the end'
it_behaves_like
'able to move existing items to the end'
end
end
context
'at least one position is free'
do
where
(
:free_space
,
:index
)
do
range
.
to_a
.
product
((
0
..
).
take
(
range
.
size
-
1
).
to_a
)
end
with_them
do
let!
(
:issues
)
do
range
.
reject
{
|
x
|
x
==
free_space
}.
map
{
|
pos
|
create_issue
(
pos
)
}
end
it_behaves_like
'able to place a new item at the end'
it_behaves_like
'able to move existing items to the end'
end
end
end
describe
'#move_to_start'
do
shared_examples
'able to place a new item at the start'
do
it
'can place any new item'
do
new_item
=
create_issue
(
nil
)
subject
.
move_to_start
(
new_item
)
new_item
.
save!
expect
(
new_item
.
relative_position
).
to
eq
(
project
.
issues
.
minimum
(
:relative_position
))
end
end
shared_examples
'able to move existing items to the start'
do
it
'can move any existing item'
do
issue
=
issues
[
index
]
subject
.
move_to_start
(
issue
)
issue
.
save!
project
.
reset
expect
(
project
.
issues
.
pluck
(
:relative_position
)).
to
all
(
be_between
(
range
.
first
,
range
.
last
))
expect
(
issue
.
relative_position
).
to
eq
(
project
.
issues
.
minimum
(
:relative_position
))
end
end
context
'all positions are taken'
do
let_it_be
(
:issues
)
do
range
.
map
{
|
pos
|
create_issue
(
pos
)
}
end
it
'raises an error when placing a new item'
do
new_item
=
create
(
:issue
,
project:
project
,
relative_position:
nil
)
expect
{
subject
.
move_to_start
(
new_item
)
}.
to
raise_error
(
RelativePositioning
::
NoSpaceLeft
)
end
where
(
:index
)
{
indices
}
with_them
do
it_behaves_like
'able to move existing items to the start'
end
end
context
'there are no siblings'
do
it_behaves_like
'able to place a new item at the start'
end
context
'there is only one sibling'
do
where
(
:pos
)
{
range
.
to_a
}
with_them
do
let!
(
:issues
)
{
[
create_issue
(
pos
)]
}
let
(
:index
)
{
0
}
it_behaves_like
'able to place a new item at the start'
it_behaves_like
'able to move existing items to the start'
end
end
context
'at least one position is free'
do
where
(
:free_space
,
:index
)
do
range
.
to_a
.
product
((
0
..
).
take
(
range
.
size
-
1
).
to_a
)
end
with_them
do
let!
(
:issues
)
do
range
.
reject
{
|
x
|
x
==
free_space
}.
map
{
|
pos
|
create_issue
(
pos
)
}
end
it_behaves_like
'able to place a new item at the start'
it_behaves_like
'able to move existing items to the start'
end
end
end
describe
'#move'
do
shared_examples
'able to move a new item'
do
it
'can place any new item betwen two others'
do
new_item
=
create_issue
(
nil
)
subject
.
move
(
new_item
,
lhs
,
rhs
)
new_item
.
save!
lhs
.
reset
rhs
.
reset
expect
(
new_item
.
relative_position
).
to
be_between
(
range
.
first
,
range
.
last
)
expect
(
new_item
.
relative_position
).
to
be_between
(
lhs
.
relative_position
,
rhs
.
relative_position
)
end
it
'can place any new item after another'
do
new_item
=
create_issue
(
nil
)
subject
.
move
(
new_item
,
lhs
,
nil
)
new_item
.
save!
lhs
.
reset
expect
(
new_item
.
relative_position
).
to
be_between
(
range
.
first
,
range
.
last
)
expect
(
new_item
.
relative_position
).
to
be
>
lhs
.
relative_position
end
it
'can place any new item before another'
do
new_item
=
create_issue
(
nil
)
subject
.
move
(
new_item
,
nil
,
rhs
)
new_item
.
save!
rhs
.
reset
expect
(
new_item
.
relative_position
).
to
be_between
(
range
.
first
,
range
.
last
)
expect
(
new_item
.
relative_position
).
to
be
<
rhs
.
relative_position
end
end
shared_examples
'able to move an existing item'
do
let
(
:item
)
{
issues
[
index
]
}
let
(
:positions
)
{
project
.
reset
.
issues
.
pluck
(
:relative_position
)
}
it
'can place any item betwen two others'
do
subject
.
move
(
item
,
lhs
,
rhs
)
item
.
save!
lhs
.
reset
rhs
.
reset
expect
(
positions
).
to
all
(
be_between
(
range
.
first
,
range
.
last
))
expect
(
positions
).
to
match_array
(
positions
.
uniq
)
expect
(
item
.
relative_position
).
to
be_between
(
lhs
.
relative_position
,
rhs
.
relative_position
)
end
it
'can place any item after another'
do
subject
.
move
(
item
,
lhs
,
nil
)
item
.
save!
lhs
.
reset
expect
(
positions
).
to
all
(
be_between
(
range
.
first
,
range
.
last
))
expect
(
positions
).
to
match_array
(
positions
.
uniq
)
expect
(
item
.
relative_position
).
to
be
>=
lhs
.
relative_position
expected_sequence
=
[
lhs
,
item
].
uniq
sequence
=
project
.
issues
.
reorder
(
:relative_position
)
.
where
(
relative_position:
(
expected_sequence
.
first
.
relative_position
..
expected_sequence
.
last
.
relative_position
))
expect
(
sequence
).
to
eq
(
expected_sequence
)
end
it
'can place any item before another'
do
subject
.
move
(
item
,
nil
,
rhs
)
item
.
save!
rhs
.
reset
expect
(
positions
).
to
all
(
be_between
(
range
.
first
,
range
.
last
))
expect
(
positions
).
to
match_array
(
positions
.
uniq
)
expect
(
item
.
relative_position
).
to
be
<=
rhs
.
relative_position
expected_sequence
=
[
item
,
rhs
].
uniq
sequence
=
project
.
issues
.
reorder
(
:relative_position
)
.
where
(
relative_position:
(
expected_sequence
.
first
.
relative_position
..
expected_sequence
.
last
.
relative_position
))
expect
(
sequence
).
to
eq
(
expected_sequence
)
end
end
context
'all positions are taken'
do
let_it_be
(
:issues
)
{
range
.
map
{
|
pos
|
create_issue
(
pos
)
}
}
where
(
:idx_a
,
:idx_b
)
do
indices
.
product
(
indices
).
select
{
|
a
,
b
|
a
<
b
}
end
with_them
do
let
(
:lhs
)
{
issues
[
idx_a
]
}
let
(
:rhs
)
{
issues
[
idx_b
]
}
before
do
issues
.
each
(
&
:reset
)
end
it
'raises an error when placing a new item anywhere'
do
new_item
=
create_issue
(
nil
)
expect
{
subject
.
move
(
new_item
,
lhs
,
rhs
)
}
.
to
raise_error
(
RelativePositioning
::
NoSpaceLeft
)
expect
{
subject
.
move
(
new_item
,
nil
,
rhs
)
}
.
to
raise_error
(
RelativePositioning
::
NoSpaceLeft
)
expect
{
subject
.
move
(
new_item
,
lhs
,
nil
)
}
.
to
raise_error
(
RelativePositioning
::
NoSpaceLeft
)
end
where
(
:index
)
{
indices
}
with_them
do
it_behaves_like
'able to move an existing item'
end
end
end
context
'there are no siblings'
do
it
'raises an ArgumentError when both first and last are nil'
do
new_item
=
create_issue
(
nil
)
expect
{
subject
.
move
(
new_item
,
nil
,
nil
)
}.
to
raise_error
(
ArgumentError
)
end
end
context
'there are a couple of siblings'
do
where
(
:pos_a
,
:pos_b
)
{
range
.
to_a
.
product
(
range
.
to_a
).
reject
{
|
x
,
y
|
x
==
y
}
}
with_them
do
let!
(
:issues
)
{
[
range
.
first
,
pos_a
,
pos_b
].
sort
.
map
{
|
p
|
create_issue
(
p
)
}
}
let
(
:index
)
{
0
}
let
(
:lhs
)
{
issues
[
1
]
}
let
(
:rhs
)
{
issues
[
2
]
}
it_behaves_like
'able to move a new item'
it_behaves_like
'able to move an existing item'
end
end
context
'at least one position is free'
do
where
(
:free_space
,
:index
,
:pos_a
,
:pos_b
)
do
is
=
indices
.
reverse
.
drop
(
1
)
range
.
to_a
.
product
(
is
).
product
(
is
).
product
(
is
)
.
map
(
&
:flatten
)
.
select
{
|
_
,
_
,
a
,
b
|
a
<
b
}
end
with_them
do
let!
(
:issues
)
do
range
.
reject
{
|
x
|
x
==
free_space
}.
map
{
|
pos
|
create_issue
(
pos
)
}
end
let
(
:lhs
)
{
issues
[
pos_a
]
}
let
(
:rhs
)
{
issues
[
pos_b
]
}
it_behaves_like
'able to move a new item'
it_behaves_like
'able to move an existing item'
end
end
end
end
end
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