xref: /webtrees/resources/views/lists/families-table.phtml (revision 054771e94ab58244acfab4da8ab0752d96fe7fb5)
1dd6b2bfcSGreg Roach<?php
2d70512abSGreg Roach
31b860509SRico Sonntagdeclare(strict_types=1);
41b860509SRico Sonntag
5*054771e9SGreg Roachuse Fisharebest\Webtrees\Age;
65373aac2SGreg Roachuse Fisharebest\Webtrees\Carbon;
71b860509SRico Sonntaguse Fisharebest\Webtrees\Date;
8*054771e9SGreg Roachuse Fisharebest\Webtrees\Family;
91b860509SRico Sonntaguse Fisharebest\Webtrees\GedcomTag;
101b860509SRico Sonntaguse Fisharebest\Webtrees\I18N;
111b860509SRico Sonntaguse Fisharebest\Webtrees\Individual;
12*054771e9SGreg Roachuse Fisharebest\Webtrees\Tree;
131b860509SRico Sonntaguse Fisharebest\Webtrees\View;
14*054771e9SGreg Roachuse Illuminate\Support\Collection;
151b860509SRico Sonntaguse Ramsey\Uuid\Uuid;
161b860509SRico Sonntag
17dd6b2bfcSGreg Roach$table_id = 'table-fam-' . Uuid::uuid4()->toString(); // lists requires a unique ID in case there are multiple lists per page
181b860509SRico Sonntag
19b9597e06SGreg Roach$today_jd          = Carbon::now()->julianDay();
205373aac2SGreg Roach$hundred_years_ago = Carbon::now()->subYears(100)->julianDay();
215373aac2SGreg Roach
22*054771e9SGreg Roach/**
23*054771e9SGreg Roach * @var Tree               $tree
24*054771e9SGreg Roach * @var Collection<Family> $families
25*054771e9SGreg Roach */
26*054771e9SGreg Roach
27dd6b2bfcSGreg Roach?>
28dd6b2bfcSGreg Roach
29dd6b2bfcSGreg Roach<?php View::push('javascript') ?>
30dd6b2bfcSGreg Roach<script>
3132a5dd8dSGreg Roach$("#<?= e($table_id) ?> > .wt-table-family").dataTable({
32dd6b2bfcSGreg Roach    processing: true,
33dd6b2bfcSGreg Roach    retrieve: true,
34dd6b2bfcSGreg Roach    columns: [
35dd6b2bfcSGreg Roach        /* Given names       */ { type: "text" },
36dd6b2bfcSGreg Roach        /* Surnames          */ { type: "text" },
37dd6b2bfcSGreg Roach        /* Age               */ { type: "num" },
38dd6b2bfcSGreg Roach        /* Given names       */ { type: "text" },
39dd6b2bfcSGreg Roach        /* Surnames          */ { type: "text" },
40dd6b2bfcSGreg Roach        /* Age               */ { type: "num" },
41dd6b2bfcSGreg Roach        /* Marriage date     */ { type: "num" },
42dd6b2bfcSGreg Roach        /* Anniversary       */ { type: "num" },
43dd6b2bfcSGreg Roach        /* Marriage place    */ { type: "text" },
44dd6b2bfcSGreg Roach        /* Children          */ { type: "num" },
45dd6b2bfcSGreg Roach        /* Last change       */ { visible: <?= json_encode((bool) $tree->getPreference('SHOW_LAST_CHANGE')) ?> },
46dd6b2bfcSGreg Roach        /* Filter marriage   */ { sortable: false },
47dd6b2bfcSGreg Roach        /* Filter alive/dead */ { sortable: false },
48dd6b2bfcSGreg Roach        /* Filter tree       */ { sortable: false }
49dd6b2bfcSGreg Roach    ],
501b860509SRico Sonntag    sorting: [
511b860509SRico Sonntag        [1, "asc"]
52b6c326d8SGreg Roach    ]
531b860509SRico Sonntag});
541b860509SRico Sonntag
551b860509SRico Sonntag$("#<?= e($table_id) ?>")
56dd6b2bfcSGreg Roach    /* Hide/show parents */
574843b94fSGreg Roach    .on("click", "#btn-toggle-parents", function() {
585e6816beSGreg Roach        $(".wt-individual-list-parents").slideToggle();
59dd6b2bfcSGreg Roach    })
60dd6b2bfcSGreg Roach    /* Hide/show statistics */
614843b94fSGreg Roach    .on("click", "#btn-toggle-statistics", function() {
621b860509SRico Sonntag        $("#family-charts-<?= e($table_id) ?>").slideToggle({
631b860509SRico Sonntag            complete: function () {
641b860509SRico Sonntag                // Trigger resize to redraw the chart
651b860509SRico Sonntag                $('div[id^="google-chart-"]').resize();
661b860509SRico Sonntag            }
671b860509SRico Sonntag        });
68dd6b2bfcSGreg Roach    })
69dd6b2bfcSGreg Roach    /* Filter buttons in table header */
70604bfd4bSGreg Roach    .on("click", "input[data-filter-column]", function() {
71604bfd4bSGreg Roach        let checkbox = $(this);
72604bfd4bSGreg Roach        let siblings = checkbox.parent().siblings();
731b860509SRico Sonntag
74604bfd4bSGreg Roach        // Deselect other options
75604bfd4bSGreg Roach        siblings.children().prop("checked", false).removeAttr("checked");
76604bfd4bSGreg Roach        siblings.removeClass('active');
77604bfd4bSGreg Roach
78604bfd4bSGreg Roach        // Apply (or clear) this filter
79604bfd4bSGreg Roach        let checked = checkbox.prop("checked");
80604bfd4bSGreg Roach        let filter  = checked ? checkbox.data("filter-value") : "";
8132a5dd8dSGreg Roach        let column  = $("#<?= e($table_id) ?> .wt-table-family").DataTable().column(checkbox.data("filter-column"));
82604bfd4bSGreg Roach        column.search(filter).draw();
83604bfd4bSGreg Roach    });
84dd6b2bfcSGreg Roach</script>
85dd6b2bfcSGreg Roach<?php View::endpush() ?>
86dd6b2bfcSGreg Roach
87dd6b2bfcSGreg Roach<?php
88dd6b2bfcSGreg Roach$max_age = (int) $tree->getPreference('MAX_ALIVE_AGE');
89dd6b2bfcSGreg Roach
90dd6b2bfcSGreg Roach// init chart data
91dd6b2bfcSGreg Roach$marr_by_age = [];
92dd6b2bfcSGreg Roachfor ($age = 0; $age <= $max_age; $age++) {
931b860509SRico Sonntag    $marr_by_age[$age]['M'] = 0;
941b860509SRico Sonntag    $marr_by_age[$age]['F'] = 0;
951b860509SRico Sonntag    $marr_by_age[$age]['U'] = 0;
96dd6b2bfcSGreg Roach}
97dd6b2bfcSGreg Roach$birt_by_decade = [];
98dd6b2bfcSGreg Roach$marr_by_decade = [];
991b860509SRico Sonntagfor ($year = 1400; $year < 2050; $year += 10) {
1001b860509SRico Sonntag    $birt_by_decade[$year]['M'] = 0;
1011b860509SRico Sonntag    $birt_by_decade[$year]['F'] = 0;
1021b860509SRico Sonntag    $birt_by_decade[$year]['U'] = 0;
1031b860509SRico Sonntag    $marr_by_decade[$year]['M'] = 0;
1041b860509SRico Sonntag    $marr_by_decade[$year]['F'] = 0;
1051b860509SRico Sonntag    $marr_by_decade[$year]['U'] = 0;
106dd6b2bfcSGreg Roach}
1071b860509SRico Sonntag
1081b860509SRico Sonntag$birthData = [
1091b860509SRico Sonntag    [
1101b860509SRico Sonntag        [
1111b860509SRico Sonntag            'label' => I18N::translate('Century'),
1121b860509SRico Sonntag            'type'  => 'date',
1131b860509SRico Sonntag        ], [
1141b860509SRico Sonntag            'label' => I18N::translate('Males'),
1151b860509SRico Sonntag            'type'  => 'number',
1161b860509SRico Sonntag        ], [
1171b860509SRico Sonntag            'label' => I18N::translate('Females'),
1181b860509SRico Sonntag            'type'  => 'number',
1191b860509SRico Sonntag        ],
1201b860509SRico Sonntag    ]
1211b860509SRico Sonntag];
1221b860509SRico Sonntag
1231b860509SRico Sonntag$marriageData = [
1241b860509SRico Sonntag    [
1251b860509SRico Sonntag        [
1261b860509SRico Sonntag            'label' => I18N::translate('Century'),
1271b860509SRico Sonntag            'type'  => 'date',
1281b860509SRico Sonntag        ], [
1291b860509SRico Sonntag            'label' => I18N::translate('Males'),
1301b860509SRico Sonntag            'type'  => 'number',
1311b860509SRico Sonntag        ], [
1321b860509SRico Sonntag            'label' => I18N::translate('Females'),
1331b860509SRico Sonntag            'type'  => 'number',
1341b860509SRico Sonntag        ],
1351b860509SRico Sonntag    ]
1361b860509SRico Sonntag];
1371b860509SRico Sonntag
1381b860509SRico Sonntag$marriageAgeData = [
1391b860509SRico Sonntag    [
1401b860509SRico Sonntag        I18N::translate('Age'),
1411b860509SRico Sonntag        I18N::translate('Males'),
1421b860509SRico Sonntag        I18N::translate('Females'),
1431b860509SRico Sonntag        I18N::translate('Average age'),
1441b860509SRico Sonntag    ]
1451b860509SRico Sonntag];
1461b860509SRico Sonntag
147dd6b2bfcSGreg Roach?>
148dd6b2bfcSGreg Roach
14932a5dd8dSGreg Roach<div id="<?= e($table_id) ?>">
15032a5dd8dSGreg Roach    <table class="table table-bordered table-sm wt-table-family"
151b6c326d8SGreg Roach        <?= view('lists/datatables-attributes') ?>
152b6c326d8SGreg Roach    >
153dd6b2bfcSGreg Roach        <thead>
154dd6b2bfcSGreg Roach            <tr>
155dd6b2bfcSGreg Roach                <th colspan="14">
156dd6b2bfcSGreg Roach                    <div class="btn-toolbar d-flex justify-content-between mb-2">
157604bfd4bSGreg Roach                        <div class="btn-group btn-group-toggle btn-group-sm" data-toggle="buttons">
158af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary btn-sm" title="' . I18N::translate('Show individuals who are alive or couples where both partners are alive.') ?>">
159604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="12" data-filter-value="N">
160dd6b2bfcSGreg Roach                                <?= I18N::translate('Both alive') ?>
161604bfd4bSGreg Roach                            </label>
162af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show couples where only the female partner is dead.') ?>">
163604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="12" data-filter-value="W">
164dd6b2bfcSGreg Roach                                <?= I18N::translate('Widower') ?>
165604bfd4bSGreg Roach                            </label>
166af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show couples where only the male partner is dead.') ?>">
167604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="12" data-filter-value="H">
168dd6b2bfcSGreg Roach                                <?= I18N::translate('Widow') ?>
169604bfd4bSGreg Roach                            </label>
170af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show individuals who are dead or couples where both partners are dead.') ?>">
171604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="12" data-filter-value="Y">
172dd6b2bfcSGreg Roach                                <?= I18N::translate('Both dead') ?>
173604bfd4bSGreg Roach                            </label>
174dd6b2bfcSGreg Roach                        </div>
175604bfd4bSGreg Roach
176604bfd4bSGreg Roach                        <div class="btn-group btn-group-toggle btn-group-sm" data-toggle="buttons">
177af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show “roots” couples or individuals. These individuals may also be called “patriarchs”. They are individuals who have no parents recorded in the database.') ?>">
178604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="13" data-filter-value="R">
179dd6b2bfcSGreg Roach                                <?= I18N::translate('Roots') ?>
180604bfd4bSGreg Roach                            </label>
181af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show “leaves” couples or individuals. These are individuals who are alive but have no children recorded in the database.') ?>">
182604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="13" data-filter-value="L">
183dd6b2bfcSGreg Roach                                <?= I18N::translate('Leaves') ?>
184604bfd4bSGreg Roach                            </label>
185dd6b2bfcSGreg Roach                        </div>
186604bfd4bSGreg Roach
187604bfd4bSGreg Roach                        <div class="btn-group btn-group-toggle btn-group-sm" data-toggle="buttons">
188af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show couples with an unknown marriage date.') ?>">
189604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="11" data-filter-value="U">
190b9597e06SGreg Roach                                <?= I18N::translate('Not married') ?>
191604bfd4bSGreg Roach                            </label>
192af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show couples who married more than 100 years ago.') ?>">
193604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="11" data-filter-value="YES">
194dd6b2bfcSGreg Roach                                <?= I18N::translate('Marriage') ?>&gt;100
195604bfd4bSGreg Roach                            </label>
196af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show couples who married within the last 100 years.') ?>">
197604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="11" data-filter-value="Y100">
198dd6b2bfcSGreg Roach                                <?= I18N::translate('Marriage') ?>&lt;=100
199604bfd4bSGreg Roach                            </label>
200af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show divorced couples.') ?>">
201604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="11" data-filter-value="D">
202dd6b2bfcSGreg Roach                                <?= I18N::translate('Divorce') ?>
203604bfd4bSGreg Roach                            </label>
204af8b52f0SGreg Roach                            <label class="btn btn-outline-secondary" title="<?= I18N::translate('Show couples where either partner married more than once.') ?>">
205604bfd4bSGreg Roach                                <input type="checkbox" data-filter-column="11" data-filter-value="M">
206dd6b2bfcSGreg Roach                                <?= I18N::translate('Multiple marriages') ?>
207604bfd4bSGreg Roach                            </label>
208dd6b2bfcSGreg Roach                        </div>
209dd6b2bfcSGreg Roach                    </div>
210dd6b2bfcSGreg Roach                </th>
211dd6b2bfcSGreg Roach            </tr>
212dd6b2bfcSGreg Roach            <tr>
213dd6b2bfcSGreg Roach                <th><?= I18N::translate('Given names') ?></th>
214dd6b2bfcSGreg Roach                <th><?= I18N::translate('Surname') ?></th>
215dd6b2bfcSGreg Roach                <th><?= I18N::translate('Age') ?></th>
216dd6b2bfcSGreg Roach                <th><?= I18N::translate('Given names') ?></th>
217dd6b2bfcSGreg Roach                <th><?= I18N::translate('Surname') ?></th>
218dd6b2bfcSGreg Roach                <th><?= I18N::translate('Age') ?></th>
219dd6b2bfcSGreg Roach                <th><?= I18N::translate('Marriage') ?></th>
220e39fd5c6SGreg Roach                <th>
221e39fd5c6SGreg Roach                    <span title="<?= I18N::translate('Anniversary') ?>">
222e39fd5c6SGreg Roach                        <?= view('icons/anniversary') ?>
223e39fd5c6SGreg Roach                    </span>
224e39fd5c6SGreg Roach                </th>
225dd6b2bfcSGreg Roach                <th><?= I18N::translate('Place') ?></th>
226dd6b2bfcSGreg Roach                <th><i class="icon-children" title="<?= I18N::translate('Children') ?>"></i></th>
227dd6b2bfcSGreg Roach                <th><?= I18N::translate('Last change') ?></th>
228dd6b2bfcSGreg Roach                <th hidden></th>
229dd6b2bfcSGreg Roach                <th hidden></th>
230dd6b2bfcSGreg Roach                <th hidden></th>
231dd6b2bfcSGreg Roach            </tr>
232dd6b2bfcSGreg Roach        </thead>
233dd6b2bfcSGreg Roach
234dd6b2bfcSGreg Roach        <tbody>
235dd6b2bfcSGreg Roach        <?php foreach ($families as $family) : ?>
23639ca88baSGreg Roach            <?php $husb = $family->husband() ?? new Individual('H', '0 @H@ INDI', null, $family->tree()) ?>
23739ca88baSGreg Roach            <?php $wife = $family->wife() ?? new Individual('W', '0 @W@ INDI', null, $family->tree()) ?>
238dd6b2bfcSGreg Roach
23917dd427eSGreg Roach            <tr class="<?= $family->isPendingDeletion() ? 'wt-old' : ($family->isPendingAddition() ? 'wt-new' : '') ?>">
240dd6b2bfcSGreg Roach                <!-- Husband name -->
24139ca88baSGreg Roach                <td colspan="2" data-sort="<?= e(str_replace([',', '@P.N.', '@N.N.'], 'AAAA', implode(',', array_reverse(explode(',', $husb->sortName()))))) ?>">
242dd6b2bfcSGreg Roach                    <?php foreach ($husb->getAllNames() as $num => $name) : ?>
24322d65e5aSGreg Roach                        <?php if ($name['type'] !== '_MARNM' || $num == $husb->getPrimaryName()) : ?>
244e840bf40SGreg Roach                        <a title="<?= $name['type'] === 'NAME' ? '' : strip_tags(GedcomTag::getLabel($name['type'], $husb)) ?>" href="<?= e($family->url()) ?>" class="<?= $num === $husb->getPrimaryName() ? 'name2' : '' ?>">
245dd6b2bfcSGreg Roach                            <?= $name['full'] ?>
246dd6b2bfcSGreg Roach                        </a>
247dd6b2bfcSGreg Roach                            <?php if ($num === $husb->getPrimaryName()) : ?>
24808362db4SGreg Roach                                <small><?= view('icons/sex', ['sex' => $husb->sex()]) ?></small>
249dd6b2bfcSGreg Roach                            <?php endif ?>
250dd6b2bfcSGreg Roach                        <br>
251dd6b2bfcSGreg Roach                        <?php endif ?>
252dd6b2bfcSGreg Roach                    <?php endforeach ?>
2535e6816beSGreg Roach                    <?= view('lists/individual-table-parents', ['individual' => $husb]) ?>
254dd6b2bfcSGreg Roach                </td>
255dd6b2bfcSGreg Roach
25639ca88baSGreg Roach                <td hidden data-sort="<?= e(str_replace([',', '@P.N.', '@N.N.'], 'AAAA', $husb->sortName())) ?>"></td>
257dd6b2bfcSGreg Roach
258dd6b2bfcSGreg Roach                <!-- Husband age -->
259dd6b2bfcSGreg Roach                <?php
260dd6b2bfcSGreg Roach                $mdate = $family->getMarriageDate();
261dd6b2bfcSGreg Roach                $hdate = $husb->getBirthDate();
262*054771e9SGreg Roach
263dd6b2bfcSGreg Roach                if ($hdate->isOK() && $mdate->isOK()) {
264*054771e9SGreg Roach                    $gregorian_year = $hdate->gregorianYear();
265*054771e9SGreg Roach
266*054771e9SGreg Roach                    if ($gregorian_year >= 1550 && $gregorian_year < 2030) {
267*054771e9SGreg Roach                        ++$birt_by_decade[(int) ($gregorian_year / 10) * 10][$husb->sex()];
268dd6b2bfcSGreg Roach                    }
269*054771e9SGreg Roach
270*054771e9SGreg Roach                    $husband_age         = new Age($hdate, $mdate);
271*054771e9SGreg Roach                    $husband_age_sort    = $husband_age->agedays();
272*054771e9SGreg Roach                    $husband_age_years   = $husband_age->ageYears();
273*054771e9SGreg Roach                    $husband_age_display = I18N::number($husband_age_years);
274*054771e9SGreg Roach
275*054771e9SGreg Roach                    if ($husband_age_years >= 0 && $husband_age_years <= $max_age) {
276*054771e9SGreg Roach                        ++$marr_by_age[$husband_age_years][$husb->sex()];
277dd6b2bfcSGreg Roach                    }
278*054771e9SGreg Roach                } else {
279*054771e9SGreg Roach                    $husband_age_sort    = 0;
280*054771e9SGreg Roach                    $husband_age_display = '';
281dd6b2bfcSGreg Roach                }
282dd6b2bfcSGreg Roach                ?>
283*054771e9SGreg Roach                <td class="text-center" data-sort="<?= $husband_age_sort ?>">
284*054771e9SGreg Roach                    <?= $husband_age_display ?>
285dd6b2bfcSGreg Roach                </td>
286dd6b2bfcSGreg Roach
287dd6b2bfcSGreg Roach                <!-- Wife name -->
28839ca88baSGreg Roach                <td colspan="2" data-sort="<?= e(str_replace([',', '@P.N.', '@N.N.'], 'AAAA', implode(',', array_reverse(explode(',', $wife->sortName()))))) ?>">
289dd6b2bfcSGreg Roach                    <?php foreach ($wife->getAllNames() as $num => $name) : ?>
29022d65e5aSGreg Roach                        <?php if ($name['type'] !== '_MARNM' || $num == $wife->getPrimaryName()) : ?>
291e840bf40SGreg Roach                            <a title="<?= $name['type'] === 'NAME' ? '' : strip_tags(GedcomTag::getLabel($name['type'], $wife)) ?>" href="<?= e($family->url()) ?>" class="<?= $num === $wife->getPrimaryName() ? 'name2' : '' ?>">
292dd6b2bfcSGreg Roach                                <?= $name['full'] ?>
293dd6b2bfcSGreg Roach                            </a>
294dd6b2bfcSGreg Roach                            <?php if ($num === $wife->getPrimaryName()) : ?>
29508362db4SGreg Roach                                <small><?= view('icons/sex', ['sex' => $wife->sex()]) ?></small>
296dd6b2bfcSGreg Roach                            <?php endif ?>
297dd6b2bfcSGreg Roach                            <br>
298dd6b2bfcSGreg Roach                        <?php endif ?>
299dd6b2bfcSGreg Roach                    <?php endforeach ?>
3005e6816beSGreg Roach                    <?= view('lists/individual-table-parents', ['individual' => $wife]) ?>
301dd6b2bfcSGreg Roach                </td>
302dd6b2bfcSGreg Roach
30339ca88baSGreg Roach                <td hidden data-sort="<?= e(str_replace([',', '@P.N.', '@N.N.'], 'AAAA', $wife->sortName())) ?>"></td>
304dd6b2bfcSGreg Roach
305dd6b2bfcSGreg Roach                <!-- Wife age -->
306dd6b2bfcSGreg Roach                <?php
307dd6b2bfcSGreg Roach                $wdate = $wife->getBirthDate();
308dd6b2bfcSGreg Roach                if ($wdate->isOK() && $mdate->isOK()) {
309*054771e9SGreg Roach                    $gregorian_year = $wdate->gregorianYear();
310*054771e9SGreg Roach
311*054771e9SGreg Roach                    if ($gregorian_year >= 1550 && $gregorian_year < 2030) {
312*054771e9SGreg Roach                        ++$birt_by_decade[(int) ($gregorian_year / 10) * 10][$husb->sex()];
313dd6b2bfcSGreg Roach                    }
314*054771e9SGreg Roach
315*054771e9SGreg Roach                    $wife_age         = new Age($wdate, $mdate);
316*054771e9SGreg Roach                    $wife_age_sort    = $wife_age->ageDays();
317*054771e9SGreg Roach                    $wife_age_years   = $wife_age->ageYears();
318*054771e9SGreg Roach                    $wife_age_display = I18N::number($wife_age_years);
319*054771e9SGreg Roach
320*054771e9SGreg Roach                    if ($wife_age_years >= 0 && $wife_age_years <= $max_age) {
321*054771e9SGreg Roach                        ++$marr_by_age[$wife_age_years][$husb->sex()];
322dd6b2bfcSGreg Roach                    }
323*054771e9SGreg Roach                } else {
324*054771e9SGreg Roach                    $wife_age_sort    = 0;
325*054771e9SGreg Roach                    $wife_age_display = '';
326dd6b2bfcSGreg Roach                }
327dd6b2bfcSGreg Roach                ?>
328*054771e9SGreg Roach                <td class="text-center" data-sort="<?= $wife_age_sort ?>">
329*054771e9SGreg Roach                    <?= $wife_age_display ?>
330dd6b2bfcSGreg Roach                </td>
331dd6b2bfcSGreg Roach
332dd6b2bfcSGreg Roach                <!-- Marriage date -->
333dd6b2bfcSGreg Roach                <td data-sort="<?= $family->getMarriageDate()->julianDay() ?>">
334dd6b2bfcSGreg Roach                    <?php if ($marriage_dates = $family->getAllMarriageDates()) : ?>
335dd6b2bfcSGreg Roach                        <?php foreach ($marriage_dates as $n => $marriage_date) : ?>
336dd6b2bfcSGreg Roach                            <div><?= $marriage_date->display(true) ?></div>
337dd6b2bfcSGreg Roach                        <?php endforeach ?>
338dd6b2bfcSGreg Roach                        <?php if ($marriage_dates[0]->gregorianYear() >= 1550 && $marriage_dates[0]->gregorianYear() < 2030) : ?>
3391b860509SRico Sonntag                            <?php
3401b860509SRico Sonntag                                ++$marr_by_decade[(int) ($marriage_dates[0]->gregorianYear() / 10) * 10][$husb->sex()];
3411b860509SRico Sonntag                                ++$marr_by_decade[(int) ($marriage_dates[0]->gregorianYear() / 10) * 10][$wife->sex()];
3421b860509SRico Sonntag                            ?>
343dd6b2bfcSGreg Roach                        <?php endif ?>
34439ca88baSGreg Roach                    <?php elseif ($family->facts(['_NMR'])->isNotEmpty()) : ?>
345dd6b2bfcSGreg Roach                        <?= I18N::translate('no') ?>
34639ca88baSGreg Roach                    <?php elseif ($family->facts(['MARR'])->isNotEmpty()) : ?>
347dd6b2bfcSGreg Roach                            <?= I18N::translate('yes') ?>
348dd6b2bfcSGreg Roach                    <?php endif ?>
349dd6b2bfcSGreg Roach                </td>
350dd6b2bfcSGreg Roach
351dd6b2bfcSGreg Roach                <!-- Marriage anniversary -->
352*054771e9SGreg Roach                <?php
353*054771e9SGreg Roach                    $marriage_date = $family->getMarriageDate();
354*054771e9SGreg Roach                    if ($marriage_date->isOK()) {
355*054771e9SGreg Roach                        $marriage_anniversary         = new Age($marriage_date, new Date(strtoupper(date('d M Y'))));
356*054771e9SGreg Roach                        $marriage_anniversary_sort    = $marriage_anniversary->ageDays();
357*054771e9SGreg Roach                        $marriage_anniversary_display = I18N::number($marriage_anniversary->ageYears());
358*054771e9SGreg Roach                    } else {
359*054771e9SGreg Roach                        $marriage_anniversary_sort    = 0;
360*054771e9SGreg Roach                        $marriage_anniversary_display = '';
361*054771e9SGreg Roach                    }
362*054771e9SGreg Roach                ?>
363*054771e9SGreg Roach                <td class="text-center" data-sort="<?= $marriage_anniversary_sort ?>">
364*054771e9SGreg Roach                    <?= $marriage_anniversary_display ?>
365dd6b2bfcSGreg Roach                </td>
366dd6b2bfcSGreg Roach
367dd6b2bfcSGreg Roach                <!-- Marriage place -->
368dd6b2bfcSGreg Roach                <td>
369dd6b2bfcSGreg Roach                    <?php foreach ($family->getAllMarriagePlaces() as $n => $marriage_place) : ?>
370392561bbSGreg Roach                        <?= $marriage_place->shortName(true) ?>
371dd6b2bfcSGreg Roach                        <br>
372dd6b2bfcSGreg Roach                    <?php endforeach ?>
373dd6b2bfcSGreg Roach                </td>
374dd6b2bfcSGreg Roach
375dd6b2bfcSGreg Roach                <!-- Number of children -->
37639ca88baSGreg Roach                <td class="text-center" data-sort="<?= $family->numberOfChildren() ?>">
37739ca88baSGreg Roach                    <?= I18N::number($family->numberOfChildren()) ?>
378dd6b2bfcSGreg Roach                </td>
379dd6b2bfcSGreg Roach
380dd6b2bfcSGreg Roach                <!-- Last change -->
3814459dc9aSGreg Roach                <td data-sort="<?= $family->lastChangeTimestamp()->unix() ?>">
3824459dc9aSGreg Roach                    <?= view('components/datetime', ['timestamp' => $family->lastChangeTimestamp()]) ?>
383dd6b2bfcSGreg Roach                </td>
384dd6b2bfcSGreg Roach
385dd6b2bfcSGreg Roach                <!-- Filter by marriage date -->
386dd6b2bfcSGreg Roach                <td hidden>
387b9597e06SGreg Roach                    <?php if ($mdate->maximumJulianDay() > $hundred_years_ago && $mdate->maximumJulianDay() <= $today_jd) : ?>
388dd6b2bfcSGreg Roach                        Y100
389b9597e06SGreg Roach                    <?php elseif ($family->facts(['MARR'])->isNotEmpty()) : ?>
390dd6b2bfcSGreg Roach                        YES
391b9597e06SGreg Roach                    <?php else : ?>
392b9597e06SGreg Roach                        U
393dd6b2bfcSGreg Roach                    <?php endif ?>
394b9597e06SGreg Roach                    <?php if ($family->facts(['DIV'])->isNotEmpty()) : ?>
395dd6b2bfcSGreg Roach                        D
396dd6b2bfcSGreg Roach                    <?php endif ?>
39739ca88baSGreg Roach                    <?php if (count($husb->spouseFamilies()) > 1 || count($wife->spouseFamilies()) > 1) : ?>
398dd6b2bfcSGreg Roach                        M
399dd6b2bfcSGreg Roach                    <?php endif ?>
400dd6b2bfcSGreg Roach                </td>
401dd6b2bfcSGreg Roach
402dd6b2bfcSGreg Roach                <!-- Filter by alive/dead -->
403dd6b2bfcSGreg Roach                <td hidden>
404dd6b2bfcSGreg Roach                    <?php if ($husb->isDead() && $wife->isDead()) : ?>
405dd6b2bfcSGreg Roach                        Y
406dd6b2bfcSGreg Roach                    <?php endif ?>
407dd6b2bfcSGreg Roach                    <?php if ($husb->isDead() && !$wife->isDead()) : ?>
40822d65e5aSGreg Roach                        <?php if ($wife->sex() === 'F') : ?>
409dd6b2bfcSGreg Roach                            H
410dd6b2bfcSGreg Roach                        <?php endif ?>
41122d65e5aSGreg Roach                        <?php if ($wife->sex() === 'M') : ?>
412dd6b2bfcSGreg Roach                            W
413dd6b2bfcSGreg Roach                        <?php endif ?>
414dd6b2bfcSGreg Roach                    <?php endif ?>
415dd6b2bfcSGreg Roach                    <?php if (!$husb->isDead() && $wife->isDead()) : ?>
41622d65e5aSGreg Roach                        <?php if ($husb->sex() === 'M') : ?>
417dd6b2bfcSGreg Roach                            W
418dd6b2bfcSGreg Roach                        <?php endif ?>
41922d65e5aSGreg Roach                        <?php if ($husb->sex() === 'F') : ?>
420dd6b2bfcSGreg Roach                            H
421dd6b2bfcSGreg Roach                        <?php endif ?>
422dd6b2bfcSGreg Roach                    <?php endif ?>
423dd6b2bfcSGreg Roach                    <?php if (!$husb->isDead() && !$wife->isDead()) : ?>
424dd6b2bfcSGreg Roach                        N
425dd6b2bfcSGreg Roach                    <?php endif ?>
426dd6b2bfcSGreg Roach                </td>
427dd6b2bfcSGreg Roach
428dd6b2bfcSGreg Roach                <!-- Filter by roots/leaves -->
429dd6b2bfcSGreg Roach                <td hidden>
43039ca88baSGreg Roach                    <?php if (!$husb->childFamilies() && !$wife->childFamilies()) : ?>
431dd6b2bfcSGreg Roach                        R
43239ca88baSGreg Roach                    <?php elseif (!$husb->isDead() && !$wife->isDead() && $family->numberOfChildren() === 0) : ?>
433dd6b2bfcSGreg Roach                        L
434dd6b2bfcSGreg Roach                    <?php endif ?>
435dd6b2bfcSGreg Roach                </td>
436dd6b2bfcSGreg Roach            </tr>
437dd6b2bfcSGreg Roach        <?php endforeach ?>
438dd6b2bfcSGreg Roach        </tbody>
4397039fd97SGreg Roach
4407039fd97SGreg Roach        <tfoot>
4417039fd97SGreg Roach            <tr>
4427039fd97SGreg Roach                <th colspan="14">
443604bfd4bSGreg Roach                    <div class="btn-group btn-group-sm">
444af8b52f0SGreg Roach                        <button id="btn-toggle-parents" class="btn btn-outline-secondary" data-toggle="button" data-persist="show-parents">
4457039fd97SGreg Roach                            <?= I18N::translate('Show parents') ?>
4467039fd97SGreg Roach                        </button>
447af8b52f0SGreg Roach                        <button id="btn-toggle-statistics" class="btn btn-outline-secondary" data-toggle="button" data-persist="show-statistics">
4487039fd97SGreg Roach                            <?= I18N::translate('Show statistics charts') ?>
4497039fd97SGreg Roach                        </button>
4507039fd97SGreg Roach                    </div>
4517039fd97SGreg Roach                </th>
4527039fd97SGreg Roach            </tr>
4537039fd97SGreg Roach        </tfoot>
454dd6b2bfcSGreg Roach    </table>
4551b860509SRico Sonntag</div>
456dd6b2bfcSGreg Roach
4571b860509SRico Sonntag<div id="family-charts-<?= e($table_id) ?>" style="display: none;">
4581b860509SRico Sonntag    <div class="mb-3">
4591b860509SRico Sonntag        <div class="card-deck">
4601b860509SRico Sonntag            <div class="col-lg-12 col-md-12 mb-3">
4611b860509SRico Sonntag                <div class="card m-0">
4621b860509SRico Sonntag                    <div class="card-header">
4631b860509SRico Sonntag                        <?= I18N::translate('Decade of birth') ?>
4641b860509SRico Sonntag                    </div><div class="card-body">
4651b860509SRico Sonntag                        <?php
4661b860509SRico Sonntag                        foreach ($birt_by_decade as $century => $values) {
4671b860509SRico Sonntag                            if (($values['M'] + $values['F']) > 0) {
4681b860509SRico Sonntag                                $birthData[] = [
4691b860509SRico Sonntag                                    [
4701b860509SRico Sonntag                                        'v' => 'Date(' . $century . ', 0, 1)',
4711b860509SRico Sonntag                                        'f' => $century,
4721b860509SRico Sonntag                                    ],
4731b860509SRico Sonntag                                    $values['M'],
4741b860509SRico Sonntag                                    $values['F'],
4751b860509SRico Sonntag                                ];
4761b860509SRico Sonntag                            }
4771b860509SRico Sonntag                        }
4781b860509SRico Sonntag                        ?>
4791b860509SRico Sonntag                        <?= view('lists/chart-by-decade', ['data' => $birthData, 'title' => I18N::translate('Decade of birth')]) ?>
4801b860509SRico Sonntag                    </div>
4811b860509SRico Sonntag                </div>
4821b860509SRico Sonntag            </div>
4831b860509SRico Sonntag        </div>
4841b860509SRico Sonntag        <div class="card-deck">
4851b860509SRico Sonntag            <div class="col-lg-12 col-md-12 mb-3">
4861b860509SRico Sonntag                <div class="card m-0">
4871b860509SRico Sonntag                    <div class="card-header">
4881b860509SRico Sonntag                        <?= I18N::translate('Decade of marriage') ?>
4891b860509SRico Sonntag                    </div><div class="card-body">
4901b860509SRico Sonntag                        <?php
4911b860509SRico Sonntag                        foreach ($marr_by_decade as $century => $values) {
4921b860509SRico Sonntag                            if (($values['M'] + $values['F']) > 0) {
4931b860509SRico Sonntag                                $marriageData[] = [
4941b860509SRico Sonntag                                    [
4951b860509SRico Sonntag                                        'v' => 'Date(' . $century . ', 0, 1)',
4961b860509SRico Sonntag                                        'f' => $century,
4971b860509SRico Sonntag                                    ],
4981b860509SRico Sonntag                                    $values['M'],
4991b860509SRico Sonntag                                    $values['F'],
5001b860509SRico Sonntag                                ];
5011b860509SRico Sonntag                            }
5021b860509SRico Sonntag                        }
5031b860509SRico Sonntag                        ?>
5041b860509SRico Sonntag                        <?= view('lists/chart-by-decade', ['data' => $marriageData, 'title' => I18N::translate('Decade of marriage')]) ?>
5051b860509SRico Sonntag                    </div>
5061b860509SRico Sonntag                </div>
5071b860509SRico Sonntag            </div>
5081b860509SRico Sonntag        </div>
5091b860509SRico Sonntag        <div class="card-deck">
5101b860509SRico Sonntag            <div class="col-lg-12 col-md-12 mb-3">
5111b860509SRico Sonntag                <div class="card m-0">
5121b860509SRico Sonntag                    <div class="card-header">
5131b860509SRico Sonntag                        <?= I18N::translate('Age in year of marriage') ?>
5141b860509SRico Sonntag                    </div>
5151b860509SRico Sonntag                    <div class="card-body">
5161b860509SRico Sonntag                        <?php
5171b860509SRico Sonntag                            $totalAge = 0;
5181b860509SRico Sonntag                            $totalSum = 0;
5191b860509SRico Sonntag                            $max      = 0;
5201b860509SRico Sonntag
5211b860509SRico Sonntag                        foreach ($marr_by_age as $age => $values) {
5221b860509SRico Sonntag                            if (($values['M'] + $values['F']) > 0) {
5231b860509SRico Sonntag                                if (($values['M'] + $values['F']) > $max) {
5241b860509SRico Sonntag                                    $max = $values['M'] + $values['F'];
5251b860509SRico Sonntag                                }
5261b860509SRico Sonntag
5271b860509SRico Sonntag                                $totalAge += $age * ($values['M'] + $values['F']);
5281b860509SRico Sonntag                                $totalSum += $values['M'] + $values['F'];
5291b860509SRico Sonntag
5301b860509SRico Sonntag                                $marriageAgeData[] = [
5311b860509SRico Sonntag                                    $age,
5321b860509SRico Sonntag                                    $values['M'],
5331b860509SRico Sonntag                                    $values['F'],
5341b860509SRico Sonntag                                    null,
5351b860509SRico Sonntag                                ];
5361b860509SRico Sonntag                            }
5371b860509SRico Sonntag                        }
5381b860509SRico Sonntag
5391b860509SRico Sonntag                        if ($totalSum > 0) {
5401b860509SRico Sonntag                            $marriageAgeData[] = [
5411b860509SRico Sonntag                                round($totalAge / $totalSum, 1),
5421b860509SRico Sonntag                                null,
5431b860509SRico Sonntag                                null,
5441b860509SRico Sonntag                                0,
5451b860509SRico Sonntag                            ];
5461b860509SRico Sonntag
5471b860509SRico Sonntag                            $marriageAgeData[] = [
5481b860509SRico Sonntag                                round($totalAge / $totalSum, 1),
5491b860509SRico Sonntag                                null,
5501b860509SRico Sonntag                                null,
5511b860509SRico Sonntag                                $max,
5521b860509SRico Sonntag                            ];
5531b860509SRico Sonntag                        }
5541b860509SRico Sonntag                        ?>
5551b860509SRico Sonntag                        <?= view('lists/chart-by-age', ['data' => $marriageAgeData, 'title' => I18N::translate('Age in year of marriage')]) ?>
5561b860509SRico Sonntag                    </div>
5571b860509SRico Sonntag                </div>
5581b860509SRico Sonntag            </div>
5591b860509SRico Sonntag        </div>
560dd6b2bfcSGreg Roach    </div>
561dd6b2bfcSGreg Roach</div>
562