GeneIT/geneit_app/src/widgets/complex_family_tree/ComplexFamilyTree.tsx

272 lines
7.2 KiB
TypeScript
Raw Normal View History

2023-08-22 14:12:55 +00:00
import React from "react";
2023-08-22 15:17:36 +00:00
import f3, { f3Data } from "family-chart";
2023-08-22 14:12:55 +00:00
import "./family-chart.css";
2023-08-23 08:27:51 +00:00
import { FamilyTreeNode } from "../../utils/family_tree";
2023-08-23 08:58:25 +00:00
import { Member, fmtDate } from "../../api/MemberApi";
2023-08-23 09:25:33 +00:00
import { Couple } from "../../api/CoupleApi";
2023-08-23 12:02:05 +00:00
import { useDarkTheme } from "../../hooks/context_providers/DarkThemeProvider";
2023-08-23 12:56:50 +00:00
import { IconButton } from "@mui/material";
import PictureAsPdfIcon from "@mui/icons-material/PictureAsPdf";
import { jsPDF } from "jspdf";
import "svg2pdf.js";
2023-08-22 14:12:55 +00:00
2023-08-23 08:27:51 +00:00
export function ComplexFamilyTree(p: {
tree: FamilyTreeNode;
2023-08-23 08:58:25 +00:00
isUp: boolean;
2023-08-23 08:27:51 +00:00
}): React.ReactElement {
2023-08-23 12:02:05 +00:00
const darkTheme = useDarkTheme();
2023-08-23 13:18:23 +00:00
const applyTree = (container: HTMLDivElement) => {
if (!container) return;
2023-08-22 14:12:55 +00:00
const store = f3.createStore({
2023-08-23 08:58:25 +00:00
data: treeToF3Data(p.tree, p.isUp),
2023-08-22 14:12:55 +00:00
node_separation: 250,
level_separation: 150,
}),
view = f3.d3AnimationView({
store,
2023-08-23 13:18:23 +00:00
cont: container,
2023-08-22 14:12:55 +00:00
}),
Card = f3.elements.Card({
store,
svg: view.svg,
card_dim: {
2023-08-23 09:25:33 +00:00
w: 230,
2023-08-22 14:12:55 +00:00
h: 70,
text_x: 75,
text_y: 15,
img_w: 60,
img_h: 60,
img_x: 5,
img_y: 5,
},
card_display: [
2023-08-23 09:25:33 +00:00
(d) =>
`${d.data.first_name || ""} ${d.data.last_name || ""} ${
d.data.dead ? "✝" : ""
}`,
(d) => {
let s = `${d.data.birthday || ""} ${
d.data.deathday ? " ✝ " + d.data.deathday : ""
}`;
if (d.data.wedding_state || d.data.dateOfWedding) {
let weddingInfo = [];
if (d.data.wedding_state) weddingInfo.push(d.data.wedding_state);
if (d.data.dateOfWedding)
weddingInfo.push("Mariage : " + d.data.dateOfWedding);
s += `</tspan> <tspan x="0" dy="14" font-size="10">${weddingInfo.join(
" - "
)}`;
}
return s;
},
2023-08-22 14:12:55 +00:00
],
mini_tree: true,
link_break: false,
});
view.setCard(Card);
store.setOnUpdate((props) => view.update(props || {}));
2023-08-23 13:18:23 +00:00
store.update.tree({ initial: false, transition_time: 0 });
};
2023-08-22 14:12:55 +00:00
2023-08-23 12:56:50 +00:00
const exportPDF = async () => {
const doc = new jsPDF({
orientation: "l",
2023-08-23 13:32:02 +00:00
format: [351, 351],
2023-08-23 12:56:50 +00:00
});
// Clone the SVG to manipulate it
2023-08-23 13:18:23 +00:00
const container = document.createElement("div");
container.classList.add("f3", "f3-export");
document.body.appendChild(container);
applyTree(container);
const target = container.children[0];
await new Promise((res) => setTimeout(() => res(null), 100));
2023-08-23 12:56:50 +00:00
// SVG manipulations (adaptations to export)
2023-08-23 13:18:23 +00:00
let dstSVG = target.innerHTML.replaceAll(
2023-08-23 12:56:50 +00:00
`<path class="link" fill="none" stroke="#fff"`,
`<path class="link" fill="none" stroke="#000"`
);
dstSVG = dstSVG.replaceAll(
`class="card-body-rect"`,
`class="card-body-rect" fill="white"`
);
dstSVG = dstSVG.replaceAll(
`class="text-overflow-mask"`,
`class="text-overflow-mask" fill="transparent"`
);
2023-08-23 13:32:02 +00:00
dstSVG = dstSVG.replaceAll(`>UNKNOWN<`, `fill="#000">INCONNU<`);
2023-08-23 13:18:23 +00:00
target.innerHTML = dstSVG;
2023-08-23 12:56:50 +00:00
2023-08-23 13:18:23 +00:00
await doc.svg(target, {
2023-08-23 13:32:02 +00:00
height: 351,
width: 351,
2023-08-23 12:56:50 +00:00
});
2023-08-23 13:18:23 +00:00
container.remove();
2023-08-23 12:56:50 +00:00
// Save the created pdf
doc.save("myPDF.pdf");
};
2023-08-23 12:02:05 +00:00
return (
2023-08-23 12:56:50 +00:00
<div>
<div style={{ textAlign: "right" }}>
<IconButton onClick={exportPDF}>
<PictureAsPdfIcon />
</IconButton>
</div>
<div
className={`f3 ${darkTheme.enabled ? "f3-dark" : "f3-light"}`}
id="FamilyChart"
2023-08-23 13:18:23 +00:00
ref={applyTree}
2023-08-23 12:56:50 +00:00
></div>
</div>
2023-08-23 12:02:05 +00:00
);
}
2023-08-22 14:57:41 +00:00
2023-08-23 08:58:25 +00:00
function treeToF3Data(node: FamilyTreeNode, isUp: boolean): f3Data[] {
2023-08-23 08:27:51 +00:00
const availableMembers = new Set<number>();
getAvailableMembers(node, availableMembers);
const list: f3Data[] = [];
2023-08-23 09:25:33 +00:00
if (isUp) treeToF3DataUpRecurse(node, list, availableMembers);
else treeToF3DataDownRecurse(node, list, availableMembers);
2023-08-23 08:27:51 +00:00
return list;
}
function getAvailableMembers(t: FamilyTreeNode, s: Set<number>) {
s.add(t.member.id);
t.couples?.forEach((c) => {
s.add(c.member.id);
c.down.forEach((e) => getAvailableMembers(e, s));
});
t.down?.forEach((e) => getAvailableMembers(e, s));
}
2023-08-23 09:25:33 +00:00
function memberData(m: Member, c?: Couple): f3.f3DataData {
2023-08-23 08:58:25 +00:00
return {
first_name: m.first_name ?? "_",
last_name: m.last_name ?? "_",
gender: m.sex ?? "M",
avatar: m.thumbnailURL ?? undefined,
2023-08-23 09:25:33 +00:00
dead: m.dead,
2023-08-23 08:58:25 +00:00
birthday: m.dateOfBirth ? fmtDate(m.dateOfBirth) : undefined,
2023-08-23 09:25:33 +00:00
deathday: m.dateOfDeath ? fmtDate(m.dateOfDeath) : undefined,
wedding_state: c?.stateFr,
dateOfWedding: c?.dateOfWedding ? fmtDate(c?.dateOfWedding) : undefined,
2023-08-23 08:58:25 +00:00
};
}
function treeToF3DataUpRecurse(
node: FamilyTreeNode,
array: f3Data[],
availableMembers: Set<number>,
child?: number,
spouses?: number[]
) {
array.push({
data: memberData(node.member),
id: node.member.id.toString(),
rels: {
father:
node.member.father && availableMembers.has(node.member.father)
? node.member.father.toString()
: undefined,
mother:
node.member.mother && availableMembers.has(node.member.mother)
? node.member.mother.toString()
: undefined,
spouses: spouses
?.filter((c) => c !== node.member.id)
.map((c) => c.toString()),
children: child ? [child.toString()] : undefined,
},
});
const parentSpouses = node.down?.map((c) => c.member.id);
node.down?.forEach((d) =>
treeToF3DataUpRecurse(
d,
array,
availableMembers,
node.member.id,
parentSpouses
)
);
}
function treeToF3DataDownRecurse(
2023-08-23 08:27:51 +00:00
node: FamilyTreeNode,
array: f3Data[],
availableMembers: Set<number>
) {
// Get all members ids
const children = node?.down?.map((c) => c.member.id.toString()) ?? [];
node.couples?.map((c) =>
c.down.forEach((m) => children.push(m.member.id.toString()))
);
array.push({
2023-08-23 08:58:25 +00:00
data: memberData(node.member),
2023-08-23 08:27:51 +00:00
id: node.member.id.toString(),
2023-08-22 15:17:36 +00:00
rels: {
2023-08-23 08:27:51 +00:00
father:
node.member.father && availableMembers.has(node.member.father)
? node.member.father.toString()
: undefined,
mother:
node.member.mother && availableMembers.has(node.member.mother)
? node.member.mother.toString()
: undefined,
spouses: node.couples?.map((c) => c.member.id.toString()),
children: children,
2023-08-22 15:17:36 +00:00
},
2023-08-23 08:27:51 +00:00
});
2023-08-23 08:58:25 +00:00
node?.down?.forEach((e) =>
treeToF3DataDownRecurse(e, array, availableMembers)
);
2023-08-23 08:27:51 +00:00
if (node.couples) {
for (const c of node.couples) {
array.push({
2023-08-23 09:25:33 +00:00
data: memberData(c.member, c.couple),
2023-08-23 08:27:51 +00:00
id: c.member.id.toString(),
rels: {
father:
c.member.father && availableMembers.has(c.member.father)
? c.member.father.toString()
: undefined,
mother:
c.member.mother && availableMembers.has(c.member.mother)
? c.member.mother.toString()
: undefined,
spouses: [node.member.id.toString()],
children: c.down.map((c) => c.member.id.toString()),
},
});
2023-08-23 08:58:25 +00:00
c.down.forEach((e) =>
treeToF3DataDownRecurse(e, array, availableMembers)
);
2023-08-23 08:27:51 +00:00
}
}
}