Total Complexity | 257 |
Total Lines | 1954 |
Duplicated Lines | 0 % |
Changes | 0 |
Complex classes like FunctionsPrintLists often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes.
Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.
While breaking up the class, it is a good idea to analyze how other classes use FunctionsPrintLists, and based on these observations, apply Extract Interface, too.
1 | <?php |
||
39 | class FunctionsPrintLists |
||
40 | { |
||
41 | /** |
||
42 | * Generate a SURN,GIVN and GIVN,SURN sortable name for an individual. |
||
43 | * This allows table data to sort by surname or given names. |
||
44 | * |
||
45 | * Use AAAA as a separator (instead of ","), as Javascript localeCompare() |
||
46 | * ignores punctuation and "ANN,ROACH" would sort after "ANNE,ROACH", |
||
47 | * instead of before it. |
||
48 | * |
||
49 | * @param Individual $individual |
||
50 | * |
||
51 | * @return string[] |
||
52 | */ |
||
53 | private static function sortableNames(Individual $individual) |
||
54 | { |
||
55 | $names = $individual->getAllNames(); |
||
56 | $primary = $individual->getPrimaryName(); |
||
57 | |||
58 | list($surn, $givn) = explode(',', $names[$primary]['sort']); |
||
59 | |||
60 | $givn = str_replace('@P.N.', 'AAAA', $givn); |
||
61 | $surn = str_replace('@N.N.', 'AAAA', $surn); |
||
62 | |||
63 | return array( |
||
64 | $surn . 'AAAA' . $givn, |
||
65 | $givn . 'AAAA' . $surn, |
||
66 | ); |
||
67 | } |
||
68 | |||
69 | /** |
||
70 | * Print a table of individuals |
||
71 | * |
||
72 | * @param Individual[] $indiviudals |
||
73 | * @param string $option |
||
74 | * |
||
75 | * @return string |
||
76 | */ |
||
77 | public static function individualTable($indiviudals, $option = '') |
||
78 | { |
||
79 | global $controller, $WT_TREE; |
||
80 | |||
81 | $table_id = 'table-indi-' . Uuid::uuid4(); // lists requires a unique ID in case there are multiple lists per page |
||
82 | |||
83 | $controller |
||
84 | ->addExternalJavascript(WT_JQUERY_DATATABLES_JS_URL) |
||
85 | ->addInlineJavascript(' |
||
86 | jQuery.fn.dataTableExt.oSort["text-asc"] = textCompareAsc; |
||
87 | jQuery.fn.dataTableExt.oSort["text-desc"] = textCompareDesc; |
||
88 | jQuery("#' . $table_id . '").dataTable( { |
||
89 | dom: \'<"H"<"filtersH_' . $table_id . '">T<"dt-clear">pf<"dt-clear">irl>t<"F"pl<"dt-clear"><"filtersF_' . $table_id . '">>\', |
||
90 | ' . I18N::datatablesI18N() . ', |
||
91 | jQueryUI: true, |
||
92 | autoWidth: false, |
||
93 | processing: true, |
||
94 | retrieve: true, |
||
95 | columns: [ |
||
96 | /* Given names */ { type: "text" }, |
||
97 | /* Surnames */ { type: "text" }, |
||
98 | /* SOSA numnber */ { type: "num", visible: ' . ($option === 'sosa' ? 'true' : 'false') . ' }, |
||
99 | /* Birth date */ { type: "num" }, |
||
100 | /* Anniversary */ { type: "num" }, |
||
101 | /* Birthplace */ { type: "text" }, |
||
102 | /* Children */ { type: "num" }, |
||
103 | /* Deate date */ { type: "num" }, |
||
104 | /* Anniversary */ { type: "num" }, |
||
105 | /* Age */ { type: "num" }, |
||
106 | /* Death place */ { type: "text" }, |
||
107 | /* Last change */ { visible: ' . ($WT_TREE->getPreference('SHOW_LAST_CHANGE') ? 'true' : 'false') . ' }, |
||
108 | /* Filter sex */ { sortable: false }, |
||
109 | /* Filter birth */ { sortable: false }, |
||
110 | /* Filter death */ { sortable: false }, |
||
111 | /* Filter tree */ { sortable: false } |
||
112 | ], |
||
113 | sorting: [[' . ($option === 'sosa' ? '4, "asc"' : '1, "asc"') . ']], |
||
114 | displayLength: 20, |
||
115 | pagingType: "full_numbers" |
||
116 | }); |
||
117 | |||
118 | jQuery("#' . $table_id . '") |
||
119 | /* Hide/show parents */ |
||
120 | .on("click", ".btn-toggle-parents", function() { |
||
121 | jQuery(this).toggleClass("ui-state-active"); |
||
122 | jQuery(".parents", jQuery(this).closest("table").DataTable().rows().nodes()).slideToggle(); |
||
123 | }) |
||
124 | /* Hide/show statistics */ |
||
125 | .on("click", ".btn-toggle-statistics", function() { |
||
126 | jQuery(this).toggleClass("ui-state-active"); |
||
127 | jQuery("#indi_list_table-charts_' . $table_id . '").slideToggle(); |
||
128 | }) |
||
129 | /* Filter buttons in table header */ |
||
130 | .on("click", "button[data-filter-column]", function() { |
||
131 | var btn = jQuery(this); |
||
132 | // De-activate the other buttons in this button group |
||
133 | btn.siblings().removeClass("ui-state-active"); |
||
134 | // Apply (or clear) this filter |
||
135 | var col = jQuery("#' . $table_id . '").DataTable().column(btn.data("filter-column")); |
||
136 | if (btn.hasClass("ui-state-active")) { |
||
137 | btn.removeClass("ui-state-active"); |
||
138 | col.search("").draw(); |
||
139 | } else { |
||
140 | btn.addClass("ui-state-active"); |
||
141 | col.search(btn.data("filter-value")).draw(); |
||
142 | } |
||
143 | }); |
||
144 | |||
145 | jQuery(".indi-list").css("visibility", "visible"); |
||
146 | jQuery(".loading-image").css("display", "none"); |
||
147 | '); |
||
148 | |||
149 | $max_age = (int) $WT_TREE->getPreference('MAX_ALIVE_AGE'); |
||
150 | |||
151 | // Inititialise chart data |
||
152 | $deat_by_age = array(); |
||
153 | for ($age = 0; $age <= $max_age; $age++) { |
||
154 | $deat_by_age[$age] = ''; |
||
155 | } |
||
156 | $birt_by_decade = array(); |
||
157 | $deat_by_decade = array(); |
||
158 | for ($year = 1550; $year < 2030; $year += 10) { |
||
159 | $birt_by_decade[$year] = ''; |
||
160 | $deat_by_decade[$year] = ''; |
||
161 | } |
||
162 | |||
163 | $html = ' |
||
164 | <div class="loading-image"></div> |
||
165 | <div class="indi-list"> |
||
166 | <table id="' . $table_id . '"> |
||
167 | <thead> |
||
168 | <tr> |
||
169 | <th colspan="16"> |
||
170 | <div class="btn-toolbar"> |
||
171 | <div class="btn-group"> |
||
172 | <button |
||
173 | class="ui-state-default" |
||
174 | data-filter-column="12" |
||
175 | data-filter-value="M" |
||
176 | title="' . I18N::translate('Show only males.') . '" |
||
177 | type="button" |
||
178 | > |
||
179 | ' . Individual::sexImage('M', 'large') . ' |
||
180 | </button> |
||
181 | <button |
||
182 | class="ui-state-default" |
||
183 | data-filter-column="12" |
||
184 | data-filter-value="F" |
||
185 | title="' . I18N::translate('Show only females.') . '" |
||
186 | type="button" |
||
187 | > |
||
188 | ' . Individual::sexImage('F', 'large') . ' |
||
189 | </button> |
||
190 | <button |
||
191 | class="ui-state-default" |
||
192 | data-filter-column="12" |
||
193 | data-filter-value="U" |
||
194 | title="' . I18N::translate('Show only individuals for whom the gender is not known.') . '" |
||
195 | type="button" |
||
196 | > |
||
197 | ' . Individual::sexImage('U', 'large') . ' |
||
198 | </button> |
||
199 | </div> |
||
200 | <div class="btn-group"> |
||
201 | <button |
||
202 | class="ui-state-default" |
||
203 | data-filter-column="14" |
||
204 | data-filter-value="N" |
||
205 | title="' . I18N::translate('Show individuals who are alive or couples where both partners are alive.') . '" |
||
206 | type="button" |
||
207 | > |
||
208 | ' . I18N::translate('Alive') . ' |
||
209 | </button> |
||
210 | <button |
||
211 | class="ui-state-default" |
||
212 | data-filter-column="14" |
||
213 | data-filter-value="Y" |
||
214 | title="' . I18N::translate('Show individuals who are dead or couples where both partners are dead.') . '" |
||
215 | type="button" |
||
216 | > |
||
217 | ' . I18N::translate('Dead') . ' |
||
218 | </button> |
||
219 | <button |
||
220 | class="ui-state-default" |
||
221 | data-filter-column="14" |
||
222 | data-filter-value="YES" |
||
223 | title="' . I18N::translate('Show individuals who died more than 100 years ago.') . '" |
||
224 | type="button" |
||
225 | > |
||
226 | ' . GedcomTag::getLabel('DEAT') . '>100 |
||
227 | </button> |
||
228 | <button |
||
229 | class="ui-state-default" |
||
230 | data-filter-column="14" |
||
231 | data-filter-value="Y100" |
||
232 | title="' . I18N::translate('Show individuals who died within the last 100 years.') . '" |
||
233 | type="button" |
||
234 | > |
||
235 | ' . GedcomTag::getLabel('DEAT') . '<=100 |
||
236 | </button> |
||
237 | </div> |
||
238 | <div class="btn-group"> |
||
239 | <button |
||
240 | class="ui-state-default" |
||
241 | data-filter-column="13" |
||
242 | data-filter-value="YES" |
||
243 | title="' . I18N::translate('Show individuals born more than 100 years ago.') . '" |
||
244 | type="button" |
||
245 | > |
||
246 | ' . GedcomTag::getLabel('BIRT') . '>100 |
||
247 | </button> |
||
248 | <button |
||
249 | class="ui-state-default" |
||
250 | data-filter-column="13" |
||
251 | data-filter-value="Y100" |
||
252 | title="' . I18N::translate('Show individuals born within the last 100 years.') . '" |
||
253 | type="button" |
||
254 | > |
||
255 | ' . GedcomTag::getLabel('BIRT') . '<=100 |
||
256 | </button> |
||
257 | </div> |
||
258 | <div class="btn-group"> |
||
259 | <button |
||
260 | class="ui-state-default" |
||
261 | data-filter-column="15" |
||
262 | data-filter-value="R" |
||
263 | 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.') . '" |
||
264 | type="button" |
||
265 | > |
||
266 | ' . I18N::translate('Roots') . ' |
||
267 | </button> |
||
268 | <button |
||
269 | class="ui-state-default" |
||
270 | data-filter-column="15" |
||
271 | data-filter-value="L" |
||
272 | title="' . I18N::translate('Show “leaves” couples or individuals. These are individuals who are alive but have no children recorded in the database.') . '" |
||
273 | type="button" |
||
274 | > |
||
275 | ' . I18N::translate('Leaves') . ' |
||
276 | </button> |
||
277 | </div> |
||
278 | </div> |
||
279 | </th> |
||
280 | </tr> |
||
281 | <tr> |
||
282 | <th>' . GedcomTag::getLabel('GIVN') . '</th> |
||
283 | <th>' . GedcomTag::getLabel('SURN') . '</th> |
||
284 | <th>' . /* I18N: Abbreviation for “Sosa-Stradonitz number”. This is an individual’s surname, so may need transliterating into non-latin alphabets. */ I18N::translate('Sosa') . '</th> |
||
285 | <th>' . GedcomTag::getLabel('BIRT') . '</th> |
||
286 | <th><i class="icon-reminder" title="' . I18N::translate('Anniversary') . '"></i></th> |
||
287 | <th>' . GedcomTag::getLabel('PLAC') . '</th> |
||
288 | <th><i class="icon-children" title="' . I18N::translate('Children') . '"></i></th> |
||
289 | <th>' . GedcomTag::getLabel('DEAT') . '</th> |
||
290 | <th><i class="icon-reminder" title="' . I18N::translate('Anniversary') . '"></i></th> |
||
291 | <th>' . GedcomTag::getLabel('AGE') . '</th> |
||
292 | <th>' . GedcomTag::getLabel('PLAC') . '</th> |
||
293 | <th>' . GedcomTag::getLabel('CHAN') . '</th> |
||
294 | <th hidden></th> |
||
295 | <th hidden></th> |
||
296 | <th hidden></th> |
||
297 | <th hidden></th> |
||
298 | </tr> |
||
299 | </thead> |
||
300 | <tfoot> |
||
301 | <tr> |
||
302 | <th colspan="16"> |
||
303 | <div class="btn-toolbar"> |
||
304 | <div class="btn-group"> |
||
305 | <button type="button" class="ui-state-default btn-toggle-parents"> |
||
306 | ' . I18N::translate('Show parents') . ' |
||
307 | </button> |
||
308 | <button type="button" class="ui-state-default btn-toggle-statistics"> |
||
309 | ' . I18N::translate('Show statistics charts') . ' |
||
310 | </button> |
||
311 | </div> |
||
312 | </div> |
||
313 | </th> |
||
314 | </tr> |
||
315 | </tfoot> |
||
316 | <tbody>'; |
||
317 | |||
318 | $hundred_years_ago = new Date(date('Y') - 100); |
||
319 | $unique_indis = array(); // Don't double-count indis with multiple names. |
||
320 | |||
321 | foreach ($indiviudals as $key => $individual) { |
||
322 | if (!$individual->canShowName()) { |
||
323 | continue; |
||
324 | } |
||
325 | if ($individual->isPendingAddtion()) { |
||
326 | $class = ' class="new"'; |
||
327 | } elseif ($individual->isPendingDeletion()) { |
||
328 | $class = ' class="old"'; |
||
329 | } else { |
||
330 | $class = ''; |
||
331 | } |
||
332 | $html .= '<tr' . $class . '>'; |
||
333 | // Extract Given names and Surnames for sorting |
||
334 | list($surn_givn, $givn_surn) = self::sortableNames($individual); |
||
335 | |||
336 | $html .= '<td colspan="2" data-sort="' . Filter::escapeHtml($givn_surn) . '">'; |
||
337 | foreach ($individual->getAllNames() as $num => $name) { |
||
338 | if ($name['type'] == 'NAME') { |
||
339 | $title = ''; |
||
340 | } else { |
||
341 | $title = 'title="' . strip_tags(GedcomTag::getLabel($name['type'], $individual)) . '"'; |
||
342 | } |
||
343 | if ($num == $individual->getPrimaryName()) { |
||
344 | $class = ' class="name2"'; |
||
345 | $sex_image = $individual->getSexImage(); |
||
346 | } else { |
||
347 | $class = ''; |
||
348 | $sex_image = ''; |
||
349 | } |
||
350 | $html .= '<a ' . $title . ' href="' . $individual->getHtmlUrl() . '"' . $class . '>' . FunctionsPrint::highlightSearchHits($name['full']) . '</a>' . $sex_image . '<br>'; |
||
351 | } |
||
352 | $html .= $individual->getPrimaryParentsNames('parents details1', 'none'); |
||
353 | $html .= '</td>'; |
||
354 | |||
355 | // Hidden column for sortable name |
||
356 | $html .= '<td hidden data-sort="' . Filter::escapeHtml($surn_givn) . '"></td>'; |
||
357 | |||
358 | // SOSA |
||
359 | $html .= '<td class="center" data-sort="' . $key . '">'; |
||
360 | if ($option === 'sosa') { |
||
361 | $html .= '<a href="relationship.php?pid1=' . $indiviudals[1] . '&pid2=' . $individual->getXref() . '" title="' . I18N::translate('Relationships') . '">' . I18N::number($key) . '</a>'; |
||
362 | } |
||
363 | $html .= '</td>'; |
||
364 | |||
365 | // Birth date |
||
366 | $birth_dates = $individual->getAllBirthDates(); |
||
367 | $html .= '<td data-sort="' . $individual->getEstimatedBirthDate()->julianDay() . '">'; |
||
368 | foreach ($birth_dates as $n => $birth_date) { |
||
369 | if ($n > 0) { |
||
370 | $html .= '<br>'; |
||
371 | } |
||
372 | $html .= $birth_date->display(true); |
||
373 | } |
||
374 | $html .= '</td>'; |
||
375 | |||
376 | // Birth anniversary |
||
377 | if (isset($birth_dates[0]) && $birth_dates[0]->gregorianYear() >= 1550 && $birth_dates[0]->gregorianYear() < 2030 && !isset($unique_indis[$individual->getXref()])) { |
||
378 | $birt_by_decade[(int) ($birth_dates[0]->gregorianYear() / 10) * 10] .= $individual->getSex(); |
||
379 | $anniversary = Date::getAge($birth_dates[0], null, 2); |
||
380 | } else { |
||
381 | $anniversary = ''; |
||
382 | } |
||
383 | $html .= '<td class="center" data-sort="' . -$individual->getEstimatedBirthDate()->julianDay() . '">' . $anniversary . '</td>'; |
||
384 | |||
385 | // Birth place |
||
386 | $html .= '<td>'; |
||
387 | foreach ($individual->getAllBirthPlaces() as $n => $birth_place) { |
||
388 | $tmp = new Place($birth_place, $individual->getTree()); |
||
389 | if ($n > 0) { |
||
390 | $html .= '<br>'; |
||
391 | } |
||
392 | $html .= '<a href="' . $tmp->getURL() . '" title="' . strip_tags($tmp->getFullName()) . '">'; |
||
393 | $html .= FunctionsPrint::highlightSearchHits($tmp->getShortName()) . '</a>'; |
||
394 | } |
||
395 | $html .= '</td>'; |
||
396 | |||
397 | // Number of children |
||
398 | $number_of_children = $individual->getNumberOfChildren(); |
||
399 | $html .= '<td class="center" data-sort="' . $number_of_children . '">' . I18N::number($number_of_children) . '</td>'; |
||
400 | |||
401 | // Death date |
||
402 | $death_dates = $individual->getAllDeathDates(); |
||
403 | $html .= '<td data-sort="' . $individual->getEstimatedDeathDate()->julianDay() . '">'; |
||
404 | foreach ($death_dates as $num => $death_date) { |
||
405 | if ($num) { |
||
406 | $html .= '<br>'; |
||
407 | } |
||
408 | $html .= $death_date->display(true); |
||
409 | } |
||
410 | $html .= '</td>'; |
||
411 | |||
412 | // Death anniversary |
||
413 | if (isset($death_dates[0]) && $death_dates[0]->gregorianYear() >= 1550 && $death_dates[0]->gregorianYear() < 2030 && !isset($unique_indis[$individual->getXref()])) { |
||
414 | $birt_by_decade[(int) ($death_dates[0]->gregorianYear() / 10) * 10] .= $individual->getSex(); |
||
415 | $anniversary = Date::getAge($death_dates[0], null, 2); |
||
416 | } else { |
||
417 | $anniversary = ''; |
||
418 | } |
||
419 | $html .= '<td class="center" data-sort="' . -$individual->getEstimatedDeathDate()->julianDay() . '">' . $anniversary . '</td>'; |
||
420 | |||
421 | // Age at death |
||
422 | if (isset($birth_dates[0]) && isset($death_dates[0])) { |
||
423 | $age_at_death = Date::getAge($birth_dates[0], $death_dates[0], 0); |
||
424 | $age_at_death_sort = Date::getAge($birth_dates[0], $death_dates[0], 2); |
||
425 | if (!isset($unique_indis[$individual->getXref()]) && $age >= 0 && $age <= $max_age) { |
||
426 | $deat_by_age[$age_at_death] .= $individual->getSex(); |
||
427 | } |
||
428 | } else { |
||
429 | $age_at_death = ''; |
||
430 | $age_at_death_sort = PHP_INT_MAX; |
||
431 | } |
||
432 | $html .= '<td class="center" data-sort="' . $age_at_death_sort . '">' . $age_at_death . '</td>'; |
||
433 | |||
434 | // Death place |
||
435 | $html .= '<td>'; |
||
436 | foreach ($individual->getAllDeathPlaces() as $n => $death_place) { |
||
437 | $tmp = new Place($death_place, $individual->getTree()); |
||
438 | if ($n > 0) { |
||
439 | $html .= '<br>'; |
||
440 | } |
||
441 | $html .= '<a href="' . $tmp->getURL() . '" title="' . strip_tags($tmp->getFullName()) . '">'; |
||
442 | $html .= FunctionsPrint::highlightSearchHits($tmp->getShortName()) . '</a>'; |
||
443 | } |
||
444 | $html .= '</td>'; |
||
445 | |||
446 | // Last change |
||
447 | $html .= '<td data-sort="' . $individual->lastChangeTimestamp(true) . '">' . $individual->lastChangeTimestamp() . '</td>'; |
||
448 | |||
449 | // Filter by sex |
||
450 | $html .= '<td hidden>' . $individual->getSex() . '</td>'; |
||
451 | |||
452 | // Filter by birth date |
||
453 | $html .= '<td hidden>'; |
||
454 | if (!$individual->canShow() || Date::compare($individual->getEstimatedBirthDate(), $hundred_years_ago) > 0) { |
||
455 | $html .= 'Y100'; |
||
456 | } else { |
||
457 | $html .= 'YES'; |
||
458 | } |
||
459 | $html .= '</td>'; |
||
460 | |||
461 | // Filter by death date |
||
462 | $html .= '<td hidden>'; |
||
463 | // Died in last 100 years? Died? Not dead? |
||
464 | if (isset($death_dates[0]) && Date::compare($death_dates[0], $hundred_years_ago) > 0) { |
||
465 | $html .= 'Y100'; |
||
466 | } elseif ($individual->isDead()) { |
||
467 | $html .= 'YES'; |
||
468 | } else { |
||
469 | $html .= 'N'; |
||
470 | } |
||
471 | $html .= '</td>'; |
||
472 | |||
473 | // Filter by roots/leaves |
||
474 | $html .= '<td hidden>'; |
||
475 | if (!$individual->getChildFamilies()) { |
||
476 | $html .= 'R'; |
||
477 | } elseif (!$individual->isDead() && $individual->getNumberOfChildren() < 1) { |
||
478 | $html .= 'L'; |
||
479 | $html .= ' '; |
||
480 | } |
||
481 | $html .= '</td>'; |
||
482 | $html .= '</tr>'; |
||
483 | |||
484 | $unique_indis[$individual->getXref()] = true; |
||
485 | } |
||
486 | $html .= ' |
||
487 | </tbody> |
||
488 | </table> |
||
489 | <div id="indi_list_table-charts_' . $table_id . '" style="display:none"> |
||
490 | <table class="list-charts"> |
||
491 | <tr> |
||
492 | <td> |
||
493 | ' . self::chartByDecade($birt_by_decade, I18N::translate('Decade of birth')) . ' |
||
494 | </td> |
||
495 | <td> |
||
496 | ' . self::chartByDecade($deat_by_decade, I18N::translate('Decade of death')) . ' |
||
497 | </td> |
||
498 | </tr> |
||
499 | <tr> |
||
500 | <td colspan="2"> |
||
501 | ' . self::chartByAge($deat_by_age, I18N::translate('Age related to death year')) . ' |
||
502 | </td> |
||
503 | </tr> |
||
504 | </table> |
||
505 | </div> |
||
506 | </div>'; |
||
507 | |||
508 | return $html; |
||
509 | } |
||
510 | |||
511 | /** |
||
512 | * Print a table of families |
||
513 | * |
||
514 | * @param Family[] $families |
||
515 | * |
||
516 | * @return string |
||
517 | */ |
||
518 | public static function familyTable($families) |
||
519 | { |
||
520 | global $WT_TREE, $controller; |
||
521 | |||
522 | $table_id = 'table-fam-' . Uuid::uuid4(); // lists requires a unique ID in case there are multiple lists per page |
||
523 | |||
524 | $controller |
||
525 | ->addExternalJavascript(WT_JQUERY_DATATABLES_JS_URL) |
||
526 | ->addInlineJavascript(' |
||
527 | jQuery.fn.dataTableExt.oSort["text-asc"] = textCompareAsc; |
||
528 | jQuery.fn.dataTableExt.oSort["text-desc"] = textCompareDesc; |
||
529 | jQuery("#' . $table_id . '").dataTable( { |
||
530 | dom: \'<"H"<"filtersH_' . $table_id . '"><"dt-clear">pf<"dt-clear">irl>t<"F"pl<"dt-clear"><"filtersF_' . $table_id . '">>\', |
||
531 | ' . I18N::datatablesI18N() . ', |
||
532 | jQueryUI: true, |
||
533 | autoWidth: false, |
||
534 | processing: true, |
||
535 | retrieve: true, |
||
536 | columns: [ |
||
537 | /* Given names */ { type: "text" }, |
||
538 | /* Surnames */ { type: "text" }, |
||
539 | /* Age */ { type: "num" }, |
||
540 | /* Given names */ { type: "text" }, |
||
541 | /* Surnames */ { type: "text" }, |
||
542 | /* Age */ { type: "num" }, |
||
543 | /* Marriage date */ { type: "num" }, |
||
544 | /* Anniversary */ { type: "num" }, |
||
545 | /* Marriage place */ { type: "text" }, |
||
546 | /* Children */ { type: "num" }, |
||
547 | /* Last change */ { visible: ' . ($WT_TREE->getPreference('SHOW_LAST_CHANGE') ? 'true' : 'false') . ' }, |
||
548 | /* Filter marriage */ { sortable: false }, |
||
549 | /* Filter alive/dead */ { sortable: false }, |
||
550 | /* Filter tree */ { sortable: false } |
||
551 | ], |
||
552 | sorting: [[1, "asc"]], |
||
553 | displayLength: 20, |
||
554 | pagingType: "full_numbers" |
||
555 | }); |
||
556 | |||
557 | jQuery("#' . $table_id . '") |
||
558 | /* Hide/show parents */ |
||
559 | .on("click", ".btn-toggle-parents", function() { |
||
560 | jQuery(this).toggleClass("ui-state-active"); |
||
561 | jQuery(".parents", jQuery(this).closest("table").DataTable().rows().nodes()).slideToggle(); |
||
562 | }) |
||
563 | /* Hide/show statistics */ |
||
564 | .on("click", ".btn-toggle-statistics", function() { |
||
565 | jQuery(this).toggleClass("ui-state-active"); |
||
566 | jQuery("#fam_list_table-charts_' . $table_id . '").slideToggle(); |
||
567 | }) |
||
568 | /* Filter buttons in table header */ |
||
569 | .on("click", "button[data-filter-column]", function() { |
||
570 | var btn = $(this); |
||
571 | // De-activate the other buttons in this button group |
||
572 | btn.siblings().removeClass("ui-state-active"); |
||
573 | // Apply (or clear) this filter |
||
574 | var col = jQuery("#' . $table_id . '").DataTable().column(btn.data("filter-column")); |
||
575 | if (btn.hasClass("ui-state-active")) { |
||
576 | btn.removeClass("ui-state-active"); |
||
577 | col.search("").draw(); |
||
578 | } else { |
||
579 | btn.addClass("ui-state-active"); |
||
580 | col.search(btn.data("filter-value")).draw(); |
||
581 | } |
||
582 | }); |
||
583 | |||
584 | jQuery(".fam-list").css("visibility", "visible"); |
||
585 | jQuery(".loading-image").css("display", "none"); |
||
586 | '); |
||
587 | |||
588 | $max_age = (int) $WT_TREE->getPreference('MAX_ALIVE_AGE'); |
||
589 | |||
590 | // init chart data |
||
591 | $marr_by_age = array(); |
||
592 | for ($age = 0; $age <= $max_age; $age++) { |
||
593 | $marr_by_age[$age] = ''; |
||
594 | } |
||
595 | $birt_by_decade = array(); |
||
596 | $marr_by_decade = array(); |
||
597 | for ($year = 1550; $year < 2030; $year += 10) { |
||
598 | $birt_by_decade[$year] = ''; |
||
599 | $marr_by_decade[$year] = ''; |
||
600 | } |
||
601 | |||
602 | $html = ' |
||
603 | <div class="loading-image"></div> |
||
604 | <div class="fam-list"> |
||
605 | <table id="' . $table_id . '"> |
||
606 | <thead> |
||
607 | <tr> |
||
608 | <th colspan="14"> |
||
609 | <div class="btn-toolbar"> |
||
610 | <div class="btn-group"> |
||
611 | <button |
||
612 | type="button" |
||
613 | data-filter-column="12" |
||
614 | data-filter-value="N" |
||
615 | class="ui-state-default" |
||
616 | title="' . I18N::translate('Show individuals who are alive or couples where both partners are alive.') . '" |
||
617 | > |
||
618 | ' . I18N::translate('Both alive') . ' |
||
619 | </button> |
||
620 | <button |
||
621 | type="button" |
||
622 | data-filter-column="12" |
||
623 | data-filter-value="W" |
||
624 | class="ui-state-default" |
||
625 | title="' . I18N::translate('Show couples where only the female partner is dead.') . '" |
||
626 | > |
||
627 | ' . I18N::translate('Widower') . ' |
||
628 | </button> |
||
629 | <button |
||
630 | type="button" |
||
631 | data-filter-column="12" |
||
632 | data-filter-value="H" |
||
633 | class="ui-state-default" |
||
634 | title="' . I18N::translate('Show couples where only the male partner is dead.') . '" |
||
635 | > |
||
636 | ' . I18N::translate('Widow') . ' |
||
637 | </button> |
||
638 | <button |
||
639 | type="button" |
||
640 | data-filter-column="12" |
||
641 | data-filter-value="Y" |
||
642 | class="ui-state-default" |
||
643 | title="' . I18N::translate('Show individuals who are dead or couples where both partners are dead.') . '" |
||
644 | > |
||
645 | ' . I18N::translate('Both dead') . ' |
||
646 | </button> |
||
647 | </div> |
||
648 | <div class="btn-group"> |
||
649 | <button |
||
650 | type="button" |
||
651 | data-filter-column="13" |
||
652 | data-filter-value="R" |
||
653 | class="ui-state-default" |
||
654 | 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.') . '" |
||
655 | > |
||
656 | ' . I18N::translate('Roots') . ' |
||
657 | </button> |
||
658 | <button |
||
659 | type="button" |
||
660 | data-filter-column="13" |
||
661 | data-filter-value="L" |
||
662 | class="ui-state-default" |
||
663 | title="' . I18N::translate('Show “leaves” couples or individuals. These are individuals who are alive but have no children recorded in the database.') . '" |
||
664 | > |
||
665 | ' . I18N::translate('Leaves') . ' |
||
666 | </button> |
||
667 | </div> |
||
668 | <div class="btn-group"> |
||
669 | <button |
||
670 | type="button" |
||
671 | data-filter-column="11" |
||
672 | data-filter-value="U" |
||
673 | class="ui-state-default" |
||
674 | title="' . I18N::translate('Show couples with an unknown marriage date.') . '" |
||
675 | > |
||
676 | ' . GedcomTag::getLabel('MARR') . ' |
||
677 | </button> |
||
678 | <button |
||
679 | type="button" |
||
680 | data-filter-column="11" |
||
681 | data-filter-value="YES" |
||
682 | class="ui-state-default" |
||
683 | title="' . I18N::translate('Show couples who married more than 100 years ago.') . '" |
||
684 | > |
||
685 | ' . GedcomTag::getLabel('MARR') . '>100 |
||
686 | </button> |
||
687 | <button |
||
688 | type="button" |
||
689 | data-filter-column="11" |
||
690 | data-filter-value="Y100" |
||
691 | class="ui-state-default" |
||
692 | title="' . I18N::translate('Show couples who married within the last 100 years.') . '" |
||
693 | > |
||
694 | ' . GedcomTag::getLabel('MARR') . '<=100 |
||
695 | </button> |
||
696 | <button |
||
697 | type="button" |
||
698 | data-filter-column="11" |
||
699 | data-filter-value="D" |
||
700 | class="ui-state-default" |
||
701 | title="' . I18N::translate('Show divorced couples.') . '" |
||
702 | > |
||
703 | ' . GedcomTag::getLabel('DIV') . ' |
||
704 | </button> |
||
705 | <button |
||
706 | type="button" |
||
707 | data-filter-column="11" |
||
708 | data-filter-value="M" |
||
709 | class="ui-state-default" |
||
710 | title="' . I18N::translate('Show couples where either partner married more than once.') . '" |
||
711 | > |
||
712 | ' . I18N::translate('Multiple marriages') . ' |
||
713 | </button> |
||
714 | </div> |
||
715 | </div> |
||
716 | </th> |
||
717 | </tr> |
||
718 | <tr> |
||
719 | <th>' . GedcomTag::getLabel('GIVN') . '</th> |
||
720 | <th>' . GedcomTag::getLabel('SURN') . '</th> |
||
721 | <th>' . GedcomTag::getLabel('AGE') . '</th> |
||
722 | <th>' . GedcomTag::getLabel('GIVN') . '</th> |
||
723 | <th>' . GedcomTag::getLabel('SURN') . '</th> |
||
724 | <th>' . GedcomTag::getLabel('AGE') . '</th> |
||
725 | <th>' . GedcomTag::getLabel('MARR') . '</th> |
||
726 | <th><i class="icon-reminder" title="' . I18N::translate('Anniversary') . '"></i></th> |
||
727 | <th>' . GedcomTag::getLabel('PLAC') . '</th> |
||
728 | <th><i class="icon-children" title="' . I18N::translate('Children') . '"></i></th> |
||
729 | <th>' . GedcomTag::getLabel('CHAN') . '</th> |
||
730 | <th hidden></th> |
||
731 | <th hidden></th> |
||
732 | <th hidden></th> |
||
733 | </tr> |
||
734 | </thead> |
||
735 | <tfoot> |
||
736 | <tr> |
||
737 | <th colspan="14"> |
||
738 | <div class="btn-toolbar"> |
||
739 | <div class="btn-group"> |
||
740 | <button type="button" class="ui-state-default btn-toggle-parents"> |
||
741 | ' . I18N::translate('Show parents') . ' |
||
742 | </button> |
||
743 | <button type="button" class="ui-state-default btn-toggle-statistics"> |
||
744 | ' . I18N::translate('Show statistics charts') . ' |
||
745 | </button> |
||
746 | </div> |
||
747 | </div> |
||
748 | </th> |
||
749 | </tr> |
||
750 | </tfoot> |
||
751 | <tbody>'; |
||
752 | |||
753 | $hundred_years_ago = new Date(date('Y') - 100); |
||
754 | |||
755 | foreach ($families as $family) { |
||
756 | // Retrieve husband and wife |
||
757 | $husb = $family->getHusband(); |
||
758 | if ($husb === null) { |
||
759 | $husb = new Individual('H', '0 @H@ INDI', null, $family->getTree()); |
||
760 | } |
||
761 | $wife = $family->getWife(); |
||
762 | if ($wife === null) { |
||
763 | $wife = new Individual('W', '0 @W@ INDI', null, $family->getTree()); |
||
764 | } |
||
765 | if (!$family->canShow()) { |
||
766 | continue; |
||
767 | } |
||
768 | if ($family->isPendingAddtion()) { |
||
769 | $class = ' class="new"'; |
||
770 | } elseif ($family->isPendingDeletion()) { |
||
771 | $class = ' class="old"'; |
||
772 | } else { |
||
773 | $class = ''; |
||
774 | } |
||
775 | $html .= '<tr' . $class . '>'; |
||
776 | // Husband name(s) |
||
777 | // Extract Given names and Surnames for sorting |
||
778 | list($surn_givn, $givn_surn) = self::sortableNames($husb); |
||
779 | |||
780 | $html .= '<td colspan="2" data-sort="' . Filter::escapeHtml($givn_surn) . '">'; |
||
781 | foreach ($husb->getAllNames() as $num => $name) { |
||
782 | if ($name['type'] == 'NAME') { |
||
783 | $title = ''; |
||
784 | } else { |
||
785 | $title = 'title="' . strip_tags(GedcomTag::getLabel($name['type'], $husb)) . '"'; |
||
786 | } |
||
787 | if ($num == $husb->getPrimaryName()) { |
||
788 | $class = ' class="name2"'; |
||
789 | $sex_image = $husb->getSexImage(); |
||
790 | } else { |
||
791 | $class = ''; |
||
792 | $sex_image = ''; |
||
793 | } |
||
794 | // Only show married names if they are the name we are filtering by. |
||
795 | if ($name['type'] != '_MARNM' || $num == $husb->getPrimaryName()) { |
||
796 | $html .= '<a ' . $title . ' href="' . $family->getHtmlUrl() . '"' . $class . '>' . FunctionsPrint::highlightSearchHits($name['full']) . '</a>' . $sex_image . '<br>'; |
||
797 | } |
||
798 | } |
||
799 | // Husband parents |
||
800 | $html .= $husb->getPrimaryParentsNames('parents details1', 'none'); |
||
801 | $html .= '</td>'; |
||
802 | |||
803 | // Hidden column for sortable name |
||
804 | $html .= '<td hidden data-sort="' . Filter::escapeHtml($surn_givn) . '"></td>'; |
||
805 | |||
806 | // Husband age |
||
807 | $mdate = $family->getMarriageDate(); |
||
808 | $hdate = $husb->getBirthDate(); |
||
809 | if ($hdate->isOK() && $mdate->isOK()) { |
||
810 | if ($hdate->gregorianYear() >= 1550 && $hdate->gregorianYear() < 2030) { |
||
811 | $birt_by_decade[(int) ($hdate->gregorianYear() / 10) * 10] .= $husb->getSex(); |
||
812 | } |
||
813 | $hage = Date::getAge($hdate, $mdate, 0); |
||
814 | if ($hage >= 0 && $hage <= $max_age) { |
||
815 | $marr_by_age[$hage] .= $husb->getSex(); |
||
816 | } |
||
817 | } |
||
818 | $html .= '<td class="center" data=-sort="' . Date::getAge($hdate, $mdate, 1) . '">' . Date::getAge($hdate, $mdate, 2) . '</td>'; |
||
819 | |||
820 | // Wife name(s) |
||
821 | // Extract Given names and Surnames for sorting |
||
822 | list($surn_givn, $givn_surn) = self::sortableNames($wife); |
||
823 | $html .= '<td colspan="2" data-sort="' . Filter::escapeHtml($givn_surn) . '">'; |
||
824 | foreach ($wife->getAllNames() as $num => $name) { |
||
825 | if ($name['type'] == 'NAME') { |
||
826 | $title = ''; |
||
827 | } else { |
||
828 | $title = 'title="' . strip_tags(GedcomTag::getLabel($name['type'], $wife)) . '"'; |
||
829 | } |
||
830 | if ($num == $wife->getPrimaryName()) { |
||
831 | $class = ' class="name2"'; |
||
832 | $sex_image = $wife->getSexImage(); |
||
833 | } else { |
||
834 | $class = ''; |
||
835 | $sex_image = ''; |
||
836 | } |
||
837 | // Only show married names if they are the name we are filtering by. |
||
838 | if ($name['type'] != '_MARNM' || $num == $wife->getPrimaryName()) { |
||
839 | $html .= '<a ' . $title . ' href="' . $family->getHtmlUrl() . '"' . $class . '>' . FunctionsPrint::highlightSearchHits($name['full']) . '</a>' . $sex_image . '<br>'; |
||
840 | } |
||
841 | } |
||
842 | // Wife parents |
||
843 | $html .= $wife->getPrimaryParentsNames('parents details1', 'none'); |
||
844 | $html .= '</td>'; |
||
845 | |||
846 | // Hidden column for sortable name |
||
847 | $html .= '<td hidden data-sort="' . Filter::escapeHtml($surn_givn) . '"></td>'; |
||
848 | |||
849 | // Wife age |
||
850 | $mdate = $family->getMarriageDate(); |
||
851 | $wdate = $wife->getBirthDate(); |
||
852 | if ($wdate->isOK() && $mdate->isOK()) { |
||
853 | if ($wdate->gregorianYear() >= 1550 && $wdate->gregorianYear() < 2030) { |
||
854 | $birt_by_decade[(int) ($wdate->gregorianYear() / 10) * 10] .= $wife->getSex(); |
||
855 | } |
||
856 | $wage = Date::getAge($wdate, $mdate, 0); |
||
857 | if ($wage >= 0 && $wage <= $max_age) { |
||
858 | $marr_by_age[$wage] .= $wife->getSex(); |
||
859 | } |
||
860 | } |
||
861 | $html .= '<td class="center" data-sort="' . Date::getAge($wdate, $mdate, 1) . '">' . Date::getAge($wdate, $mdate, 2) . '</td>'; |
||
862 | |||
863 | // Marriage date |
||
864 | $html .= '<td data-sort="' . $family->getMarriageDate()->julianDay() . '">'; |
||
865 | if ($marriage_dates = $family->getAllMarriageDates()) { |
||
866 | foreach ($marriage_dates as $n => $marriage_date) { |
||
867 | if ($n) { |
||
868 | $html .= '<br>'; |
||
869 | } |
||
870 | $html .= '<div>' . $marriage_date->display(true) . '</div>'; |
||
871 | } |
||
872 | if ($marriage_dates[0]->gregorianYear() >= 1550 && $marriage_dates[0]->gregorianYear() < 2030) { |
||
873 | $marr_by_decade[(int) ($marriage_dates[0]->gregorianYear() / 10) * 10] .= $husb->getSex() . $wife->getSex(); |
||
874 | } |
||
875 | } elseif ($family->getFacts('_NMR')) { |
||
876 | $html .= I18N::translate('no'); |
||
877 | } elseif ($family->getFacts('MARR')) { |
||
878 | $html .= I18N::translate('yes'); |
||
879 | } else { |
||
880 | $html .= ' '; |
||
881 | } |
||
882 | $html .= '</td>'; |
||
883 | |||
884 | // Marriage anniversary |
||
885 | $html .= '<td class="center" data-sort="' . -$family->getMarriageDate()->julianDay() . '">' . Date::getAge($family->getMarriageDate(), null, 2) . '</td>'; |
||
886 | |||
887 | // Marriage place |
||
888 | $html .= '<td>'; |
||
889 | foreach ($family->getAllMarriagePlaces() as $n => $marriage_place) { |
||
890 | $tmp = new Place($marriage_place, $family->getTree()); |
||
891 | if ($n) { |
||
892 | $html .= '<br>'; |
||
893 | } |
||
894 | $html .= '<a href="' . $tmp->getURL() . '" title="' . strip_tags($tmp->getFullName()) . '">'; |
||
895 | $html .= FunctionsPrint::highlightSearchHits($tmp->getShortName()) . '</a>'; |
||
896 | } |
||
897 | $html .= '</td>'; |
||
898 | |||
899 | // Number of children |
||
900 | $html .= '<td class="center" data-sort="' . $family->getNumberOfChildren() . '">' . I18N::number($family->getNumberOfChildren()) . '</td>'; |
||
901 | |||
902 | // Last change |
||
903 | $html .= '<td data-sort="' . $family->lastChangeTimestamp(true) . '">' . $family->lastChangeTimestamp() . '</td>'; |
||
904 | |||
905 | // Filter by marriage date |
||
906 | $html .= '<td hidden>'; |
||
907 | if (!$family->canShow() || !$mdate->isOK()) { |
||
908 | $html .= 'U'; |
||
909 | } else { |
||
910 | if (Date::compare($mdate, $hundred_years_ago) > 0) { |
||
911 | $html .= 'Y100'; |
||
912 | } else { |
||
913 | $html .= 'YES'; |
||
914 | } |
||
915 | } |
||
916 | if ($family->getFacts(WT_EVENTS_DIV)) { |
||
917 | $html .= 'D'; |
||
918 | } |
||
919 | if (count($husb->getSpouseFamilies()) > 1 || count($wife->getSpouseFamilies()) > 1) { |
||
920 | $html .= 'M'; |
||
921 | } |
||
922 | $html .= '</td>'; |
||
923 | |||
924 | // Filter by alive/dead |
||
925 | $html .= '<td hidden>'; |
||
926 | if ($husb->isDead() && $wife->isDead()) { |
||
927 | $html .= 'Y'; |
||
928 | } |
||
929 | if ($husb->isDead() && !$wife->isDead()) { |
||
930 | if ($wife->getSex() == 'F') { |
||
931 | $html .= 'H'; |
||
932 | } |
||
933 | if ($wife->getSex() == 'M') { |
||
934 | $html .= 'W'; |
||
935 | } // male partners |
||
936 | } |
||
937 | if (!$husb->isDead() && $wife->isDead()) { |
||
938 | if ($husb->getSex() == 'M') { |
||
939 | $html .= 'W'; |
||
940 | } |
||
941 | if ($husb->getSex() == 'F') { |
||
942 | $html .= 'H'; |
||
943 | } // female partners |
||
944 | } |
||
945 | if (!$husb->isDead() && !$wife->isDead()) { |
||
946 | $html .= 'N'; |
||
947 | } |
||
948 | $html .= '</td>'; |
||
949 | |||
950 | // Filter by roots/leaves |
||
951 | $html .= '<td hidden>'; |
||
952 | if (!$husb->getChildFamilies() && !$wife->getChildFamilies()) { |
||
953 | $html .= 'R'; |
||
954 | } elseif (!$husb->isDead() && !$wife->isDead() && $family->getNumberOfChildren() === 0) { |
||
955 | $html .= 'L'; |
||
956 | } |
||
957 | $html .= '</td> |
||
958 | </tr>'; |
||
959 | } |
||
960 | |||
961 | $html .= ' |
||
962 | </tbody> |
||
963 | </table> |
||
964 | <div id="fam_list_table-charts_' . $table_id . '" style="display:none"> |
||
965 | <table class="list-charts"> |
||
966 | <tr> |
||
967 | <td>' . self::chartByDecade($birt_by_decade, I18N::translate('Decade of birth')) . '</td> |
||
968 | <td>' . self::chartByDecade($marr_by_decade, I18N::translate('Decade of marriage')) . '</td> |
||
969 | </tr> |
||
970 | <tr> |
||
971 | <td colspan="2">' . self::chartByAge($marr_by_age, I18N::translate('Age in year of marriage')) . '</td> |
||
972 | </tr> |
||
973 | </table> |
||
974 | </div> |
||
975 | </div>'; |
||
976 | |||
977 | return $html; |
||
978 | } |
||
979 | |||
980 | /** |
||
981 | * Print a table of sources |
||
982 | * |
||
983 | * @param Source[] $sources |
||
984 | * |
||
985 | * @return string |
||
986 | */ |
||
987 | public static function sourceTable($sources) |
||
988 | { |
||
989 | global $WT_TREE, $controller; |
||
990 | |||
991 | // Count the number of linked records. These numbers include private records. |
||
992 | // It is not good to bypass privacy, but many servers do not have the resources |
||
993 | // to process privacy for every record in the tree |
||
994 | $count_individuals = Database::prepare( |
||
995 | "SELECT CONCAT(l_to, '@', l_file), COUNT(*) FROM `##individuals` JOIN `##link` ON l_from = i_id AND l_file = i_file AND l_type = 'SOUR' GROUP BY l_to, l_file" |
||
996 | )->fetchAssoc(); |
||
997 | $count_families = Database::prepare( |
||
998 | "SELECT CONCAT(l_to, '@', l_file), COUNT(*) FROM `##families` JOIN `##link` ON l_from = f_id AND l_file = f_file AND l_type = 'SOUR' GROUP BY l_to, l_file" |
||
999 | )->fetchAssoc(); |
||
1000 | $count_media = Database::prepare( |
||
1001 | "SELECT CONCAT(l_to, '@', l_file), COUNT(*) FROM `##media` JOIN `##link` ON l_from = m_id AND l_file = m_file AND l_type = 'SOUR' GROUP BY l_to, l_file" |
||
1002 | )->fetchAssoc(); |
||
1003 | $count_notes = Database::prepare( |
||
1004 | "SELECT CONCAT(l_to, '@', l_file), COUNT(*) FROM `##other` JOIN `##link` ON l_from = o_id AND l_file = o_file AND o_type = 'NOTE' AND l_type = 'SOUR' GROUP BY l_to, l_file" |
||
1005 | )->fetchAssoc(); |
||
1006 | |||
1007 | $html = ''; |
||
1008 | $table_id = 'table-sour-' . Uuid::uuid4(); // lists requires a unique ID in case there are multiple lists per page |
||
1009 | $controller |
||
1010 | ->addExternalJavascript(WT_JQUERY_DATATABLES_JS_URL) |
||
1011 | ->addInlineJavascript(' |
||
1012 | jQuery.fn.dataTableExt.oSort["text-asc"] = textCompareAsc; |
||
1013 | jQuery.fn.dataTableExt.oSort["text-desc"] = textCompareDesc; |
||
1014 | jQuery("#' . $table_id . '").dataTable( { |
||
1015 | dom: \'<"H"pf<"dt-clear">irl>t<"F"pl>\', |
||
1016 | ' . I18N::datatablesI18N() . ', |
||
1017 | jQueryUI: true, |
||
1018 | autoWidth: false, |
||
1019 | processing: true, |
||
1020 | columns: [ |
||
1021 | /* Title */ { type: "text" }, |
||
1022 | /* Author */ { type: "text" }, |
||
1023 | /* Individuals */ { type: "num" }, |
||
1024 | /* Families */ { type: "num" }, |
||
1025 | /* Media objects */ { type: "num" }, |
||
1026 | /* Notes */ { type: "num" }, |
||
1027 | /* Last change */ { visible: ' . ($WT_TREE->getPreference('SHOW_LAST_CHANGE') ? 'true' : 'false') . ' }, |
||
1028 | /* Delete */ { visible: ' . (Auth::isManager($WT_TREE) ? 'true' : 'false') . ', sortable: false } |
||
1029 | ], |
||
1030 | displayLength: 20, |
||
1031 | pagingType: "full_numbers" |
||
1032 | }); |
||
1033 | jQuery(".source-list").css("visibility", "visible"); |
||
1034 | jQuery(".loading-image").css("display", "none"); |
||
1035 | '); |
||
1036 | |||
1037 | $html .= '<div class="loading-image"></div>'; |
||
1038 | $html .= '<div class="source-list">'; |
||
1039 | $html .= '<table id="' . $table_id . '"><thead><tr>'; |
||
1040 | $html .= '<th>' . GedcomTag::getLabel('TITL') . '</th>'; |
||
1041 | $html .= '<th>' . GedcomTag::getLabel('AUTH') . '</th>'; |
||
1042 | $html .= '<th>' . I18N::translate('Individuals') . '</th>'; |
||
1043 | $html .= '<th>' . I18N::translate('Families') . '</th>'; |
||
1044 | $html .= '<th>' . I18N::translate('Media objects') . '</th>'; |
||
1045 | $html .= '<th>' . I18N::translate('Shared notes') . '</th>'; |
||
1046 | $html .= '<th>' . GedcomTag::getLabel('CHAN') . '</th>'; |
||
1047 | $html .= '<th>' . I18N::translate('Delete') . '</th>'; |
||
1048 | $html .= '</tr></thead>'; |
||
1049 | $html .= '<tbody>'; |
||
1050 | |||
1051 | foreach ($sources as $source) { |
||
1052 | if (!$source->canShow()) { |
||
1053 | continue; |
||
1054 | } |
||
1055 | if ($source->isPendingAddtion()) { |
||
1056 | $class = ' class="new"'; |
||
1057 | } elseif ($source->isPendingDeletion()) { |
||
1058 | $class = ' class="old"'; |
||
1059 | } else { |
||
1060 | $class = ''; |
||
1061 | } |
||
1062 | $html .= '<tr' . $class . '>'; |
||
1063 | // Source name(s) |
||
1064 | $html .= '<td data-sort="' . Filter::escapeHtml($source->getSortName()) . '">'; |
||
1065 | foreach ($source->getAllNames() as $n => $name) { |
||
1066 | if ($n) { |
||
1067 | $html .= '<br>'; |
||
1068 | } |
||
1069 | if ($n == $source->getPrimaryName()) { |
||
1070 | $html .= '<a class="name2" href="' . $source->getHtmlUrl() . '">' . FunctionsPrint::highlightSearchHits($name['full']) . '</a>'; |
||
1071 | } else { |
||
1072 | $html .= '<a href="' . $source->getHtmlUrl() . '">' . FunctionsPrint::highlightSearchHits($name['full']) . '</a>'; |
||
1073 | } |
||
1074 | } |
||
1075 | $html .= '</td>'; |
||
1076 | // Author |
||
1077 | $auth = $source->getFirstFact('AUTH'); |
||
1078 | if ($auth) { |
||
1079 | $author = $auth->getValue(); |
||
1080 | } else { |
||
1081 | $author = ''; |
||
1082 | } |
||
1083 | $html .= '<td data-sort="' . Filter::escapeHtml($author) . '">' . FunctionsPrint::highlightSearchHits($author) . '</td>'; |
||
1084 | $key = $source->getXref() . '@' . $source->getTree()->getTreeId(); |
||
1085 | // Count of linked individuals |
||
1086 | $num = array_key_exists($key, $count_individuals) ? $count_individuals[$key] : 0; |
||
1087 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
|
|||
1088 | // Count of linked families |
||
1089 | $num = array_key_exists($key, $count_families) ? $count_families[$key] : 0; |
||
1090 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1091 | // Count of linked media objects |
||
1092 | $num = array_key_exists($key, $count_media) ? $count_media[$key] : 0; |
||
1093 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1094 | // Count of linked notes |
||
1095 | $num = array_key_exists($key, $count_notes) ? $count_notes[$key] : 0; |
||
1096 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1097 | // Last change |
||
1098 | $html .= '<td data-sort="' . $source->lastChangeTimestamp(true) . '">' . $source->lastChangeTimestamp() . '</td>'; |
||
1099 | // Delete |
||
1100 | $html .= '<td><a href="#" title="' . I18N::translate('Delete') . '" class="deleteicon" onclick="return delete_record(\'' . I18N::translate('Are you sure you want to delete “%s”?', Filter::escapeJs(Filter::unescapeHtml($source->getFullName()))) . "', '" . $source->getXref() . '\');"><span class="link_text">' . I18N::translate('Delete') . '</span></a></td>'; |
||
1101 | $html .= '</tr>'; |
||
1102 | } |
||
1103 | $html .= '</tbody></table></div>'; |
||
1104 | |||
1105 | return $html; |
||
1106 | } |
||
1107 | |||
1108 | /** |
||
1109 | * Print a table of shared notes |
||
1110 | * |
||
1111 | * @param Note[] $notes |
||
1112 | * |
||
1113 | * @return string |
||
1114 | */ |
||
1115 | public static function noteTable($notes) |
||
1116 | { |
||
1117 | global $WT_TREE, $controller; |
||
1118 | |||
1119 | // Count the number of linked records. These numbers include private records. |
||
1120 | // It is not good to bypass privacy, but many servers do not have the resources |
||
1121 | // to process privacy for every record in the tree |
||
1122 | $count_individuals = Database::prepare( |
||
1123 | "SELECT CONCAT(l_to, '@', l_file), COUNT(*) FROM `##individuals` JOIN `##link` ON l_from = i_id AND l_file = i_file AND l_type = 'NOTE' GROUP BY l_to, l_file" |
||
1124 | )->fetchAssoc(); |
||
1125 | $count_families = Database::prepare( |
||
1126 | "SELECT CONCAT(l_to, '@', l_file), COUNT(*) FROM `##families` JOIN `##link` ON l_from = f_id AND l_file = f_file AND l_type = 'NOTE' GROUP BY l_to, l_file" |
||
1127 | )->fetchAssoc(); |
||
1128 | $count_media = Database::prepare( |
||
1129 | "SELECT CONCAT(l_to, '@', l_file), COUNT(*) FROM `##media` JOIN `##link` ON l_from = m_id AND l_file = m_file AND l_type = 'NOTE' GROUP BY l_to, l_file" |
||
1130 | )->fetchAssoc(); |
||
1131 | $count_sources = Database::prepare( |
||
1132 | "SELECT CONCAT(l_to, '@', l_file), COUNT(*) FROM `##sources` JOIN `##link` ON l_from = s_id AND l_file = s_file AND l_type = 'NOTE' GROUP BY l_to, l_file" |
||
1133 | )->fetchAssoc(); |
||
1134 | |||
1135 | $html = ''; |
||
1136 | $table_id = 'table-note-' . Uuid::uuid4(); // lists requires a unique ID in case there are multiple lists per page |
||
1137 | $controller |
||
1138 | ->addExternalJavascript(WT_JQUERY_DATATABLES_JS_URL) |
||
1139 | ->addInlineJavascript(' |
||
1140 | jQuery.fn.dataTableExt.oSort["text-asc"] = textCompareAsc; |
||
1141 | jQuery.fn.dataTableExt.oSort["text-desc"] = textCompareDesc; |
||
1142 | jQuery("#' . $table_id . '").dataTable({ |
||
1143 | dom: \'<"H"pf<"dt-clear">irl>t<"F"pl>\', |
||
1144 | ' . I18N::datatablesI18N() . ', |
||
1145 | jQueryUI: true, |
||
1146 | autoWidth: false, |
||
1147 | processing: true, |
||
1148 | columns: [ |
||
1149 | /* Title */ { type: "text" }, |
||
1150 | /* Individuals */ { type: "num" }, |
||
1151 | /* Families */ { type: "num" }, |
||
1152 | /* Media objects */ { type: "num" }, |
||
1153 | /* Sources */ { type: "num" }, |
||
1154 | /* Last change */ { type: "num", visible: ' . ($WT_TREE->getPreference('SHOW_LAST_CHANGE') ? 'true' : 'false') . ' }, |
||
1155 | /* Delete */ { visible: ' . (Auth::isManager($WT_TREE) ? 'true' : 'false') . ', sortable: false } |
||
1156 | ], |
||
1157 | displayLength: 20, |
||
1158 | pagingType: "full_numbers" |
||
1159 | }); |
||
1160 | jQuery(".note-list").css("visibility", "visible"); |
||
1161 | jQuery(".loading-image").css("display", "none"); |
||
1162 | '); |
||
1163 | |||
1164 | $html .= '<div class="loading-image"></div>'; |
||
1165 | $html .= '<div class="note-list">'; |
||
1166 | $html .= '<table id="' . $table_id . '"><thead><tr>'; |
||
1167 | $html .= '<th>' . GedcomTag::getLabel('TITL') . '</th>'; |
||
1168 | $html .= '<th>' . I18N::translate('Individuals') . '</th>'; |
||
1169 | $html .= '<th>' . I18N::translate('Families') . '</th>'; |
||
1170 | $html .= '<th>' . I18N::translate('Media objects') . '</th>'; |
||
1171 | $html .= '<th>' . I18N::translate('Sources') . '</th>'; |
||
1172 | $html .= '<th>' . GedcomTag::getLabel('CHAN') . '</th>'; |
||
1173 | $html .= '<th>' . I18N::translate('Delete') . '</th>'; |
||
1174 | $html .= '</tr></thead>'; |
||
1175 | $html .= '<tbody>'; |
||
1176 | |||
1177 | foreach ($notes as $note) { |
||
1178 | if (!$note->canShow()) { |
||
1179 | continue; |
||
1180 | } |
||
1181 | if ($note->isPendingAddtion()) { |
||
1182 | $class = ' class="new"'; |
||
1183 | } elseif ($note->isPendingDeletion()) { |
||
1184 | $class = ' class="old"'; |
||
1185 | } else { |
||
1186 | $class = ''; |
||
1187 | } |
||
1188 | $html .= '<tr' . $class . '>'; |
||
1189 | // Count of linked notes |
||
1190 | $html .= '<td data-sort="' . Filter::escapeHtml($note->getSortName()) . '"><a class="name2" href="' . $note->getHtmlUrl() . '">' . FunctionsPrint::highlightSearchHits($note->getFullName()) . '</a></td>'; |
||
1191 | $key = $note->getXref() . '@' . $note->getTree()->getTreeId(); |
||
1192 | // Count of linked individuals |
||
1193 | $num = array_key_exists($key, $count_individuals) ? $count_individuals[$key] : 0; |
||
1194 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1195 | // Count of linked families |
||
1196 | $num = array_key_exists($key, $count_families) ? $count_families[$key] : 0; |
||
1197 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1198 | // Count of linked media objects |
||
1199 | $num = array_key_exists($key, $count_media) ? $count_media[$key] : 0; |
||
1200 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1201 | // Count of linked sources |
||
1202 | $num = array_key_exists($key, $count_sources) ? $count_sources[$key] : 0; |
||
1203 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1204 | // Last change |
||
1205 | $html .= '<td data-sort="' . $note->lastChangeTimestamp(true) . '">' . $note->lastChangeTimestamp() . '</td>'; |
||
1206 | // Delete |
||
1207 | $html .= '<td><a href="#" title="' . I18N::translate('Delete') . '" class="deleteicon" onclick="return delete_record(\'' . I18N::translate('Are you sure you want to delete “%s”?', Filter::escapeJs(Filter::unescapeHtml($note->getFullName()))) . "', '" . $note->getXref() . '\');"><span class="link_text">' . I18N::translate('Delete') . '</span></a></td>'; |
||
1208 | $html .= '</tr>'; |
||
1209 | } |
||
1210 | $html .= '</tbody></table></div>'; |
||
1211 | |||
1212 | return $html; |
||
1213 | } |
||
1214 | |||
1215 | /** |
||
1216 | * Print a table of repositories |
||
1217 | * |
||
1218 | * @param Repository[] $repositories |
||
1219 | * |
||
1220 | * @return string |
||
1221 | */ |
||
1222 | public static function repositoryTable($repositories) |
||
1223 | { |
||
1224 | global $WT_TREE, $controller; |
||
1225 | |||
1226 | // Count the number of linked records. These numbers include private records. |
||
1227 | // It is not good to bypass privacy, but many servers do not have the resources |
||
1228 | // to process privacy for every record in the tree |
||
1229 | $count_sources = Database::prepare( |
||
1230 | "SELECT CONCAT(l_to, '@', l_file), COUNT(*) FROM `##sources` JOIN `##link` ON l_from = s_id AND l_file = s_file AND l_type = 'REPO' GROUP BY l_to, l_file" |
||
1231 | )->fetchAssoc(); |
||
1232 | |||
1233 | $html = ''; |
||
1234 | $table_id = 'table-repo-' . Uuid::uuid4(); // lists requires a unique ID in case there are multiple lists per page |
||
1235 | $controller |
||
1236 | ->addExternalJavascript(WT_JQUERY_DATATABLES_JS_URL) |
||
1237 | ->addInlineJavascript(' |
||
1238 | jQuery.fn.dataTableExt.oSort["text-asc"] = textCompareAsc; |
||
1239 | jQuery.fn.dataTableExt.oSort["text-desc"] = textCompareDesc; |
||
1240 | jQuery("#' . $table_id . '").dataTable({ |
||
1241 | dom: \'<"H"pf<"dt-clear">irl>t<"F"pl>\', |
||
1242 | ' . I18N::datatablesI18N() . ', |
||
1243 | jQueryUI: true, |
||
1244 | autoWidth: false, |
||
1245 | processing: true, |
||
1246 | columns: [ |
||
1247 | /* Name */ { type: "text" }, |
||
1248 | /* Sources */ { type: "num" }, |
||
1249 | /* Last change */ { visible: ' . ($WT_TREE->getPreference('SHOW_LAST_CHANGE') ? 'true' : 'false') . ' }, |
||
1250 | /* Delete */ { visible: ' . (Auth::isManager($WT_TREE) ? 'true' : 'false') . ', sortable: false } |
||
1251 | ], |
||
1252 | displayLength: 20, |
||
1253 | pagingType: "full_numbers" |
||
1254 | }); |
||
1255 | jQuery(".repo-list").css("visibility", "visible"); |
||
1256 | jQuery(".loading-image").css("display", "none"); |
||
1257 | '); |
||
1258 | |||
1259 | $html .= '<div class="loading-image"></div>'; |
||
1260 | $html .= '<div class="repo-list">'; |
||
1261 | $html .= '<table id="' . $table_id . '"><thead><tr>'; |
||
1262 | $html .= '<th>' . I18N::translate('Repository name') . '</th>'; |
||
1263 | $html .= '<th>' . I18N::translate('Sources') . '</th>'; |
||
1264 | $html .= '<th>' . GedcomTag::getLabel('CHAN') . '</th>'; |
||
1265 | $html .= '<th>' . I18N::translate('Delete') . '</th>'; |
||
1266 | $html .= '</tr></thead>'; |
||
1267 | $html .= '<tbody>'; |
||
1268 | |||
1269 | foreach ($repositories as $repository) { |
||
1270 | if (!$repository->canShow()) { |
||
1271 | continue; |
||
1272 | } |
||
1273 | if ($repository->isPendingAddtion()) { |
||
1274 | $class = ' class="new"'; |
||
1275 | } elseif ($repository->isPendingDeletion()) { |
||
1276 | $class = ' class="old"'; |
||
1277 | } else { |
||
1278 | $class = ''; |
||
1279 | } |
||
1280 | $html .= '<tr' . $class . '>'; |
||
1281 | // Repository name(s) |
||
1282 | $html .= '<td data-sort="' . Filter::escapeHtml($repository->getSortName()) . '">'; |
||
1283 | foreach ($repository->getAllNames() as $n => $name) { |
||
1284 | if ($n) { |
||
1285 | $html .= '<br>'; |
||
1286 | } |
||
1287 | if ($n == $repository->getPrimaryName()) { |
||
1288 | $html .= '<a class="name2" href="' . $repository->getHtmlUrl() . '">' . FunctionsPrint::highlightSearchHits($name['full']) . '</a>'; |
||
1289 | } else { |
||
1290 | $html .= '<a href="' . $repository->getHtmlUrl() . '">' . FunctionsPrint::highlightSearchHits($name['full']) . '</a>'; |
||
1291 | } |
||
1292 | } |
||
1293 | $html .= '</td>'; |
||
1294 | $key = $repository->getXref() . '@' . $repository->getTree()->getTreeId(); |
||
1295 | // Count of linked sources |
||
1296 | $num = array_key_exists($key, $count_sources) ? $count_sources[$key] : 0; |
||
1297 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1298 | // Last change |
||
1299 | $html .= '<td data-sort="' . $repository->lastChangeTimestamp(true) . '">' . $repository->lastChangeTimestamp() . '</td>'; |
||
1300 | // Delete |
||
1301 | $html .= '<td><a href="#" title="' . I18N::translate('Delete') . '" class="deleteicon" onclick="return delete_record(\'' . I18N::translate('Are you sure you want to delete “%s”?', Filter::escapeJs(Filter::unescapeHtml($repository->getFullName()))) . "', '" . $repository->getXref() . '\');"><span class="link_text">' . I18N::translate('Delete') . '</span></a></td>'; |
||
1302 | $html .= '</tr>'; |
||
1303 | } |
||
1304 | $html .= '</tbody></table></div>'; |
||
1305 | |||
1306 | return $html; |
||
1307 | } |
||
1308 | |||
1309 | /** |
||
1310 | * Print a table of media objects |
||
1311 | * |
||
1312 | * @param Media[] $media_objects |
||
1313 | * |
||
1314 | * @return string |
||
1315 | */ |
||
1316 | public static function mediaTable($media_objects) |
||
1317 | { |
||
1318 | global $WT_TREE, $controller; |
||
1319 | |||
1320 | $html = ''; |
||
1321 | $table_id = 'table-obje-' . Uuid::uuid4(); // lists requires a unique ID in case there are multiple lists per page |
||
1322 | $controller |
||
1323 | ->addExternalJavascript(WT_JQUERY_DATATABLES_JS_URL) |
||
1324 | ->addInlineJavascript(' |
||
1325 | jQuery.fn.dataTableExt.oSort["text-asc"] = textCompareAsc; |
||
1326 | jQuery.fn.dataTableExt.oSort["text-desc"] = textCompareDesc; |
||
1327 | jQuery("#' . $table_id . '").dataTable({ |
||
1328 | dom: \'<"H"pf<"dt-clear">irl>t<"F"pl>\', |
||
1329 | ' . I18N::datatablesI18N() . ', |
||
1330 | jQueryUI: true, |
||
1331 | autoWidth:false, |
||
1332 | processing: true, |
||
1333 | columns: [ |
||
1334 | /* Thumbnail */ { sortable: false }, |
||
1335 | /* Title */ { type: "text" }, |
||
1336 | /* Individuals */ { type: "num" }, |
||
1337 | /* Families */ { type: "num" }, |
||
1338 | /* Sources */ { type: "num" }, |
||
1339 | /* Last change */ { visible: ' . ($WT_TREE->getPreference('SHOW_LAST_CHANGE') ? 'true' : 'false') . ' }, |
||
1340 | ], |
||
1341 | displayLength: 20, |
||
1342 | pagingType: "full_numbers" |
||
1343 | }); |
||
1344 | jQuery(".media-list").css("visibility", "visible"); |
||
1345 | jQuery(".loading-image").css("display", "none"); |
||
1346 | '); |
||
1347 | |||
1348 | $html .= '<div class="loading-image"></div>'; |
||
1349 | $html .= '<div class="media-list">'; |
||
1350 | $html .= '<table id="' . $table_id . '"><thead><tr>'; |
||
1351 | $html .= '<th>' . I18N::translate('Media') . '</th>'; |
||
1352 | $html .= '<th>' . GedcomTag::getLabel('TITL') . '</th>'; |
||
1353 | $html .= '<th>' . I18N::translate('Individuals') . '</th>'; |
||
1354 | $html .= '<th>' . I18N::translate('Families') . '</th>'; |
||
1355 | $html .= '<th>' . I18N::translate('Sources') . '</th>'; |
||
1356 | $html .= '<th>' . GedcomTag::getLabel('CHAN') . '</th>'; |
||
1357 | $html .= '</tr></thead>'; |
||
1358 | $html .= '<tbody>'; |
||
1359 | |||
1360 | foreach ($media_objects as $media_object) { |
||
1361 | if ($media_object->canShow()) { |
||
1362 | $name = $media_object->getFullName(); |
||
1363 | if ($media_object->isPendingAddtion()) { |
||
1364 | $class = ' class="new"'; |
||
1365 | } elseif ($media_object->isPendingDeletion()) { |
||
1366 | $class = ' class="old"'; |
||
1367 | } else { |
||
1368 | $class = ''; |
||
1369 | } |
||
1370 | $html .= '<tr' . $class . '>'; |
||
1371 | // Media object thumbnail |
||
1372 | $html .= '<td>' . $media_object->displayImage() . '</td>'; |
||
1373 | // Media object name(s) |
||
1374 | $html .= '<td data-sort="' . Filter::escapeHtml($media_object->getSortName()) . '">'; |
||
1375 | $html .= '<a href="' . $media_object->getHtmlUrl() . '" class="list_item name2">'; |
||
1376 | $html .= FunctionsPrint::highlightSearchHits($name) . '</a>'; |
||
1377 | if (Auth::isEditor($media_object->getTree())) { |
||
1378 | $html .= '<br><a href="' . $media_object->getHtmlUrl() . '">' . basename($media_object->getFilename()) . '</a>'; |
||
1379 | } |
||
1380 | $html .= '</td>'; |
||
1381 | |||
1382 | // Count of linked individuals |
||
1383 | $num = count($media_object->linkedIndividuals('OBJE')); |
||
1384 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1385 | // Count of linked families |
||
1386 | $num = count($media_object->linkedFamilies('OBJE')); |
||
1387 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1388 | // Count of linked sources |
||
1389 | $num = count($media_object->linkedSources('OBJE')); |
||
1390 | $html .= '<td class="center" data-sort="' . $num . '">' . I18N::number($num) . '</td>'; |
||
1391 | // Last change |
||
1392 | $html .= '<td data-sort="' . $media_object->lastChangeTimestamp(true) . '">' . $media_object->lastChangeTimestamp() . '</td>'; |
||
1393 | $html .= '</tr>'; |
||
1394 | } |
||
1395 | } |
||
1396 | $html .= '</tbody></table></div>'; |
||
1397 | |||
1398 | return $html; |
||
1399 | } |
||
1400 | |||
1401 | /** |
||
1402 | * Print a table of surnames, for the top surnames block, the indi/fam lists, etc. |
||
1403 | * |
||
1404 | * @param string[][] $surnames array (of SURN, of array of SPFX_SURN, of array of PID) |
||
1405 | * @param string $script "indilist.php" (counts of individuals) or "famlist.php" (counts of spouses) |
||
1406 | * @param Tree $tree generate links for this tree |
||
1407 | * |
||
1408 | * @return string |
||
1409 | */ |
||
1410 | public static function surnameTable($surnames, $script, Tree $tree) |
||
1489 | } |
||
1490 | |||
1491 | /** |
||
1492 | * Print a tagcloud of surnames. |
||
1493 | * |
||
1494 | * @param string[][] $surnames array (of SURN, of array of SPFX_SURN, of array of PID) |
||
1495 | * @param string $script indilist or famlist |
||
1496 | * @param bool $totals show totals after each name |
||
1497 | * @param Tree $tree generate links to this tree |
||
1498 | * |
||
1499 | * @return string |
||
1500 | */ |
||
1501 | public static function surnameTagCloud($surnames, $script, $totals, Tree $tree) |
||
1532 | } |
||
1533 | |||
1534 | /** |
||
1535 | * Print a list of surnames. |
||
1536 | * |
||
1537 | * @param string[][] $surnames array (of SURN, of array of SPFX_SURN, of array of PID) |
||
1538 | * @param int $style 1=bullet list, 2=semicolon-separated list, 3=tabulated list with up to 4 columns |
||
1539 | * @param bool $totals show totals after each name |
||
1540 | * @param string $script indilist or famlist |
||
1541 | * @param Tree $tree Link back to the individual list in this tree |
||
1542 | * |
||
1543 | * @return string |
||
1544 | */ |
||
1545 | public static function surnameList($surnames, $style, $totals, $script, Tree $tree) |
||
1546 | { |
||
1547 | $html = array(); |
||
1548 | foreach ($surnames as $surn => $surns) { |
||
1549 | // Each surname links back to the indilist |
||
1550 | if ($surn) { |
||
1551 | $url = $script . '?surname=' . urlencode($surn) . '&ged=' . $tree->getNameUrl(); |
||
1552 | } else { |
||
1553 | $url = $script . '?alpha=,&ged=' . $tree->getNameUrl(); |
||
1554 | } |
||
1555 | // If all the surnames are just case variants, then merge them into one |
||
1556 | // Comment out this block if you want SMITH listed separately from Smith |
||
1557 | $first_spfxsurn = null; |
||
1558 | foreach ($surns as $spfxsurn => $indis) { |
||
1559 | if ($first_spfxsurn) { |
||
1560 | if (I18N::strtoupper($spfxsurn) == I18N::strtoupper($first_spfxsurn)) { |
||
1561 | $surns[$first_spfxsurn] = array_merge($surns[$first_spfxsurn], $surns[$spfxsurn]); |
||
1562 | unset($surns[$spfxsurn]); |
||
1563 | } |
||
1564 | } else { |
||
1565 | $first_spfxsurn = $spfxsurn; |
||
1566 | } |
||
1567 | } |
||
1568 | $subhtml = '<a href="' . $url . '" dir="auto">' . Filter::escapeHtml(implode(I18N::$list_separator, array_keys($surns))) . '</a>'; |
||
1569 | |||
1570 | if ($totals) { |
||
1571 | $subtotal = 0; |
||
1572 | foreach ($surns as $indis) { |
||
1573 | $subtotal += count($indis); |
||
1574 | } |
||
1575 | $subhtml .= ' (' . I18N::number($subtotal) . ')'; |
||
1576 | } |
||
1577 | $html[] = $subhtml; |
||
1578 | } |
||
1579 | switch ($style) { |
||
1580 | case 1: |
||
1581 | return '<ul><li>' . implode('</li><li>', $html) . '</li></ul>'; |
||
1582 | case 2: |
||
1583 | return implode(I18N::$list_separator, $html); |
||
1584 | case 3: |
||
1585 | $i = 0; |
||
1586 | $count = count($html); |
||
1587 | if ($count > 36) { |
||
1588 | $col = 4; |
||
1589 | } elseif ($count > 18) { |
||
1590 | $col = 3; |
||
1591 | } elseif ($count > 6) { |
||
1592 | $col = 2; |
||
1593 | } else { |
||
1594 | $col = 1; |
||
1595 | } |
||
1596 | $newcol = ceil($count / $col); |
||
1597 | $html2 = '<table class="list_table"><tr>'; |
||
1598 | $html2 .= '<td class="list_value" style="padding: 14px;">'; |
||
1599 | |||
1600 | foreach ($html as $surns) { |
||
1601 | $html2 .= $surns . '<br>'; |
||
1602 | $i++; |
||
1603 | if ($i == $newcol && $i < $count) { |
||
1604 | $html2 .= '</td><td class="list_value" style="padding: 14px;">'; |
||
1605 | $newcol = $i + ceil($count / $col); |
||
1606 | } |
||
1607 | } |
||
1608 | $html2 .= '</td></tr></table>'; |
||
1609 | |||
1610 | return $html2; |
||
1611 | } |
||
1612 | } |
||
1613 | /** |
||
1614 | * Print a table of events |
||
1615 | * |
||
1616 | * @param int $startjd |
||
1617 | * @param int $endjd |
||
1618 | * @param string $events |
||
1619 | * @param bool $only_living |
||
1620 | * @param string $sort_by |
||
1621 | * |
||
1622 | * @return string |
||
1623 | */ |
||
1624 | public static function eventsTable($startjd, $endjd, $events = 'BIRT MARR DEAT', $only_living = false, $sort_by = 'anniv') |
||
1625 | { |
||
1626 | global $controller, $WT_TREE; |
||
1627 | |||
1628 | $html = ''; |
||
1629 | $table_id = 'table-even-' . Uuid::uuid4(); // lists requires a unique ID in case there are multiple lists per page |
||
1630 | $controller |
||
1631 | ->addExternalJavascript(WT_JQUERY_DATATABLES_JS_URL) |
||
1632 | ->addInlineJavascript(' |
||
1633 | jQuery.fn.dataTableExt.oSort["text-asc"] = textCompareAsc; |
||
1634 | jQuery.fn.dataTableExt.oSort["text-desc"] = textCompareDesc; |
||
1635 | jQuery("#' . $table_id . '").dataTable({ |
||
1636 | dom: "t", |
||
1637 | ' . I18N::datatablesI18N() . ', |
||
1638 | autoWidth: false, |
||
1639 | paging: false, |
||
1640 | lengthChange: false, |
||
1641 | filter: false, |
||
1642 | info: true, |
||
1643 | jQueryUI: true, |
||
1644 | sorting: [[ ' . ($sort_by == 'alpha' ? 0 : 1) . ', "asc"]], |
||
1645 | columns: [ |
||
1646 | /* Name */ { type: "text" }, |
||
1647 | /* Date */ { type: "num" }, |
||
1648 | /* Anniversary */ { type: "num" }, |
||
1649 | /* Event */ { type: "text" } |
||
1650 | ] |
||
1651 | }); |
||
1652 | '); |
||
1653 | |||
1654 | // Did we have any output? Did we skip anything? |
||
1655 | $filter = 0; |
||
1656 | $filtered_events = array(); |
||
1657 | |||
1658 | foreach (FunctionsDb::getEventsList($startjd, $endjd, $events, $WT_TREE) as $fact) { |
||
1659 | $record = $fact->getParent(); |
||
1660 | // Only living people ? |
||
1661 | if ($only_living) { |
||
1662 | if ($record instanceof Individual && $record->isDead()) { |
||
1663 | $filter++; |
||
1664 | continue; |
||
1665 | } |
||
1666 | if ($record instanceof Family) { |
||
1667 | $husb = $record->getHusband(); |
||
1668 | if ($husb === null || $husb->isDead()) { |
||
1669 | $filter++; |
||
1670 | continue; |
||
1671 | } |
||
1672 | $wife = $record->getWife(); |
||
1673 | if ($wife === null || $wife->isDead()) { |
||
1674 | $filter++; |
||
1675 | continue; |
||
1676 | } |
||
1677 | } |
||
1678 | } |
||
1679 | |||
1680 | $filtered_events[] = $fact; |
||
1681 | } |
||
1682 | |||
1683 | if (!empty($filtered_events)) { |
||
1684 | $html .= '<table id="' . $table_id . '" class="width100">'; |
||
1685 | $html .= '<thead><tr>'; |
||
1686 | $html .= '<th>' . I18N::translate('Record') . '</th>'; |
||
1687 | $html .= '<th>' . GedcomTag::getLabel('DATE') . '</th>'; |
||
1688 | $html .= '<th><i class="icon-reminder" title="' . I18N::translate('Anniversary') . '"></i></th>'; |
||
1689 | $html .= '<th>' . GedcomTag::getLabel('EVEN') . '</th>'; |
||
1690 | $html .= '</tr></thead><tbody>'; |
||
1691 | |||
1692 | foreach ($filtered_events as $n => $fact) { |
||
1693 | $record = $fact->getParent(); |
||
1694 | $html .= '<tr>'; |
||
1695 | $html .= '<td data-sort="' . Filter::escapeHtml($record->getSortName()) . '">'; |
||
1696 | $html .= '<a href="' . $record->getHtmlUrl() . '">' . $record->getFullName() . '</a>'; |
||
1697 | if ($record instanceof Individual) { |
||
1698 | $html .= $record->getSexImage(); |
||
1699 | } |
||
1700 | $html .= '</td>'; |
||
1701 | $html .= '<td data-sort="' . $fact->getDate()->minimumJulianDay() . '">'; |
||
1702 | $html .= $fact->getDate()->display(); |
||
1703 | $html .= '</td>'; |
||
1704 | $html .= '<td class="center" data-sort="' . $fact->anniv . '">'; |
||
1705 | $html .= ($fact->anniv ? I18N::number($fact->anniv) : ''); |
||
1706 | $html .= '</td>'; |
||
1707 | $html .= '<td class="center">' . $fact->getLabel() . '</td>'; |
||
1708 | $html .= '</tr>'; |
||
1709 | } |
||
1710 | |||
1711 | $html .= '</tbody></table>'; |
||
1712 | } else { |
||
1713 | if ($endjd === WT_CLIENT_JD) { |
||
1714 | // We're dealing with the Today’s Events block |
||
1715 | if ($filter === 0) { |
||
1716 | $html .= I18N::translate('No events exist for today.'); |
||
1717 | } else { |
||
1718 | $html .= I18N::translate('No events for living individuals exist for today.'); |
||
1719 | } |
||
1720 | } else { |
||
1721 | // We're dealing with the Upcoming Events block |
||
1722 | if ($filter === 0) { |
||
1723 | if ($endjd === $startjd) { |
||
1724 | $html .= I18N::translate('No events exist for tomorrow.'); |
||
1725 | } else { |
||
1726 | $html .= /* I18N: translation for %s==1 is unused; it is translated separately as “tomorrow” */ I18N::plural('No events exist for the next %s day.', 'No events exist for the next %s days.', $endjd - $startjd + 1, I18N::number($endjd - $startjd + 1)); |
||
1727 | } |
||
1728 | } else { |
||
1729 | if ($endjd === $startjd) { |
||
1730 | $html .= I18N::translate('No events for living individuals exist for tomorrow.'); |
||
1731 | } else { |
||
1732 | // I18N: translation for %s==1 is unused; it is translated separately as “tomorrow” |
||
1733 | $html .= I18N::plural('No events for living people exist for the next %s day.', 'No events for living people exist for the next %s days.', $endjd - $startjd + 1, I18N::number($endjd - $startjd + 1)); |
||
1734 | } |
||
1735 | } |
||
1736 | } |
||
1737 | } |
||
1738 | |||
1739 | return $html; |
||
1740 | } |
||
1741 | |||
1742 | /** |
||
1743 | * Print a list of events |
||
1744 | * |
||
1745 | * This performs the same function as print_events_table(), but formats the output differently. |
||
1746 | * |
||
1747 | * @param int $startjd |
||
1748 | * @param int $endjd |
||
1749 | * @param string $events |
||
1750 | * @param bool $only_living |
||
1751 | * @param string $sort_by |
||
1752 | * |
||
1753 | * @return string |
||
1754 | */ |
||
1755 | public static function eventsList($startjd, $endjd, $events = 'BIRT MARR DEAT', $only_living = false, $sort_by = 'anniv') |
||
1856 | } |
||
1857 | |||
1858 | /** |
||
1859 | * Print a chart by age using Google chart API |
||
1860 | * |
||
1861 | * @param int[] $data |
||
1862 | * @param string $title |
||
1863 | * |
||
1864 | * @return string |
||
1865 | */ |
||
1866 | public static function chartByAge($data, $title) |
||
1867 | { |
||
1868 | $count = 0; |
||
1869 | $agemax = 0; |
||
1870 | $vmax = 0; |
||
1871 | $avg = 0; |
||
1872 | foreach ($data as $age => $v) { |
||
1873 | $n = strlen($v); |
||
1874 | $vmax = max($vmax, $n); |
||
1875 | $agemax = max($agemax, $age); |
||
1876 | $count += $n; |
||
1877 | $avg += $age * $n; |
||
1878 | } |
||
1879 | if ($count < 1) { |
||
1880 | return ''; |
||
1881 | } |
||
1882 | $avg = round($avg / $count); |
||
1883 | $chart_url = "https://chart.googleapis.com/chart?cht=bvs"; // chart type |
||
1884 | $chart_url .= "&chs=725x150"; // size |
||
1885 | $chart_url .= "&chbh=3,2,2"; // bvg : 4,1,2 |
||
1886 | $chart_url .= "&chf=bg,s,FFFFFF99"; //background color |
||
1887 | $chart_url .= "&chco=0000FF,FFA0CB,FF0000"; // bar color |
||
1888 | $chart_url .= "&chdl=" . rawurlencode(I18N::translate('Males')) . "|" . rawurlencode(I18N::translate('Females')) . "|" . rawurlencode(I18N::translate('Average age') . ": " . $avg); // legend & average age |
||
1889 | $chart_url .= "&chtt=" . rawurlencode($title); // title |
||
1890 | $chart_url .= "&chxt=x,y,r"; // axis labels specification |
||
1891 | $chart_url .= "&chm=V,FF0000,0," . ($avg - 0.3) . ",1"; // average age line marker |
||
1892 | $chart_url .= "&chxl=0:|"; // label |
||
1893 | for ($age = 0; $age <= $agemax; $age += 5) { |
||
1894 | $chart_url .= $age . "|||||"; // x axis |
||
1895 | } |
||
1896 | $chart_url .= "|1:||" . rawurlencode(I18N::percentage($vmax / $count)); // y axis |
||
1897 | $chart_url .= "|2:||"; |
||
1898 | $step = $vmax; |
||
1899 | for ($d = $vmax; $d > 0; $d--) { |
||
1900 | if ($vmax < ($d * 10 + 1) && ($vmax % $d) == 0) { |
||
1901 | $step = $d; |
||
1902 | } |
||
1903 | } |
||
1904 | if ($step == $vmax) { |
||
1905 | for ($d = $vmax - 1; $d > 0; $d--) { |
||
1906 | if (($vmax - 1) < ($d * 10 + 1) && (($vmax - 1) % $d) == 0) { |
||
1907 | $step = $d; |
||
1908 | } |
||
1909 | } |
||
1910 | } |
||
1911 | for ($n = $step; $n < $vmax; $n += $step) { |
||
1912 | $chart_url .= $n . "|"; |
||
1913 | } |
||
1914 | $chart_url .= rawurlencode($vmax . " / " . $count); // r axis |
||
1915 | $chart_url .= "&chg=100," . round(100 * $step / $vmax, 1) . ",1,5"; // grid |
||
1916 | $chart_url .= "&chd=s:"; // data : simple encoding from A=0 to 9=61 |
||
1917 | $CHART_ENCODING61 = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"; |
||
1918 | for ($age = 0; $age <= $agemax; $age++) { |
||
1919 | $chart_url .= $CHART_ENCODING61[(int) (substr_count($data[$age], "M") * 61 / $vmax)]; |
||
1920 | } |
||
1921 | $chart_url .= ","; |
||
1922 | for ($age = 0; $age <= $agemax; $age++) { |
||
1923 | $chart_url .= $CHART_ENCODING61[(int) (substr_count($data[$age], "F") * 61 / $vmax)]; |
||
1924 | } |
||
1925 | $html = '<img src="' . $chart_url . '" alt="' . $title . '" title="' . $title . '" class="gchart">'; |
||
1926 | |||
1927 | return $html; |
||
1928 | } |
||
1929 | |||
1930 | /** |
||
1931 | * Print a chart by decade using Google chart API |
||
1932 | * |
||
1933 | * @param int[] $data |
||
1934 | * @param string $title |
||
1935 | * |
||
1936 | * @return string |
||
1937 | */ |
||
1938 | public static function chartByDecade($data, $title) |
||
1993 | } |
||
1994 | } |
||
1995 |