uikit
Version:
UIkit is a lightweight and modular front-end framework for developing fast and powerful web interfaces.
141 lines (101 loc) • 3.61 kB
JavaScript
import Class from '../mixin/class';
import {default as Togglable, toggleHeight} from '../mixin/togglable';
import {$, $$, attr, filter, getIndex, hasClass, includes, index, isInView, scrollIntoView, toggleClass, unwrap, wrapAll} from 'uikit-util';
export default {
mixins: [Class, Togglable],
props: {
targets: String,
active: null,
collapsible: Boolean,
multiple: Boolean,
toggle: String,
content: String,
transition: String,
offset: Number
},
data: {
targets: '> *',
active: false,
animation: [true],
collapsible: true,
multiple: false,
clsOpen: 'uk-open',
toggle: '> .uk-accordion-title',
content: '> .uk-accordion-content',
transition: 'ease',
offset: 0
},
computed: {
items: {
get({targets}, $el) {
return $$(targets, $el);
},
watch(items, prev) {
items.forEach(el => hide($(this.content, el), !hasClass(el, this.clsOpen)));
if (prev || hasClass(items, this.clsOpen)) {
return;
}
const active = this.active !== false && items[Number(this.active)]
|| !this.collapsible && items[0];
if (active) {
this.toggle(active, false);
}
},
immediate: true
},
toggles({toggle}) {
return this.items.map(item => $(toggle, item));
}
},
events: [
{
name: 'click',
delegate() {
return `${this.targets} ${this.$props.toggle}`;
},
handler(e) {
e.preventDefault();
this.toggle(index(this.toggles, e.current));
}
}
],
methods: {
toggle(item, animate) {
let items = [this.items[getIndex(item, this.items)]];
const activeItems = filter(this.items, `.${this.clsOpen}`);
if (!this.multiple && !includes(activeItems, items[0])) {
items = items.concat(activeItems);
}
if (!this.collapsible && activeItems.length < 2 && !filter(items, `:not(.${this.clsOpen})`).length) {
return;
}
items.forEach(el => this.toggleElement(el, !hasClass(el, this.clsOpen), (el, show) => {
toggleClass(el, this.clsOpen, show);
attr($(this.$props.toggle, el), 'aria-expanded', show);
const content = $(`${el._wrapper ? '> * ' : ''}${this.content}`, el);
if (animate === false || !this.hasTransition) {
hide(content, !show);
return;
}
if (!el._wrapper) {
el._wrapper = wrapAll(content, `<div${show ? ' hidden' : ''}>`);
}
hide(content, false);
return toggleHeight(this)(el._wrapper, show).then(() => {
hide(content, !show);
delete el._wrapper;
unwrap(content);
if (show) {
const toggle = $(this.$props.toggle, el);
if (!isInView(toggle)) {
scrollIntoView(toggle, {offset: this.offset});
}
}
});
}));
}
}
};
function hide(el, hide) {
el && (el.hidden = hide);
}