2021-09-25 08:51:32 +00:00
|
|
|
import Cell from '../Cell';
|
|
|
|
import { getColumnsKey } from '../utils/valueUtil';
|
|
|
|
import type { CustomizeComponent, GetComponentProps, Key, GetRowKey } from '../interface';
|
|
|
|
import ExpandedRow from './ExpandedRow';
|
|
|
|
import { computed, defineComponent, ref, watchEffect } from 'vue';
|
|
|
|
import { useInjectTable } from '../context/TableContext';
|
|
|
|
import { useInjectBody } from '../context/BodyContext';
|
|
|
|
import classNames from '../../_util/classNames';
|
|
|
|
import { parseStyleText } from '../../_util/props-util';
|
|
|
|
|
|
|
|
export interface BodyRowProps<RecordType> {
|
|
|
|
record: RecordType;
|
|
|
|
index: number;
|
|
|
|
recordKey: Key;
|
|
|
|
expandedKeys: Set<Key>;
|
|
|
|
rowComponent: CustomizeComponent;
|
|
|
|
cellComponent: CustomizeComponent;
|
|
|
|
customRow: GetComponentProps<RecordType>;
|
|
|
|
rowExpandable: (record: RecordType) => boolean;
|
|
|
|
indent?: number;
|
|
|
|
rowKey: Key;
|
|
|
|
getRowKey: GetRowKey<RecordType>;
|
|
|
|
childrenColumnName: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export default defineComponent<BodyRowProps<unknown>>({
|
|
|
|
name: 'BodyRow',
|
|
|
|
inheritAttrs: false,
|
|
|
|
props: [
|
|
|
|
'record',
|
|
|
|
'index',
|
|
|
|
'recordKey',
|
|
|
|
'expandedKeys',
|
|
|
|
'rowComponent',
|
|
|
|
'cellComponent',
|
|
|
|
'customRow',
|
|
|
|
'rowExpandable',
|
|
|
|
'indent',
|
|
|
|
'rowKey',
|
|
|
|
'getRowKey',
|
|
|
|
'childrenColumnName',
|
|
|
|
] as any,
|
|
|
|
setup(props, { attrs }) {
|
|
|
|
const tableContext = useInjectTable();
|
|
|
|
const bodyContext = useInjectBody();
|
|
|
|
const expandRended = ref(false);
|
|
|
|
|
|
|
|
const expanded = computed(() => props.expandedKeys && props.expandedKeys.has(props.recordKey));
|
|
|
|
|
|
|
|
watchEffect(() => {
|
|
|
|
if (expanded.value) {
|
|
|
|
expandRended.value = true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
const rowSupportExpand = computed(
|
|
|
|
() =>
|
|
|
|
bodyContext.expandableType === 'row' &&
|
|
|
|
(!props.rowExpandable || props.rowExpandable(props.record)),
|
|
|
|
);
|
|
|
|
// Only when row is not expandable and `children` exist in record
|
|
|
|
const nestExpandable = computed(() => bodyContext.expandableType === 'nest');
|
|
|
|
const hasNestChildren = computed(
|
|
|
|
() => props.childrenColumnName && props.record && props.record[props.childrenColumnName],
|
|
|
|
);
|
|
|
|
const mergedExpandable = computed(() => rowSupportExpand.value || nestExpandable.value);
|
|
|
|
|
|
|
|
const onInternalTriggerExpand = (record, event) => {
|
|
|
|
bodyContext.onTriggerExpand(record, event);
|
|
|
|
};
|
|
|
|
|
|
|
|
// =========================== onRow ===========================
|
|
|
|
const additionalProps = computed<Record<string, any>>(
|
|
|
|
() => props.customRow?.(props.record, props.index) || {},
|
|
|
|
);
|
|
|
|
|
|
|
|
const onClick = (event, ...args) => {
|
|
|
|
if (bodyContext.expandRowByClick && mergedExpandable.value) {
|
|
|
|
onInternalTriggerExpand(props.record, event);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (additionalProps.value?.onClick) {
|
|
|
|
additionalProps.value.onClick(event, ...args);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const computeRowClassName = computed(() => {
|
|
|
|
const { record, index, indent } = props;
|
|
|
|
const { rowClassName } = bodyContext;
|
|
|
|
if (typeof rowClassName === 'string') {
|
|
|
|
return rowClassName;
|
|
|
|
} else if (typeof rowClassName === 'function') {
|
|
|
|
return rowClassName(record, index, indent);
|
|
|
|
}
|
|
|
|
return '';
|
|
|
|
});
|
|
|
|
|
|
|
|
const columnsKey = computed(() => getColumnsKey(bodyContext.flattenColumns));
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
const { class: className, style } = attrs as any;
|
|
|
|
const {
|
|
|
|
record,
|
|
|
|
index,
|
|
|
|
rowKey,
|
|
|
|
indent = 0,
|
|
|
|
rowComponent: RowComponent,
|
|
|
|
cellComponent,
|
|
|
|
} = props;
|
|
|
|
const { prefixCls, fixedInfoList, transformCellText } = tableContext;
|
|
|
|
const {
|
|
|
|
fixHeader,
|
|
|
|
fixColumn,
|
|
|
|
horizonScroll,
|
|
|
|
componentWidth,
|
|
|
|
flattenColumns,
|
|
|
|
expandedRowClassName,
|
|
|
|
indentSize,
|
|
|
|
expandIcon,
|
|
|
|
expandedRowRender,
|
|
|
|
expandIconColumnIndex,
|
|
|
|
} = bodyContext;
|
|
|
|
const baseRowNode = (
|
|
|
|
<RowComponent
|
|
|
|
{...additionalProps.value}
|
|
|
|
data-row-key={rowKey}
|
|
|
|
class={classNames(
|
|
|
|
className,
|
|
|
|
`${prefixCls}-row`,
|
|
|
|
`${prefixCls}-row-level-${indent}`,
|
|
|
|
computeRowClassName.value,
|
|
|
|
additionalProps.value.class,
|
|
|
|
)}
|
|
|
|
style={{
|
|
|
|
...style,
|
|
|
|
...parseStyleText(additionalProps.value.style),
|
|
|
|
}}
|
|
|
|
onClick={onClick}
|
|
|
|
>
|
|
|
|
{flattenColumns.map((column, colIndex) => {
|
|
|
|
const { customRender, dataIndex, className: columnClassName } = column;
|
|
|
|
|
|
|
|
const key = columnsKey[colIndex];
|
|
|
|
const fixedInfo = fixedInfoList[colIndex];
|
|
|
|
|
|
|
|
let additionalCellProps;
|
|
|
|
if (column.customCell) {
|
2021-12-17 13:23:56 +00:00
|
|
|
additionalCellProps = column.customCell(record, index, column);
|
2021-09-25 08:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Cell
|
|
|
|
cellType="body"
|
|
|
|
class={columnClassName}
|
|
|
|
ellipsis={column.ellipsis}
|
|
|
|
align={column.align}
|
|
|
|
component={cellComponent}
|
|
|
|
prefixCls={prefixCls}
|
|
|
|
key={key}
|
|
|
|
record={record}
|
|
|
|
index={index}
|
|
|
|
dataIndex={dataIndex}
|
|
|
|
customRender={customRender}
|
|
|
|
{...fixedInfo}
|
|
|
|
additionalProps={additionalCellProps}
|
|
|
|
column={column}
|
|
|
|
transformCellText={transformCellText}
|
|
|
|
v-slots={{
|
|
|
|
// ============= Used for nest expandable =============
|
|
|
|
appendNode:
|
|
|
|
colIndex === (expandIconColumnIndex || 0) && nestExpandable.value
|
|
|
|
? () => {
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<span
|
|
|
|
style={{ paddingLeft: `${indentSize * indent}px` }}
|
|
|
|
class={`${prefixCls}-row-indent indent-level-${indent}`}
|
|
|
|
/>
|
|
|
|
{expandIcon({
|
|
|
|
prefixCls,
|
|
|
|
expanded: expanded.value,
|
|
|
|
expandable: hasNestChildren.value,
|
|
|
|
record,
|
|
|
|
onExpand: onInternalTriggerExpand,
|
|
|
|
})}
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
: undefined,
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
</RowComponent>
|
|
|
|
);
|
|
|
|
|
|
|
|
// ======================== Expand Row =========================
|
|
|
|
let expandRowNode;
|
|
|
|
if (rowSupportExpand.value && (expandRended.value || expanded.value)) {
|
|
|
|
const expandContent = expandedRowRender({
|
|
|
|
record,
|
|
|
|
index,
|
|
|
|
indent: indent + 1,
|
|
|
|
expanded: expanded.value,
|
|
|
|
});
|
|
|
|
const computedExpandedRowClassName =
|
|
|
|
expandedRowClassName && expandedRowClassName(record, index, indent);
|
|
|
|
expandRowNode = (
|
|
|
|
<ExpandedRow
|
|
|
|
expanded={expanded.value}
|
|
|
|
class={classNames(
|
|
|
|
`${prefixCls}-expanded-row`,
|
|
|
|
`${prefixCls}-expanded-row-level-${indent + 1}`,
|
|
|
|
computedExpandedRowClassName,
|
|
|
|
)}
|
|
|
|
prefixCls={prefixCls}
|
|
|
|
fixHeader={fixHeader}
|
|
|
|
fixColumn={fixColumn}
|
|
|
|
horizonScroll={horizonScroll}
|
|
|
|
component={RowComponent}
|
|
|
|
componentWidth={componentWidth}
|
|
|
|
cellComponent={cellComponent}
|
|
|
|
colSpan={flattenColumns.length}
|
|
|
|
>
|
|
|
|
{expandContent}
|
|
|
|
</ExpandedRow>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
{baseRowNode}
|
|
|
|
{expandRowNode}
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
},
|
|
|
|
});
|