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
6cc8fe28
Commit
6cc8fe28
authored
Jul 28, 2021
by
Eulyeon Ko
Committed by
Nicolò Maria Mezzopera
Jul 28, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Resolve "Boards: reordering lists sending incorrect position" [RUN AS-IF-FOSS]
parent
b8b6fb29
Changes
16
Show whitespace changes
Inline
Side-by-side
Showing
16 changed files
with
215 additions
and
189 deletions
+215
-189
app/assets/javascripts/boards/components/board_column.vue
app/assets/javascripts/boards/components/board_column.vue
+1
-1
app/assets/javascripts/boards/components/board_content.vue
app/assets/javascripts/boards/components/board_content.vue
+1
-14
app/assets/javascripts/boards/graphql/board_lists.query.graphql
...sets/javascripts/boards/graphql/board_lists.query.graphql
+2
-0
app/assets/javascripts/boards/stores/actions.js
app/assets/javascripts/boards/stores/actions.js
+45
-21
app/assets/javascripts/boards/stores/mutation_types.js
app/assets/javascripts/boards/stores/mutation_types.js
+1
-2
app/assets/javascripts/boards/stores/mutations.js
app/assets/javascripts/boards/stores/mutations.js
+7
-10
ee/app/assets/javascripts/boards/components/epics_swimlanes.vue
.../assets/javascripts/boards/components/epics_swimlanes.vue
+1
-13
ee/app/assets/javascripts/boards/stores/actions.js
ee/app/assets/javascripts/boards/stores/actions.js
+10
-9
ee/app/assets/javascripts/boards/stores/mutations.js
ee/app/assets/javascripts/boards/stores/mutations.js
+0
-4
ee/spec/features/boards/boards_spec.rb
ee/spec/features/boards/boards_spec.rb
+2
-2
ee/spec/features/epic_boards/epic_boards_spec.rb
ee/spec/features/epic_boards/epic_boards_spec.rb
+1
-1
ee/spec/frontend/boards/stores/actions_spec.js
ee/spec/frontend/boards/stores/actions_spec.js
+2
-2
locale/gitlab.pot
locale/gitlab.pot
+1
-1
spec/features/boards/boards_spec.rb
spec/features/boards/boards_spec.rb
+27
-15
spec/frontend/boards/stores/actions_spec.js
spec/frontend/boards/stores/actions_spec.js
+101
-67
spec/frontend/boards/stores/mutations_spec.js
spec/frontend/boards/stores/mutations_spec.js
+13
-27
No files found.
app/assets/javascripts/boards/components/board_column.vue
View file @
6cc8fe28
...
@@ -79,7 +79,7 @@ export default {
...
@@ -79,7 +79,7 @@ export default {
'is-collapsed': list.collapsed,
'is-collapsed': list.collapsed,
'board-type-assignee': list.listType === 'assignee',
'board-type-assignee': list.listType === 'assignee',
}"
}"
:data-id="list.id"
:data-
list-
id="list.id"
class="board gl-display-inline-block gl-h-full gl-px-3 gl-vertical-align-top gl-white-space-normal is-expandable"
class="board gl-display-inline-block gl-h-full gl-px-3 gl-vertical-align-top gl-white-space-normal is-expandable"
data-qa-selector="board_list"
data-qa-selector="board_list"
>
>
...
...
app/assets/javascripts/boards/components/board_content.vue
View file @
6cc8fe28
...
@@ -76,19 +76,6 @@ export default {
...
@@ -76,19 +76,6 @@ export default {
const
el
=
this
.
canDragColumns
?
this
.
$refs
.
list
.
$el
:
this
.
$refs
.
list
;
const
el
=
this
.
canDragColumns
?
this
.
$refs
.
list
.
$el
:
this
.
$refs
.
list
;
el
.
scrollTo
({
left
:
el
.
scrollWidth
,
behavior
:
'
smooth
'
});
el
.
scrollTo
({
left
:
el
.
scrollWidth
,
behavior
:
'
smooth
'
});
},
},
handleDragOnEnd
(
params
)
{
const
{
item
,
newIndex
,
oldIndex
,
to
}
=
params
;
const
listId
=
item
.
dataset
.
id
;
const
replacedListId
=
to
.
children
[
newIndex
].
dataset
.
id
;
this
.
moveList
({
listId
,
replacedListId
,
newIndex
,
adjustmentValue
:
newIndex
<
oldIndex
?
1
:
-
1
,
});
},
},
},
};
};
</
script
>
</
script
>
...
@@ -104,7 +91,7 @@ export default {
...
@@ -104,7 +91,7 @@ export default {
ref=
"list"
ref=
"list"
v-bind=
"draggableOptions"
v-bind=
"draggableOptions"
class=
"boards-list gl-w-full gl-py-5 gl-px-3 gl-white-space-nowrap"
class=
"boards-list gl-w-full gl-py-5 gl-px-3 gl-white-space-nowrap"
@
end=
"
handleDragOnEnd
"
@
end=
"
moveList
"
>
>
<component
<component
:is=
"boardColumnComponent"
:is=
"boardColumnComponent"
...
...
app/assets/javascripts/boards/graphql/board_lists.query.graphql
View file @
6cc8fe28
...
@@ -9,6 +9,7 @@ query ListIssues(
...
@@ -9,6 +9,7 @@ query ListIssues(
)
{
)
{
group
(
fullPath
:
$fullPath
)
@include
(
if
:
$isGroup
)
{
group
(
fullPath
:
$fullPath
)
@include
(
if
:
$isGroup
)
{
board
(
id
:
$boardId
)
{
board
(
id
:
$boardId
)
{
hideBacklogList
lists
(
issueFilters
:
$filters
)
{
lists
(
issueFilters
:
$filters
)
{
nodes
{
nodes
{
...
BoardListFragment
...
BoardListFragment
...
@@ -18,6 +19,7 @@ query ListIssues(
...
@@ -18,6 +19,7 @@ query ListIssues(
}
}
project
(
fullPath
:
$fullPath
)
@include
(
if
:
$isProject
)
{
project
(
fullPath
:
$fullPath
)
@include
(
if
:
$isProject
)
{
board
(
id
:
$boardId
)
{
board
(
id
:
$boardId
)
{
hideBacklogList
lists
(
issueFilters
:
$filters
)
{
lists
(
issueFilters
:
$filters
)
{
nodes
{
nodes
{
...
BoardListFragment
...
BoardListFragment
...
...
app/assets/javascripts/boards/stores/actions.js
View file @
6cc8fe28
import
*
as
Sentry
from
'
@sentry/browser
'
;
import
*
as
Sentry
from
'
@sentry/browser
'
;
import
{
sortBy
}
from
'
lodash
'
;
import
{
import
{
BoardType
,
BoardType
,
ListType
,
ListType
,
...
@@ -216,33 +217,48 @@ export default {
...
@@ -216,33 +217,48 @@ export default {
},
},
moveList
:
(
moveList
:
(
{
state
,
commit
,
dispatch
},
{
state
:
{
boardLists
},
commit
,
dispatch
},
{
listId
,
replacedListId
,
newIndex
,
adjustmentValue
},
{
item
:
{
dataset
:
{
listId
:
movedListId
},
},
newIndex
,
to
:
{
children
},
},
)
=>
{
)
=>
{
if
(
listId
===
replacedListId
)
{
const
displacedListId
=
children
[
newIndex
].
dataset
.
listId
;
if
(
movedListId
===
displacedListId
)
{
return
;
return
;
}
}
const
{
boardLists
}
=
state
;
const
listIds
=
sortBy
(
const
backupList
=
{
...
boardLists
};
Object
.
keys
(
boardLists
).
filter
(
const
movedList
=
boardLists
[
listId
];
(
listId
)
=>
listId
!==
movedListId
&&
boardLists
[
listId
].
listType
!==
ListType
.
backlog
&&
boardLists
[
listId
].
listType
!==
ListType
.
closed
,
),
(
i
)
=>
boardLists
[
i
].
position
,
);
const
newPosition
=
newIndex
-
1
;
const
targetPosition
=
boardLists
[
displacedListId
].
position
;
const
listAtNewIndex
=
boardLists
[
replacedListId
];
// When the dragged list moves left, displaced list should shift right.
const
shiftOffset
=
Number
(
boardLists
[
movedListId
].
position
<
targetPosition
);
const
displacedListIndex
=
listIds
.
findIndex
((
listId
)
=>
listId
===
displacedListId
);
movedList
.
position
=
newPosition
;
commit
(
listAtNewIndex
.
position
+=
adjustmentValue
;
types
.
MOVE_LISTS
,
commit
(
types
.
MOVE_LIST
,
{
listIds
movedList
,
.
slice
(
0
,
displacedListIndex
+
shiftOffset
)
listAtNewIndex
,
.
concat
([
movedListId
],
listIds
.
slice
(
displacedListIndex
+
shiftOffset
))
});
.
map
((
listId
,
index
)
=>
({
listId
,
position
:
index
})),
);
dispatch
(
'
updateList
'
,
{
listId
,
position
:
newPosition
,
backupList
});
dispatch
(
'
updateList
'
,
{
listId
:
movedListId
,
position
:
targetPosition
});
},
},
updateList
:
(
updateList
:
(
{
commit
,
state
:
{
issuableType
,
boardItemsByListId
=
{}
},
dispatch
},
{
state
:
{
issuableType
,
boardItemsByListId
=
{}
},
dispatch
},
{
listId
,
position
,
collapsed
,
backupList
},
{
listId
,
position
,
collapsed
},
)
=>
{
)
=>
{
gqlClient
gqlClient
.
mutate
({
.
mutate
({
...
@@ -255,8 +271,7 @@ export default {
...
@@ -255,8 +271,7 @@ export default {
})
})
.
then
(({
data
})
=>
{
.
then
(({
data
})
=>
{
if
(
data
?.
updateBoardList
?.
errors
.
length
)
{
if
(
data
?.
updateBoardList
?.
errors
.
length
)
{
commit
(
types
.
UPDATE_LIST_FAILURE
,
backupList
);
throw
new
Error
();
return
;
}
}
// Only fetch when board items havent been fetched on a collapsed list
// Only fetch when board items havent been fetched on a collapsed list
...
@@ -265,10 +280,19 @@ export default {
...
@@ -265,10 +280,19 @@ export default {
}
}
})
})
.
catch
(()
=>
{
.
catch
(()
=>
{
commit
(
types
.
UPDATE_LIST_FAILURE
,
backupList
);
dispatch
(
'
handleUpdateListFailure
'
);
});
});
},
},
handleUpdateListFailure
:
({
dispatch
,
commit
})
=>
{
dispatch
(
'
fetchLists
'
);
commit
(
types
.
SET_ERROR
,
s__
(
'
Boards|An error occurred while updating the board list. Please try again.
'
),
);
},
toggleListCollapsed
:
({
commit
},
{
listId
,
collapsed
})
=>
{
toggleListCollapsed
:
({
commit
},
{
listId
,
collapsed
})
=>
{
commit
(
types
.
TOGGLE_LIST_COLLAPSED
,
{
listId
,
collapsed
});
commit
(
types
.
TOGGLE_LIST_COLLAPSED
,
{
listId
,
collapsed
});
},
},
...
...
app/assets/javascripts/boards/stores/mutation_types.js
View file @
6cc8fe28
...
@@ -10,8 +10,7 @@ export const RECEIVE_BOARD_LISTS_SUCCESS = 'RECEIVE_BOARD_LISTS_SUCCESS';
...
@@ -10,8 +10,7 @@ export const RECEIVE_BOARD_LISTS_SUCCESS = 'RECEIVE_BOARD_LISTS_SUCCESS';
export
const
RECEIVE_BOARD_LISTS_FAILURE
=
'
RECEIVE_BOARD_LISTS_FAILURE
'
;
export
const
RECEIVE_BOARD_LISTS_FAILURE
=
'
RECEIVE_BOARD_LISTS_FAILURE
'
;
export
const
SHOW_PROMOTION_LIST
=
'
SHOW_PROMOTION_LIST
'
;
export
const
SHOW_PROMOTION_LIST
=
'
SHOW_PROMOTION_LIST
'
;
export
const
RECEIVE_ADD_LIST_SUCCESS
=
'
RECEIVE_ADD_LIST_SUCCESS
'
;
export
const
RECEIVE_ADD_LIST_SUCCESS
=
'
RECEIVE_ADD_LIST_SUCCESS
'
;
export
const
MOVE_LIST
=
'
MOVE_LIST
'
;
export
const
MOVE_LISTS
=
'
MOVE_LISTS
'
;
export
const
UPDATE_LIST_FAILURE
=
'
UPDATE_LIST_FAILURE
'
;
export
const
TOGGLE_LIST_COLLAPSED
=
'
TOGGLE_LIST_COLLAPSED
'
;
export
const
TOGGLE_LIST_COLLAPSED
=
'
TOGGLE_LIST_COLLAPSED
'
;
export
const
REMOVE_LIST
=
'
REMOVE_LIST
'
;
export
const
REMOVE_LIST
=
'
REMOVE_LIST
'
;
export
const
REMOVE_LIST_FAILURE
=
'
REMOVE_LIST_FAILURE
'
;
export
const
REMOVE_LIST_FAILURE
=
'
REMOVE_LIST_FAILURE
'
;
...
...
app/assets/javascripts/boards/stores/mutations.js
View file @
6cc8fe28
import
{
pull
,
union
}
from
'
lodash
'
;
import
{
cloneDeep
,
pull
,
union
}
from
'
lodash
'
;
import
Vue
from
'
vue
'
;
import
Vue
from
'
vue
'
;
import
{
getIdFromGraphQLId
}
from
'
~/graphql_shared/utils
'
;
import
{
getIdFromGraphQLId
}
from
'
~/graphql_shared/utils
'
;
import
{
s__
}
from
'
~/locale
'
;
import
{
s__
}
from
'
~/locale
'
;
...
@@ -103,15 +103,12 @@ export default {
...
@@ -103,15 +103,12 @@ export default {
Vue
.
set
(
state
.
boardLists
,
list
.
id
,
list
);
Vue
.
set
(
state
.
boardLists
,
list
.
id
,
list
);
},
},
[
mutationTypes
.
MOVE_LIST
]:
(
state
,
{
movedList
,
listAtNewIndex
})
=>
{
[
mutationTypes
.
MOVE_LISTS
]:
(
state
,
movedLists
)
=>
{
const
{
boardLists
}
=
state
;
const
updatedBoardList
=
movedLists
.
reduce
((
acc
,
{
listId
,
position
})
=>
{
Vue
.
set
(
boardLists
,
movedList
.
id
,
movedList
);
acc
[
listId
].
position
=
position
;
Vue
.
set
(
boardLists
,
listAtNewIndex
.
id
,
listAtNewIndex
);
return
acc
;
},
},
cloneDeep
(
state
.
boardLists
));
Vue
.
set
(
state
,
'
boardLists
'
,
updatedBoardList
);
[
mutationTypes
.
UPDATE_LIST_FAILURE
]:
(
state
,
backupList
)
=>
{
state
.
error
=
s__
(
'
Boards|An error occurred while updating the list. Please try again.
'
);
Vue
.
set
(
state
,
'
boardLists
'
,
backupList
);
},
},
[
mutationTypes
.
TOGGLE_LIST_COLLAPSED
]:
(
state
,
{
listId
,
collapsed
})
=>
{
[
mutationTypes
.
TOGGLE_LIST_COLLAPSED
]:
(
state
,
{
listId
,
collapsed
})
=>
{
...
...
ee/app/assets/javascripts/boards/components/epics_swimlanes.vue
View file @
6cc8fe28
...
@@ -148,18 +148,6 @@ export default {
...
@@ -148,18 +148,6 @@ export default {
'
fetchItemsForList
'
,
'
fetchItemsForList
'
,
'
doneLoadingSwimlanesItems
'
,
'
doneLoadingSwimlanesItems
'
,
]),
]),
handleDragOnEnd
(
params
)
{
const
{
newIndex
,
oldIndex
,
item
,
to
}
=
params
;
const
{
listId
}
=
item
.
dataset
;
const
replacedListId
=
to
.
children
[
newIndex
].
dataset
.
listId
;
this
.
moveList
({
listId
,
replacedListId
,
newIndex
,
adjustmentValue
:
newIndex
<
oldIndex
?
1
:
-
1
,
});
},
fetchMoreEpics
()
{
fetchMoreEpics
()
{
this
.
fetchEpicsSwimlanes
({
fetchNext
:
true
});
this
.
fetchEpicsSwimlanes
({
fetchNext
:
true
});
},
},
...
@@ -215,7 +203,7 @@ export default {
...
@@ -215,7 +203,7 @@ export default {
v-bind=
"treeRootOptions"
v-bind=
"treeRootOptions"
class=
"board-swimlanes-headers gl-display-table gl-sticky gl-pt-5 gl-mb-5 gl-bg-white gl-top-0 gl-z-index-3"
class=
"board-swimlanes-headers gl-display-table gl-sticky gl-pt-5 gl-mb-5 gl-bg-white gl-top-0 gl-z-index-3"
data-testid=
"board-swimlanes-headers"
data-testid=
"board-swimlanes-headers"
@
end=
"
handleDragOnEnd
"
@
end=
"
moveList
"
>
>
<div
<div
v-for=
"list in lists"
v-for=
"list in lists"
...
...
ee/app/assets/javascripts/boards/stores/actions.js
View file @
6cc8fe28
...
@@ -225,7 +225,7 @@ export default {
...
@@ -225,7 +225,7 @@ export default {
commit
(
types
.
SET_SHOW_LABELS
,
val
);
commit
(
types
.
SET_SHOW_LABELS
,
val
);
},
},
updateListWipLimit
({
commit
,
getters
},
{
maxIssueCount
,
listId
})
{
updateListWipLimit
({
commit
,
getters
,
dispatch
},
{
maxIssueCount
,
listId
})
{
if
(
getters
.
shouldUseGraphQL
)
{
if
(
getters
.
shouldUseGraphQL
)
{
return
gqlClient
return
gqlClient
.
mutate
({
.
mutate
({
...
@@ -239,16 +239,17 @@ export default {
...
@@ -239,16 +239,17 @@ export default {
})
})
.
then
(({
data
})
=>
{
.
then
(({
data
})
=>
{
if
(
data
?.
boardListUpdateLimitMetrics
?.
errors
.
length
)
{
if
(
data
?.
boardListUpdateLimitMetrics
?.
errors
.
length
)
{
commit
(
types
.
UPDATE_LIST_FAILURE
);
throw
new
Error
(
);
}
else
{
}
const
list
=
data
.
boardListUpdateLimitMetrics
?.
list
;
commit
(
types
.
UPDATE_LIST_SUCCESS
,
{
commit
(
types
.
UPDATE_LIST_SUCCESS
,
{
listId
,
listId
,
list
,
list
:
data
.
boardListUpdateLimitMetrics
?.
list
,
});
});
}
})
})
.
catch
(()
=>
commit
(
types
.
UPDATE_LIST_FAILURE
));
.
catch
(()
=>
{
dispatch
(
'
handleUpdateListFailure
'
);
});
}
}
return
axios
.
put
(
`
${
boardsStoreEE
.
store
.
state
.
endpoints
.
listsEndpoint
}
/
${
listId
}
`
,
{
return
axios
.
put
(
`
${
boardsStoreEE
.
store
.
state
.
endpoints
.
listsEndpoint
}
/
${
listId
}
`
,
{
...
...
ee/app/assets/javascripts/boards/stores/mutations.js
View file @
6cc8fe28
...
@@ -17,10 +17,6 @@ export default {
...
@@ -17,10 +17,6 @@ export default {
Vue
.
set
(
state
.
boardLists
,
listId
,
list
);
Vue
.
set
(
state
.
boardLists
,
listId
,
list
);
},
},
[
mutationTypes
.
UPDATE_LIST_FAILURE
]:
(
state
)
=>
{
state
.
error
=
s__
(
'
Boards|An error occurred while updating the list. Please try again.
'
);
},
[
mutationTypes
.
RECEIVE_ITEMS_FOR_LIST_SUCCESS
]:
(
[
mutationTypes
.
RECEIVE_ITEMS_FOR_LIST_SUCCESS
]:
(
state
,
state
,
{
listItems
,
listPageInfo
,
listId
,
noEpicIssues
},
{
listItems
,
listPageInfo
,
listId
,
noEpicIssues
},
...
...
ee/spec/features/boards/boards_spec.rb
View file @
6cc8fe28
...
@@ -345,11 +345,11 @@ RSpec.describe 'Project issue boards', :js do
...
@@ -345,11 +345,11 @@ RSpec.describe 'Project issue boards', :js do
end
end
def
list_weight_badge
(
list
)
def
list_weight_badge
(
list
)
find
(
".board[data-id='gid://gitlab/List/
#{
list
.
id
}
'] [data-testid='issue-count-badge']"
)
find
(
".board[data-
list-
id='gid://gitlab/List/
#{
list
.
id
}
'] [data-testid='issue-count-badge']"
)
end
end
def
card_weight_badge
(
list
)
def
card_weight_badge
(
list
)
find
(
".board[data-id='gid://gitlab/List/
#{
list
.
id
}
'] [data-testid='board-card-weight']"
)
find
(
".board[data-
list-
id='gid://gitlab/List/
#{
list
.
id
}
'] [data-testid='board-card-weight']"
)
end
end
def
visit_board_page
def
visit_board_page
...
...
ee/spec/features/epic_boards/epic_boards_spec.rb
View file @
6cc8fe28
...
@@ -328,7 +328,7 @@ RSpec.describe 'epic boards', :js do
...
@@ -328,7 +328,7 @@ RSpec.describe 'epic boards', :js do
end
end
def
list_header
(
list
)
def
list_header
(
list
)
find
(
".board[data-id='gid://gitlab/Boards::EpicList/
#{
list
.
id
}
'] .board-header"
)
find
(
".board[data-
list-
id='gid://gitlab/Boards::EpicList/
#{
list
.
id
}
'] .board-header"
)
end
end
def
drag
(
selector:
'.board-list'
,
list_from_index:
0
,
from_index:
0
,
to_index:
0
,
list_to_index:
0
,
perform_drop:
true
)
def
drag
(
selector:
'.board-list'
,
list_from_index:
0
,
from_index:
0
,
to_index:
0
,
list_to_index:
0
,
perform_drop:
true
)
...
...
ee/spec/frontend/boards/stores/actions_spec.js
View file @
6cc8fe28
...
@@ -533,7 +533,7 @@ describe('updateListWipLimit', () => {
...
@@ -533,7 +533,7 @@ describe('updateListWipLimit', () => {
);
);
});
});
it
(
'
graphql -
commit UPDATE_LIST_FAILURE mutation
on failure
'
,
()
=>
{
it
(
'
graphql -
dispatch handleUpdateListFailure
on failure
'
,
()
=>
{
const
maxIssueCount
=
0
;
const
maxIssueCount
=
0
;
const
activeId
=
1
;
const
activeId
=
1
;
getters
.
shouldUseGraphQL
=
true
;
getters
.
shouldUseGraphQL
=
true
;
...
@@ -543,8 +543,8 @@ describe('updateListWipLimit', () => {
...
@@ -543,8 +543,8 @@ describe('updateListWipLimit', () => {
actions
.
updateListWipLimit
,
actions
.
updateListWipLimit
,
{
maxIssueCount
,
listId
:
activeId
},
{
maxIssueCount
,
listId
:
activeId
},
{
isShowingEpicsSwimlanes
:
true
,
...
getters
},
{
isShowingEpicsSwimlanes
:
true
,
...
getters
},
[{
type
:
types
.
UPDATE_LIST_FAILURE
}],
[],
[],
[{
type
:
'
handleUpdateListFailure
'
}],
);
);
});
});
});
});
...
...
locale/gitlab.pot
View file @
6cc8fe28
...
@@ -5371,7 +5371,7 @@ msgstr ""
...
@@ -5371,7 +5371,7 @@ msgstr ""
msgid "Boards|An error occurred while removing the list. Please try again."
msgid "Boards|An error occurred while removing the list. Please try again."
msgstr ""
msgstr ""
msgid "Boards|An error occurred while updating the list. Please try again."
msgid "Boards|An error occurred while updating the
board
list. Please try again."
msgstr ""
msgstr ""
msgid "Boards|Blocked by %{blockedByCount} %{issuableType}"
msgid "Boards|Blocked by %{blockedByCount} %{issuableType}"
...
...
spec/features/boards/boards_spec.rb
View file @
6cc8fe28
...
@@ -70,13 +70,6 @@ RSpec.describe 'Project issue boards', :js do
...
@@ -70,13 +70,6 @@ RSpec.describe 'Project issue boards', :js do
stub_feature_flags
(
board_new_list:
false
)
stub_feature_flags
(
board_new_list:
false
)
visit_project_board_path_without_query_limit
(
project
,
board
)
visit_project_board_path_without_query_limit
(
project
,
board
)
wait_for_requests
expect
(
page
).
to
have_selector
(
'.board'
,
count:
4
)
expect
(
find
(
'.board:nth-child(2)'
)).
to
have_selector
(
'.board-card'
)
expect
(
find
(
'.board:nth-child(3)'
)).
to
have_selector
(
'.board-card'
)
expect
(
find
(
'.board:nth-child(4)'
)).
to
have_selector
(
'.board-card'
)
end
end
it
'shows description tooltip on list title'
,
:quarantine
do
it
'shows description tooltip on list title'
,
:quarantine
do
...
@@ -221,18 +214,35 @@ RSpec.describe 'Project issue boards', :js do
...
@@ -221,18 +214,35 @@ RSpec.describe 'Project issue boards', :js do
it
'changes position of list'
do
it
'changes position of list'
do
drag
(
list_from_index:
2
,
list_to_index:
1
,
selector:
'.board-header'
)
drag
(
list_from_index:
2
,
list_to_index:
1
,
selector:
'.board-header'
)
wait_for_board_cards
(
2
,
2
)
expect
(
find
(
'.board:nth-child(2) [data-testid="board-list-header"]'
)).
to
have_content
(
development
.
title
)
wait_for_board_cards
(
3
,
8
)
expect
(
find
(
'.board:nth-child(3) [data-testid="board-list-header"]'
)).
to
have_content
(
planning
.
title
)
wait_for_board_cards
(
4
,
1
)
# Make sure list positions are preserved after a reload
visit_project_board_path_without_query_limit
(
project
,
board
)
expect
(
find
(
'.board:nth-child(2)'
)).
to
have_content
(
development
.
title
)
expect
(
find
(
'.board:nth-child(2) [data-testid="board-list-header"]'
)).
to
have_content
(
development
.
title
)
expect
(
find
(
'.board:nth-child(3)'
)).
to
have_content
(
planning
.
title
)
expect
(
find
(
'.board:nth-child(3) [data-testid="board-list-header"]'
)).
to
have_content
(
planning
.
title
)
end
context
'without backlog and closed lists'
do
let_it_be
(
:board
)
{
create
(
:board
,
project:
project
,
hide_backlog_list:
true
,
hide_closed_list:
true
)
}
let_it_be
(
:list1
)
{
create
(
:list
,
board:
board
,
label:
planning
,
position:
0
)
}
let_it_be
(
:list2
)
{
create
(
:list
,
board:
board
,
label:
development
,
position:
1
)
}
it
'changes position of list'
do
visit_project_board_path_without_query_limit
(
project
,
board
)
drag
(
list_from_index:
0
,
list_to_index:
1
,
selector:
'.board-header'
)
expect
(
find
(
'.board:nth-child(1) [data-testid="board-list-header"]'
)).
to
have_content
(
development
.
title
)
expect
(
find
(
'.board:nth-child(2) [data-testid="board-list-header"]'
)).
to
have_content
(
planning
.
title
)
# Make sure list positions are preserved after a reload
# Make sure list positions are preserved after a reload
visit_project_board_path_without_query_limit
(
project
,
board
)
visit_project_board_path_without_query_limit
(
project
,
board
)
expect
(
find
(
'.board:nth-child(2)'
)).
to
have_content
(
development
.
title
)
expect
(
find
(
'.board:nth-child(1) [data-testid="board-list-header"]'
)).
to
have_content
(
development
.
title
)
expect
(
find
(
'.board:nth-child(3)'
)).
to
have_content
(
planning
.
title
)
expect
(
find
(
'.board:nth-child(2) [data-testid="board-list-header"]'
)).
to
have_content
(
planning
.
title
)
end
end
end
it
'dragging does not duplicate list'
do
it
'dragging does not duplicate list'
do
...
@@ -682,6 +692,8 @@ RSpec.describe 'Project issue boards', :js do
...
@@ -682,6 +692,8 @@ RSpec.describe 'Project issue boards', :js do
def
visit_project_board_path_without_query_limit
(
project
,
board
)
def
visit_project_board_path_without_query_limit
(
project
,
board
)
inspect_requests
(
inject_headers:
{
'X-GITLAB-DISABLE-SQL-QUERY-LIMIT'
=>
'https://gitlab.com/gitlab-org/gitlab/-/issues/323426'
})
do
inspect_requests
(
inject_headers:
{
'X-GITLAB-DISABLE-SQL-QUERY-LIMIT'
=>
'https://gitlab.com/gitlab-org/gitlab/-/issues/323426'
})
do
visit
project_board_path
(
project
,
board
)
visit
project_board_path
(
project
,
board
)
wait_for_requests
end
end
end
end
end
end
spec/frontend/boards/stores/actions_spec.js
View file @
6cc8fe28
import
*
as
Sentry
from
'
@sentry/browser
'
;
import
*
as
Sentry
from
'
@sentry/browser
'
;
import
{
cloneDeep
}
from
'
lodash
'
;
import
{
import
{
inactiveId
,
inactiveId
,
ISSUABLE
,
ISSUABLE
,
...
@@ -419,75 +420,94 @@ describe('fetchLabels', () => {
...
@@ -419,75 +420,94 @@ describe('fetchLabels', () => {
});
});
describe
(
'
moveList
'
,
()
=>
{
describe
(
'
moveList
'
,
()
=>
{
it
(
'
should commit MOVE_LIST mutation and dispatch updateList action
'
,
(
done
)
=>
{
const
backlogListId
=
'
gid://1
'
;
const
initialBoardListsState
=
{
const
closedListId
=
'
gid://5
'
;
'
gid://gitlab/List/1
'
:
mockLists
[
0
],
'
gid://gitlab/List/2
'
:
mockLists
[
1
],
const
boardLists1
=
{
'
gid://3
'
:
{
listType
:
''
,
position
:
0
},
'
gid://4
'
:
{
listType
:
''
,
position
:
1
},
'
gid://5
'
:
{
listType
:
''
,
position
:
2
},
};
};
const
state
=
{
const
boardLists2
=
{
fullPath
:
'
gitlab-org
'
,
[
backlogListId
]:
{
listType
:
ListType
.
backlog
,
position
:
-
Infinity
},
fullBoardId
:
'
gid://gitlab/Board/1
'
,
[
closedListId
]:
{
listType
:
ListType
.
closed
,
position
:
Infinity
},
boardType
:
'
group
'
,
...
cloneDeep
(
boardLists1
),
disabled
:
false
,
boardLists
:
initialBoardListsState
,
};
};
testAction
(
const
movableListsOrder
=
[
'
gid://3
'
,
'
gid://4
'
,
'
gid://5
'
];
actions
.
moveList
,
const
allListsOrder
=
[
backlogListId
,
...
movableListsOrder
,
closedListId
];
{
listId
:
'
gid://gitlab/List/1
'
,
describe
.
each
`
replacedListId
:
'
gid://gitlab/List/2
'
,
draggableFrom | draggableTo | boardLists | boardListsOrder | expectedMovableListsOrder
newIndex
:
1
,
${
0
}
|
${
2
}
|
${
boardLists1
}
|
${
movableListsOrder
}
|
${[
'
gid://4
'
,
'
gid://5
'
,
'
gid://3
'
]}
adjustmentValue
:
1
,
${
2
}
|
${
0
}
|
${
boardLists1
}
|
${
movableListsOrder
}
|
${[
'
gid://5
'
,
'
gid://3
'
,
'
gid://4
'
]}
${
0
}
|
${
1
}
|
${
boardLists1
}
|
${
movableListsOrder
}
|
${[
'
gid://4
'
,
'
gid://3
'
,
'
gid://5
'
]}
${
1
}
|
${
2
}
|
${
boardLists1
}
|
${
movableListsOrder
}
|
${[
'
gid://3
'
,
'
gid://5
'
,
'
gid://4
'
]}
${
2
}
|
${
1
}
|
${
boardLists1
}
|
${
movableListsOrder
}
|
${[
'
gid://3
'
,
'
gid://5
'
,
'
gid://4
'
]}
${
1
}
|
${
3
}
|
${
boardLists2
}
|
${
allListsOrder
}
|
${[
'
gid://4
'
,
'
gid://5
'
,
'
gid://3
'
]}
${
3
}
|
${
1
}
|
${
boardLists2
}
|
${
allListsOrder
}
|
${[
'
gid://5
'
,
'
gid://3
'
,
'
gid://4
'
]}
${
1
}
|
${
2
}
|
${
boardLists2
}
|
${
allListsOrder
}
|
${[
'
gid://4
'
,
'
gid://3
'
,
'
gid://5
'
]}
${
2
}
|
${
3
}
|
${
boardLists2
}
|
${
allListsOrder
}
|
${[
'
gid://3
'
,
'
gid://5
'
,
'
gid://4
'
]}
${
3
}
|
${
2
}
|
${
boardLists2
}
|
${
allListsOrder
}
|
${[
'
gid://3
'
,
'
gid://5
'
,
'
gid://4
'
]}
`
(
'
when moving a list from position $draggableFrom to $draggableTo with lists $boardListsOrder
'
,
({
draggableFrom
,
draggableTo
,
boardLists
,
boardListsOrder
,
expectedMovableListsOrder
})
=>
{
const
movedListId
=
boardListsOrder
[
draggableFrom
];
const
displacedListId
=
boardListsOrder
[
draggableTo
];
const
buildDraggablePayload
=
()
=>
{
return
{
item
:
{
dataset
:
{
listId
:
boardListsOrder
[
draggableFrom
]
}
},
newIndex
:
draggableTo
,
to
:
{
children
:
boardListsOrder
.
map
((
listId
)
=>
({
dataset
:
{
listId
}
})),
},
},
state
,
};
[
};
it
(
'
should commit MOVE_LIST mutations and dispatch updateList action with correct payloads
'
,
()
=>
{
return
testAction
({
action
:
actions
.
moveList
,
payload
:
buildDraggablePayload
(),
state
:
{
boardLists
},
expectedMutations
:
[
{
{
type
:
types
.
MOVE_LIST
,
type
:
types
.
MOVE_LISTS
,
payload
:
{
movedList
:
mockLists
[
0
],
listAtNewIndex
:
mockLists
[
1
]
}
,
payload
:
expectedMovableListsOrder
.
map
((
listId
,
i
)
=>
({
listId
,
position
:
i
}))
,
},
},
],
],
[
expectedActions
:
[
{
{
type
:
'
updateList
'
,
type
:
'
updateList
'
,
payload
:
{
payload
:
{
listId
:
'
gid://gitlab/List/1
'
,
listId
:
movedListId
,
position
:
0
,
position
:
movableListsOrder
.
findIndex
((
i
)
=>
i
===
displacedListId
),
backupList
:
initialBoardListsState
,
},
},
},
},
],
],
done
,
);
});
});
});
},
);
it
(
'
should not commit MOVE_LIST or dispatch updateList if listId and replacedListId are the same
'
,
()
=>
{
describe
(
'
when moving from and to the same position
'
,
()
=>
{
const
initialBoardListsState
=
{
it
(
'
should not commit MOVE_LIST and should not dispatch updateList
'
,
()
=>
{
'
gid://gitlab/List/1
'
:
mockLists
[
0
],
const
listId
=
'
gid://1000
'
;
'
gid://gitlab/List/2
'
:
mockLists
[
1
],
};
const
state
=
{
fullPath
:
'
gitlab-org
'
,
fullBoardId
:
'
gid://gitlab/Board/1
'
,
boardType
:
'
group
'
,
disabled
:
false
,
boardLists
:
initialBoardListsState
,
};
testAction
(
return
testAction
({
actions
.
moveList
,
action
:
actions
.
moveList
,
{
payload
:
{
listId
:
'
gid://gitlab/List/1
'
,
item
:
{
dataset
:
{
listId
}
}
,
replacedListId
:
'
gid://gitlab/List/1
'
,
newIndex
:
0
,
newIndex
:
1
,
to
:
{
adjustmentValue
:
1
,
children
:
[{
dataset
:
{
listId
}
}]
,
},
},
state
,
},
[],
state
:
{
boardLists
:
{
[
listId
]:
{
position
:
0
}
}
},
[],
expectedMutations
:
[],
);
expectedActions
:
[],
});
});
});
});
});
});
...
@@ -549,7 +569,7 @@ describe('updateList', () => {
...
@@ -549,7 +569,7 @@ describe('updateList', () => {
});
});
});
});
it
(
'
should
commit UPDATE_LIST_FAILURE mutation when API returns an error
'
,
(
done
)
=>
{
it
(
'
should
dispatch handleUpdateListFailure when API returns an error
'
,
(
)
=>
{
jest
.
spyOn
(
gqlClient
,
'
mutate
'
).
mockResolvedValue
({
jest
.
spyOn
(
gqlClient
,
'
mutate
'
).
mockResolvedValue
({
data
:
{
data
:
{
updateBoardList
:
{
updateBoardList
:
{
...
@@ -559,17 +579,31 @@ describe('updateList', () => {
...
@@ -559,17 +579,31 @@ describe('updateList', () => {
},
},
});
});
testAction
(
return
testAction
(
actions
.
updateList
,
actions
.
updateList
,
{
listId
:
'
gid://gitlab/List/1
'
,
position
:
1
},
{
listId
:
'
gid://gitlab/List/1
'
,
position
:
1
},
createState
(),
createState
(),
[{
type
:
types
.
UPDATE_LIST_FAILURE
}],
[],
[],
done
,
[{
type
:
'
handleUpdateListFailure
'
}]
,
);
);
});
});
});
});
describe
(
'
handleUpdateListFailure
'
,
()
=>
{
it
(
'
should dispatch fetchLists action and commit SET_ERROR mutation
'
,
async
()
=>
{
await
testAction
({
action
:
actions
.
handleUpdateListFailure
,
expectedMutations
:
[
{
type
:
types
.
SET_ERROR
,
payload
:
'
An error occurred while updating the board list. Please try again.
'
,
},
],
expectedActions
:
[{
type
:
'
fetchLists
'
}],
});
});
});
describe
(
'
toggleListCollapsed
'
,
()
=>
{
describe
(
'
toggleListCollapsed
'
,
()
=>
{
it
(
'
should commit TOGGLE_LIST_COLLAPSED mutation
'
,
async
()
=>
{
it
(
'
should commit TOGGLE_LIST_COLLAPSED mutation
'
,
async
()
=>
{
const
payload
=
{
listId
:
'
gid://gitlab/List/1
'
,
collapsed
:
true
};
const
payload
=
{
listId
:
'
gid://gitlab/List/1
'
,
collapsed
:
true
};
...
...
spec/frontend/boards/stores/mutations_spec.js
View file @
6cc8fe28
...
@@ -165,40 +165,26 @@ describe('Board Store Mutations', () => {
...
@@ -165,40 +165,26 @@ describe('Board Store Mutations', () => {
});
});
});
});
describe
(
'
MOVE_LIST
'
,
()
=>
{
describe
(
'
MOVE_LIST
S
'
,
()
=>
{
it
(
'
updates
boardLists state with reordere
d lists
'
,
()
=>
{
it
(
'
updates
the positions of boar
d lists
'
,
()
=>
{
state
=
{
state
=
{
...
state
,
...
state
,
boardLists
:
initialBoardListsState
,
boardLists
:
initialBoardListsState
,
};
};
mutations
.
MOVE_LIST
(
state
,
{
mutations
.
MOVE_LISTS
(
state
,
[
movedList
:
mockLists
[
0
],
{
listAtNewIndex
:
mockLists
[
1
],
listId
:
mockLists
[
0
].
id
,
});
position
:
1
,
expect
(
state
.
boardLists
).
toEqual
({
'
gid://gitlab/List/2
'
:
mockLists
[
1
],
'
gid://gitlab/List/1
'
:
mockLists
[
0
],
});
});
});
describe
(
'
UPDATE_LIST_FAILURE
'
,
()
=>
{
it
(
'
updates boardLists state with previous order and sets error message
'
,
()
=>
{
state
=
{
...
state
,
boardLists
:
{
'
gid://gitlab/List/2
'
:
mockLists
[
1
],
'
gid://gitlab/List/1
'
:
mockLists
[
0
],
},
},
error
:
undefined
,
{
};
listId
:
mockLists
[
1
].
id
,
position
:
0
,
mutations
.
UPDATE_LIST_FAILURE
(
state
,
initialBoardListsState
);
},
]);
expect
(
state
.
boardLists
).
toEqual
(
initialBoardListsState
);
expect
(
state
.
boardLists
[
mockLists
[
0
].
id
].
position
).
toBe
(
1
);
expect
(
state
.
error
).
toEqual
(
'
An error occurred while updating the list. Please try again.
'
);
expect
(
state
.
boardLists
[
mockLists
[
1
].
id
].
position
).
toBe
(
0
);
});
});
});
});
...
...
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