Chris Hut a6c990c6fe
Cc 5545: Upgrade HDS packages and modifiers (#19226)
* Upgrade @hashicorp/design-system-tokens to 1.9.0

* Upgrade @hashicorp/design-system-components to 1.8.1

* Upgrade @hashicorp/design-system-components and ember-in-viewport

* Explicitly install ember-modifier@4.1.0

* rename copy-button

* Fix how cleanup is done in with-copyable

* Update aria-menu modifier for new structure

* Update css-prop modifier to new structure

* Convert did-upsert to regular class modifier

* Update notification modifier for new structure

* Update on-oustside modifier for new structure

* Move destroy handler registration in with-copyable

* Update style modifier for new structure

* Update validate modifier for new structure

* Guard against setting on destroyed object

* Upgrade @hashicorp/design-system-components to 2.14.1

* Remove debugger

* Guard against null in aria-menu

* Fix undefined hash in validate addon

* Upgrade ember-on-resize-modifier

* Fix copy button import, missing import and array destructuring

---------

Co-authored-by: wenincode <tyler.wendlandt@hashicorp.com>
2023-10-17 07:27:42 -06:00

127 lines
3.4 KiB
JavaScript

/**
* Copyright (c) HashiCorp, Inc.
* SPDX-License-Identifier: BUSL-1.1
*/
import Modifier from 'ember-modifier';
import { action } from '@ember/object';
import { registerDestructor } from '@ember/destroyable';
class ValidationError extends Error {}
function cleanup(instance) {
if (instance && instance?.element) {
instance?.element?.removeEventListener('input', instance?.listen);
instance?.element?.removeEventListener('blur', instance?.reset);
}
}
export default class ValidateModifier extends Modifier {
item = null;
hash = null;
validate(value, validations = {}) {
if (Object.keys(validations).length === 0) {
return;
}
const errors = {};
Object.entries(this.hash.validations)
// filter out strings, for now these are helps, but ain't great if someone has a item.help
.filter(([key, value]) => typeof value !== 'string')
.forEach(([key, item]) => {
// optionally set things for you
if (this.item) {
this.item[key] = value;
}
(item || []).forEach((validation) => {
const re = new RegExp(validation.test);
if (!re.test(value)) {
errors[key] = new ValidationError(validation.error);
}
});
});
const state = this.hash.chart.state || {};
if (state.context == null) {
state.context = {};
}
if (Object.keys(errors).length > 0) {
state.context.errors = errors;
this.hash.chart.dispatch('ERROR', state.context);
} else {
state.context.errors = null;
this.hash.chart.dispatch('RESET', state.context);
}
}
@action
reset(e) {
if (e.target.value.length === 0) {
const state = this.hash.chart.state;
if (!state.context) {
state.context = {};
}
if (!state.context.errors) {
state.context.errors = {};
}
Object.entries(this.hash.validations)
// filter out strings, for now these are helps, but ain't great if someone has a item.help
.filter(([key, value]) => typeof value !== 'string')
.forEach(([key, item]) => {
if (typeof state.context.errors[key] !== 'undefined') {
delete state.context.errors[key];
}
});
if (Object.keys(state.context.errors).length === 0) {
state.context.errors = null;
this.hash.chart.dispatch('RESET', state.context);
}
}
}
@action
listen(e) {
this.validate(e.target.value, this.hash.validations);
}
constructor(owner, args) {
super(owner, args);
registerDestructor(this, cleanup);
}
async modify(element, positional, named) {
cleanup.call(this);
this.element = element;
this.hash = named;
this.item = positional[0];
if (typeof this.hash.chart === 'undefined') {
this.hash.chart = {
state: {
context: {},
},
dispatch: (state) => {
switch (state) {
case 'ERROR':
this.hash.onchange(this.hash.chart.state.context.errors);
break;
case 'RESET':
this.hash.onchange();
break;
}
},
};
}
this.element.addEventListener('input', this.listen);
this.element.addEventListener('blur', this.reset);
if (this.element.value.length > 0) {
await Promise.resolve();
if (this && this.element) {
this.validate(this.element.value, this.hash.validations);
}
}
}
}