81f209d71e
Adds support for ACL Roles and Service Identities CRUD, along with necessary changes to Tokens, and the CSS improvements required. Also includes refinements/improvements for easier testing of deeply nested components. 1. ember-data adapter/serializer/model triplet for Roles 2. repository, form/validations and searching filter for Roles 3. Moves potentially, repeated, or soon to to repeated functionality into a mixin (mainly for 'many policy' relationships) 4. A few styling tweaks for little edge cases around roles 5. Router additions, Route, Controller and templates for Roles Also see: * UI: ACL Roles cont. plus Service Identities (#5661 and #5720)
124 lines
3.8 KiB
JavaScript
124 lines
3.8 KiB
JavaScript
import { get, set } from '@ember/object';
|
|
import { inject as service } from '@ember/service';
|
|
import DomBufferComponent from 'consul-ui/components/dom-buffer';
|
|
import SlotsMixin from 'block-slots';
|
|
import WithResizing from 'consul-ui/mixins/with-resizing';
|
|
|
|
import templatize from 'consul-ui/utils/templatize';
|
|
export default DomBufferComponent.extend(SlotsMixin, WithResizing, {
|
|
dom: service('dom'),
|
|
checked: true,
|
|
height: null,
|
|
// dialog is a reference to the modal-dialog 'panel' so its 'window'
|
|
dialog: null,
|
|
overflowingClass: 'overflowing',
|
|
onclose: function() {},
|
|
onopen: function() {},
|
|
_open: function(e) {
|
|
set(this, 'checked', true);
|
|
if (get(this, 'height') === null) {
|
|
if (this.element) {
|
|
const dialogPanel = get(this, 'dom').element('[role="dialog"] > div > div', this.element);
|
|
const rect = dialogPanel.getBoundingClientRect();
|
|
set(this, 'dialog', dialogPanel);
|
|
set(this, 'height', rect.height);
|
|
}
|
|
}
|
|
this.didAppear();
|
|
this.onopen(e);
|
|
},
|
|
didAppear: function() {
|
|
this._super(...arguments);
|
|
if (get(this, 'checked')) {
|
|
get(this, 'dom')
|
|
.root()
|
|
.classList.add(...templatize(['with-modal']));
|
|
}
|
|
},
|
|
_close: function(e) {
|
|
set(this, 'checked', false);
|
|
const dialogPanel = get(this, 'dialog');
|
|
if (dialogPanel) {
|
|
const overflowing = get(this, 'overflowingClass');
|
|
if (dialogPanel.classList.contains(overflowing)) {
|
|
dialogPanel.classList.remove(overflowing);
|
|
}
|
|
}
|
|
// TODO: should we make a didDisappear?
|
|
get(this, 'dom')
|
|
.root()
|
|
.classList.remove(...templatize(['with-modal']));
|
|
this.onclose(e);
|
|
},
|
|
didReceiveAttrs: function() {
|
|
this._super(...arguments);
|
|
// TODO: Why does setting name mean checked it false?
|
|
// It's because if it has a name then it is likely to be linked
|
|
// to HTML state rather than just being added via HTMLBars
|
|
// and therefore likely to be immediately on the page
|
|
// It's not our usecase just yet, but this should check the state
|
|
// of the thing its linked to, incase that has a `checked` of true
|
|
// right now we know ours is always false.
|
|
if (get(this, 'name')) {
|
|
set(this, 'checked', false);
|
|
}
|
|
if (this.element) {
|
|
if (get(this, 'checked')) {
|
|
// TODO: probably need an event here
|
|
// possibly this.element for the target
|
|
// or find the input
|
|
this._open({ target: {} });
|
|
}
|
|
}
|
|
},
|
|
didInsertElement: function() {
|
|
this._super(...arguments);
|
|
if (get(this, 'checked')) {
|
|
// TODO: probably need an event here
|
|
// possibly this.element for the target
|
|
// or find the input
|
|
this._open({ target: {} });
|
|
}
|
|
},
|
|
didDestroyElement: function() {
|
|
this._super(...arguments);
|
|
get(this, 'dom')
|
|
.root()
|
|
.classList.remove(...templatize(['with-modal']));
|
|
},
|
|
resize: function(e) {
|
|
if (get(this, 'checked')) {
|
|
const height = get(this, 'height');
|
|
if (height !== null) {
|
|
const dialogPanel = get(this, 'dialog');
|
|
const overflowing = get(this, 'overflowingClass');
|
|
if (height > e.detail.height) {
|
|
if (!dialogPanel.classList.contains(overflowing)) {
|
|
dialogPanel.classList.add(overflowing);
|
|
}
|
|
return;
|
|
} else {
|
|
if (dialogPanel.classList.contains(overflowing)) {
|
|
dialogPanel.classList.remove(overflowing);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
actions: {
|
|
change: function(e) {
|
|
if (get(e, 'target.checked')) {
|
|
this._open(e);
|
|
} else {
|
|
this._close(e);
|
|
}
|
|
},
|
|
close: function() {
|
|
const $close = get(this, 'dom').element('#modal_close');
|
|
$close.checked = true;
|
|
const $input = get(this, 'dom').element('input[name="modal"]', this.element);
|
|
$input.onchange({ target: $input });
|
|
},
|
|
},
|
|
});
|