mirror of
https://github.com/primefaces/primevue.git
synced 2025-05-09 00:42:36 +00:00
Fixed #5643 - Reimplement: Accordion
This commit is contained in:
parent
6cbb5e0993
commit
10f530f7b1
30 changed files with 1049 additions and 243 deletions
147
components/lib/accordionheader/AccordionHeader.d.ts
vendored
Executable file
147
components/lib/accordionheader/AccordionHeader.d.ts
vendored
Executable file
|
@ -0,0 +1,147 @@
|
|||
/**
|
||||
*
|
||||
* AccordionHeader is a helper component for Accordion component.
|
||||
*
|
||||
* [Live Demo](https://www.primevue.org/accordion/)
|
||||
*
|
||||
* @module accordionheader
|
||||
*
|
||||
*/
|
||||
import { VNode } from 'vue';
|
||||
import { ComponentHooks } from '../basecomponent';
|
||||
import { PassThroughOptions } from '../passthrough';
|
||||
import { ClassComponent, DesignToken, GlobalComponentConstructor, PassThrough } from '../ts-helpers';
|
||||
|
||||
export declare type AccordionHeaderPassThroughOptionType = AccordionHeaderPassThroughAttributes | ((options: AccordionHeaderPassThroughMethodOptions) => AccordionHeaderPassThroughAttributes | string) | string | null | undefined;
|
||||
|
||||
/**
|
||||
* Custom passthrough(pt) option method.
|
||||
*/
|
||||
export interface AccordionHeaderPassThroughMethodOptions {
|
||||
/**
|
||||
* Defines instance.
|
||||
*/
|
||||
instance: any;
|
||||
/**
|
||||
* Defines valid properties.
|
||||
*/
|
||||
props: AccordionHeaderProps;
|
||||
/**
|
||||
* Defines current options.
|
||||
*/
|
||||
context: AccordionHeaderContext;
|
||||
/**
|
||||
* Defines valid attributes.
|
||||
*/
|
||||
attrs: any;
|
||||
/**
|
||||
* Defines parent options.
|
||||
*/
|
||||
parent: any;
|
||||
/**
|
||||
* Defines passthrough(pt) options in global config.
|
||||
*/
|
||||
global: object | undefined;
|
||||
}
|
||||
|
||||
/**
|
||||
* Custom passthrough(pt) options.
|
||||
* @see {@link AccordionHeaderProps.pt}
|
||||
*/
|
||||
export interface AccordionHeaderPassThroughOptions {
|
||||
/**
|
||||
* Used to pass attributes to the root's DOM element.
|
||||
*/
|
||||
root?: AccordionHeaderPassThroughOptionType;
|
||||
/**
|
||||
* Used to pass attributes to the root's DOM element.
|
||||
*/
|
||||
toggleicon?: AccordionHeaderPassThroughOptionType;
|
||||
/**
|
||||
* Used to manage all lifecycle hooks.
|
||||
* @see {@link BaseComponent.ComponentHooks}
|
||||
*/
|
||||
hooks?: ComponentHooks;
|
||||
}
|
||||
|
||||
export interface AccordionHeaderPassThroughAttributes {
|
||||
[key: string]: any;
|
||||
}
|
||||
|
||||
/**
|
||||
* Defines valid properties in AccordionHeader component.
|
||||
*/
|
||||
export interface AccordionHeaderProps {
|
||||
/**
|
||||
* Use to change the HTML tag of root element.
|
||||
* @defaultValue BUTTON
|
||||
*/
|
||||
as?: string | undefined;
|
||||
/**
|
||||
* When enabled, it changes the default rendered element for the one passed as a child element.
|
||||
* @defaultValue false
|
||||
*/
|
||||
asChild?: boolean | undefined;
|
||||
/**
|
||||
* It generates scoped CSS variables using design tokens for the component.
|
||||
*/
|
||||
dt?: DesignToken<any>;
|
||||
/**
|
||||
* Used to pass attributes to DOM elements inside the component.
|
||||
* @type {AccordionHeaderPassThroughOptions}
|
||||
*/
|
||||
pt?: PassThrough<AccordionHeaderPassThroughOptions>;
|
||||
/**
|
||||
* Used to configure passthrough(pt) options of the component.
|
||||
* @type {PassThroughOptions}
|
||||
*/
|
||||
ptOptions?: PassThroughOptions;
|
||||
}
|
||||
|
||||
/**
|
||||
* Defines current options in AccordionHeader component.
|
||||
*/
|
||||
export interface AccordionHeaderContext {
|
||||
/**
|
||||
* Whether the item is active.
|
||||
*/
|
||||
active: boolean;
|
||||
}
|
||||
|
||||
/**
|
||||
* Defines valid slots in AccordionHeader slots.
|
||||
*/
|
||||
export interface AccordionHeaderSlots {
|
||||
/**
|
||||
* Custom content template.
|
||||
*/
|
||||
default(): VNode[];
|
||||
/**
|
||||
* Custom toggleicon template.
|
||||
*/
|
||||
toggleicon(): VNode[];
|
||||
}
|
||||
|
||||
export interface AccordionHeaderEmits {}
|
||||
|
||||
/**
|
||||
* **PrimeVue - AccordionHeader**
|
||||
*
|
||||
* _AccordionHeader is a helper component for Accordion component._
|
||||
*
|
||||
* [Live Demo](https://www.primevue.org/accordion/)
|
||||
* --- ---
|
||||
* 
|
||||
*
|
||||
* @group Component
|
||||
*
|
||||
*/
|
||||
declare class AccordionHeader extends ClassComponent<AccordionHeaderProps, AccordionHeaderSlots, AccordionHeaderEmits> {}
|
||||
|
||||
declare module 'vue' {
|
||||
export interface GlobalComponents {
|
||||
AccordionHeader: GlobalComponentConstructor<AccordionHeader>;
|
||||
}
|
||||
}
|
||||
|
||||
export default AccordionHeader;
|
171
components/lib/accordionheader/AccordionHeader.vue
Normal file
171
components/lib/accordionheader/AccordionHeader.vue
Normal file
|
@ -0,0 +1,171 @@
|
|||
<template>
|
||||
<component v-if="!asChild" :is="as" v-ripple :class="cx('root')" @click="onClick" v-bind="attrs">
|
||||
<slot :active="$pcAccordionPanel.active"></slot>
|
||||
<slot name="toggleicon" :active="$pcAccordionPanel.active" :class="cx('toggleicon')">
|
||||
<component
|
||||
v-if="$pcAccordionPanel.active"
|
||||
:is="$pcAccordion.$slots.collapseicon ? $pcAccordion.$slots.collapseicon : $pcAccordion.collapseIcon ? 'span' : 'ChevronDownIcon'"
|
||||
:class="[$pcAccordion.collapseIcon, cx('toggleicon')]"
|
||||
aria-hidden="true"
|
||||
v-bind="ptm('toggleicon', ptParams)"
|
||||
></component>
|
||||
<component
|
||||
v-else
|
||||
:is="$pcAccordion.$slots.expandicon ? $pcAccordion.$slots.expandicon : $pcAccordion.expandIcon ? 'span' : 'ChevronUpIcon'"
|
||||
:class="[$pcAccordion.expandIcon, cx('toggleicon')]"
|
||||
aria-hidden="true"
|
||||
v-bind="ptm('toggleicon', ptParams)"
|
||||
></component>
|
||||
</slot>
|
||||
</component>
|
||||
<slot v-else :class="cx('root')" :active="$pcAccordionPanel.active" :a11yAttrs="a11yAttrs" :onClick="onClick"></slot>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import ChevronDownIcon from 'primevue/icons/chevrondown';
|
||||
import ChevronUpIcon from 'primevue/icons/chevronup';
|
||||
import Ripple from 'primevue/ripple';
|
||||
import { DomHandler } from 'primevue/utils';
|
||||
import { mergeProps } from 'vue';
|
||||
import BaseAccordionHeader from './BaseAccordionHeader.vue';
|
||||
|
||||
export default {
|
||||
name: 'AccordionHeader',
|
||||
extends: BaseAccordionHeader,
|
||||
inheritAttrs: false,
|
||||
inject: ['$pcAccordion', '$pcAccordionPanel'],
|
||||
methods: {
|
||||
onFocus() {
|
||||
this.$pcAccordion.selectOnFocus && this.changeActiveValue();
|
||||
},
|
||||
onClick() {
|
||||
this.changeActiveValue();
|
||||
},
|
||||
onKeydown(event) {
|
||||
switch (event.code) {
|
||||
case 'ArrowDown':
|
||||
this.onArrowDownKey(event);
|
||||
break;
|
||||
|
||||
case 'ArrowUp':
|
||||
this.onArrowUpKey(event);
|
||||
break;
|
||||
|
||||
case 'Home':
|
||||
this.onHomeKey(event);
|
||||
break;
|
||||
|
||||
case 'End':
|
||||
this.onEndKey(event);
|
||||
break;
|
||||
|
||||
case 'Enter':
|
||||
case 'NumpadEnter':
|
||||
case 'Space':
|
||||
this.onEnterKey(event);
|
||||
break;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
},
|
||||
onArrowDownKey(event) {
|
||||
const nextPanel = this.findNextPanel(this.findPanel(event.currentTarget));
|
||||
|
||||
nextPanel ? this.changeFocusedPanel(event, nextPanel) : this.onHomeKey(event);
|
||||
event.preventDefault();
|
||||
},
|
||||
onArrowUpKey(event) {
|
||||
const prevPanel = this.findPrevPanel(this.findPanel(event.currentTarget));
|
||||
|
||||
prevPanel ? this.changeFocusedPanel(event, prevPanel) : this.onEndKey(event);
|
||||
event.preventDefault();
|
||||
},
|
||||
onHomeKey(event) {
|
||||
const firstPanel = this.findFirstPanel();
|
||||
|
||||
this.changeFocusedPanel(event, firstPanel);
|
||||
event.preventDefault();
|
||||
},
|
||||
onEndKey(event) {
|
||||
const lastPanel = this.findLastPanel();
|
||||
|
||||
this.changeFocusedPanel(event, lastPanel);
|
||||
event.preventDefault();
|
||||
},
|
||||
onEnterKey(event) {
|
||||
this.changeActiveValue();
|
||||
event.preventDefault();
|
||||
},
|
||||
findPanel(headerElement) {
|
||||
return headerElement?.closest('[data-pc-name="accordionpanel"]');
|
||||
},
|
||||
findHeader(panelElement) {
|
||||
return DomHandler.findSingle(panelElement, '[data-pc-name="accordionheader"]');
|
||||
},
|
||||
findNextPanel(panelElement, selfCheck = false) {
|
||||
const element = selfCheck ? panelElement : panelElement.nextElementSibling;
|
||||
|
||||
return element ? (DomHandler.getAttribute(element, 'data-p-disabled') ? this.findNextPanel(element) : this.findHeader(element)) : null;
|
||||
},
|
||||
findPrevPanel(panelElement, selfCheck = false) {
|
||||
const element = selfCheck ? panelElement : panelElement.previousElementSibling;
|
||||
|
||||
return element ? (DomHandler.getAttribute(element, 'data-p-disabled') ? this.findPrevPanel(element) : this.findHeader(element)) : null;
|
||||
},
|
||||
findFirstPanel() {
|
||||
return this.findNextPanel(this.$pcAccordion.$el.firstElementChild, true);
|
||||
},
|
||||
findLastPanel() {
|
||||
return this.findPrevPanel(this.$pcAccordion.$el.lastElementChild, true);
|
||||
},
|
||||
changeActiveValue() {
|
||||
this.$pcAccordion.updateValue(this.$pcAccordionPanel.value);
|
||||
},
|
||||
changeFocusedPanel(event, element) {
|
||||
DomHandler.focus(this.findHeader(element));
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
id() {
|
||||
return `${this.$pcAccordion.id}_accordionheader_${this.$pcAccordionPanel.value}`;
|
||||
},
|
||||
ariaControls() {
|
||||
return `${this.$pcAccordion.id}_accordioncontent_${this.$pcAccordionPanel.value}`;
|
||||
},
|
||||
attrs() {
|
||||
return mergeProps(this.asAttrs, this.a11yAttrs, this.ptmi('root', this.ptParams));
|
||||
},
|
||||
asAttrs() {
|
||||
return this.as === 'BUTTON' ? { type: 'button', disabled: this.$pcAccordionPanel.disabled } : undefined;
|
||||
},
|
||||
a11yAttrs() {
|
||||
return {
|
||||
id: this.id,
|
||||
tabindex: this.$pcAccordion.tabindex,
|
||||
'aria-expanded': this.$pcAccordionPanel.active,
|
||||
'aria-controls': this.ariaControls,
|
||||
'data-pc-name': 'accordionheader',
|
||||
'data-p-disabled': this.$pcAccordionPanel.disabled,
|
||||
'data-p-active': this.$pcAccordionPanel.active,
|
||||
onFocus: this.onFocus,
|
||||
onKeydown: this.onKeydown
|
||||
};
|
||||
},
|
||||
ptParams() {
|
||||
return {
|
||||
context: {
|
||||
active: this.$pcAccordionPanel.active
|
||||
}
|
||||
};
|
||||
}
|
||||
},
|
||||
components: {
|
||||
ChevronUpIcon,
|
||||
ChevronDownIcon
|
||||
},
|
||||
directives: {
|
||||
ripple: Ripple
|
||||
}
|
||||
};
|
||||
</script>
|
25
components/lib/accordionheader/BaseAccordionHeader.vue
Normal file
25
components/lib/accordionheader/BaseAccordionHeader.vue
Normal file
|
@ -0,0 +1,25 @@
|
|||
<script>
|
||||
import AccordionHeaderStyle from 'primevue/accordionheader/style';
|
||||
import BaseComponent from 'primevue/basecomponent';
|
||||
|
||||
export default {
|
||||
name: 'BaseAccordionHeader',
|
||||
extends: BaseComponent,
|
||||
props: {
|
||||
as: {
|
||||
type: String,
|
||||
default: 'BUTTON'
|
||||
},
|
||||
asChild: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
style: AccordionHeaderStyle,
|
||||
provide() {
|
||||
return {
|
||||
$parentInstance: this
|
||||
};
|
||||
}
|
||||
};
|
||||
</script>
|
9
components/lib/accordionheader/package.json
Normal file
9
components/lib/accordionheader/package.json
Normal file
|
@ -0,0 +1,9 @@
|
|||
{
|
||||
"main": "./accordionheader.cjs.js",
|
||||
"module": "./accordionheader.esm.js",
|
||||
"unpkg": "./accordionheader.min.js",
|
||||
"types": "./AccordionHeader.d.ts",
|
||||
"browser": {
|
||||
"./sfc": "./AccordionHeader.vue"
|
||||
}
|
||||
}
|
3
components/lib/accordionheader/style/AccordionHeaderStyle.d.ts
vendored
Normal file
3
components/lib/accordionheader/style/AccordionHeaderStyle.d.ts
vendored
Normal file
|
@ -0,0 +1,3 @@
|
|||
import { BaseStyle } from '../../base/style/BaseStyle';
|
||||
|
||||
export interface AccordionHeaderStyle extends BaseStyle {}
|
11
components/lib/accordionheader/style/AccordionHeaderStyle.js
Normal file
11
components/lib/accordionheader/style/AccordionHeaderStyle.js
Normal file
|
@ -0,0 +1,11 @@
|
|||
import BaseStyle from 'primevue/base/style';
|
||||
|
||||
const classes = {
|
||||
root: 'p-accordionheader',
|
||||
toggleicon: 'p-accordionheader-toggle-icon'
|
||||
};
|
||||
|
||||
export default BaseStyle.extend({
|
||||
name: 'accordionheader',
|
||||
classes
|
||||
});
|
6
components/lib/accordionheader/style/package.json
Normal file
6
components/lib/accordionheader/style/package.json
Normal file
|
@ -0,0 +1,6 @@
|
|||
{
|
||||
"main": "./accordionheaderstyle.cjs.js",
|
||||
"module": "./accordionheaderstyle.esm.js",
|
||||
"unpkg": "./accordionheaderstyle.min.js",
|
||||
"types": "./AccordionHeaderStyle.d.ts"
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue