Skip to content

Commit 373b94a

Browse files
committed
adding error on invalid email-address for contact-editor
Signed-off-by: TimedIn <[email protected]>
1 parent 9d71828 commit 373b94a

File tree

4 files changed

+141
-3
lines changed

4 files changed

+141
-3
lines changed

src/components/Properties/PropertyText.vue

+48-2
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,23 @@
5757
@mousemove="resizeHeight"
5858
@keypress="resizeHeight" />
5959

60+
<!-- email with validation-->
61+
<NcTextField v-else-if="propName === 'email'"
62+
ref="email"
63+
:class="{'property__value--with-ext': haveExtHandler}"
64+
autocapitalize="none"
65+
autocomplete="email"
66+
:inputmode="inputmode"
67+
:readonly="isReadOnly"
68+
:error="(hasError || !!helperText) && !isReadonly"
69+
:helper-text="helperTextWithNonConfirmed"
70+
label-outside
71+
:placeholder="placeholder"
72+
:value.sync="localValue"
73+
:success="isSuccess"
74+
type="email"
75+
@update:value="updateEmailValue" />
76+
6077
<!-- OR default to input -->
6178
<NcTextField v-else
6279
:value.sync="localValue"
@@ -90,10 +107,11 @@
90107
<script>
91108
import { NcSelect, NcTextArea, NcTextField } from '@nextcloud/vue'
92109
import debounce from 'debounce'
110+
import OpenInNewIcon from 'vue-material-design-icons/OpenInNew.vue'
93111
import PropertyMixin from '../../mixins/PropertyMixin.js'
94-
import PropertyTitle from './PropertyTitle.vue'
112+
import { validateEmail } from '../../utils/validate.js'
95113
import PropertyActions from './PropertyActions.vue'
96-
import OpenInNewIcon from 'vue-material-design-icons/OpenInNew.vue'
114+
import PropertyTitle from './PropertyTitle.vue'
97115

98116
export default {
99117
name: 'PropertyText',
@@ -122,6 +140,14 @@ export default {
122140
},
123141
},
124142

143+
data() {
144+
return {
145+
hasError: false,
146+
helperText: null,
147+
isSuccess: false,
148+
}
149+
},
150+
125151
computed: {
126152
showProperty() {
127153
return (this.isReadOnly && this.localValue) || !this.isReadOnly
@@ -162,6 +188,13 @@ export default {
162188
return this.externalHandler.trim() !== '' && this.localValue && this.localValue.length > 0
163189
},
164190

191+
helperTextWithNonConfirmed() {
192+
if (this.helperText || this.hasError || this.isSuccess) {
193+
return this.helperText || ''
194+
}
195+
return this.isNotConfirmedHelperText
196+
},
197+
165198
/**
166199
* Return the selected type placeholder if any
167200
* or the propModel default placeholder
@@ -181,6 +214,19 @@ export default {
181214
},
182215

183216
methods: {
217+
updateEmailValue(e) {
218+
// TODO: provide method to get native input in NcTextField
219+
this.helperText = this.$refs.email.$refs.inputField.$refs.input.validationMessage || null
220+
if (this.helperText !== null) {
221+
return
222+
}
223+
value = e.trim()
224+
225+
if (validateEmail(value) || value === '') {
226+
this.updateValue(value)
227+
}
228+
},
229+
184230
/**
185231
* Watch textarea resize and update the gridSize accordingly
186232
*/
+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
/**
2+
* SPDX-FileCopyrightText: 2021 Nextcloud GmbH and Nextcloud contributors
3+
* SPDX-License-Identifier: AGPL-3.0-or-later
4+
*/
5+
6+
/**
7+
* Email validation regex
8+
*
9+
* Sourced from https://github.com/mpyw/FILTER_VALIDATE_EMAIL.js/blob/71e62ca48841d2246a1b531e7e84f5a01f15e615/src/regexp/ascii.ts*
10+
*/
11+
// eslint-disable-next-line no-control-regex
12+
export const VALIDATE_EMAIL_REGEX = /^(?!(?:(?:\x22?\x5C[\x00-\x7E]\x22?)|(?:\x22?[^\x5C\x22]\x22?)){255,})(?!(?:(?:\x22?\x5C[\x00-\x7E]\x22?)|(?:\x22?[^\x5C\x22]\x22?)){65,}@)(?:(?:[\x21\x23-\x27\x2A\x2B\x2D\x2F-\x39\x3D\x3F\x5E-\x7E]+)|(?:\x22(?:[\x01-\x08\x0B\x0C\x0E-\x1F\x21\x23-\x5B\x5D-\x7F]|(?:\x5C[\x00-\x7F]))*\x22))(?:\.(?:(?:[\x21\x23-\x27\x2A\x2B\x2D\x2F-\x39\x3D\x3F\x5E-\x7E]+)|(?:\x22(?:[\x01-\x08\x0B\x0C\x0E-\x1F\x21\x23-\x5B\x5D-\x7F]|(?:\x5C[\x00-\x7F]))*\x22)))*@(?:(?:(?!.*[^.]{64,})(?:(?:(?:xn--)?[a-z0-9]+(?:-+[a-z0-9]+)*\.){1,126}){1,}(?:(?:[a-z][a-z0-9]*)|(?:(?:xn--)[a-z0-9]+))(?:-+[a-z0-9]+)*)|(?:\[(?:(?:IPv6:(?:(?:[a-f0-9]{1,4}(?::[a-f0-9]{1,4}){7})|(?:(?!(?:.*[a-f0-9][:\]]){7,})(?:[a-f0-9]{1,4}(?::[a-f0-9]{1,4}){0,5})?::(?:[a-f0-9]{1,4}(?::[a-f0-9]{1,4}){0,5})?)))|(?:(?:IPv6:(?:(?:[a-f0-9]{1,4}(?::[a-f0-9]{1,4}){5}:)|(?:(?!(?:.*[a-f0-9]:){5,})(?:[a-f0-9]{1,4}(?::[a-f0-9]{1,4}){0,3})?::(?:[a-f0-9]{1,4}(?::[a-f0-9]{1,4}){0,3}:)?)))?(?:(?:25[0-5])|(?:2[0-4][0-9])|(?:1[0-9]{2})|(?:[1-9]?[0-9]))(?:\.(?:(?:25[0-5])|(?:2[0-4][0-9])|(?:1[0-9]{2})|(?:[1-9]?[0-9]))){3}))\]))$/i

src/css/Properties/Properties.scss

+1-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ $property-row-gap: $contact-details-row-gap;
2525
&__row {
2626
margin-top: var(--default-grid-baseline);
2727
display: flex;
28-
align-items: center;
28+
align-items: flex-start;
2929
gap: $property-row-gap;
3030

3131
// fix default margin from server stylesheet causing slight misalignment

src/utils/validate.js

+80
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
/**
2+
* SPDX-FileCopyrightText: 2021 Nextcloud GmbH and Nextcloud contributors
3+
* SPDX-License-Identifier: AGPL-3.0-or-later
4+
*/
5+
6+
/*
7+
* Frontend validators, less strict than backend validators
8+
*
9+
* TODO add nice validation errors for Profile page settings modal
10+
*/
11+
12+
import { VALIDATE_EMAIL_REGEX } from '../constants/AccountPropertyConstants.ts'
13+
14+
/**
15+
* Validate the email input
16+
*
17+
* Compliant with PHP core FILTER_VALIDATE_EMAIL validator*
18+
*
19+
* Reference implementation https://github.com/mpyw/FILTER_VALIDATE_EMAIL.js/blob/71e62ca48841d2246a1b531e7e84f5a01f15e615/src/index.ts*
20+
*
21+
* @param {string} input the input
22+
* @return {boolean}
23+
*/
24+
export function validateEmail(input) {
25+
return typeof input === 'string'
26+
&& VALIDATE_EMAIL_REGEX.test(input)
27+
&& input.slice(-1) !== '\n'
28+
&& input.length <= 320
29+
&& encodeURIComponent(input).replace(/%../g, 'x').length <= 320
30+
}
31+
32+
/**
33+
* Validate the URL input
34+
*
35+
* @param {string} input the input
36+
* @return {boolean}
37+
*/
38+
export function validateUrl(input) {
39+
try {
40+
// eslint-disable-next-line no-new
41+
new URL(input)
42+
return true
43+
} catch (e) {
44+
return false
45+
}
46+
}
47+
48+
/**
49+
* Validate the language input
50+
*
51+
* @param {object} input the input
52+
* @return {boolean}
53+
*/
54+
export function validateLanguage(input) {
55+
return input.code !== ''
56+
&& input.name !== ''
57+
&& input.name !== undefined
58+
}
59+
60+
/**
61+
* Validate the locale input
62+
*
63+
* @param {object} input the input
64+
* @return {boolean}
65+
*/
66+
export function validateLocale(input) {
67+
return input.code !== ''
68+
&& input.name !== ''
69+
&& input.name !== undefined
70+
}
71+
72+
/**
73+
* Validate boolean input
74+
*
75+
* @param {boolean} input the input
76+
* @return {boolean}
77+
*/
78+
export function validateBoolean(input) {
79+
return typeof input === 'boolean'
80+
}

0 commit comments

Comments
 (0)