Commit c2f8bdc7 authored by Clement Ho's avatar Clement Ho

Merge branch 'ee-proj-settings-ok-leftovers' into 'master'

Improve project settings page (leftovers)

See merge request gitlab-org/gitlab-ee!10630
parents 51eb9527 4f12ef0f
import _ from 'underscore'; import _ from 'underscore';
import $ from 'jquery';
class DirtySubmitForm { class DirtySubmitForm {
constructor(form) { constructor(form) {
...@@ -26,6 +27,7 @@ class DirtySubmitForm { ...@@ -26,6 +27,7 @@ class DirtySubmitForm {
); );
this.form.addEventListener('input', throttledUpdateDirtyInput); this.form.addEventListener('input', throttledUpdateDirtyInput);
this.form.addEventListener('change', throttledUpdateDirtyInput); this.form.addEventListener('change', throttledUpdateDirtyInput);
$(this.form).on('change.select2', throttledUpdateDirtyInput);
this.form.addEventListener('submit', event => this.formSubmit(event)); this.form.addEventListener('submit', event => this.formSubmit(event));
} }
......
...@@ -3,17 +3,24 @@ import initSettingsPanels from '~/settings_panels'; ...@@ -3,17 +3,24 @@ import initSettingsPanels from '~/settings_panels';
import setupProjectEdit from '~/project_edit'; import setupProjectEdit from '~/project_edit';
import initConfirmDangerModal from '~/confirm_danger_modal'; import initConfirmDangerModal from '~/confirm_danger_modal';
import mountBadgeSettings from '~/pages/shared/mount_badge_settings'; import mountBadgeSettings from '~/pages/shared/mount_badge_settings';
import dirtySubmitFactory from '~/dirty_submit/dirty_submit_factory';
import initAvatarPicker from '~/avatar_picker'; import initAvatarPicker from '~/avatar_picker';
import initProjectLoadingSpinner from '../shared/save_project_loader'; import initProjectLoadingSpinner from '../shared/save_project_loader';
import initProjectPermissionsSettings from '../shared/permissions'; import initProjectPermissionsSettings from '../shared/permissions';
document.addEventListener('DOMContentLoaded', () => { document.addEventListener('DOMContentLoaded', () => {
initProjectLoadingSpinner();
setupProjectEdit();
// Initialize expandable settings panels
initSettingsPanels();
initAvatarPicker(); initAvatarPicker();
initProjectPermissionsSettings();
initConfirmDangerModal(); initConfirmDangerModal();
initSettingsPanels();
mountBadgeSettings(PROJECT_BADGE); mountBadgeSettings(PROJECT_BADGE);
initProjectLoadingSpinner();
initProjectPermissionsSettings();
setupProjectEdit();
dirtySubmitFactory(
document.querySelectorAll(
'.js-general-settings-form, .js-mr-settings-form, .js-mr-approvals-form',
),
);
}); });
...@@ -157,6 +157,10 @@ label { ...@@ -157,6 +157,10 @@ label {
padding-left: 10px; padding-left: 10px;
padding-right: 10px; padding-right: 10px;
appearance: none; appearance: none;
/* stylelint-disable property-no-vendor-prefix */
-webkit-appearance: none;
-moz-appearance: none;
/* stylelint-enable property-no-vendor-prefix */
&::-ms-expand { &::-ms-expand {
display: none; display: none;
......
...@@ -39,7 +39,7 @@ ...@@ -39,7 +39,7 @@
.settings-header { .settings-header {
position: relative; position: relative;
padding: 20px 110px 10px 0; padding: 20px 110px 0 0;
h4 { h4 {
margin-top: 0; margin-top: 0;
......
- if ::Gitlab::ExternalAuthorization.enabled? - if ::Gitlab::ExternalAuthorization.enabled?
.form-group .form-group.col-md-9
= f.label :external_authorization_classification_label, class: 'label-bold' do = f.label :external_authorization_classification_label, _('Classification Label (optional)'), class: 'label-bold'
= s_('ExternalAuthorizationService|Classification Label')
%span.light (optional)
= f.text_field :external_authorization_classification_label, class: "form-control" = f.text_field :external_authorization_classification_label, class: "form-control"
%span.form-text.text-muted %span.form-text.text-muted
= external_classification_label_help_message = external_classification_label_help_message
- return unless Gitlab::CurrentSettings.project_export_enabled? - return unless Gitlab::CurrentSettings.project_export_enabled?
- project = local_assigns.fetch(:project) - project = local_assigns.fetch(:project)
- expanded = Rails.env.test?
%section.settings.no-animate#js-export-project{ class: ('expanded' if expanded) } .sub-section
.settings-header %h4= _('Export project')
%h4 %p= _('Export this project with all its related data in order to move your project to a new GitLab instance. Once the export is finished, you can import the file from the "New Project" page.')
Export project
%button.btn.js-settings-toggle{ type: 'button' } .bs-callout.bs-callout-info
= expanded ? 'Collapse' : 'Expand' %p.append-bottom-0
%p %p= _('The following items will be exported:')
Export this project with all its related data in order to move your project to a new GitLab instance. Once the export is finished, you can import the file from the "New Project" page. %ul
.settings-content %li= _('Project and wiki repositories')
.bs-callout.bs-callout-info %li= _('Project uploads')
%p.append-bottom-0 %li= _('Project configuration, including services')
%p %li= _('Issues with comments, merge requests with diffs and comments, labels, milestones, snippets, and other project entities')
The following items will be exported: %li= _('LFS objects')
%ul %p= _('The following items will NOT be exported:')
%li Project and wiki repositories %ul
%li Project uploads %li= _('Job traces and artifacts')
%li Project configuration, including services %li= _('Container registry images')
%li Issues with comments, merge requests with diffs and comments, labels, milestones, snippets, and other project entities %li= _('CI variables')
%li LFS objects %li= _('Webhooks')
%p %li= _('Any encrypted tokens')
The following items will NOT be exported: %p= _('Once the exported file is ready, you will receive a notification email with a download link, or you can download it from this page.')
%ul - if project.export_status == :finished
%li Job traces and artifacts = link_to _('Download export'), download_export_project_path(project),
%li Container registry images rel: 'nofollow', download: '', method: :get, class: "btn btn-default"
%li CI variables = link_to _('Generate new export'), generate_new_export_project_path(project),
%li Webhooks method: :post, class: "btn btn-default"
%li Any encrypted tokens - else
%p = link_to _('Export project'), export_project_path(project),
Once the exported file is ready, you will receive a notification email with a download link, or you can download it from this page. method: :post, class: "btn btn-default"
- if project.export_status == :finished
= link_to 'Download export', download_export_project_path(project),
rel: 'nofollow', download: '', method: :get, class: "btn btn-default"
= link_to 'Generate new export', generate_new_export_project_path(project),
method: :post, class: "btn btn-default"
- else
= link_to 'Export project', export_project_path(project),
method: :post, class: "btn btn-default"
...@@ -3,210 +3,155 @@ ...@@ -3,210 +3,155 @@
- @content_class = "limit-container-width" unless fluid_layout - @content_class = "limit-container-width" unless fluid_layout
- expanded = Rails.env.test? - expanded = Rails.env.test?
.project-edit-container %section.settings.general-settings.no-animate.expanded#js-general-settings
%section.settings.general-settings.no-animate#js-general-project-settings{ class: ('expanded' if expanded) } .settings-header
.settings-header %h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Naming, topics, avatar')
%h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Naming, tags, avatar') %button.btn.btn-default.js-settings-toggle{ type: 'button' }= _('Collapse')
%button.btn.js-settings-toggle{ type: 'button' }= expanded ? _('Collapse') : _('Expand') %p= _('Update your project name, topics, description and avatar.')
%p= _('Update your project name, tags, description and avatar.') .settings-content= render 'projects/settings/general'
.settings-content %section.settings.sharing-permissions.no-animate#js-shared-permissions{ class: ('expanded' if expanded) }
.project-edit-errors .settings-header
= form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { multipart: true, class: "edit-project" }, authenticity_token: true do |f| %h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Visibility, project features, permissions')
%input{ name: 'update_section', type: 'hidden', value: 'js-general-project-settings' } %button.btn.btn-default.js-settings-toggle{ type: 'button' }= expanded ? _('Collapse') : _('Expand')
%fieldset %p= _('Choose visibility level, enable/disable project features (issues, repository, wiki, snippets) and set permissions.')
.row
.form-group.col-md-9 .settings-content
= f.label :name, class: 'label-bold', for: 'project_name_edit' do = form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { multipart: true, class: "sharing-permissions-form" }, authenticity_token: true do |f|
Project name %input{ name: 'update_section', type: 'hidden', value: 'js-shared-permissions' }
= f.text_field :name, class: "form-control", id: "project_name_edit" %template.js-project-permissions-form-data{ type: "application/json" }= project_permissions_panel_data_json(@project)
.js-project-permissions-form
.form-group.col-md-3 = f.submit _('Save changes'), class: "btn btn-success"
= f.label :id, class: 'label-bold' do
Project ID %section.qa-merge-request-settings.settings.merge-requests-feature.no-animate#js-merge-request-settings{ class: [('expanded' if expanded), ('hidden' if @project.project_feature.send(:merge_requests_access_level) == 0)] }
= f.text_field :id, class: 'form-control', readonly: true .settings-header
%h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Merge requests')
.form-group %button.btn.btn-default.js-settings-toggle{ type: 'button' }= expanded ? _('Collapse') : _('Expand')
= f.label :description, class: 'label-bold' do %p= _('Choose your merge method, set up a default merge request description template.')
Project description
%span.light (optional) .settings-content
= f.text_area :description, class: "form-control", rows: 3, maxlength: 250 = render_if_exists 'shared/promotions/promote_mr_features'
= render 'projects/classification_policy_settings', f: f = form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { multipart: true, class: "merge-request-settings-form js-mr-settings-form" }, authenticity_token: true do |f|
%input{ name: 'update_section', type: 'hidden', value: 'js-merge-request-settings' }
= render_if_exists 'shared/repository_size_limit_setting', form: f, type: :project = render 'projects/merge_request_settings', form: f
= f.submit _('Save changes'), class: "btn btn-success qa-save-merge-request-changes"
.form-group
= f.label :tag_list, "Topics", class: 'label-bold' = render_if_exists 'projects/merge_request_approvals_settings', expanded: expanded
= f.text_field :tag_list, value: @project.tag_list.join(', '), maxlength: 2000, class: "form-control"
%p.form-text.text-muted Separate topics with commas.
%section.settings.no-animate{ class: ('expanded' if expanded) }
.form-group.prepend-top-default.append-bottom-20 .settings-header
.avatar-container.s90 %h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only
= project_icon(@project, alt: _('Project avatar'), class: 'avatar project-avatar s90') = s_('ProjectSettings|Badges')
= f.label :avatar, _('Project avatar'), class: 'label-bold d-block' %button.btn.btn-default.js-settings-toggle{ type: 'button' }
= render 'shared/choose_avatar_button', f: f = expanded ? _('Collapse') : _('Expand')
- if @project.avatar? %p
%hr = s_('ProjectSettings|Customize your project badges.')
= link_to _('Remove avatar'), project_avatar_path(@project), data: { confirm: _('Avatar will be removed. Are you sure?')}, method: :delete, class: 'btn btn-link' = link_to s_('ProjectSettings|Learn more about badges.'), help_page_path('user/project/badges')
.settings-content
= f.submit 'Save changes', class: "btn btn-success js-btn-success-general-project-settings" = render 'shared/badges/badge_settings'
%section.settings.sharing-permissions.no-animate#js-shared-permissions{ class: ('expanded' if expanded) } = render_if_exists 'projects/settings/default_issue_template'
.settings-header
%h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Visibility, project features, permissions') = render_if_exists 'projects/service_desk_settings'
%button.btn.js-settings-toggle{ type: 'button' }= expanded ? _('Collapse') : _('Expand')
%p= _('Choose visibility level, enable/disable project features (issues, repository, wiki, snippets) and set permissions.') %section.qa-advanced-settings.settings.advanced-settings.no-animate#js-project-advanced-settings{ class: ('expanded' if expanded) }
.settings-header
.settings-content %h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Advanced')
= form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { multipart: true, class: "sharing-permissions-form" }, authenticity_token: true do |f| %button.btn.btn-default.js-settings-toggle{ type: 'button' }= expanded ? _('Collapse') : _('Expand')
%input{ name: 'update_section', type: 'hidden', value: 'js-shared-permissions' } %p= _('Housekeeping, export, path, transfer, remove, archive.')
-# haml-lint:disable InlineJavaScript
%script.js-project-permissions-form-data{ type: "application/json" }= project_permissions_panel_data_json(@project) .settings-content
.js-project-permissions-form .sub-section
= f.submit 'Save changes', class: "btn btn-success" %h4= _('Housekeeping')
%p= _('Runs a number of housekeeping tasks within the current repository, such as compressing file revisions and removing unreachable objects.')
= render_if_exists 'projects/issues_settings' = link_to _('Run housekeeping'), housekeeping_project_path(@project),
method: :post, class: "btn btn-default"
%section.qa-merge-request-settings.settings.merge-requests-feature.no-animate#js-merge-request-settings{ class: [('expanded' if expanded), ('hidden' if @project.project_feature.send(:merge_requests_access_level) == 0)] }
.settings-header = render 'export', project: @project
%h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Merge requests')
%button.btn.js-settings-toggle{ type: 'button' }= expanded ? _('Collapse') : _('Expand') - if can? current_user, :archive_project, @project
%p= _('Choose your merge method, set up a default merge request description template.')
.settings-content
= render_if_exists 'shared/promotions/promote_mr_features'
= form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { multipart: true, class: "merge-request-settings-form" }, authenticity_token: true do |f|
%input{ name: 'update_section', type: 'hidden', value: 'js-merge-request-settings' }
= render 'projects/merge_request_settings', form: f
= f.submit 'Save changes', class: "btn btn-success qa-save-merge-request-changes"
= render_if_exists 'projects/merge_request_approvals_settings', expanded: expanded
%section.settings.no-animate{ class: ('expanded' if expanded) }
.settings-header
%h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only
= s_('ProjectSettings|Badges')
%button.btn.js-settings-toggle{ type: 'button' }
= expanded ? 'Collapse' : 'Expand'
%p
= s_('ProjectSettings|Customize your project badges.')
= link_to s_('ProjectSettings|Learn more about badges.'), help_page_path('user/project/badges')
.settings-content
= render 'shared/badges/badge_settings'
= render_if_exists 'projects/service_desk_settings'
= render 'export', project: @project
%section.qa-advanced-settings.settings.advanced-settings.no-animate#js-project-advanced-settings{ class: ('expanded' if expanded) }
.settings-header
%h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Advanced')
%button.btn.js-settings-toggle{ type: 'button' }= expanded ? _('Collapse') : _('Expand')
%p= _('Housekeeping, export, path, transfer, remove, archive.')
.settings-content
.sub-section .sub-section
%h4 Housekeeping %h4.warning-title
%p
Runs a number of housekeeping tasks within the current repository, such as compressing file revisions and removing unreachable objects.
= link_to 'Run housekeeping', housekeeping_project_path(@project),
method: :post, class: "btn btn-default"
- if can? current_user, :archive_project, @project
.sub-section
%h4.warning-title
- if @project.archived?
Unarchive project
- else
Archive project
- if @project.archived? - if @project.archived?
%p = _('Unarchive project')
Unarchiving the project will restore people's ability to make changes to it.
The repository can be committed to, and issues, comments and other entities can be created.
%strong Once active this project shows up in the search and on the dashboard.
= link_to 'Unarchive project', unarchive_project_path(@project),
data: { confirm: "Are you sure that you want to unarchive this project?" },
method: :post, class: "btn btn-success"
- else - else
%p = _('Archive project')
Archiving the project will make it entirely read-only. It is hidden from the dashboard and doesn't show up in searches. - if @project.archived?
%strong The repository cannot be committed to, and no issues, comments or other entities can be created. %p= _("Unarchiving the project will restore people's ability to make changes to it. The repository can be committed to, and issues, comments and other entities can be created. <strong>Once active this project shows up in the search and on the dashboard.</strong>").html_safe
= link_to 'Archive project', archive_project_path(@project), = link_to _('Unarchive project'), unarchive_project_path(@project),
data: { confirm: "Are you sure that you want to archive this project?" }, data: { confirm: _("Are you sure that you want to unarchive this project?") },
method: :post, class: "btn btn-warning" method: :post, class: "btn btn-success"
.sub-section.rename-repository - else
%h4.warning-title %p= _("Archiving the project will make it entirely read-only. It is hidden from the dashboard and doesn't show up in searches. <strong>The repository cannot be committed to, and no issues, comments or other entities can be created.</strong>").html_safe
Rename repository = link_to _('Archive project'), archive_project_path(@project),
= render 'projects/errors' data: { confirm: _("Are you sure that you want to archive this project?") },
= form_for([@project.namespace.becomes(Namespace), @project]) do |f| method: :post, class: "btn btn-warning"
.form-group.project_name_holder .sub-section.rename-repository
= f.label :name, class: 'label-bold' do %h4.warning-title= _('Change path')
Project name = render 'projects/errors'
.form-group = form_for([@project.namespace.becomes(Namespace), @project]) do |f|
= f.text_field :name, class: "form-control" .form-group
= f.label :path, _('Path'), class: 'label-bold'
.form-group
.input-group
.input-group-prepend
.input-group-text
#{Gitlab::Utils.append_path(root_url, @project.namespace.full_path)}/
= f.text_field :path, class: 'form-control qa-project-path-field h-auto'
%ul
%li= _("Be careful. Renaming a project's repository can have unintended side effects.")
%li= _('You will need to update your local repositories to point to the new location.')
- if @project.deployment_platform.present?
%li= _('Your deployment services will be broken, you will need to manually fix the services after renaming.')
= f.submit _('Change path'), class: "btn btn-warning qa-change-path-button"
- if can?(current_user, :change_namespace, @project)
.sub-section
%h4.danger-title= _('Transfer project')
= form_for([@project.namespace.becomes(Namespace), @project], url: transfer_project_path(@project), method: :put, remote: true, html: { class: 'js-project-transfer-form' } ) do |f|
.form-group .form-group
= f.label :path, class: 'label-bold' do = label_tag :new_namespace_id, nil, class: 'label-bold' do
%span Path %span= _('Select a new namespace')
.form-group .form-group
.input-group = select_tag :new_namespace_id, namespaces_options(nil), include_blank: true, class: 'select2'
.input-group-prepend
.input-group-text
#{Gitlab::Utils.append_path(root_url, @project.namespace.full_path)}/
= f.text_field :path, class: 'form-control'
%ul %ul
%li Be careful. Renaming a project's repository can have unintended side effects. %li= _("Be careful. Changing the project's namespace can have unintended side effects.")
%li You will need to update your local repositories to point to the new location. %li= _('You can only transfer the project to namespaces you manage.')
- if @project.deployment_platform.present? %li= _('You will need to update your local repositories to point to the new location.')
%li Your deployment services will be broken, you will need to manually fix the services after renaming. %li= _('Project visibility level will be changed to match namespace rules when transferring to a group.')
= f.submit 'Rename project', class: "btn btn-warning" = f.submit 'Transfer project', class: "btn btn-remove js-confirm-danger", data: { "confirm-danger-message" => transfer_project_message(@project) }
- if can?(current_user, :change_namespace, @project)
.sub-section - if @project.forked? && can?(current_user, :remove_fork_project, @project)
%h4.danger-title .sub-section
Transfer project %h4.danger-title= _('Remove fork relationship')
= form_for([@project.namespace.becomes(Namespace), @project], url: transfer_project_path(@project), method: :put, remote: true, html: { class: 'js-project-transfer-form' } ) do |f| %p
.form-group = _('This will remove the fork relationship to source project')
= label_tag :new_namespace_id, nil, class: 'label-bold' do = succeed "." do
%span Select a new namespace - if @project.fork_source
.form-group = link_to(fork_source_name(@project), project_path(@project.fork_source))
= select_tag :new_namespace_id, namespaces_options(nil), include_blank: true, class: 'select2' - else
%ul = fork_source_name(@project)
%li Be careful. Changing the project's namespace can have unintended side effects. = form_for([@project.namespace.becomes(Namespace), @project], url: remove_fork_project_path(@project), method: :delete, remote: true, html: { class: 'transfer-project' }) do |f|
%li You can only transfer the project to namespaces you manage.
%li You will need to update your local repositories to point to the new location.
%li Project visibility level will be changed to match namespace rules when transferring to a group.
= f.submit 'Transfer project', class: "btn btn-remove js-confirm-danger", data: { "confirm-danger-message" => transfer_project_message(@project) }
- if @project.forked? && can?(current_user, :remove_fork_project, @project)
.sub-section
%h4.danger-title
Remove fork relationship
%p %p
This will remove the fork relationship to source project %strong= _('Once removed, the fork relationship cannot be restored and you will no longer be able to send merge requests to the source.')
= succeed "." do = button_to _('Remove fork relationship'), '#', class: "btn btn-remove js-confirm-danger", data: { "confirm-danger-message" => remove_fork_project_message(@project) }
- if @project.fork_source
= link_to(fork_source_name(@project), project_path(@project.fork_source)) - if can?(current_user, :remove_project, @project)
- else .sub-section
= fork_source_name(@project) %h4.danger-title= _('Remove project')
= form_for([@project.namespace.becomes(Namespace), @project], url: remove_fork_project_path(@project), method: :delete, remote: true, html: { class: 'transfer-project' }) do |f| %p= _('Removing the project will delete its repository and all related resources including issues, merge requests etc.')
%p = form_tag(project_path(@project), method: :delete) do
%strong Once removed, the fork relationship cannot be restored and you will no longer be able to send merge requests to the source.
= button_to 'Remove fork relationship', '#', class: "btn btn-remove js-confirm-danger", data: { "confirm-danger-message" => remove_fork_project_message(@project) }
- if can?(current_user, :remove_project, @project)
.sub-section
%h4.danger-title
Remove project
%p %p
Removing the project will delete its repository and all related resources including issues, merge requests etc. %strong= _('Removed projects cannot be restored!')
= form_tag(project_path(@project), method: :delete) do = button_to _('Remove project'), '#', class: "btn btn-remove js-confirm-danger", data: { "confirm-danger-message" => remove_project_message(@project) }
%p
%strong Removed projects cannot be restored!
= button_to 'Remove project', '#', class: "btn btn-remove js-confirm-danger", data: { "confirm-danger-message" => remove_project_message(@project) }
.save-project-loader.hide .save-project-loader.hide
.center .center
%h2 %h2
%i.fa.fa-spinner.fa-spin %i.fa.fa-spinner.fa-spin
Saving project. = _('Saving project.')
%p Please wait a moment, this page will automatically refresh when ready. %p= _('Please wait a moment, this page will automatically refresh when ready.')
= render 'shared/confirm_modal', phrase: @project.path = render 'shared/confirm_modal', phrase: @project.path
= form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { multipart: true, class: "edit-project js-general-settings-form" }, authenticity_token: true do |f|
%input{ name: 'update_section', type: 'hidden', value: 'js-general-settings' }
= form_errors(@project)
%fieldset
.row
.form-group.col-md-5
= f.label :name, class: 'label-bold', for: 'project_name_edit' do
= _('Project name')
= f.text_field :name, class: 'form-control qa-project-name-field', id: "project_name_edit"
.form-group.col-md-7
= f.label :id, class: 'label-bold' do
= _('Project ID')
= f.text_field :id, class: 'form-control w-auto', readonly: true
.row
.form-group.col-md-9
= f.label :tag_list, _('Topics'), class: 'label-bold'
= f.text_field :tag_list, value: @project.tag_list.join(', '), maxlength: 2000, class: "form-control"
%p.form-text.text-muted= _('Separate topics with commas.')
.row
.form-group.col-md-9
= f.label :description, _('Project description (optional)'), class: 'label-bold'
= f.text_area :description, class: 'form-control', rows: 3, maxlength: 250
.row= render_if_exists 'projects/classification_policy_settings', f: f
.row= render_if_exists 'shared/repository_size_limit_setting', form: f, type: :project
.form-group.prepend-top-default.append-bottom-20
.avatar-container.s90
= project_icon(@project, alt: _('Project avatar'), class: 'avatar project-avatar s90')
= f.label :avatar, _('Project avatar'), class: 'label-bold d-block'
= render 'shared/choose_avatar_button', f: f
- if @project.avatar?
%hr
= link_to _('Remove avatar'), project_avatar_path(@project), data: { confirm: _('Avatar will be removed. Are you sure?')}, method: :delete, class: 'btn btn-link'
= f.submit _('Save changes'), class: "btn btn-success mt-4 qa-save-naming-topics-avatar-button"
...@@ -2,8 +2,7 @@ ...@@ -2,8 +2,7 @@
.modal-dialog .modal-dialog
.modal-content .modal-content
.modal-header .modal-header
%h3.page-title %h3.page-title= _('Confirmation required')
Confirmation required
%button.close{ type: "button", "data-dismiss": "modal", "aria-label" => _('Close') } %button.close{ type: "button", "data-dismiss": "modal", "aria-label" => _('Close') }
%span{ "aria-hidden": true } &times; %span{ "aria-hidden": true } &times;
...@@ -11,9 +10,7 @@ ...@@ -11,9 +10,7 @@
%p.text-danger.js-confirm-text %p.text-danger.js-confirm-text
%p %p
%span.js-warning-text %span.js-warning-text= _('This action can lead to data loss. To prevent accidental actions we ask you to confirm your intention.')
This action can lead to data loss.
To prevent accidental actions we ask you to confirm your intention.
%br %br
Please type Please type
%code.js-confirm-danger-match= phrase %code.js-confirm-danger-match= phrase
...@@ -22,4 +19,4 @@ ...@@ -22,4 +19,4 @@
.form-group .form-group
= text_field_tag 'confirm_name_input', '', class: 'form-control js-confirm-danger-input' = text_field_tag 'confirm_name_input', '', class: 'form-control js-confirm-danger-input'
.form-actions .form-actions
= submit_tag 'Confirm', class: "btn btn-danger js-confirm-danger-submit" = submit_tag _('Confirm'), class: "btn btn-danger js-confirm-danger-submit"
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
= link_to _("Learn more about approvals."), help_page_path("user/project/merge_requests/merge_request_approvals"), target: '_blank' = link_to _("Learn more about approvals."), help_page_path("user/project/merge_requests/merge_request_approvals"), target: '_blank'
.settings-content .settings-content
= form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { class: "merge-request-approval-settings-form" }, authenticity_token: true do |f| = form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { class: "merge-request-approval-settings-form js-mr-approvals-form" }, authenticity_token: true do |f|
%input{ name: 'update_section', type: 'hidden', value: 'js-merge-request-approval-settings' } %input{ name: 'update_section', type: 'hidden', value: 'js-merge-request-approval-settings' }
= render 'projects/merge_request_approvals_settings_form', form: f, project: @project = render 'projects/merge_request_approvals_settings_form', form: f, project: @project
= f.submit _("Save changes"), class: "btn btn-success" = f.submit _("Save changes"), class: "btn btn-success"
...@@ -2,18 +2,20 @@ ...@@ -2,18 +2,20 @@
- expanded = Rails.env.test? - expanded = Rails.env.test?
%section.settings.issues-feature.no-animate#js-issue-settings{ class: [('expanded' if expanded), ('hidden' if @project.project_feature.send(:issues_access_level) == 0)] } %section.settings.issues-feature.no-animate#js-issue-settings{ class: [('expanded' if expanded), ('hidden' if @project.project_feature.send(:issues_access_level) == 0)] }
.settings-header .settings-header
%h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Issue settings') %h4.settings-title.js-settings-toggle.js-settings-toggle-trigger-only= _('Default issue template')
%button.btn.js-settings-toggle= expanded ? _('Collapse') : _('Expand') %button.btn.btn-default.js-settings-toggle= expanded ? _('Collapse') : _('Expand')
%p= _('Customize your issue restrictions.') %p= _('Set a default template for issue descriptions.')
.settings-content .settings-content
= form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { multipart: true, class: "issue-settings-form" }, authenticity_token: true do |f| = form_for [@project.namespace.becomes(Namespace), @project], remote: true, html: { multipart: true, class: "issue-settings-form" }, authenticity_token: true do |f|
%input{ type: 'hidden', name: 'update_section', value: 'js-issue-settings' } %input{ type: 'hidden', name: 'update_section', value: 'js-issue-settings' }
.form-group .row
= f.label :issues_template, class: 'label-bold' do .form-group.col-md-9
Default description template for issues = f.label :issues_template, class: 'label-bold' do
= link_to icon('question-circle'), help_page_path('user/project/description_templates', anchor: 'setting-a-default-template-for-issues-and-merge-requests'), target: '_blank' = _('Default description template for issues')
= f.text_area :issues_template, class: "form-control", rows: 3 = link_to icon('question-circle'), help_page_path('user/project/description_templates', anchor: 'setting-a-default-template-for-issues-and-merge-requests'), target: '_blank'
.hint = f.text_area :issues_template, class: "form-control", rows: 3
Description parsed with #{link_to "GitLab Flavored Markdown", help_page_path('user/markdown'), target: '_blank'}. .text-secondary
= f.submit 'Save changes', class: "btn btn-success" - link_start = '<a href="%{url}" target="_blank" rel="noopener noreferrer">'.html_safe % { url: help_page_path('user/markdown') }
= _('Description parsed with %{link_start}GitLab Flavored Markdown%{link_end}').html_safe % { link_start: link_start, link_end: '</a>'.html_safe }
= f.submit _('Save changes'), class: "btn btn-success"
- return unless current_user.admin? && License.feature_available?(:repository_size_limit) - return unless current_user.admin? && License.feature_available?(:repository_size_limit)
- form = local_assigns.fetch(:form) - form = local_assigns.fetch(:form)
- type = local_assigns.fetch(:type) - is_project = local_assigns.fetch(:type) == :project
- form_group_class = type === :group ? 'col-md-9' : ''
.form-group{ class: form_group_class } .form-group.col-md-9
= form.label :repository_size_limit, class: 'label-bold' do = form.label :repository_size_limit, class: 'label-bold' do
Repository size limit (MB) Repository size limit (MB)
= form.number_field :repository_size_limit, value: form.object.repository_size_limit.try(:to_mb), class: 'form-control', min: 0 = form.number_field :repository_size_limit, value: form.object.repository_size_limit.try(:to_mb), class: 'form-control', min: 0
%span.form-text.text-muted#repository_size_limit_help_block %span.form-text.text-muted#repository_size_limit_help_block
= type === :project ? size_limit_message(@project) : size_limit_message_for_group(@group) = is_project ? size_limit_message(@project) : size_limit_message_for_group(@group)
---
title: Improve project settings page layout and UX
merge_request: 10388
author:
type: other
...@@ -17,14 +17,14 @@ describe 'Project settings > Issues', :js do ...@@ -17,14 +17,14 @@ describe 'Project settings > Issues', :js do
end end
it 'shows the Issues settings' do it 'shows the Issues settings' do
expect(page).to have_content('Customize your issue restrictions') expect(page).to have_content('Set a default template for issue descriptions.')
within('.sharing-permissions-form') do within('.sharing-permissions-form') do
find('.project-feature-controls[data-for="project[project_feature_attributes][issues_access_level]"] .project-feature-toggle').click find('.project-feature-controls[data-for="project[project_feature_attributes][issues_access_level]"] .project-feature-toggle').click
click_on('Save changes') click_on('Save changes')
end end
expect(page).not_to have_content('Customize your issue restrictions') expect(page).not_to have_content('Set a default template for issue descriptions.')
end end
end end
end end
...@@ -36,14 +36,14 @@ describe 'Project settings > Issues', :js do ...@@ -36,14 +36,14 @@ describe 'Project settings > Issues', :js do
end end
it 'does not show the Issues settings' do it 'does not show the Issues settings' do
expect(page).not_to have_content('Customize your issue restrictions') expect(page).not_to have_content('Set a default template for issue descriptions.')
within('.sharing-permissions-form') do within('.sharing-permissions-form') do
find('.project-feature-controls[data-for="project[project_feature_attributes][issues_access_level]"] .project-feature-toggle').click find('.project-feature-controls[data-for="project[project_feature_attributes][issues_access_level]"] .project-feature-toggle').click
click_on('Save changes') click_on('Save changes')
end end
expect(page).to have_content('Customize your issue restrictions') expect(page).to have_content('Set a default template for issue descriptions.')
end end
end end
......
...@@ -3,16 +3,19 @@ require 'spec_helper' ...@@ -3,16 +3,19 @@ require 'spec_helper'
describe 'EE > Projects > Settings > User manages approval rule settings' do describe 'EE > Projects > Settings > User manages approval rule settings' do
let(:project) { create(:project) } let(:project) { create(:project) }
let(:user) { project.owner } let(:user) { project.owner }
let(:path) { edit_project_path(project) }
before do before do
sign_in(user) sign_in(user)
stub_licensed_features(licensed_features) stub_licensed_features(licensed_features)
visit edit_project_path(project) visit path
end end
context 'when `code_owner_approval_required` is available' do context 'when `code_owner_approval_required` is available' do
let(:licensed_features) { { code_owner_approval_required: true } } let(:licensed_features) { { code_owner_approval_required: true } }
it_behaves_like 'dirty submit form', [{ form: '#js-merge-request-approval-settings', input: '#project_merge_requests_author_approval' }]
it 'allows the user to enforce code owner approval' do it 'allows the user to enforce code owner approval' do
within('.require-code-owner-approval') do within('.require-code-owner-approval') do
check('Require approval from code owners') check('Require approval from code owners')
......
...@@ -1138,6 +1138,9 @@ msgstr "" ...@@ -1138,6 +1138,9 @@ msgstr ""
msgid "Any Label" msgid "Any Label"
msgstr "" msgstr ""
msgid "Any encrypted tokens"
msgstr ""
msgid "Appearance" msgid "Appearance"
msgstr "" msgstr ""
...@@ -1230,15 +1233,27 @@ msgstr "" ...@@ -1230,15 +1233,27 @@ msgstr ""
msgid "Archive jobs" msgid "Archive jobs"
msgstr "" msgstr ""
msgid "Archive project"
msgstr ""
msgid "Archived project! Repository and other project resources are read-only" msgid "Archived project! Repository and other project resources are read-only"
msgstr "" msgstr ""
msgid "Archived projects" msgid "Archived projects"
msgstr "" msgstr ""
msgid "Archiving the project will make it entirely read-only. It is hidden from the dashboard and doesn't show up in searches. <strong>The repository cannot be committed to, and no issues, comments or other entities can be created.</strong>"
msgstr ""
msgid "Are you sure" msgid "Are you sure"
msgstr "" msgstr ""
msgid "Are you sure that you want to archive this project?"
msgstr ""
msgid "Are you sure that you want to unarchive this project?"
msgstr ""
msgid "Are you sure you want to delete this pipeline schedule?" msgid "Are you sure you want to delete this pipeline schedule?"
msgstr "" msgstr ""
...@@ -1578,6 +1593,12 @@ msgstr "" ...@@ -1578,6 +1593,12 @@ msgstr ""
msgid "BatchComments|You're about to discard your review which will delete all of your pending comments. The deleted comments %{strong_start}cannot%{strong_end} be restored." msgid "BatchComments|You're about to discard your review which will delete all of your pending comments. The deleted comments %{strong_start}cannot%{strong_end} be restored."
msgstr "" msgstr ""
msgid "Be careful. Changing the project's namespace can have unintended side effects."
msgstr ""
msgid "Be careful. Renaming a project's repository can have unintended side effects."
msgstr ""
msgid "Begin with the selected commit" msgid "Begin with the selected commit"
msgstr "" msgstr ""
...@@ -1872,6 +1893,9 @@ msgstr "" ...@@ -1872,6 +1893,9 @@ msgstr ""
msgid "CI Lint" msgid "CI Lint"
msgstr "" msgstr ""
msgid "CI variables"
msgstr ""
msgid "CI will run using the credentials assigned above." msgid "CI will run using the credentials assigned above."
msgstr "" msgstr ""
...@@ -1983,6 +2007,9 @@ msgstr "" ...@@ -1983,6 +2007,9 @@ msgstr ""
msgid "Change Weight" msgid "Change Weight"
msgstr "" msgstr ""
msgid "Change path"
msgstr ""
msgid "Change permissions" msgid "Change permissions"
msgstr "" msgstr ""
...@@ -2211,6 +2238,9 @@ msgstr "" ...@@ -2211,6 +2238,9 @@ msgstr ""
msgid "CiVariable|Validation failed" msgid "CiVariable|Validation failed"
msgstr "" msgstr ""
msgid "Classification Label (optional)"
msgstr ""
msgid "ClassificationLabelUnavailable|is unavailable: %{reason}" msgid "ClassificationLabelUnavailable|is unavailable: %{reason}"
msgstr "" msgstr ""
...@@ -2963,6 +2993,12 @@ msgstr "" ...@@ -2963,6 +2993,12 @@ msgstr ""
msgid "Configure the way a user creates a new account." msgid "Configure the way a user creates a new account."
msgstr "" msgstr ""
msgid "Confirm"
msgstr ""
msgid "Confirmation required"
msgstr ""
msgid "Connect" msgid "Connect"
msgstr "" msgstr ""
...@@ -2993,6 +3029,9 @@ msgstr "" ...@@ -2993,6 +3029,9 @@ msgstr ""
msgid "Container Registry" msgid "Container Registry"
msgstr "" msgstr ""
msgid "Container registry images"
msgstr ""
msgid "ContainerRegistry|Created" msgid "ContainerRegistry|Created"
msgstr "" msgstr ""
...@@ -3332,9 +3371,6 @@ msgstr "" ...@@ -3332,9 +3371,6 @@ msgstr ""
msgid "Customize language and region related settings." msgid "Customize language and region related settings."
msgstr "" msgstr ""
msgid "Customize your issue restrictions."
msgstr ""
msgid "Customize your pipeline configuration, view your pipeline status and coverage report." msgid "Customize your pipeline configuration, view your pipeline status and coverage report."
msgstr "" msgstr ""
...@@ -3410,12 +3446,18 @@ msgstr "" ...@@ -3410,12 +3446,18 @@ msgstr ""
msgid "Default classification label" msgid "Default classification label"
msgstr "" msgstr ""
msgid "Default description template for issues"
msgstr ""
msgid "Default first day of the week" msgid "Default first day of the week"
msgstr "" msgstr ""
msgid "Default first day of the week in calendars and date pickers." msgid "Default first day of the week in calendars and date pickers."
msgstr "" msgstr ""
msgid "Default issue template"
msgstr ""
msgid "Default: Directly import the Google Code email address or username" msgid "Default: Directly import the Google Code email address or username"
msgstr "" msgstr ""
...@@ -3796,6 +3838,9 @@ msgstr "" ...@@ -3796,6 +3838,9 @@ msgstr ""
msgid "Download asset" msgid "Download asset"
msgstr "" msgstr ""
msgid "Download export"
msgstr ""
msgid "Download license" msgid "Download license"
msgstr "" msgstr ""
...@@ -4462,6 +4507,12 @@ msgstr "" ...@@ -4462,6 +4507,12 @@ msgstr ""
msgid "Export issues" msgid "Export issues"
msgstr "" msgstr ""
msgid "Export project"
msgstr ""
msgid "Export this project with all its related data in order to move your project to a new GitLab instance. Once the export is finished, you can import the file from the \"New Project\" page."
msgstr ""
msgid "External Classification Policy Authorization" msgid "External Classification Policy Authorization"
msgstr "" msgstr ""
...@@ -4480,9 +4531,6 @@ msgstr "" ...@@ -4480,9 +4531,6 @@ msgstr ""
msgid "External authorization request timeout" msgid "External authorization request timeout"
msgstr "" msgstr ""
msgid "ExternalAuthorizationService|Classification Label"
msgstr ""
msgid "ExternalAuthorizationService|Classification label" msgid "ExternalAuthorizationService|Classification label"
msgstr "" msgstr ""
...@@ -4938,6 +4986,9 @@ msgstr "" ...@@ -4938,6 +4986,9 @@ msgstr ""
msgid "Generate key" msgid "Generate key"
msgstr "" msgstr ""
msgid "Generate new export"
msgstr ""
msgid "Geo" msgid "Geo"
msgstr "" msgstr ""
...@@ -5801,6 +5852,9 @@ msgstr "" ...@@ -5801,6 +5852,9 @@ msgstr ""
msgid "Hook was successfully updated." msgid "Hook was successfully updated."
msgstr "" msgstr ""
msgid "Housekeeping"
msgstr ""
msgid "Housekeeping successfully started" msgid "Housekeeping successfully started"
msgstr "" msgstr ""
...@@ -6169,9 +6223,6 @@ msgstr "" ...@@ -6169,9 +6223,6 @@ msgstr ""
msgid "Issue events" msgid "Issue events"
msgstr "" msgstr ""
msgid "Issue settings"
msgstr ""
msgid "Issue template (optional)" msgid "Issue template (optional)"
msgstr "" msgstr ""
...@@ -6205,6 +6256,9 @@ msgstr "" ...@@ -6205,6 +6256,9 @@ msgstr ""
msgid "Issues closed" msgid "Issues closed"
msgstr "" msgstr ""
msgid "Issues with comments, merge requests with diffs and comments, labels, milestones, snippets, and other project entities"
msgstr ""
msgid "Issues, merge requests, pushes, and comments." msgid "Issues, merge requests, pushes, and comments."
msgstr "" msgstr ""
...@@ -6271,6 +6325,9 @@ msgstr "" ...@@ -6271,6 +6325,9 @@ msgstr ""
msgid "Job is stuck. Check runners." msgid "Job is stuck. Check runners."
msgstr "" msgstr ""
msgid "Job traces and artifacts"
msgstr ""
msgid "Job was retried" msgid "Job was retried"
msgstr "" msgstr ""
...@@ -6382,6 +6439,9 @@ msgstr "" ...@@ -6382,6 +6439,9 @@ msgstr ""
msgid "LFS" msgid "LFS"
msgstr "" msgstr ""
msgid "LFS objects"
msgstr ""
msgid "LFSStatus|Disabled" msgid "LFSStatus|Disabled"
msgstr "" msgstr ""
...@@ -7253,7 +7313,7 @@ msgstr "" ...@@ -7253,7 +7313,7 @@ msgstr ""
msgid "Namespaces to index" msgid "Namespaces to index"
msgstr "" msgstr ""
msgid "Naming, tags, avatar" msgid "Naming, topics, avatar"
msgstr "" msgstr ""
msgid "Naming, visibility" msgid "Naming, visibility"
...@@ -7669,6 +7729,12 @@ msgstr "" ...@@ -7669,6 +7729,12 @@ msgstr ""
msgid "Once imported, repositories can be mirrored over SSH. Read more %{ssh_link}" msgid "Once imported, repositories can be mirrored over SSH. Read more %{ssh_link}"
msgstr "" msgstr ""
msgid "Once removed, the fork relationship cannot be restored and you will no longer be able to send merge requests to the source."
msgstr ""
msgid "Once the exported file is ready, you will receive a notification email with a download link, or you can download it from this page."
msgstr ""
msgid "One more item" msgid "One more item"
msgid_plural "%d more items" msgid_plural "%d more items"
msgstr[0] "" msgstr[0] ""
...@@ -7875,6 +7941,9 @@ msgstr "" ...@@ -7875,6 +7941,9 @@ msgstr ""
msgid "Paste your public SSH key, which is usually contained in the file '~/.ssh/id_rsa.pub' and begins with 'ssh-rsa'. Don't use your private SSH key." msgid "Paste your public SSH key, which is usually contained in the file '~/.ssh/id_rsa.pub' and begins with 'ssh-rsa'. Don't use your private SSH key."
msgstr "" msgstr ""
msgid "Path"
msgstr ""
msgid "Path, transfer, remove" msgid "Path, transfer, remove"
msgstr "" msgstr ""
...@@ -8202,6 +8271,9 @@ msgstr "" ...@@ -8202,6 +8271,9 @@ msgstr ""
msgid "Please use this form to report users to GitLab who create spam issues, comments or behave inappropriately." msgid "Please use this form to report users to GitLab who create spam issues, comments or behave inappropriately."
msgstr "" msgstr ""
msgid "Please wait a moment, this page will automatically refresh when ready."
msgstr ""
msgid "Please wait while we connect to your repository. Refresh at will." msgid "Please wait while we connect to your repository. Refresh at will."
msgstr "" msgstr ""
...@@ -8556,15 +8628,27 @@ msgstr "" ...@@ -8556,15 +8628,27 @@ msgstr ""
msgid "Project Badges" msgid "Project Badges"
msgstr "" msgstr ""
msgid "Project ID"
msgstr ""
msgid "Project URL" msgid "Project URL"
msgstr "" msgstr ""
msgid "Project access must be granted explicitly to each user." msgid "Project access must be granted explicitly to each user."
msgstr "" msgstr ""
msgid "Project and wiki repositories"
msgstr ""
msgid "Project avatar" msgid "Project avatar"
msgstr "" msgstr ""
msgid "Project configuration, including services"
msgstr ""
msgid "Project description (optional)"
msgstr ""
msgid "Project details" msgid "Project details"
msgstr "" msgstr ""
...@@ -8595,6 +8679,12 @@ msgstr "" ...@@ -8595,6 +8679,12 @@ msgstr ""
msgid "Project slug" msgid "Project slug"
msgstr "" msgstr ""
msgid "Project uploads"
msgstr ""
msgid "Project visibility level will be changed to match namespace rules when transferring to a group."
msgstr ""
msgid "Project:" msgid "Project:"
msgstr "" msgstr ""
...@@ -9092,6 +9182,9 @@ msgstr "" ...@@ -9092,6 +9182,9 @@ msgstr ""
msgid "Remove avatar" msgid "Remove avatar"
msgstr "" msgstr ""
msgid "Remove fork relationship"
msgstr ""
msgid "Remove group" msgid "Remove group"
msgstr "" msgstr ""
...@@ -9116,12 +9209,18 @@ msgstr "" ...@@ -9116,12 +9209,18 @@ msgstr ""
msgid "Removed group can not be restored!" msgid "Removed group can not be restored!"
msgstr "" msgstr ""
msgid "Removed projects cannot be restored!"
msgstr ""
msgid "Removing group will cause all child projects and resources to be removed." msgid "Removing group will cause all child projects and resources to be removed."
msgstr "" msgstr ""
msgid "Removing license…" msgid "Removing license…"
msgstr "" msgstr ""
msgid "Removing the project will delete its repository and all related resources including issues, merge requests etc."
msgstr ""
msgid "Rename" msgid "Rename"
msgstr "" msgstr ""
...@@ -9398,6 +9497,9 @@ msgstr "" ...@@ -9398,6 +9497,9 @@ msgstr ""
msgid "Run CI/CD pipelines for external repositories" msgid "Run CI/CD pipelines for external repositories"
msgstr "" msgstr ""
msgid "Run housekeeping"
msgstr ""
msgid "Run tests against your code live using the Web Terminal" msgid "Run tests against your code live using the Web Terminal"
msgstr "" msgstr ""
...@@ -9461,6 +9563,9 @@ msgstr "" ...@@ -9461,6 +9563,9 @@ msgstr ""
msgid "Running…" msgid "Running…"
msgstr "" msgstr ""
msgid "Runs a number of housekeeping tasks within the current repository, such as compressing file revisions and removing unreachable objects."
msgstr ""
msgid "SAML SSO" msgid "SAML SSO"
msgstr "" msgstr ""
...@@ -9512,6 +9617,9 @@ msgstr "" ...@@ -9512,6 +9617,9 @@ msgstr ""
msgid "Save variables" msgid "Save variables"
msgstr "" msgstr ""
msgid "Saving project."
msgstr ""
msgid "Schedule a new pipeline" msgid "Schedule a new pipeline"
msgstr "" msgstr ""
...@@ -9719,6 +9827,9 @@ msgstr "" ...@@ -9719,6 +9827,9 @@ msgstr ""
msgid "Select a namespace to fork the project" msgid "Select a namespace to fork the project"
msgstr "" msgstr ""
msgid "Select a new namespace"
msgstr ""
msgid "Select a project to read Insights configuration file" msgid "Select a project to read Insights configuration file"
msgstr "" msgstr ""
...@@ -9788,6 +9899,9 @@ msgstr "" ...@@ -9788,6 +9899,9 @@ msgstr ""
msgid "Sep" msgid "Sep"
msgstr "" msgstr ""
msgid "Separate topics with commas."
msgstr ""
msgid "September" msgid "September"
msgstr "" msgstr ""
...@@ -9866,6 +9980,9 @@ msgstr "" ...@@ -9866,6 +9980,9 @@ msgstr ""
msgid "Session expiration, projects limit and attachment size." msgid "Session expiration, projects limit and attachment size."
msgstr "" msgstr ""
msgid "Set a default template for issue descriptions."
msgstr ""
msgid "Set a number of approvals required, the approvers and other approval settings." msgid "Set a number of approvals required, the approvers and other approval settings."
msgstr "" msgstr ""
...@@ -10866,6 +10983,12 @@ msgstr "" ...@@ -10866,6 +10983,12 @@ msgstr ""
msgid "The file has been successfully deleted." msgid "The file has been successfully deleted."
msgstr "" msgstr ""
msgid "The following items will NOT be exported:"
msgstr ""
msgid "The following items will be exported:"
msgstr ""
msgid "The fork relationship has been removed." msgid "The fork relationship has been removed."
msgstr "" msgstr ""
...@@ -11124,6 +11247,9 @@ msgstr "" ...@@ -11124,6 +11247,9 @@ msgstr ""
msgid "This GitLab instance does not provide any shared Runners yet. Instance administrators can register shared Runners in the admin area." msgid "This GitLab instance does not provide any shared Runners yet. Instance administrators can register shared Runners in the admin area."
msgstr "" msgstr ""
msgid "This action can lead to data loss. To prevent accidental actions we ask you to confirm your intention."
msgstr ""
msgid "This application was created by %{link_to_owner}." msgid "This application was created by %{link_to_owner}."
msgstr "" msgstr ""
...@@ -11340,6 +11466,9 @@ msgstr "" ...@@ -11340,6 +11466,9 @@ msgstr ""
msgid "This will redirect you to an external sign in page." msgid "This will redirect you to an external sign in page."
msgstr "" msgstr ""
msgid "This will remove the fork relationship to source project"
msgstr ""
msgid "Those emails automatically become issues (with the comments becoming the email conversation) listed here." msgid "Those emails automatically become issues (with the comments becoming the email conversation) listed here."
msgstr "" msgstr ""
...@@ -11677,6 +11806,9 @@ msgstr "" ...@@ -11677,6 +11806,9 @@ msgstr ""
msgid "Too many changes to show." msgid "Too many changes to show."
msgstr "" msgstr ""
msgid "Topics"
msgstr ""
msgid "Total" msgid "Total"
msgstr "" msgstr ""
...@@ -11704,6 +11836,9 @@ msgstr "" ...@@ -11704,6 +11836,9 @@ msgstr ""
msgid "Track time with quick actions" msgid "Track time with quick actions"
msgstr "" msgstr ""
msgid "Transfer project"
msgstr ""
msgid "Tree view" msgid "Tree view"
msgstr "" msgstr ""
...@@ -11791,6 +11926,12 @@ msgstr "" ...@@ -11791,6 +11926,12 @@ msgstr ""
msgid "Unable to update this epic at this time." msgid "Unable to update this epic at this time."
msgstr "" msgstr ""
msgid "Unarchive project"
msgstr ""
msgid "Unarchiving the project will restore people's ability to make changes to it. The repository can be committed to, and issues, comments and other entities can be created. <strong>Once active this project shows up in the search and on the dashboard.</strong>"
msgstr ""
msgid "Unblock" msgid "Unblock"
msgstr "" msgstr ""
...@@ -11881,7 +12022,7 @@ msgstr "" ...@@ -11881,7 +12022,7 @@ msgstr ""
msgid "Update your group name, description, avatar, and visibility." msgid "Update your group name, description, avatar, and visibility."
msgstr "" msgstr ""
msgid "Update your project name, tags, description and avatar." msgid "Update your project name, topics, description and avatar."
msgstr "" msgstr ""
msgid "Updated" msgid "Updated"
...@@ -12334,6 +12475,9 @@ msgstr "" ...@@ -12334,6 +12475,9 @@ msgstr ""
msgid "Web terminal" msgid "Web terminal"
msgstr "" msgstr ""
msgid "Webhooks"
msgstr ""
msgid "Webhooks Help" msgid "Webhooks Help"
msgstr "" msgstr ""
...@@ -12627,6 +12771,9 @@ msgstr "" ...@@ -12627,6 +12771,9 @@ msgstr ""
msgid "You can only merge once the items above are resolved" msgid "You can only merge once the items above are resolved"
msgstr "" msgstr ""
msgid "You can only transfer the project to namespaces you manage."
msgstr ""
msgid "You can resolve the merge conflict using either the Interactive mode, by choosing %{use_ours} or %{use_theirs} buttons, or by editing the files directly. Commit these changes into %{branch_name}" msgid "You can resolve the merge conflict using either the Interactive mode, by choosing %{use_ours} or %{use_theirs} buttons, or by editing the files directly. Commit these changes into %{branch_name}"
msgstr "" msgstr ""
...@@ -12738,6 +12885,9 @@ msgstr "" ...@@ -12738,6 +12885,9 @@ msgstr ""
msgid "You will lose all the unstaged changes you've made in this project. This action cannot be undone." msgid "You will lose all the unstaged changes you've made in this project. This action cannot be undone."
msgstr "" msgstr ""
msgid "You will need to update your local repositories to point to the new location."
msgstr ""
msgid "You will not get any notifications via email" msgid "You will not get any notifications via email"
msgstr "" msgstr ""
...@@ -12840,6 +12990,9 @@ msgstr "" ...@@ -12840,6 +12990,9 @@ msgstr ""
msgid "Your comment will not be visible to the public." msgid "Your comment will not be visible to the public."
msgstr "" msgstr ""
msgid "Your deployment services will be broken, you will need to manually fix the services after renaming."
msgstr ""
msgid "Your device was successfully set up! Give it a name and register it with the GitLab server." msgid "Your device was successfully set up! Give it a name and register it with the GitLab server."
msgstr "" msgstr ""
......
...@@ -4,27 +4,21 @@ module QA ...@@ -4,27 +4,21 @@ module QA
module Settings module Settings
class Advanced < Page::Base class Advanced < Page::Base
view 'app/views/projects/edit.html.haml' do view 'app/views/projects/edit.html.haml' do
element :project_path_field, 'text_field :path' # rubocop:disable QA/ElementWithPattern element :project_path_field
element :project_name_field, 'text_field :name' # rubocop:disable QA/ElementWithPattern element :change_path_button
element :rename_project_button, "submit 'Rename project'" # rubocop:disable QA/ElementWithPattern
end end
def rename_to(path) def update_project_path_to(path)
fill_project_name(path)
fill_project_path(path) fill_project_path(path)
rename_project! click_change_path_button
end end
def fill_project_path(path) def fill_project_path(path)
fill_in :project_path, with: path fill_element :project_path_field, path
end end
def fill_project_name(name) def click_change_path_button
fill_in :project_name, with: name click_element :change_path_button
end
def rename_project!
click_on 'Rename project'
end end
end end
end end
......
...@@ -4,14 +4,6 @@ module QA ...@@ -4,14 +4,6 @@ module QA
module Settings module Settings
module Common module Common
include QA::Page::Settings::Common include QA::Page::Settings::Common
def self.included(base)
base.class_eval do
view 'app/views/projects/edit.html.haml' do
element :advanced_settings_expand, "= expanded ? 'Collapse' : 'Expand'" # rubocop:disable QA/ElementWithPattern
end
end
end
end end
end end
end end
......
...@@ -9,6 +9,24 @@ module QA ...@@ -9,6 +9,24 @@ module QA
element :advanced_settings element :advanced_settings
end end
view 'app/views/projects/settings/_general.html.haml' do
element :project_name_field
element :save_naming_topics_avatar_button
end
def rename_project_to(name)
fill_project_name(name)
click_save_changes
end
def fill_project_name(name)
fill_element :project_name_field, name
end
def click_save_changes
click_element :save_naming_topics_avatar_button
end
def expand_advanced_settings(&block) def expand_advanced_settings(&block)
expand_section(:advanced_settings) do expand_section(:advanced_settings) do
Advanced.perform(&block) Advanced.perform(&block)
......
...@@ -34,8 +34,11 @@ module QA ...@@ -34,8 +34,11 @@ module QA
geo_project_renamed = "geo-after-rename-#{SecureRandom.hex(8)}" geo_project_renamed = "geo-after-rename-#{SecureRandom.hex(8)}"
Page::Project::Settings::Main.perform do |settings| Page::Project::Settings::Main.perform do |settings|
settings.rename_project_to(geo_project_renamed)
expect(page).to have_content "Project '#{geo_project_renamed}' was successfully updated."
settings.expand_advanced_settings do |page| settings.expand_advanced_settings do |page|
page.rename_to(geo_project_renamed) page.update_project_path_to(geo_project_renamed)
end end
end end
......
...@@ -9,24 +9,33 @@ describe 'Projects > Settings > User renames a project' do ...@@ -9,24 +9,33 @@ describe 'Projects > Settings > User renames a project' do
visit edit_project_path(project) visit edit_project_path(project)
end end
def rename_project(project, name: nil, path: nil) def change_path(project, path)
fill_in('project_name', with: name) if name within('.advanced-settings') do
fill_in('Path', with: path) if path fill_in('Path', with: path)
click_button('Rename project') click_button('Change path')
end
project.reload
wait_for_edit_project_page_reload wait_for_edit_project_page_reload
end
def change_name(project, name)
within('.general-settings') do
fill_in('Project name', with: name)
click_button('Save changes')
end
project.reload project.reload
wait_for_edit_project_page_reload
end end
def wait_for_edit_project_page_reload def wait_for_edit_project_page_reload
expect(find('.project-edit-container')).to have_content('Rename repository') expect(find('.advanced-settings')).to have_content('Change path')
end end
context 'with invalid characters' do context 'with invalid characters' do
it 'shows errors for invalid project path/name' do it 'shows errors for invalid project path' do
rename_project(project, name: 'foo&bar', path: 'foo&bar') change_path(project, 'foo&bar')
expect(page).to have_field 'Project name', with: 'foo&bar'
expect(page).to have_field 'Path', with: 'foo&bar' expect(page).to have_field 'Path', with: 'foo&bar'
expect(page).to have_content "Name can contain only letters, digits, emojis, '_', '.', dash, space. It must start with letter, digit, emoji or '_'."
expect(page).to have_content "Path can contain only letters, digits, '_', '-' and '.'. Cannot start with '-', end in '.git' or end in '.atom'" expect(page).to have_content "Path can contain only letters, digits, '_', '-' and '.'. Cannot start with '-', end in '.git' or end in '.atom'"
end end
end end
...@@ -42,13 +51,13 @@ describe 'Projects > Settings > User renames a project' do ...@@ -42,13 +51,13 @@ describe 'Projects > Settings > User renames a project' do
context 'when changing project name' do context 'when changing project name' do
it 'renames the repository' do it 'renames the repository' do
rename_project(project, name: 'bar') change_name(project, 'bar')
expect(find('.breadcrumbs')).to have_content(project.name) expect(find('.breadcrumbs')).to have_content(project.name)
end end
context 'with emojis' do context 'with emojis' do
it 'shows error for invalid project name' do it 'shows error for invalid project name' do
rename_project(project, name: '🚀 foo bar ☁️') change_name(project, '🚀 foo bar ☁️')
expect(page).to have_field 'Project name', with: '🚀 foo bar ☁️' expect(page).to have_field 'Project name', with: '🚀 foo bar ☁️'
expect(page).not_to have_content "Name can contain only letters, digits, emojis '_', '.', dash and space. It must start with letter, digit, emoji or '_'." expect(page).not_to have_content "Name can contain only letters, digits, emojis '_', '.', dash and space. It must start with letter, digit, emoji or '_'."
end end
...@@ -67,7 +76,7 @@ describe 'Projects > Settings > User renames a project' do ...@@ -67,7 +76,7 @@ describe 'Projects > Settings > User renames a project' do
end end
it 'the project is accessible via the new path' do it 'the project is accessible via the new path' do
rename_project(project, path: 'bar') change_path(project, 'bar')
new_path = namespace_project_path(project.namespace, 'bar') new_path = namespace_project_path(project.namespace, 'bar')
visit new_path visit new_path
...@@ -77,7 +86,7 @@ describe 'Projects > Settings > User renames a project' do ...@@ -77,7 +86,7 @@ describe 'Projects > Settings > User renames a project' do
it 'the project is accessible via a redirect from the old path' do it 'the project is accessible via a redirect from the old path' do
old_path = project_path(project) old_path = project_path(project)
rename_project(project, path: 'bar') change_path(project, 'bar')
new_path = namespace_project_path(project.namespace, 'bar') new_path = namespace_project_path(project.namespace, 'bar')
visit old_path visit old_path
...@@ -88,7 +97,7 @@ describe 'Projects > Settings > User renames a project' do ...@@ -88,7 +97,7 @@ describe 'Projects > Settings > User renames a project' do
context 'and a new project is added with the same path' do context 'and a new project is added with the same path' do
it 'overrides the redirect' do it 'overrides the redirect' do
old_path = project_path(project) old_path = project_path(project)
rename_project(project, path: 'bar') change_path(project, 'bar')
new_project = create(:project, namespace: user.namespace, path: 'gitlabhq', name: 'quz') new_project = create(:project, namespace: user.namespace, path: 'gitlabhq', name: 'quz')
visit old_path visit old_path
......
...@@ -373,6 +373,21 @@ describe 'Project' do ...@@ -373,6 +373,21 @@ describe 'Project' do
end end
end end
describe 'edit' do
let(:user) { create(:user) }
let(:project) { create(:project, :public) }
let(:path) { edit_project_path(project) }
before do
project.add_maintainer(user)
sign_in(user)
visit path
end
it_behaves_like 'dirty submit form', [{ form: '.js-general-settings-form', input: 'input[name="project[name]"]' },
{ form: '.qa-merge-request-settings', input: '#project_printing_merge_request_link_enabled' }]
end
def remove_with_confirm(button_text, confirm_with) def remove_with_confirm(button_text, confirm_with)
click_button button_text click_button button_text
fill_in 'confirm_name_input', with: confirm_with fill_in 'confirm_name_input', with: confirm_with
......
shared_examples 'dirty submit form' do |selector_args| shared_examples 'dirty submit form' do |selector_args|
selectors = selector_args.is_a?(Array) ? selector_args : [selector_args] selectors = selector_args.is_a?(Array) ? selector_args : [selector_args]
def expect_disabled_state(form, submit, is_disabled = true) def expect_disabled_state(form, submit_selector, is_disabled = true)
disabled_selector = is_disabled == true ? '[disabled]' : ':not([disabled])' disabled_selector = is_disabled == true ? '[disabled]' : ':not([disabled])'
form.find(".js-dirty-submit#{disabled_selector}", match: :first) form.find("#{submit_selector}#{disabled_selector}")
expect(submit.disabled?).to be is_disabled
end end
selectors.each do |selector| selectors.each do |selector|
it "disables #{selector[:form]} submit until there are changes on #{selector[:input]}", :js do it "disables #{selector[:form]} submit until there are changes on #{selector[:input]}", :js do
form = find(selector[:form]) form = find(selector[:form])
submit = form.first('.js-dirty-submit') submit_selector = selector[:submit] || 'input[type="submit"]'
submit = form.first(submit_selector)
input = form.first(selector[:input]) input = form.first(selector[:input])
is_radio = input[:type] == 'radio' is_radio = input[:type] == 'radio'
is_checkbox = input[:type] == 'checkbox' is_checkbox = input[:type] == 'checkbox'
...@@ -22,15 +21,14 @@ shared_examples 'dirty submit form' do |selector_args| ...@@ -22,15 +21,14 @@ shared_examples 'dirty submit form' do |selector_args|
original_checkable = input if is_checkbox original_checkable = input if is_checkbox
expect(submit.disabled?).to be true expect(submit.disabled?).to be true
expect(input.checked?).to be false
is_checkable ? input.click : input.set("#{original_value} changes") is_checkable ? input.click : input.set("#{original_value} changes")
expect_disabled_state(form, submit, false) expect_disabled_state(form, submit_selector, false)
is_checkable ? original_checkable.click : input.set(original_value) is_checkable ? original_checkable.click : input.set(original_value)
expect_disabled_state(form, submit) expect_disabled_state(form, submit_selector)
end end
end end
end end
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment