UNPKG

quill-delta-to-html

Version:
106 lines (90 loc) 3.88 kB
import { ListGroup, ListItem, BlockGroup, TDataGroup } from './group-types'; class ListNester { nest(groups: TDataGroup[]): TDataGroup[] { var listBlocked = this.convertListBlocksToListGroups(groups); var groupedByListGroups = this.groupConsecutiveListGroups(listBlocked); // convert grouped ones into listgroup var nested = groupedByListGroups.map((group: TDataGroup) => { if (!Array.isArray(group)) { return group; } return this.nestListSection(group); }) ._flatten(); var groupRootLists = nested._groupConsecutiveElementsWhile( (curr: TDataGroup, prev: TDataGroup) => { if (!(curr instanceof ListGroup && prev instanceof ListGroup)) { return false; } return curr.items[0].item.op.isSameListAs(prev.items[0].item.op) }); return groupRootLists.map((v: TDataGroup | ListGroup[]) => { if (!Array.isArray(v)) { return v; } var litems = v.map((g: ListGroup): ListItem[] => g.items); return new ListGroup(litems._flatten()); }); } private convertListBlocksToListGroups(items: TDataGroup[]): Array<TDataGroup> { var grouped = items._groupConsecutiveElementsWhile((g: TDataGroup, gPrev: TDataGroup) => { return g instanceof BlockGroup && gPrev instanceof BlockGroup && g.op.isList() && gPrev.op.isList() && g.op.isSameListAs(gPrev.op) && g.op.hasSameIndentationAs(gPrev.op); }); return grouped.map((item: TDataGroup | BlockGroup[]) => { if (!Array.isArray(item)) { if (item instanceof BlockGroup && item.op.isList()) { return new ListGroup([new ListItem(item)]); } return item; } return new ListGroup(item.map((g) => new ListItem(g))); }); } private groupConsecutiveListGroups(items: TDataGroup[]): Array<TDataGroup | ListGroup[]> { return items._groupConsecutiveElementsWhile((curr: TDataGroup, prev: TDataGroup) => { return curr instanceof ListGroup && prev instanceof ListGroup; }); } private nestListSection(sectionItems: ListGroup[]): ListGroup[] { var indentGroups = this.groupByIndent(sectionItems); Object.keys(indentGroups).map(Number).sort().reverse().forEach((indent) => { indentGroups[indent].forEach((lg: ListGroup) => { var idx = sectionItems.indexOf(lg); if (this.placeUnderParent(lg, sectionItems.slice(0, idx))) { sectionItems.splice(idx, 1); } }); }); return sectionItems; } private groupByIndent(items: ListGroup[]): { [index: number]: ListGroup[] } { return items.reduce((pv: { [index: number]: ListGroup[] }, cv: ListGroup) => { var indent = cv.items[0].item.op.attributes.indent; if (indent) { pv[indent] = pv[indent] || []; pv[indent].push(cv); } return pv; }, {}); } private placeUnderParent(target: ListGroup, items: ListGroup[]) { for (var i = items.length - 1; i >= 0; i--) { var elm = items[i]; if (target.items[0].item.op.hasHigherIndentThan(elm.items[0].item.op)) { var parent = elm.items[elm.items.length - 1]; if (parent.innerList) { parent.innerList.items = parent.innerList.items.concat(target.items); } else { parent.innerList = target; } return true; } } return false; } } export { ListNester };