blob: a9c0f6c536e3d30ae68be68abbe0028f549ed4f4 [file] [log] [blame]
<template>
<b-modal id="modal-user" ref="modal" @ok="onOk" @hidden="resetForm">
<template v-slot:modal-title>
<template v-if="newUser">
{{ $t('pageLocalUserManagement.addUser') }}
</template>
<template v-else>
{{ $t('pageLocalUserManagement.editUser') }}
</template>
</template>
<b-form novalidate @submit="handleSubmit">
<b-container>
<!-- Manual unlock form control -->
<b-row v-if="!newUser && manualUnlockPolicy && user.Locked">
<b-col sm="9">
<alert :show="true" variant="warning" small>
<template v-if="!$v.form.manualUnlock.$dirty">
{{ $t('pageLocalUserManagement.modal.accountLocked') }}
</template>
<template v-else>
{{
$t('pageLocalUserManagement.modal.clickSaveToUnlockAccount')
}}
</template>
</alert>
</b-col>
<b-col sm="3">
<input v-model="form.manualUnlock" type="hidden" value="false" />
<b-button
variant="primary"
:disabled="$v.form.manualUnlock.$dirty"
@click="$v.form.manualUnlock.$touch()"
>
{{ $t('pageLocalUserManagement.modal.unlock') }}
</b-button>
</b-col>
</b-row>
<b-row>
<b-col>
<b-form-group
:label="$t('pageLocalUserManagement.modal.accountStatus')"
>
<b-form-radio
v-model="form.status"
name="user-status"
:value="true"
@input="$v.form.status.$touch()"
>
{{ $t('global.status.enabled') }}
</b-form-radio>
<b-form-radio
v-model="form.status"
name="user-status"
:value="false"
@input="$v.form.status.$touch()"
>
{{ $t('global.status.disabled') }}
</b-form-radio>
</b-form-group>
<b-form-group
:label="$t('pageLocalUserManagement.modal.username')"
label-for="username"
>
<b-form-text id="username-help-block">
{{ $t('pageLocalUserManagement.modal.cannotStartWithANumber') }}
<br />
{{
$t(
'pageLocalUserManagement.modal.noSpecialCharactersExceptUnderscore'
)
}}
</b-form-text>
<b-form-input
id="username"
v-model="form.username"
type="text"
aria-describedby="username-help-block"
:state="getValidationState($v.form.username)"
:disabled="!newUser && originalUsername === 'root'"
@input="$v.form.username.$touch()"
/>
<b-form-invalid-feedback role="alert">
<template v-if="!$v.form.username.required">
{{ $t('global.form.fieldRequired') }}
</template>
<template v-else-if="!$v.form.username.maxLength">
{{
$t('global.form.lengthMustBeBetween', { min: 1, max: 16 })
}}
</template>
<template v-else-if="!$v.form.username.pattern">
{{ $t('global.form.invalidFormat') }}
</template>
</b-form-invalid-feedback>
</b-form-group>
<b-form-group
:label="$t('pageLocalUserManagement.modal.privilege')"
label-for="privilege"
>
<b-form-select
id="privilege"
v-model="form.privilege"
:options="privilegeTypes"
:state="getValidationState($v.form.privilege)"
@input="$v.form.privilege.$touch()"
>
</b-form-select>
<b-form-invalid-feedback role="alert">
<template v-if="!$v.form.privilege.required">
{{ $t('global.form.fieldRequired') }}
</template>
</b-form-invalid-feedback>
</b-form-group>
</b-col>
<b-col>
<b-form-group
:label="$t('pageLocalUserManagement.modal.userPassword')"
label-for="password"
>
<b-form-text id="password-help-block">
{{
$t('pageLocalUserManagement.modal.passwordMustBeBetween', {
min: passwordRequirements.minLength,
max: passwordRequirements.maxLength
})
}}
</b-form-text>
<input-password-toggle>
<b-form-input
id="password"
v-model="form.password"
type="password"
aria-describedby="password-help-block"
:state="getValidationState($v.form.password)"
@input="$v.form.password.$touch()"
/>
<b-form-invalid-feedback role="alert">
<template v-if="!$v.form.password.required">
{{ $t('global.form.fieldRequired') }}
</template>
<template
v-if="
!$v.form.password.minLength || !$v.form.password.maxLength
"
>
{{
$t(
'pageLocalUserManagement.modal.passwordMustBeBetween',
{
min: passwordRequirements.minLength,
max: passwordRequirements.maxLength
}
)
}}
</template>
</b-form-invalid-feedback>
</input-password-toggle>
</b-form-group>
<b-form-group
:label="$t('pageLocalUserManagement.modal.confirmUserPassword')"
label-for="password-confirmation"
>
<input-password-toggle>
<b-form-input
id="password-confirmation"
v-model="form.passwordConfirmation"
type="password"
:state="getValidationState($v.form.passwordConfirmation)"
@input="$v.form.passwordConfirmation.$touch()"
/>
<b-form-invalid-feedback role="alert">
<template v-if="!$v.form.passwordConfirmation.required">
{{ $t('global.form.fieldRequired') }}
</template>
<template
v-else-if="!$v.form.passwordConfirmation.sameAsPassword"
>
{{
$t('pageLocalUserManagement.modal.passwordsDoNotMatch')
}}
</template>
</b-form-invalid-feedback>
</input-password-toggle>
</b-form-group>
</b-col>
</b-row>
</b-container>
</b-form>
<template v-slot:modal-ok>
<template v-if="newUser">
{{ $t('pageLocalUserManagement.addUser') }}
</template>
<template v-else>
{{ $t('global.action.save') }}
</template>
</template>
</b-modal>
</template>
<script>
import {
required,
maxLength,
minLength,
sameAs,
helpers,
requiredIf
} from 'vuelidate/lib/validators';
import VuelidateMixin from '../../../components/Mixins/VuelidateMixin.js';
import InputPasswordToggle from '../../../components/Global/InputPasswordToggle';
import Alert from '../../../components/Global/Alert';
export default {
components: { Alert, InputPasswordToggle },
mixins: [VuelidateMixin],
props: {
user: {
type: Object,
default: null
},
passwordRequirements: {
type: Object,
required: true
}
},
data() {
return {
privilegeTypes: ['Administrator', 'Operator', 'ReadOnly', 'NoAccess'],
originalUsername: '',
form: {
status: true,
username: '',
privilege: '',
password: '',
passwordConfirmation: '',
manualUnlock: false
}
};
},
computed: {
newUser() {
return this.user ? false : true;
},
accountSettings() {
return this.$store.getters['localUsers/accountSettings'];
},
manualUnlockPolicy() {
return !this.accountSettings.accountLockoutDuration;
}
},
watch: {
user: function(value) {
if (value === null) return;
this.originalUsername = value.username;
this.form.username = value.username;
this.form.status = value.Enabled;
this.form.privilege = value.privilege;
}
},
validations() {
return {
form: {
status: {
required
},
username: {
required,
maxLength: maxLength(16),
pattern: helpers.regex('pattern', /^([a-zA-Z_][a-zA-Z0-9_]*)/)
},
privilege: {
required
},
password: {
required: requiredIf(function() {
return this.requirePassword();
}),
minLength: minLength(this.passwordRequirements.minLength),
maxLength: maxLength(this.passwordRequirements.maxLength)
},
passwordConfirmation: {
required: requiredIf(function() {
return this.requirePassword();
}),
sameAsPassword: sameAs('password')
},
manualUnlock: {}
}
};
},
methods: {
handleSubmit() {
let userData = {};
if (this.newUser) {
this.$v.$touch();
if (this.$v.$invalid) return;
userData.username = this.form.username;
userData.status = this.form.status;
userData.privilege = this.form.privilege;
userData.password = this.form.password;
} else {
if (this.$v.$invalid) return;
userData.originalUsername = this.originalUsername;
if (this.$v.form.status.$dirty) {
userData.status = this.form.status;
}
if (this.$v.form.username.$dirty) {
userData.username = this.form.username;
}
if (this.$v.form.privilege.$dirty) {
userData.privilege = this.form.privilege;
}
if (this.$v.form.password.$dirty) {
userData.password = this.form.password;
}
if (this.$v.form.manualUnlock.$dirty) {
// If form manualUnlock control $dirty then
// set user Locked property to false
userData.locked = false;
}
if (Object.entries(userData).length === 1) {
this.closeModal();
return;
}
}
this.$emit('ok', { isNewUser: this.newUser, userData });
this.closeModal();
},
closeModal() {
this.$nextTick(() => {
this.$refs.modal.hide();
});
},
resetForm() {
this.form.originalUsername = '';
this.form.status = true;
this.form.username = '';
this.form.privilege = '';
this.form.password = '';
this.form.passwordConfirmation = '';
this.$v.$reset();
},
requirePassword() {
if (this.newUser) return true;
if (this.$v.form.password.$dirty) return true;
if (this.$v.form.passwordConfirmation.$dirty) return true;
return false;
},
onOk(bvModalEvt) {
// prevent modal close
bvModalEvt.preventDefault();
this.handleSubmit();
}
}
};
</script>