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
6912e757
Commit
6912e757
authored
Feb 15, 2021
by
Michael Lunøe
Committed by
Jose Ivan Vargas
Feb 15, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Refactor(profile preferences): migrate rails eval
Migrate rails eval'ed js to form event listeners
parent
8822a07e
Changes
16
Show whitespace changes
Inline
Side-by-side
Showing
16 changed files
with
321 additions
and
258 deletions
+321
-258
app/assets/javascripts/profile/preferences/components/profile_preferences.vue
...ts/profile/preferences/components/profile_preferences.vue
+90
-15
app/assets/javascripts/profile/preferences/constants.js
app/assets/javascripts/profile/preferences/constants.js
+22
-0
app/assets/javascripts/profile/preferences/profile_preferences_bundle.js
...scripts/profile/preferences/profile_preferences_bundle.js
+12
-8
app/controllers/profiles/preferences_controller.rb
app/controllers/profiles/preferences_controller.rb
+10
-15
app/views/profiles/preferences/show.html.haml
app/views/profiles/preferences/show.html.haml
+5
-10
app/views/profiles/preferences/update.js.erb
app/views/profiles/preferences/update.js.erb
+0
-20
ee/spec/controllers/ee/profiles/preferences_controller_spec.rb
...ec/controllers/ee/profiles/preferences_controller_spec.rb
+6
-3
locale/gitlab.pot
locale/gitlab.pot
+21
-18
spec/controllers/profiles/preferences_controller_spec.rb
spec/controllers/profiles/preferences_controller_spec.rb
+14
-21
spec/features/profiles/user_edit_preferences_spec.rb
spec/features/profiles/user_edit_preferences_spec.rb
+9
-2
spec/features/profiles/user_visits_profile_preferences_page_spec.rb
...res/profiles/user_visits_profile_preferences_page_spec.rb
+7
-5
spec/frontend/profile/preferences/components/__snapshots__/integration_view_spec.js.snap
...es/components/__snapshots__/integration_view_spec.js.snap
+0
-67
spec/frontend/profile/preferences/components/__snapshots__/profile_preferences_spec.js.snap
...components/__snapshots__/profile_preferences_spec.js.snap
+0
-51
spec/frontend/profile/preferences/components/integration_view_spec.js
...d/profile/preferences/components/integration_view_spec.js
+0
-6
spec/frontend/profile/preferences/components/profile_preferences_spec.js
...rofile/preferences/components/profile_preferences_spec.js
+123
-17
spec/frontend/profile/preferences/mock_data.js
spec/frontend/profile/preferences/mock_data.js
+2
-0
No files found.
app/assets/javascripts/profile/preferences/components/profile_preferences.vue
View file @
6912e757
<
script
>
<
script
>
import
{
s__
}
from
'
~/locale
'
;
import
{
GlButton
}
from
'
@gitlab/ui
'
;
import
createFlash
,
{
FLASH_TYPES
}
from
'
~/flash
'
;
import
{
INTEGRATION_VIEW_CONFIGS
,
i18n
}
from
'
../constants
'
;
import
IntegrationView
from
'
./integration_view.vue
'
;
import
IntegrationView
from
'
./integration_view.vue
'
;
const
INTEGRATION_VIEW_CONFIGS
=
{
function
updateClasses
(
bodyClasses
=
''
,
applicationTheme
,
layout
)
{
sourcegraph
:
{
// Remove body class for any previous theme, re-add current one
title
:
s__
(
'
ProfilePreferences|Sourcegraph
'
),
document
.
body
.
classList
.
remove
(...
bodyClasses
.
split
(
'
'
));
label
:
s__
(
'
ProfilePreferences|Enable integrated code intelligence on code views
'
),
document
.
body
.
classList
.
add
(
applicationTheme
);
formName
:
'
sourcegraph_enabled
'
,
},
// Toggle container-fluid class
gitpod
:
{
if
(
layout
===
'
fluid
'
)
{
title
:
s__
(
'
ProfilePreferences|Gitpod
'
),
document
label
:
s__
(
'
ProfilePreferences|Enable Gitpod integration
'
),
.
querySelector
(
'
.content-wrapper .container-fluid
'
)
formName
:
'
gitpod_enabled
'
,
.
classList
.
remove
(
'
container-limited
'
);
},
}
else
{
};
document
.
querySelector
(
'
.content-wrapper .container-fluid
'
).
classList
.
add
(
'
container-limited
'
);
}
}
export
default
{
export
default
{
name
:
'
ProfilePreferences
'
,
name
:
'
ProfilePreferences
'
,
components
:
{
components
:
{
IntegrationView
,
IntegrationView
,
GlButton
,
},
},
inject
:
{
inject
:
{
integrationViews
:
{
integrationViews
:
{
default
:
[],
default
:
[],
},
},
themes
:
{
default
:
[],
},
userFields
:
{
default
:
{},
},
formEl
:
'
formEl
'
,
profilePreferencesPath
:
'
profilePreferencesPath
'
,
bodyClasses
:
'
bodyClasses
'
,
},
},
integrationViewConfigs
:
INTEGRATION_VIEW_CONFIGS
,
integrationViewConfigs
:
INTEGRATION_VIEW_CONFIGS
,
i18n
,
data
()
{
return
{
isSubmitEnabled
:
true
,
};
},
computed
:
{
applicationThemes
()
{
return
this
.
themes
.
reduce
((
themes
,
theme
)
=>
{
const
{
id
,
...
rest
}
=
theme
;
return
{
...
themes
,
[
id
]:
rest
};
},
{});
},
},
created
()
{
this
.
formEl
.
addEventListener
(
'
ajax:beforeSend
'
,
this
.
handleLoading
);
this
.
formEl
.
addEventListener
(
'
ajax:success
'
,
this
.
handleSuccess
);
this
.
formEl
.
addEventListener
(
'
ajax:error
'
,
this
.
handleError
);
},
beforeDestroy
()
{
this
.
formEl
.
removeEventListener
(
'
ajax:beforeSend
'
,
this
.
handleLoading
);
this
.
formEl
.
removeEventListener
(
'
ajax:success
'
,
this
.
handleSuccess
);
this
.
formEl
.
removeEventListener
(
'
ajax:error
'
,
this
.
handleError
);
},
methods
:
{
handleLoading
()
{
this
.
isSubmitEnabled
=
false
;
},
handleSuccess
(
customEvent
)
{
const
formData
=
new
FormData
(
this
.
formEl
);
updateClasses
(
this
.
bodyClasses
,
this
.
applicationThemes
[
formData
.
get
(
'
user[theme_id]
'
)].
css_class
,
this
.
selectedLayout
,
);
const
{
message
=
this
.
$options
.
i18n
.
defaultSuccess
,
type
=
FLASH_TYPES
.
NOTICE
}
=
customEvent
?.
detail
?.[
0
]
||
{};
createFlash
({
message
,
type
});
this
.
isSubmitEnabled
=
true
;
},
handleError
(
customEvent
)
{
const
{
message
=
this
.
$options
.
i18n
.
defaultError
,
type
=
FLASH_TYPES
.
ALERT
}
=
customEvent
?.
detail
?.[
0
]
||
{};
createFlash
({
message
,
type
});
this
.
isSubmitEnabled
=
true
;
},
},
};
};
</
script
>
</
script
>
...
@@ -36,10 +97,10 @@ export default {
...
@@ -36,10 +97,10 @@ export default {
</div>
</div>
<div
v-if=
"integrationViews.length"
class=
"col-lg-4 profile-settings-sidebar"
>
<div
v-if=
"integrationViews.length"
class=
"col-lg-4 profile-settings-sidebar"
>
<h4
class=
"gl-mt-0"
data-testid=
"profile-preferences-integrations-heading"
>
<h4
class=
"gl-mt-0"
data-testid=
"profile-preferences-integrations-heading"
>
{{
s__
(
'
ProfilePreferences|Integrations
'
)
}}
{{
$options
.
i18n
.
integrations
}}
</h4>
</h4>
<p>
<p>
{{
s__
(
'
ProfilePreferences|Customize integrations with third party services.
'
)
}}
{{
$options
.
i18n
.
integrationsDescription
}}
</p>
</p>
</div>
</div>
<div
v-if=
"integrationViews.length"
class=
"col-lg-8"
>
<div
v-if=
"integrationViews.length"
class=
"col-lg-8"
>
...
@@ -52,5 +113,19 @@ export default {
...
@@ -52,5 +113,19 @@ export default {
:config=
"$options.integrationViewConfigs[view.name]"
:config=
"$options.integrationViewConfigs[view.name]"
/>
/>
</div>
</div>
<div
class=
"col-lg-4 profile-settings-sidebar"
></div>
<div
class=
"col-lg-8"
>
<div
class=
"form-group"
>
<gl-button
variant=
"success"
name=
"commit"
type=
"submit"
:disabled=
"!isSubmitEnabled"
:value=
"$options.i18n.saveChanges"
>
{{
$options
.
i18n
.
saveChanges
}}
</gl-button>
</div>
</div>
</div>
</div>
</
template
>
</
template
>
app/assets/javascripts/profile/preferences/constants.js
0 → 100644
View file @
6912e757
import
{
s__
,
__
}
from
'
~/locale
'
;
export
const
INTEGRATION_VIEW_CONFIGS
=
{
sourcegraph
:
{
title
:
s__
(
'
Preferences|Sourcegraph
'
),
label
:
s__
(
'
Preferences|Enable integrated code intelligence on code views
'
),
formName
:
'
sourcegraph_enabled
'
,
},
gitpod
:
{
title
:
s__
(
'
Preferences|Gitpod
'
),
label
:
s__
(
'
Preferences|Enable Gitpod integration
'
),
formName
:
'
gitpod_enabled
'
,
},
};
export
const
i18n
=
{
saveChanges
:
__
(
'
Save changes
'
),
defaultSuccess
:
__
(
'
Preferences saved.
'
),
defaultError
:
s__
(
'
Preferences|Failed to save preferences.
'
),
integrations
:
s__
(
'
Preferences|Integrations
'
),
integrationsDescription
:
s__
(
'
Preferences|Customize integrations with third party services.
'
),
};
app/assets/javascripts/profile/preferences/profile_preferences_bundle.js
View file @
6912e757
...
@@ -3,16 +3,20 @@ import ProfilePreferences from './components/profile_preferences.vue';
...
@@ -3,16 +3,20 @@ import ProfilePreferences from './components/profile_preferences.vue';
export
default
()
=>
{
export
default
()
=>
{
const
el
=
document
.
querySelector
(
'
#js-profile-preferences-app
'
);
const
el
=
document
.
querySelector
(
'
#js-profile-preferences-app
'
);
const
shouldParse
=
[
'
integrationViews
'
,
'
userFields
'
];
const
formEl
=
document
.
querySelector
(
'
#profile-preferences-form
'
);
const
shouldParse
=
[
'
integrationViews
'
,
'
themes
'
,
'
userFields
'
];
const
provide
=
Object
.
keys
(
el
.
dataset
).
reduce
((
memo
,
key
)
=>
{
const
provide
=
Object
.
keys
(
el
.
dataset
).
reduce
(
(
memo
,
key
)
=>
{
let
value
=
el
.
dataset
[
key
];
let
value
=
el
.
dataset
[
key
];
if
(
shouldParse
.
includes
(
key
))
{
if
(
shouldParse
.
includes
(
key
))
{
value
=
JSON
.
parse
(
value
);
value
=
JSON
.
parse
(
value
);
}
}
return
{
...
memo
,
[
key
]:
value
};
return
{
...
memo
,
[
key
]:
value
};
},
{});
},
{
formEl
},
);
return
new
Vue
({
return
new
Vue
({
el
,
el
,
...
...
app/controllers/profiles/preferences_controller.rb
View file @
6912e757
...
@@ -9,23 +9,18 @@ class Profiles::PreferencesController < Profiles::ApplicationController
...
@@ -9,23 +9,18 @@ class Profiles::PreferencesController < Profiles::ApplicationController
end
end
def
update
def
update
begin
result
=
Users
::
UpdateService
.
new
(
current_user
,
preferences_params
.
merge
(
user:
user
)).
execute
result
=
Users
::
UpdateService
.
new
(
current_user
,
preferences_params
.
merge
(
user:
user
)).
execute
if
result
[
:status
]
==
:success
if
result
[
:status
]
==
:success
flash
[
:notice
]
=
_
(
'Preferences saved.'
)
message
=
_
(
'Preferences saved.'
)
render
json:
{
type: :notice
,
message:
message
}
else
else
flash
[
:alert
]
=
_
(
'Failed to save preferences.'
)
render
status: :bad_request
,
json:
{
type: :alert
,
message:
_
(
'Failed to save preferences.'
)
}
end
end
rescue
ArgumentError
=>
e
rescue
ArgumentError
=>
e
# Raised when `dashboard` is given an invalid value.
# Raised when `dashboard` is given an invalid value.
flash
[
:alert
]
=
_
(
"Failed to save preferences (%{error_message})."
)
%
{
error_message:
e
.
message
}
message
=
_
(
"Failed to save preferences (%{error_message})."
)
%
{
error_message:
e
.
message
}
end
render
status: :bad_request
,
json:
{
type: :alert
,
message:
message
}
respond_to
do
|
format
|
format
.
html
{
redirect_to
profile_preferences_path
}
format
.
js
end
end
end
private
private
...
...
app/views/profiles/preferences/show.html.haml
View file @
6912e757
-
page_title
_
(
'Preferences'
)
-
page_title
_
(
'Preferences'
)
-
@content_class
=
"limit-container-width"
unless
fluid_layout
-
@content_class
=
"limit-container-width"
unless
fluid_layout
-
user_fields
=
{
gitpod_enabled:
@user
.
gitpod_enabled
,
sourcegraph_enabled:
@user
.
sourcegraph_enabled
}
-
user_theme_id
=
Gitlab
::
Themes
.
for_user
(
@user
).
id
-
user_theme_id
=
Gitlab
::
Themes
.
for_user
(
@user
).
id
-
data_attributes
=
{
integration_views:
integration_views
.
to_json
,
user_fields:
user_fields
.
to_json
}
-
user_fields
=
{
theme:
user_theme_id
,
gitpod_enabled:
@user
.
gitpod_enabled
,
sourcegraph_enabled:
@user
.
sourcegraph_enabled
}.
to_json
-
@themes
=
Gitlab
::
Themes
::
THEMES
.
to_json
-
data_attributes
=
{
themes:
@themes
,
integration_views:
integration_views
.
to_json
,
user_fields:
user_fields
,
body_classes:
Gitlab
::
Themes
.
body_classes
,
profile_preferences_path:
profile_preferences_path
}
-
Gitlab
::
Themes
.
each
do
|
theme
|
-
Gitlab
::
Themes
.
each
do
|
theme
|
=
stylesheet_link_tag
"themes/
#{
theme
.
css_filename
}
"
if
theme
.
css_filename
=
stylesheet_link_tag
"themes/
#{
theme
.
css_filename
}
"
if
theme
.
css_filename
=
form_for
@user
,
url:
profile_preferences_path
,
remote:
true
,
method: :put
do
|
f
|
=
form_for
@user
,
url:
profile_preferences_path
,
remote:
true
,
method: :put
,
html:
{
id:
"profile-preferences-form"
}
do
|
f
|
.row.gl-mt-3.js-preferences-form.js-search-settings-section
.row.gl-mt-3.js-preferences-form.js-search-settings-section
.col-lg-4.application-theme
#navigation-theme
.col-lg-4.application-theme
#navigation-theme
%h4
.gl-mt-0
%h4
.gl-mt-0
...
@@ -143,10 +144,4 @@
...
@@ -143,10 +144,4 @@
.form-text.text-muted
.form-text.text-muted
=
s_
(
'Preferences|For example: 30 mins ago.'
)
=
s_
(
'Preferences|For example: 30 mins ago.'
)
#js-profile-preferences-app
{
data:
data_attributes
,
user_fields:
user_fields
.
to_json
}
#js-profile-preferences-app
{
data:
data_attributes
}
.row.gl-mt-3.js-preferences-form
.col-lg-4.profile-settings-sidebar
.col-lg-8
.form-group
=
f
.
submit
_
(
'Save changes'
),
class:
'gl-button btn btn-success'
app/views/profiles/preferences/update.js.erb
deleted
100644 → 0
View file @
8822a07e
// Remove body class for any previous theme, re-add current one
$('body').removeClass('
<%=
Gitlab
::
Themes
.
body_classes
%>
')
$('body').addClass('
<%=
user_application_theme
%>
')
// Toggle container-fluid class
if ('
<%=
current_user
.
layout
%>
' === 'fluid') {
$('.content-wrapper .container-fluid').removeClass('container-limited')
} else {
$('.content-wrapper .container-fluid').addClass('container-limited')
}
// Re-enable the "Save" button
$('input[type=submit]').enable()
// Show flash messages
<%
if
flash
.
notice
%>
new Flash({ message: '
<%=
flash
.
discard
(
:notice
)
%>
', type: 'notice'})
<%
elsif
flash
.
alert
%>
new Flash({ message: '
<%=
flash
.
discard
(
:alert
)
%>
', type: 'alert'})
<%
end
%>
ee/spec/controllers/ee/profiles/preferences_controller_spec.rb
View file @
6912e757
...
@@ -10,7 +10,7 @@ RSpec.describe Profiles::PreferencesController do
...
@@ -10,7 +10,7 @@ RSpec.describe Profiles::PreferencesController do
end
end
describe
'PATCH update'
do
describe
'PATCH update'
do
subject
{
patch
:update
,
params:
{
user:
{
group_view:
group_view
}
},
format: :js
}
subject
{
patch
:update
,
params:
{
user:
{
group_view:
group_view
}
},
format: :js
on
}
let
(
:group_view
)
{
'security_dashboard'
}
let
(
:group_view
)
{
'security_dashboard'
}
...
@@ -27,9 +27,12 @@ RSpec.describe Profiles::PreferencesController do
...
@@ -27,9 +27,12 @@ RSpec.describe Profiles::PreferencesController do
context
'and an invalid group view choice is submitted'
do
context
'and an invalid group view choice is submitted'
do
let
(
:group_view
)
{
'foo'
}
let
(
:group_view
)
{
'foo'
}
it
'
sets the flash
'
do
it
'
responds with an error message
'
do
subject
subject
expect
(
flash
[
:alert
]).
to
match
(
/Failed to save preferences/
)
expect
(
response
).
to
have_gitlab_http_status
(
:bad_request
)
expect
(
response
.
parsed_body
[
'message'
]).
to
match
(
/Failed to save preferences/
)
expect
(
response
.
parsed_body
[
'type'
]).
to
eq
(
'alert'
)
end
end
end
end
end
end
...
...
locale/gitlab.pot
View file @
6912e757
...
@@ -22218,21 +22218,39 @@ msgstr ""
...
@@ -22218,21 +22218,39 @@ msgstr ""
msgid "Preferences|Choose what content you want to see on your homepage."
msgid "Preferences|Choose what content you want to see on your homepage."
msgstr ""
msgstr ""
msgid "Preferences|Customize integrations with third party services."
msgstr ""
msgid "Preferences|Customize the appearance of the application header and navigation sidebar."
msgid "Preferences|Customize the appearance of the application header and navigation sidebar."
msgstr ""
msgstr ""
msgid "Preferences|Display time in 24-hour format"
msgid "Preferences|Display time in 24-hour format"
msgstr ""
msgstr ""
msgid "Preferences|Enable Gitpod integration"
msgstr ""
msgid "Preferences|Enable integrated code intelligence on code views"
msgstr ""
msgid "Preferences|Failed to save preferences."
msgstr ""
msgid "Preferences|For example: 30 mins ago."
msgid "Preferences|For example: 30 mins ago."
msgstr ""
msgstr ""
msgid "Preferences|Gitpod"
msgstr ""
msgid "Preferences|Homepage content"
msgid "Preferences|Homepage content"
msgstr ""
msgstr ""
msgid "Preferences|Instead of all the files changed, show only one file at a time. To switch between files, use the file browser."
msgid "Preferences|Instead of all the files changed, show only one file at a time. To switch between files, use the file browser."
msgstr ""
msgstr ""
msgid "Preferences|Integrations"
msgstr ""
msgid "Preferences|Layout width"
msgid "Preferences|Layout width"
msgstr ""
msgstr ""
...
@@ -22254,6 +22272,9 @@ msgstr ""
...
@@ -22254,6 +22272,9 @@ msgstr ""
msgid "Preferences|Show whitespace changes in diffs"
msgid "Preferences|Show whitespace changes in diffs"
msgstr ""
msgstr ""
msgid "Preferences|Sourcegraph"
msgstr ""
msgid "Preferences|Syntax highlighting theme"
msgid "Preferences|Syntax highlighting theme"
msgstr ""
msgstr ""
...
@@ -22446,24 +22467,6 @@ msgstr ""
...
@@ -22446,24 +22467,6 @@ msgstr ""
msgid "Profile Settings"
msgid "Profile Settings"
msgstr ""
msgstr ""
msgid "ProfilePreferences|Customize integrations with third party services."
msgstr ""
msgid "ProfilePreferences|Enable Gitpod integration"
msgstr ""
msgid "ProfilePreferences|Enable integrated code intelligence on code views"
msgstr ""
msgid "ProfilePreferences|Gitpod"
msgstr ""
msgid "ProfilePreferences|Integrations"
msgstr ""
msgid "ProfilePreferences|Sourcegraph"
msgstr ""
msgid "ProfileSession|on"
msgid "ProfileSession|on"
msgstr ""
msgstr ""
...
...
spec/controllers/profiles/preferences_controller_spec.rb
View file @
6912e757
...
@@ -24,7 +24,7 @@ RSpec.describe Profiles::PreferencesController do
...
@@ -24,7 +24,7 @@ RSpec.describe Profiles::PreferencesController do
end
end
describe
'PATCH update'
do
describe
'PATCH update'
do
def
go
(
params:
{},
format: :js
)
def
go
(
params:
{},
format: :js
on
)
params
.
reverse_merge!
(
params
.
reverse_merge!
(
color_scheme_id:
'1'
,
color_scheme_id:
'1'
,
dashboard:
'stars'
,
dashboard:
'stars'
,
...
@@ -35,9 +35,12 @@ RSpec.describe Profiles::PreferencesController do
...
@@ -35,9 +35,12 @@ RSpec.describe Profiles::PreferencesController do
end
end
context
'on successful update'
do
context
'on successful update'
do
it
'
sets the flash
'
do
it
'
responds with success
'
do
go
go
expect
(
flash
[
:notice
]).
to
eq
_
(
'Preferences saved.'
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
response
.
parsed_body
[
'message'
]).
to
eq
_
(
'Preferences saved.'
)
expect
(
response
.
parsed_body
[
'type'
]).
to
eq
(
'notice'
)
end
end
it
"changes the user's preferences"
do
it
"changes the user's preferences"
do
...
@@ -59,36 +62,26 @@ RSpec.describe Profiles::PreferencesController do
...
@@ -59,36 +62,26 @@ RSpec.describe Profiles::PreferencesController do
end
end
context
'on failed update'
do
context
'on failed update'
do
it
'
sets the flash
'
do
it
'
responds with error
'
do
expect
(
user
).
to
receive
(
:save
).
and_return
(
false
)
expect
(
user
).
to
receive
(
:save
).
and_return
(
false
)
go
go
expect
(
flash
[
:alert
]).
to
eq
(
_
(
'Failed to save preferences.'
))
expect
(
response
).
to
have_gitlab_http_status
(
:bad_request
)
expect
(
response
.
parsed_body
[
'message'
]).
to
eq
_
(
'Failed to save preferences.'
)
expect
(
response
.
parsed_body
[
'type'
]).
to
eq
(
'alert'
)
end
end
end
end
context
'on invalid dashboard setting'
do
context
'on invalid dashboard setting'
do
it
'
sets the flash
'
do
it
'
responds with error
'
do
prefs
=
{
dashboard:
'invalid'
}
prefs
=
{
dashboard:
'invalid'
}
go
params:
prefs
go
params:
prefs
expect
(
flash
[
:alert
]).
to
match
(
/\AFailed to save preferences \(.+\)\.\z/
)
expect
(
response
).
to
have_gitlab_http_status
(
:bad_request
)
end
expect
(
response
.
parsed_body
[
'message'
]).
to
match
(
/\AFailed to save preferences \(.+\)\.\z/
)
end
expect
(
response
.
parsed_body
[
'type'
]).
to
eq
(
'alert'
)
context
'as js'
do
it
'renders'
do
go
expect
(
response
).
to
render_template
:update
end
end
context
'as html'
do
it
'redirects'
do
go
format: :html
expect
(
response
).
to
redirect_to
(
profile_preferences_path
)
end
end
end
end
end
end
...
...
spec/features/profiles/user_edit_preferences_spec.rb
View file @
6912e757
# frozen_string_literal: true
# frozen_string_literal: true
require
'spec_helper'
require
'spec_helper'
RSpec
.
describe
'User edit preferences profile'
do
RSpec
.
describe
'User edit preferences profile'
,
:js
do
let
(
:user
)
{
create
(
:user
)
}
let
(
:user
)
{
create
(
:user
)
}
before
do
before
do
...
@@ -53,7 +53,14 @@ RSpec.describe 'User edit preferences profile' do
...
@@ -53,7 +53,14 @@ RSpec.describe 'User edit preferences profile' do
fill_in
'Tab width'
,
with:
-
1
fill_in
'Tab width'
,
with:
-
1
click_button
'Save changes'
click_button
'Save changes'
expect
(
page
).
to
have_content
(
'Failed to save preferences'
)
field
=
page
.
find_field
(
'user[tab_width]'
)
message
=
field
.
native
.
attribute
(
"validationMessage"
)
expect
(
message
).
to
eq
"Value must be greater than or equal to 1."
# User trying to hack an invalid value
page
.
execute_script
(
"document.querySelector('#user_tab_width').setAttribute('min', '-1')"
)
click_button
'Save changes'
expect
(
page
).
to
have_content
(
'Failed to save preferences.'
)
end
end
end
end
...
...
spec/features/profiles/user_visits_profile_preferences_page_spec.rb
View file @
6912e757
...
@@ -2,7 +2,7 @@
...
@@ -2,7 +2,7 @@
require
'spec_helper'
require
'spec_helper'
RSpec
.
describe
'User visits the profile preferences page'
do
RSpec
.
describe
'User visits the profile preferences page'
,
:js
do
include
Select2Helper
include
Select2Helper
let
(
:user
)
{
create
(
:user
)
}
let
(
:user
)
{
create
(
:user
)
}
...
@@ -39,7 +39,7 @@ RSpec.describe 'User visits the profile preferences page' do
...
@@ -39,7 +39,7 @@ RSpec.describe 'User visits the profile preferences page' do
describe
'User changes their default dashboard'
,
:js
do
describe
'User changes their default dashboard'
,
:js
do
it
'creates a flash message'
do
it
'creates a flash message'
do
select2
(
'stars'
,
from:
'#user_dashboard'
)
select2
(
'stars'
,
from:
'#user_dashboard'
)
click_button
'Save'
click_button
'Save
changes
'
wait_for_requests
wait_for_requests
...
@@ -48,7 +48,7 @@ RSpec.describe 'User visits the profile preferences page' do
...
@@ -48,7 +48,7 @@ RSpec.describe 'User visits the profile preferences page' do
it
'updates their preference'
do
it
'updates their preference'
do
select2
(
'stars'
,
from:
'#user_dashboard'
)
select2
(
'stars'
,
from:
'#user_dashboard'
)
click_button
'Save'
click_button
'Save
changes
'
wait_for_requests
wait_for_requests
...
@@ -67,7 +67,7 @@ RSpec.describe 'User visits the profile preferences page' do
...
@@ -67,7 +67,7 @@ RSpec.describe 'User visits the profile preferences page' do
describe
'User changes their language'
,
:js
do
describe
'User changes their language'
,
:js
do
it
'creates a flash message'
,
quarantine:
'https://gitlab.com/gitlab-org/gitlab/-/issues/31404'
do
it
'creates a flash message'
,
quarantine:
'https://gitlab.com/gitlab-org/gitlab/-/issues/31404'
do
select2
(
'en'
,
from:
'#user_preferred_language'
)
select2
(
'en'
,
from:
'#user_preferred_language'
)
click_button
'Save'
click_button
'Save
changes
'
wait_for_requests
wait_for_requests
...
@@ -77,7 +77,7 @@ RSpec.describe 'User visits the profile preferences page' do
...
@@ -77,7 +77,7 @@ RSpec.describe 'User visits the profile preferences page' do
it
'updates their preference'
do
it
'updates their preference'
do
wait_for_requests
wait_for_requests
select2
(
'pt_BR'
,
from:
'#user_preferred_language'
)
select2
(
'pt_BR'
,
from:
'#user_preferred_language'
)
click_button
'Save'
click_button
'Save
changes
'
wait_for_requests
wait_for_requests
refresh
refresh
...
@@ -94,6 +94,8 @@ RSpec.describe 'User visits the profile preferences page' do
...
@@ -94,6 +94,8 @@ RSpec.describe 'User visits the profile preferences page' do
click_button
'Save changes'
click_button
'Save changes'
wait_for_requests
expect
(
user
.
reload
.
render_whitespace_in_code
).
to
be
(
true
)
expect
(
user
.
reload
.
render_whitespace_in_code
).
to
be
(
true
)
expect
(
render_whitespace_field
).
to
be_checked
expect
(
render_whitespace_field
).
to
be_checked
end
end
...
...
spec/frontend/profile/preferences/components/__snapshots__/integration_view_spec.js.snap
deleted
100644 → 0
View file @
8822a07e
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`IntegrationView component should render IntegrationView properly 1`] = `
<div
name="sourcegraph"
>
<label
class="label-bold"
>
Foo
</label>
<gl-link-stub
class="has-tooltip"
href="http://foo.com/help"
title="More information"
>
<gl-icon-stub
class="vertical-align-middle"
name="question-o"
size="16"
/>
</gl-link-stub>
<div
class="form-group form-check"
data-testid="profile-preferences-integration-form-group"
>
<input
data-testid="profile-preferences-integration-hidden-field"
name="user[foo_enabled]"
type="hidden"
value="0"
/>
<input
class="form-check-input"
data-testid="profile-preferences-integration-checkbox"
id="user_foo_enabled"
name="user[foo_enabled]"
type="checkbox"
value="1"
/>
<label
class="form-check-label"
for="user_foo_enabled"
>
Enable foo
</label>
<gl-form-text-stub
tag="div"
textvariant="muted"
>
<integration-help-text-stub
message="Click %{linkStart}Foo%{linkEnd}!"
messageurl="http://foo.com"
/>
</gl-form-text-stub>
</div>
</div>
`;
spec/frontend/profile/preferences/components/__snapshots__/profile_preferences_spec.js.snap
deleted
100644 → 0
View file @
8822a07e
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`ProfilePreferences component should render ProfilePreferences properly 1`] = `
<div
class="row gl-mt-3 js-preferences-form"
>
<div
class="col-sm-12"
>
<hr
data-testid="profile-preferences-integrations-rule"
/>
</div>
<div
class="col-lg-4 profile-settings-sidebar"
>
<h4
class="gl-mt-0"
data-testid="profile-preferences-integrations-heading"
>
Integrations
</h4>
<p>
Customize integrations with third party services.
</p>
</div>
<div
class="col-lg-8"
>
<integration-view-stub
config="[object Object]"
helplink="http://foo.com/help"
message="Click %{linkStart}Foo%{linkEnd}!"
messageurl="http://foo.com"
/>
<integration-view-stub
config="[object Object]"
helplink="http://bar.com/help"
message="Click %{linkStart}Bar%{linkEnd}!"
messageurl="http://bar.com"
/>
</div>
</div>
`;
spec/frontend/profile/preferences/components/integration_view_spec.js
View file @
6912e757
...
@@ -115,10 +115,4 @@ describe('IntegrationView component', () => {
...
@@ -115,10 +115,4 @@ describe('IntegrationView component', () => {
expect
(
findFormGroupLabel
().
text
()).
toBe
(
'
Enable foo
'
);
expect
(
findFormGroupLabel
().
text
()).
toBe
(
'
Enable foo
'
);
});
});
it
(
'
should render IntegrationView properly
'
,
()
=>
{
wrapper
=
createComponent
();
expect
(
wrapper
.
element
).
toMatchSnapshot
();
});
});
});
spec/frontend/profile/preferences/components/profile_preferences_spec.js
View file @
6912e757
import
{
GlButton
}
from
'
@gitlab/ui
'
;
import
{
shallowMount
}
from
'
@vue/test-utils
'
;
import
{
shallowMount
}
from
'
@vue/test-utils
'
;
import
{
extendedWrapper
}
from
'
helpers/vue_test_utils_helper
'
;
import
IntegrationView
from
'
~/profile/preferences/components/integration_view.vue
'
;
import
IntegrationView
from
'
~/profile/preferences/components/integration_view.vue
'
;
import
ProfilePreferences
from
'
~/profile/preferences/components/profile_preferences.vue
'
;
import
ProfilePreferences
from
'
~/profile/preferences/components/profile_preferences.vue
'
;
import
{
integrationViews
,
userFields
}
from
'
../mock_data
'
;
import
{
i18n
}
from
'
~/profile/preferences/constants
'
;
import
{
integrationViews
,
userFields
,
bodyClasses
}
from
'
../mock_data
'
;
const
expectedUrl
=
'
/foo
'
;
describe
(
'
ProfilePreferences component
'
,
()
=>
{
describe
(
'
ProfilePreferences component
'
,
()
=>
{
let
wrapper
;
let
wrapper
;
const
defaultProvide
=
{
const
defaultProvide
=
{
integrationViews
:
[],
integrationViews
:
[],
userFields
,
userFields
,
bodyClasses
,
themes
:
[{
id
:
1
,
css_class
:
'
foo
'
}],
profilePreferencesPath
:
'
/update-profile
'
,
formEl
:
document
.
createElement
(
'
form
'
),
};
};
function
createComponent
(
options
=
{})
{
function
createComponent
(
options
=
{})
{
const
{
props
=
{},
provide
=
{}
}
=
options
;
const
{
props
=
{},
provide
=
{},
attachTo
}
=
options
;
return
shallowMount
(
ProfilePreferences
,
{
return
extendedWrapper
(
shallowMount
(
ProfilePreferences
,
{
provide
:
{
provide
:
{
...
defaultProvide
,
...
defaultProvide
,
...
provide
,
...
provide
,
},
},
propsData
:
props
,
propsData
:
props
,
});
attachTo
,
}),
);
}
function
findIntegrationsDivider
()
{
return
wrapper
.
findByTestId
(
'
profile-preferences-integrations-rule
'
);
}
function
findIntegrationsHeading
()
{
return
wrapper
.
findByTestId
(
'
profile-preferences-integrations-heading
'
);
}
function
findSubmitButton
()
{
return
wrapper
.
findComponent
(
GlButton
);
}
function
findFlashError
()
{
return
document
.
querySelector
(
'
.flash-container .flash-text
'
);
}
}
beforeEach
(()
=>
{
setFixtures
(
'
<div class="flash-container"></div>
'
);
});
afterEach
(()
=>
{
afterEach
(()
=>
{
wrapper
.
destroy
();
wrapper
.
destroy
();
wrapper
=
null
;
wrapper
=
null
;
...
@@ -30,8 +61,8 @@ describe('ProfilePreferences component', () => {
...
@@ -30,8 +61,8 @@ describe('ProfilePreferences component', () => {
it
(
'
should not render Integrations section
'
,
()
=>
{
it
(
'
should not render Integrations section
'
,
()
=>
{
wrapper
=
createComponent
();
wrapper
=
createComponent
();
const
views
=
wrapper
.
findAll
(
IntegrationView
);
const
views
=
wrapper
.
findAll
(
IntegrationView
);
const
divider
=
wrapper
.
find
(
'
[data-testid="profile-preferences-integrations-rule"]
'
);
const
divider
=
findIntegrationsDivider
(
);
const
heading
=
wrapper
.
find
(
'
[data-testid="profile-preferences-integrations-heading"]
'
);
const
heading
=
findIntegrationsHeading
(
);
expect
(
divider
.
exists
()).
toBe
(
false
);
expect
(
divider
.
exists
()).
toBe
(
false
);
expect
(
heading
.
exists
()).
toBe
(
false
);
expect
(
heading
.
exists
()).
toBe
(
false
);
...
@@ -40,8 +71,8 @@ describe('ProfilePreferences component', () => {
...
@@ -40,8 +71,8 @@ describe('ProfilePreferences component', () => {
it
(
'
should render Integration section
'
,
()
=>
{
it
(
'
should render Integration section
'
,
()
=>
{
wrapper
=
createComponent
({
provide
:
{
integrationViews
}
});
wrapper
=
createComponent
({
provide
:
{
integrationViews
}
});
const
divider
=
wrapper
.
find
(
'
[data-testid="profile-preferences-integrations-rule"]
'
);
const
divider
=
findIntegrationsDivider
(
);
const
heading
=
wrapper
.
find
(
'
[data-testid="profile-preferences-integrations-heading"]
'
);
const
heading
=
findIntegrationsHeading
(
);
const
views
=
wrapper
.
findAll
(
IntegrationView
);
const
views
=
wrapper
.
findAll
(
IntegrationView
);
expect
(
divider
.
exists
()).
toBe
(
true
);
expect
(
divider
.
exists
()).
toBe
(
true
);
...
@@ -49,9 +80,84 @@ describe('ProfilePreferences component', () => {
...
@@ -49,9 +80,84 @@ describe('ProfilePreferences component', () => {
expect
(
views
).
toHaveLength
(
integrationViews
.
length
);
expect
(
views
).
toHaveLength
(
integrationViews
.
length
);
});
});
it
(
'
should render ProfilePreferences properly
'
,
()
=>
{
describe
(
'
form submit
'
,
()
=>
{
wrapper
=
createComponent
({
provide
:
{
integrationViews
}
});
let
form
;
beforeEach
(()
=>
{
const
div
=
document
.
createElement
(
'
div
'
);
div
.
classList
.
add
(
'
container-fluid
'
);
document
.
body
.
appendChild
(
div
);
document
.
body
.
classList
.
add
(
'
content-wrapper
'
);
form
=
document
.
createElement
(
'
form
'
);
form
.
setAttribute
(
'
url
'
,
expectedUrl
);
form
.
setAttribute
(
'
method
'
,
'
put
'
);
const
input
=
document
.
createElement
(
'
input
'
);
input
.
setAttribute
(
'
name
'
,
'
user[theme_id]
'
);
input
.
setAttribute
(
'
type
'
,
'
radio
'
);
input
.
setAttribute
(
'
value
'
,
'
1
'
);
input
.
setAttribute
(
'
checked
'
,
'
checked
'
);
form
.
appendChild
(
input
);
wrapper
=
createComponent
({
provide
:
{
formEl
:
form
},
attachTo
:
document
.
body
});
const
beforeSendEvent
=
new
CustomEvent
(
'
ajax:beforeSend
'
);
form
.
dispatchEvent
(
beforeSendEvent
);
});
expect
(
wrapper
.
element
).
toMatchSnapshot
();
it
(
'
disables the submit button
'
,
async
()
=>
{
await
wrapper
.
vm
.
$nextTick
();
const
button
=
findSubmitButton
();
expect
(
button
.
props
(
'
disabled
'
)).
toBe
(
true
);
});
it
(
'
success re-enables the submit button
'
,
async
()
=>
{
const
successEvent
=
new
CustomEvent
(
'
ajax:success
'
);
form
.
dispatchEvent
(
successEvent
);
await
wrapper
.
vm
.
$nextTick
();
const
button
=
findSubmitButton
();
expect
(
button
.
props
(
'
disabled
'
)).
toBe
(
false
);
});
it
(
'
error re-enables the submit button
'
,
async
()
=>
{
const
errorEvent
=
new
CustomEvent
(
'
ajax:error
'
);
form
.
dispatchEvent
(
errorEvent
);
await
wrapper
.
vm
.
$nextTick
();
const
button
=
findSubmitButton
();
expect
(
button
.
props
(
'
disabled
'
)).
toBe
(
false
);
});
it
(
'
displays the default success message
'
,
()
=>
{
const
successEvent
=
new
CustomEvent
(
'
ajax:success
'
);
form
.
dispatchEvent
(
successEvent
);
expect
(
findFlashError
().
innerText
.
trim
()).
toEqual
(
i18n
.
defaultSuccess
);
});
it
(
'
displays the custom success message
'
,
()
=>
{
const
message
=
'
foo
'
;
const
successEvent
=
new
CustomEvent
(
'
ajax:success
'
,
{
detail
:
[{
message
}]
});
form
.
dispatchEvent
(
successEvent
);
expect
(
findFlashError
().
innerText
.
trim
()).
toEqual
(
message
);
});
it
(
'
displays the default error message
'
,
()
=>
{
const
errorEvent
=
new
CustomEvent
(
'
ajax:error
'
);
form
.
dispatchEvent
(
errorEvent
);
expect
(
findFlashError
().
innerText
.
trim
()).
toEqual
(
i18n
.
defaultError
);
});
it
(
'
displays the custom error message
'
,
()
=>
{
const
message
=
'
bar
'
;
const
errorEvent
=
new
CustomEvent
(
'
ajax:error
'
,
{
detail
:
[{
message
}]
});
form
.
dispatchEvent
(
errorEvent
);
expect
(
findFlashError
().
innerText
.
trim
()).
toEqual
(
message
);
});
});
});
});
});
spec/frontend/profile/preferences/mock_data.js
View file @
6912e757
...
@@ -16,3 +16,5 @@ export const integrationViews = [
...
@@ -16,3 +16,5 @@ export const integrationViews = [
export
const
userFields
=
{
export
const
userFields
=
{
foo_enabled
:
true
,
foo_enabled
:
true
,
};
};
export
const
bodyClasses
=
'
ui-light-indigo ui-light gl-dark
'
;
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