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
bb77e90b
Commit
bb77e90b
authored
Mar 02, 2021
by
Jarek Ostrowski
Committed by
Enrique Alcantara
Mar 10, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Simplify notifications dropdown
MR:
https://gitlab.com/gitlab-org/gitlab/-/merge_requests/55522
parent
5e29c446
Changes
6
Show whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
48 additions
and
89 deletions
+48
-89
app/assets/javascripts/notifications/components/custom_notifications_modal.vue
...s/notifications/components/custom_notifications_modal.vue
+11
-0
app/assets/javascripts/notifications/components/notifications_dropdown.vue
...ripts/notifications/components/notifications_dropdown.vue
+15
-60
changelogs/unreleased/simplify-notifications-dropdown.yml
changelogs/unreleased/simplify-notifications-dropdown.yml
+5
-0
spec/features/groups/show_spec.rb
spec/features/groups/show_spec.rb
+2
-2
spec/features/profiles/user_visits_notifications_tab_spec.rb
spec/features/profiles/user_visits_notifications_tab_spec.rb
+3
-3
spec/frontend/notifications/components/notifications_dropdown_spec.js
...d/notifications/components/notifications_dropdown_spec.js
+12
-24
No files found.
app/assets/javascripts/notifications/components/custom_notifications_modal.vue
View file @
bb77e90b
...
@@ -24,12 +24,21 @@ export default {
...
@@ -24,12 +24,21 @@ export default {
default
:
''
,
default
:
''
,
},
},
},
},
model
:
{
prop
:
'
visible
'
,
event
:
'
change
'
,
},
props
:
{
props
:
{
modalId
:
{
modalId
:
{
type
:
String
,
type
:
String
,
required
:
false
,
required
:
false
,
default
:
'
custom-notifications-modal
'
,
default
:
'
custom-notifications-modal
'
,
},
},
visible
:
{
type
:
Boolean
,
required
:
false
,
default
:
false
,
},
},
},
data
()
{
data
()
{
return
{
return
{
...
@@ -95,9 +104,11 @@ export default {
...
@@ -95,9 +104,11 @@ export default {
<
template
>
<
template
>
<gl-modal
<gl-modal
ref=
"modal"
ref=
"modal"
:visible=
"visible"
:modal-id=
"modalId"
:modal-id=
"modalId"
:title=
"$options.i18n.customNotificationsModal.title"
:title=
"$options.i18n.customNotificationsModal.title"
@
show=
"onOpen"
@
show=
"onOpen"
v-on=
"$listeners"
>
>
<div
class=
"container-fluid"
>
<div
class=
"container-fluid"
>
<div
class=
"row"
>
<div
class=
"row"
>
...
...
app/assets/javascripts/notifications/components/notifications_dropdown.vue
View file @
bb77e90b
<
script
>
<
script
>
import
{
import
{
GlDropdown
,
GlDropdownDivider
,
GlTooltipDirective
}
from
'
@gitlab/ui
'
;
GlButtonGroup
,
GlButton
,
GlDropdown
,
GlDropdownDivider
,
GlTooltipDirective
,
GlModalDirective
,
}
from
'
@gitlab/ui
'
;
import
Api
from
'
~/api
'
;
import
Api
from
'
~/api
'
;
import
{
sprintf
}
from
'
~/locale
'
;
import
{
sprintf
}
from
'
~/locale
'
;
import
{
CUSTOM_LEVEL
,
i18n
}
from
'
../constants
'
;
import
{
CUSTOM_LEVEL
,
i18n
}
from
'
../constants
'
;
...
@@ -16,8 +9,6 @@ import NotificationsDropdownItem from './notifications_dropdown_item.vue';
...
@@ -16,8 +9,6 @@ import NotificationsDropdownItem from './notifications_dropdown_item.vue';
export
default
{
export
default
{
name
:
'
NotificationsDropdown
'
,
name
:
'
NotificationsDropdown
'
,
components
:
{
components
:
{
GlButtonGroup
,
GlButton
,
GlDropdown
,
GlDropdown
,
GlDropdownDivider
,
GlDropdownDivider
,
NotificationsDropdownItem
,
NotificationsDropdownItem
,
...
@@ -25,7 +16,6 @@ export default {
...
@@ -25,7 +16,6 @@ export default {
},
},
directives
:
{
directives
:
{
GlTooltip
:
GlTooltipDirective
,
GlTooltip
:
GlTooltipDirective
,
'
gl-modal
'
:
GlModalDirective
,
},
},
inject
:
{
inject
:
{
containerClass
:
{
containerClass
:
{
...
@@ -57,6 +47,7 @@ export default {
...
@@ -57,6 +47,7 @@ export default {
return
{
return
{
selectedNotificationLevel
:
this
.
initialNotificationLevel
,
selectedNotificationLevel
:
this
.
initialNotificationLevel
,
isLoading
:
false
,
isLoading
:
false
,
notificationsModalVisible
:
false
,
};
};
},
},
computed
:
{
computed
:
{
...
@@ -95,6 +86,11 @@ export default {
...
@@ -95,6 +86,11 @@ export default {
},
},
},
},
methods
:
{
methods
:
{
openNotificationsModal
()
{
if
(
this
.
isCustomNotification
)
{
this
.
notificationsModalVisible
=
true
;
}
},
selectItem
(
level
)
{
selectItem
(
level
)
{
if
(
level
!==
this
.
selectedNotificationLevel
)
{
if
(
level
!==
this
.
selectedNotificationLevel
)
{
this
.
updateNotificationLevel
(
level
);
this
.
updateNotificationLevel
(
level
);
...
@@ -106,10 +102,7 @@ export default {
...
@@ -106,10 +102,7 @@ export default {
try
{
try
{
await
Api
.
updateNotificationSettings
(
this
.
projectId
,
this
.
groupId
,
{
level
});
await
Api
.
updateNotificationSettings
(
this
.
projectId
,
this
.
groupId
,
{
level
});
this
.
selectedNotificationLevel
=
level
;
this
.
selectedNotificationLevel
=
level
;
this
.
openNotificationsModal
();
if
(
level
===
CUSTOM_LEVEL
)
{
this
.
$refs
.
customNotificationsModal
.
open
();
}
}
catch
(
error
)
{
}
catch
(
error
)
{
this
.
$toast
.
show
(
this
.
$options
.
i18n
.
updateNotificationLevelErrorMessage
,
{
type
:
'
error
'
});
this
.
$toast
.
show
(
this
.
$options
.
i18n
.
updateNotificationLevelErrorMessage
,
{
type
:
'
error
'
});
}
finally
{
}
finally
{
...
@@ -124,55 +117,17 @@ export default {
...
@@ -124,55 +117,17 @@ export default {
</
script
>
</
script
>
<
template
>
<
template
>
<div
:class=
"containerClass"
>
<div
:class=
"containerClass"
data-testid=
"notification-button"
>
<gl-button-group
<gl-dropdown
v-if=
"isCustomNotification"
v-gl-tooltip=
"
{ title: buttonTooltip }"
v-gl-tooltip=
"
{ title: buttonTooltip }"
data-testid="notification-button"
data-testid="notification-dropdown"
:class="{ disabled: disabled }"
:size="buttonSize"
>
<gl-button
v-gl-modal=
"$options.modalId"
:size="buttonSize"
:size="buttonSize"
:icon="buttonIcon"
:icon="buttonIcon"
:loading="isLoading"
:loading="isLoading"
:disabled="disabled"
:disabled="disabled"
>
:split="isCustomNotification"
<template
v-if=
"buttonText"
>
{{
buttonText
}}
</
template
>
</gl-button>
<gl-dropdown
:size=
"buttonSize"
:disabled=
"disabled"
>
<notifications-dropdown-item
v-for=
"item in notificationLevels"
:key=
"item.level"
:level=
"item.level"
:title=
"item.title"
:description=
"item.description"
:notification-level=
"selectedNotificationLevel"
@
item-selected=
"selectItem"
/>
<gl-dropdown-divider
/>
<notifications-dropdown-item
:key=
"$options.customLevel"
:level=
"$options.customLevel"
:title=
"$options.i18n.notificationTitles.custom"
:description=
"$options.i18n.notificationDescriptions.custom"
:notification-level=
"selectedNotificationLevel"
@
item-selected=
"selectItem"
/>
</gl-dropdown>
</gl-button-group>
<gl-dropdown
v-else
v-gl-tooltip=
"{ title: buttonTooltip }"
data-testid=
"notification-button"
:text="buttonText"
:text="buttonText"
:icon=
"buttonIcon"
@click="openNotificationsModal"
:loading=
"isLoading"
:size=
"buttonSize"
:disabled=
"disabled"
:class=
"{ disabled: disabled }"
>
>
<notifications-dropdown-item
<notifications-dropdown-item
v-for=
"item in notificationLevels"
v-for=
"item in notificationLevels"
...
@@ -193,6 +148,6 @@ export default {
...
@@ -193,6 +148,6 @@ export default {
@
item-selected=
"selectItem"
@
item-selected=
"selectItem"
/>
/>
</gl-dropdown>
</gl-dropdown>
<custom-notifications-modal
ref=
"customNotificationsModal
"
:modal-id=
"$options.modalId"
/>
<custom-notifications-modal
v-model=
"notificationsModalVisible
"
:modal-id=
"$options.modalId"
/>
</div>
</div>
</
template
>
</
template
>
changelogs/unreleased/simplify-notifications-dropdown.yml
0 → 100644
View file @
bb77e90b
---
title
:
Simplify notifications dropdown
merge_request
:
55522
author
:
type
:
changed
spec/features/groups/show_spec.rb
View file @
bb77e90b
...
@@ -170,14 +170,14 @@ RSpec.describe 'Group show page' do
...
@@ -170,14 +170,14 @@ RSpec.describe 'Group show page' do
it
'is enabled by default'
do
it
'is enabled by default'
do
visit
path
visit
path
expect
(
page
).
to
have_selector
(
'[data-testid="notification-
butto
n"]:not(.disabled)'
)
expect
(
page
).
to
have_selector
(
'[data-testid="notification-
dropdow
n"]:not(.disabled)'
)
end
end
it
'is disabled if emails are disabled'
do
it
'is disabled if emails are disabled'
do
group
.
update_attribute
(
:emails_disabled
,
true
)
group
.
update_attribute
(
:emails_disabled
,
true
)
visit
path
visit
path
expect
(
page
).
to
have_selector
(
'[data-testid="notification-
butto
n"].disabled'
)
expect
(
page
).
to
have_selector
(
'[data-testid="notification-
dropdow
n"].disabled'
)
end
end
end
end
...
...
spec/features/profiles/user_visits_notifications_tab_spec.rb
View file @
bb77e90b
...
@@ -15,17 +15,17 @@ RSpec.describe 'User visits the notifications tab', :js do
...
@@ -15,17 +15,17 @@ RSpec.describe 'User visits the notifications tab', :js do
it
'changes the project notifications setting'
do
it
'changes the project notifications setting'
do
expect
(
page
).
to
have_content
(
'Notifications'
)
expect
(
page
).
to
have_content
(
'Notifications'
)
first
(
'[data-testid="notification-
butto
n"]'
).
click
first
(
'[data-testid="notification-
dropdow
n"]'
).
click
click_button
(
'On mention'
)
click_button
(
'On mention'
)
expect
(
page
).
to
have_selector
(
'[data-testid="notification-
butto
n"]'
,
text:
'On mention'
)
expect
(
page
).
to
have_selector
(
'[data-testid="notification-
dropdow
n"]'
,
text:
'On mention'
)
end
end
context
'when project emails are disabled'
do
context
'when project emails are disabled'
do
let
(
:project
)
{
create
(
:project
,
emails_disabled:
true
)
}
let
(
:project
)
{
create
(
:project
,
emails_disabled:
true
)
}
it
'notification button is disabled'
do
it
'notification button is disabled'
do
expect
(
page
).
to
have_selector
(
'[data-testid="notification-
butto
n"].disabled'
)
expect
(
page
).
to
have_selector
(
'[data-testid="notification-
dropdow
n"].disabled'
)
end
end
end
end
end
end
spec/frontend/notifications/components/notifications_dropdown_spec.js
View file @
bb77e90b
import
{
Gl
ButtonGroup
,
GlButton
,
Gl
Dropdown
,
GlDropdownItem
}
from
'
@gitlab/ui
'
;
import
{
GlDropdown
,
GlDropdownItem
}
from
'
@gitlab/ui
'
;
import
{
shallowMount
}
from
'
@vue/test-utils
'
;
import
{
shallowMount
}
from
'
@vue/test-utils
'
;
import
axios
from
'
axios
'
;
import
axios
from
'
axios
'
;
import
MockAdapter
from
'
axios-mock-adapter
'
;
import
MockAdapter
from
'
axios-mock-adapter
'
;
...
@@ -15,14 +15,10 @@ const mockToastShow = jest.fn();
...
@@ -15,14 +15,10 @@ const mockToastShow = jest.fn();
describe
(
'
NotificationsDropdown
'
,
()
=>
{
describe
(
'
NotificationsDropdown
'
,
()
=>
{
let
wrapper
;
let
wrapper
;
let
mockAxios
;
let
mockAxios
;
let
glModalDirective
;
function
createComponent
(
injectedProperties
=
{})
{
function
createComponent
(
injectedProperties
=
{})
{
glModalDirective
=
jest
.
fn
();
return
shallowMount
(
NotificationsDropdown
,
{
return
shallowMount
(
NotificationsDropdown
,
{
stubs
:
{
stubs
:
{
GlButtonGroup
,
GlDropdown
,
GlDropdown
,
GlDropdownItem
,
GlDropdownItem
,
NotificationsDropdownItem
,
NotificationsDropdownItem
,
...
@@ -30,11 +26,6 @@ describe('NotificationsDropdown', () => {
...
@@ -30,11 +26,6 @@ describe('NotificationsDropdown', () => {
},
},
directives
:
{
directives
:
{
GlTooltip
:
createMockDirective
(),
GlTooltip
:
createMockDirective
(),
glModal
:
{
bind
(
_
,
{
value
})
{
glModalDirective
(
value
);
},
},
},
},
provide
:
{
provide
:
{
dropdownItems
:
mockDropdownItems
,
dropdownItems
:
mockDropdownItems
,
...
@@ -49,13 +40,12 @@ describe('NotificationsDropdown', () => {
...
@@ -49,13 +40,12 @@ describe('NotificationsDropdown', () => {
});
});
}
}
const
findButtonGroup
=
()
=>
wrapper
.
find
(
GlButtonGroup
);
const
findButton
=
()
=>
wrapper
.
find
(
GlButton
);
const
findDropdown
=
()
=>
wrapper
.
find
(
GlDropdown
);
const
findDropdown
=
()
=>
wrapper
.
find
(
GlDropdown
);
const
findByTestId
=
(
testId
)
=>
wrapper
.
find
(
`[data-testid="
${
testId
}
"]`
);
const
findByTestId
=
(
testId
)
=>
wrapper
.
find
(
`[data-testid="
${
testId
}
"]`
);
const
findAllNotificationsDropdownItems
=
()
=>
wrapper
.
findAll
(
NotificationsDropdownItem
);
const
findAllNotificationsDropdownItems
=
()
=>
wrapper
.
findAll
(
NotificationsDropdownItem
);
const
findDropdownItemAt
=
(
index
)
=>
const
findDropdownItemAt
=
(
index
)
=>
findAllNotificationsDropdownItems
().
at
(
index
).
find
(
GlDropdownItem
);
findAllNotificationsDropdownItems
().
at
(
index
).
find
(
GlDropdownItem
);
const
findNotificationsModal
=
()
=>
wrapper
.
find
(
CustomNotificationsModal
);
const
clickDropdownItemAt
=
async
(
index
)
=>
{
const
clickDropdownItemAt
=
async
(
index
)
=>
{
const
dropdownItem
=
findDropdownItemAt
(
index
);
const
dropdownItem
=
findDropdownItemAt
(
index
);
...
@@ -83,8 +73,8 @@ describe('NotificationsDropdown', () => {
...
@@ -83,8 +73,8 @@ describe('NotificationsDropdown', () => {
});
});
});
});
it
(
'
renders
a button group
'
,
()
=>
{
it
(
'
renders
split dropdown
'
,
()
=>
{
expect
(
find
ButtonGroup
().
exists
()
).
toBe
(
true
);
expect
(
find
Dropdown
().
props
().
split
).
toBe
(
true
);
});
});
it
(
'
shows the button text when showLabel is true
'
,
()
=>
{
it
(
'
shows the button text when showLabel is true
'
,
()
=>
{
...
@@ -93,7 +83,7 @@ describe('NotificationsDropdown', () => {
...
@@ -93,7 +83,7 @@ describe('NotificationsDropdown', () => {
showLabel
:
true
,
showLabel
:
true
,
});
});
expect
(
find
Button
().
text
()
).
toBe
(
'
Custom
'
);
expect
(
find
Dropdown
().
props
().
text
).
toBe
(
'
Custom
'
);
});
});
it
(
"
doesn't show the button text when showLabel is false
"
,
()
=>
{
it
(
"
doesn't show the button text when showLabel is false
"
,
()
=>
{
...
@@ -102,7 +92,7 @@ describe('NotificationsDropdown', () => {
...
@@ -102,7 +92,7 @@ describe('NotificationsDropdown', () => {
showLabel
:
false
,
showLabel
:
false
,
});
});
expect
(
find
Button
().
text
()).
toBe
(
''
);
expect
(
find
Dropdown
().
props
().
text
).
toBe
(
null
);
});
});
it
(
'
opens the modal when the user clicks the button
'
,
async
()
=>
{
it
(
'
opens the modal when the user clicks the button
'
,
async
()
=>
{
...
@@ -113,9 +103,9 @@ describe('NotificationsDropdown', () => {
...
@@ -113,9 +103,9 @@ describe('NotificationsDropdown', () => {
initialNotificationLevel
:
'
custom
'
,
initialNotificationLevel
:
'
custom
'
,
});
});
findButto
n
().
vm
.
$emit
(
'
click
'
);
await
findDropdow
n
().
vm
.
$emit
(
'
click
'
);
expect
(
glModalDirective
).
toHaveBeenCalled
(
);
expect
(
findNotificationsModal
().
props
().
visible
).
toBe
(
true
);
});
});
});
});
...
@@ -126,8 +116,8 @@ describe('NotificationsDropdown', () => {
...
@@ -126,8 +116,8 @@ describe('NotificationsDropdown', () => {
});
});
});
});
it
(
'
does not render a button group
'
,
()
=>
{
it
(
'
renders unified dropdown
'
,
()
=>
{
expect
(
find
ButtonGroup
().
exists
()
).
toBe
(
false
);
expect
(
find
Dropdown
().
props
().
split
).
toBe
(
false
);
});
});
it
(
'
shows the button text when showLabel is true
'
,
()
=>
{
it
(
'
shows the button text when showLabel is true
'
,
()
=>
{
...
@@ -162,7 +152,7 @@ describe('NotificationsDropdown', () => {
...
@@ -162,7 +152,7 @@ describe('NotificationsDropdown', () => {
initialNotificationLevel
:
level
,
initialNotificationLevel
:
level
,
});
});
const
tooltipElement
=
findByTestId
(
'
notification-
butto
n
'
);
const
tooltipElement
=
findByTestId
(
'
notification-
dropdow
n
'
);
const
tooltip
=
getBinding
(
tooltipElement
.
element
,
'
gl-tooltip
'
);
const
tooltip
=
getBinding
(
tooltipElement
.
element
,
'
gl-tooltip
'
);
expect
(
tooltip
.
value
.
title
).
toBe
(
`
${
tooltipTitlePrefix
}
-
${
title
}
`
);
expect
(
tooltip
.
value
.
title
).
toBe
(
`
${
tooltipTitlePrefix
}
-
${
title
}
`
);
...
@@ -264,11 +254,9 @@ describe('NotificationsDropdown', () => {
...
@@ -264,11 +254,9 @@ describe('NotificationsDropdown', () => {
mockAxios
.
onPut
(
'
/api/v4/notification_settings
'
).
reply
(
httpStatus
.
OK
,
{});
mockAxios
.
onPut
(
'
/api/v4/notification_settings
'
).
reply
(
httpStatus
.
OK
,
{});
wrapper
=
createComponent
();
wrapper
=
createComponent
();
const
mockModalShow
=
jest
.
spyOn
(
wrapper
.
vm
.
$refs
.
customNotificationsModal
,
'
open
'
);
await
clickDropdownItemAt
(
5
);
await
clickDropdownItemAt
(
5
);
expect
(
mockModalShow
).
toHaveBeenCalled
(
);
expect
(
findNotificationsModal
().
props
().
visible
).
toBe
(
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