Skip to content

Commit c19fac5

Browse files
committed
feat: validate contact email
Signed-off-by: TimedIn <[email protected]>
1 parent 9d71828 commit c19fac5

File tree

4 files changed

+65
-3
lines changed

4 files changed

+65
-3
lines changed

package-lock.json

+14
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,7 @@
6969
"pinia": "^2.3.1",
7070
"string-natural-compare": "^3.0.1",
7171
"uuid": "^11.1.0",
72+
"validator": "^13.12.0",
7273
"vue": "~2.7.16",
7374
"vue-click-outside": "^1.1.0",
7475
"vue-cropperjs": "^4.2.0",

src/components/Properties/PropertyText.vue

+49-2
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,22 @@
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="!!helperText && !isReadonly"
69+
:helper-text="!helperText || isReadonly ? '' : helperText"
70+
label-outside
71+
:placeholder="placeholder"
72+
:value.sync="localValue"
73+
type="email"
74+
@update:value="updateEmailValue" />
75+
6076
<!-- OR default to input -->
6177
<NcTextField v-else
6278
:value.sync="localValue"
@@ -90,10 +106,11 @@
90106
<script>
91107
import { NcSelect, NcTextArea, NcTextField } from '@nextcloud/vue'
92108
import debounce from 'debounce'
109+
import isEmail from 'validator/lib/isEmail'
110+
import OpenInNewIcon from 'vue-material-design-icons/OpenInNew.vue'
93111
import PropertyMixin from '../../mixins/PropertyMixin.js'
94-
import PropertyTitle from './PropertyTitle.vue'
95112
import PropertyActions from './PropertyActions.vue'
96-
import OpenInNewIcon from 'vue-material-design-icons/OpenInNew.vue'
113+
import PropertyTitle from './PropertyTitle.vue'
97114

98115
export default {
99116
name: 'PropertyText',
@@ -122,6 +139,13 @@ export default {
122139
},
123140
},
124141

142+
data() {
143+
return {
144+
helperText: null,
145+
valueValid: false,
146+
}
147+
},
148+
125149
computed: {
126150
showProperty() {
127151
return (this.isReadOnly && this.localValue) || !this.isReadOnly
@@ -176,11 +200,34 @@ export default {
176200
},
177201
},
178202

203+
watch: {
204+
isReadOnly(newValue) {
205+
if (newValue && this.propName === 'email') {
206+
// If value invalid restore saved valid value
207+
if (!this.valueValid) {
208+
this.localValue = this.value
209+
this.helperText = null
210+
}
211+
}
212+
},
213+
},
214+
179215
mounted() {
180216
this.resizeHeight()
181217
},
182218

183219
methods: {
220+
updateEmailValue() {
221+
// If email valid or empty
222+
this.valueValid = this.localValue === '' || isEmail(this.localValue)
223+
if (this.valueValid) {
224+
this.helperText = null
225+
this.updateValue(this.localValue)
226+
return
227+
}
228+
this.helperText = this.$refs.email.$refs.inputField.$refs.input.validationMessage || null
229+
},
230+
184231
/**
185232
* Watch textarea resize and update the gridSize accordingly
186233
*/

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

0 commit comments

Comments
 (0)