52 lines
1.6 KiB
JavaScript
Raw Normal View History

2024-01-29 09:26:07 +08:00
import { Fragment, createTextVNode, Comment } from 'vue';
export function getSlot(scope, slots, slotName = 'default') {
const slot = slots[slotName];
if (slot === undefined) {
throw new Error(`[vueuc/${scope}]: slot[${slotName}] is empty.`);
}
return slot();
}
// o(n) flatten
export function flatten(vNodes, filterCommentNode = true, result = []) {
vNodes.forEach((vNode) => {
if (vNode === null)
return;
if (typeof vNode !== 'object') {
if (typeof vNode === 'string' || typeof vNode === 'number') {
result.push(createTextVNode(String(vNode)));
}
return;
}
if (Array.isArray(vNode)) {
flatten(vNode, filterCommentNode, result);
return;
}
if (vNode.type === Fragment) {
if (vNode.children === null)
return;
if (Array.isArray(vNode.children)) {
flatten(vNode.children, filterCommentNode, result);
}
// rawSlot
}
else if (vNode.type !== Comment) {
result.push(vNode);
}
});
return result;
}
export function getFirstVNode(scope, slots, slotName = 'default') {
const slot = slots[slotName];
if (slot === undefined) {
throw new Error(`[vueuc/${scope}]: slot[${slotName}] is empty.`);
}
const content = flatten(slot());
// vue will normalize the slot, so slot must be an array
if (content.length === 1) {
return content[0];
}
else {
throw new Error(`[vueuc/${scope}]: slot[${slotName}] should have exactly one child.`);
}
}