1b6f35a76SGreg Roach<?php 2b6f35a76SGreg Roach 3b6f35a76SGreg Roach/** 4b6f35a76SGreg Roach * webtrees: online genealogy 5d11be702SGreg Roach * Copyright (C) 2023 webtrees development team 6b6f35a76SGreg Roach * This program is free software: you can redistribute it and/or modify 7b6f35a76SGreg Roach * it under the terms of the GNU General Public License as published by 8b6f35a76SGreg Roach * the Free Software Foundation, either version 3 of the License, or 9b6f35a76SGreg Roach * (at your option) any later version. 10b6f35a76SGreg Roach * This program is distributed in the hope that it will be useful, 11b6f35a76SGreg Roach * but WITHOUT ANY WARRANTY; without even the implied warranty of 12b6f35a76SGreg Roach * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13b6f35a76SGreg Roach * GNU General Public License for more details. 14b6f35a76SGreg Roach * You should have received a copy of the GNU General Public License 1589f7189bSGreg Roach * along with this program. If not, see <https://www.gnu.org/licenses/>. 16b6f35a76SGreg Roach */ 17b6f35a76SGreg Roach 18b6f35a76SGreg Roachdeclare(strict_types=1); 19b6f35a76SGreg Roach 20b6f35a76SGreg Roachnamespace Fisharebest\Webtrees\Report; 21b6f35a76SGreg Roach 22b6f35a76SGreg Roachuse Fisharebest\Webtrees\I18N; 23b6f35a76SGreg Roachuse Fisharebest\Webtrees\MediaFile; 24b6f35a76SGreg Roachuse Fisharebest\Webtrees\Webtrees; 25b6f35a76SGreg Roach 26b6f35a76SGreg Roach/** 27b6f35a76SGreg Roach * Class AbstractRenderer - base for PDF and HTML reports 28b6f35a76SGreg Roach */ 29b6f35a76SGreg Roachabstract class AbstractRenderer 30b6f35a76SGreg Roach{ 31b6f35a76SGreg Roach // Reports layouts are measured in points. 32b6f35a76SGreg Roach protected const UNITS = 'pt'; 33b6f35a76SGreg Roach 34b6f35a76SGreg Roach // A point is 1/72 of an inch 35b6f35a76SGreg Roach protected const INCH_TO_POINTS = 72.0; 36b6f35a76SGreg Roach protected const MM_TO_POINTS = 72.0 / 25.4; 37b6f35a76SGreg Roach 38b6f35a76SGreg Roach protected const PAPER_SIZES = [ 39b6f35a76SGreg Roach // ISO 216 40b6f35a76SGreg Roach 'A0' => [841.0 * self::MM_TO_POINTS, 1189.0 * self::MM_TO_POINTS], 41b6f35a76SGreg Roach 'A1' => [594.0 * self::MM_TO_POINTS, 841.0 * self::MM_TO_POINTS], 42b6f35a76SGreg Roach 'A2' => [420.0 * self::MM_TO_POINTS, 594.0 * self::MM_TO_POINTS], 43b6f35a76SGreg Roach 'A3' => [297.0 * self::MM_TO_POINTS, 420.0 * self::MM_TO_POINTS], 44b6f35a76SGreg Roach 'A4' => [210.0 * self::MM_TO_POINTS, 297.0 * self::MM_TO_POINTS], 45b6f35a76SGreg Roach // US 46b6f35a76SGreg Roach 'US-Letter' => [8.5 * self::INCH_TO_POINTS, 11.0 * self::INCH_TO_POINTS], 47b6f35a76SGreg Roach 'US-Legal' => [8.5 * self::INCH_TO_POINTS, 14.0 * self::INCH_TO_POINTS], 48b6f35a76SGreg Roach 'US-Tabloid' => [11.0 * self::INCH_TO_POINTS, 17.0 * self::INCH_TO_POINTS], 49b6f35a76SGreg Roach ]; 50b6f35a76SGreg Roach 51345fcc8aSGreg Roach public float $left_margin = 18.0 * self::MM_TO_POINTS; 52b6f35a76SGreg Roach 53345fcc8aSGreg Roach public float $right_margin = 9.9 * self::MM_TO_POINTS; 54b6f35a76SGreg Roach 55345fcc8aSGreg Roach public float $top_margin = 26.8 * self::MM_TO_POINTS; 56b6f35a76SGreg Roach 57345fcc8aSGreg Roach public float $bottom_margin = 21.6 * self::MM_TO_POINTS; 58b6f35a76SGreg Roach 59345fcc8aSGreg Roach public float $header_margin = 4.9 * self::MM_TO_POINTS; 60b6f35a76SGreg Roach 61345fcc8aSGreg Roach public float $footer_margin = 9.9 * self::MM_TO_POINTS; 62b6f35a76SGreg Roach 63b6f35a76SGreg Roach /** @var string Page orientation (portrait, landscape) */ 64345fcc8aSGreg Roach public string $orientation = 'portrait'; 65b6f35a76SGreg Roach 66b6f35a76SGreg Roach /** @var string Page format name */ 67345fcc8aSGreg Roach public string $page_format = 'A4'; 68b6f35a76SGreg Roach 69b6f35a76SGreg Roach /** @var float Height of page format in points */ 70345fcc8aSGreg Roach public float $page_height = 0.0; 71b6f35a76SGreg Roach 72b6f35a76SGreg Roach /** @var float Width of page format in points */ 73345fcc8aSGreg Roach public float $page_width = 0.0; 74b6f35a76SGreg Roach 75*f315390bSGreg Roach /** @var array<array{'name': string, 'font': string, 'style': string, 'size': float}> Styles elements found in the document */ 76345fcc8aSGreg Roach public array $styles = []; 77b6f35a76SGreg Roach 78b6f35a76SGreg Roach /** @var string The default Report font name */ 79345fcc8aSGreg Roach public string $default_font = 'dejavusans'; 80b6f35a76SGreg Roach 81b6f35a76SGreg Roach /** @var float The default Report font size */ 82345fcc8aSGreg Roach public float $default_font_size = 12.0; 83b6f35a76SGreg Roach 84a393a2a1SGreg Roach /** @var string Header (H), Body (B) or Footer (F) */ 85345fcc8aSGreg Roach public string $processing = 'H'; 86b6f35a76SGreg Roach 87b6f35a76SGreg Roach /** @var bool RTL Language (false=LTR, true=RTL) */ 88345fcc8aSGreg Roach public bool $rtl = false; 89b6f35a76SGreg Roach 90b6f35a76SGreg Roach /** @var bool Show the Generated by... (true=show the text) */ 910a709a28SGreg Roach public bool $show_generated_by = true; 92b6f35a76SGreg Roach 93b6f35a76SGreg Roach /** @var string Generated By... text */ 94345fcc8aSGreg Roach public string $generated_by = ''; 95b6f35a76SGreg Roach 96b6f35a76SGreg Roach /** @var string The report title */ 97345fcc8aSGreg Roach public string $title = ''; 98b6f35a76SGreg Roach 99b6f35a76SGreg Roach /** @var string Author of the report, the users full name */ 100345fcc8aSGreg Roach public string $rauthor = Webtrees::NAME . ' ' . Webtrees::VERSION; 101b6f35a76SGreg Roach 102b6f35a76SGreg Roach /** @var string Keywords */ 103345fcc8aSGreg Roach public string $rkeywords = ''; 104b6f35a76SGreg Roach 105b6f35a76SGreg Roach /** @var string Report Description / Subject */ 106345fcc8aSGreg Roach public string $rsubject = ''; 107345fcc8aSGreg Roach 108345fcc8aSGreg Roach /** @var array<ReportBaseElement|string> */ 109345fcc8aSGreg Roach public array $headerElements = []; 110345fcc8aSGreg Roach 111345fcc8aSGreg Roach /** @var array<ReportBaseElement|string> */ 11277bab461SGreg Roach public array $footerElements = []; 113345fcc8aSGreg Roach 114345fcc8aSGreg Roach /** @var array<ReportBaseElement|string> */ 115345fcc8aSGreg Roach public array $bodyElements = []; 116345fcc8aSGreg Roach 117345fcc8aSGreg Roach public string $currentStyle = ''; 118b6f35a76SGreg Roach 119b6f35a76SGreg Roach /** 120b6f35a76SGreg Roach * Clear the Header. 121b6f35a76SGreg Roach * 122b6f35a76SGreg Roach * @return void 123b6f35a76SGreg Roach */ 124b6f35a76SGreg Roach abstract public function clearHeader(): void; 125b6f35a76SGreg Roach 126b6f35a76SGreg Roach /** 127b6f35a76SGreg Roach * @param ReportBaseElement|string $element 128b6f35a76SGreg Roach * 129b6f35a76SGreg Roach * @return void 130b6f35a76SGreg Roach */ 131345fcc8aSGreg Roach public function addElement($element): void 132345fcc8aSGreg Roach { 133345fcc8aSGreg Roach if ($this->processing === 'B') { 134345fcc8aSGreg Roach $this->addElementToBody($element); 135345fcc8aSGreg Roach } elseif ($this->processing === 'H') { 136345fcc8aSGreg Roach $this->addElementToHeader($element); 137345fcc8aSGreg Roach } elseif ($this->processing === 'F') { 138345fcc8aSGreg Roach $this->addElementToFooter($element); 139345fcc8aSGreg Roach } 140345fcc8aSGreg Roach } 141345fcc8aSGreg Roach 142345fcc8aSGreg Roach /** 143345fcc8aSGreg Roach * @param ReportBaseElement|string $element 144345fcc8aSGreg Roach * 145345fcc8aSGreg Roach * @return void 146345fcc8aSGreg Roach */ 147345fcc8aSGreg Roach public function addElementToHeader($element): void 148345fcc8aSGreg Roach { 149345fcc8aSGreg Roach $this->headerElements[] = $element; 150345fcc8aSGreg Roach } 151345fcc8aSGreg Roach 152345fcc8aSGreg Roach /** 153345fcc8aSGreg Roach * @param ReportBaseElement|string $element 154345fcc8aSGreg Roach * 155345fcc8aSGreg Roach * @return void 156345fcc8aSGreg Roach */ 157345fcc8aSGreg Roach public function addElementToBody($element): void 158345fcc8aSGreg Roach { 159345fcc8aSGreg Roach $this->bodyElements[] = $element; 160345fcc8aSGreg Roach } 161345fcc8aSGreg Roach 162345fcc8aSGreg Roach /** 163345fcc8aSGreg Roach * @param ReportBaseElement|string $element 164345fcc8aSGreg Roach * 165345fcc8aSGreg Roach * @return void 166345fcc8aSGreg Roach */ 167345fcc8aSGreg Roach public function addElementToFooter($element): void 168345fcc8aSGreg Roach { 169345fcc8aSGreg Roach $this->footerElements[] = $element; 170345fcc8aSGreg Roach } 171b6f35a76SGreg Roach 172b6f35a76SGreg Roach /** 173b6f35a76SGreg Roach * Run the report. 174b6f35a76SGreg Roach * 175b6f35a76SGreg Roach * @return void 176b6f35a76SGreg Roach */ 177b6f35a76SGreg Roach abstract public function run(): void; 178b6f35a76SGreg Roach 179b6f35a76SGreg Roach /** 180b6f35a76SGreg Roach * Create a new Cell object. 181b6f35a76SGreg Roach * 18277bab461SGreg Roach * @param float $width cell width (expressed in points) 18377bab461SGreg Roach * @param float $height cell height (expressed in points) 18477bab461SGreg Roach * @param string $border Border style 185e63974caSStefan Weil * @param string $align Text alignment 186b6f35a76SGreg Roach * @param string $bgcolor Background color code 187b6f35a76SGreg Roach * @param string $style The name of the text style 188b6f35a76SGreg Roach * @param int $ln Indicates where the current position should go after the call 189*f315390bSGreg Roach * @param float $top Y-position 190*f315390bSGreg Roach * @param float $left X-position 191*f315390bSGreg Roach * @param bool $fill Indicates if the cell background must be painted (1) or transparent (0). Default value: 1 192e63974caSStefan Weil * @param int $stretch Stretch character mode 193b6f35a76SGreg Roach * @param string $bocolor Border color 194b6f35a76SGreg Roach * @param string $tcolor Text color 195b6f35a76SGreg Roach * @param bool $reseth 196b6f35a76SGreg Roach * 197b6f35a76SGreg Roach * @return ReportBaseCell 198b6f35a76SGreg Roach */ 19924f2a3afSGreg Roach abstract public function createCell( 20077bab461SGreg Roach float $width, 20177bab461SGreg Roach float $height, 20277bab461SGreg Roach string $border, 20324f2a3afSGreg Roach string $align, 20424f2a3afSGreg Roach string $bgcolor, 20524f2a3afSGreg Roach string $style, 20624f2a3afSGreg Roach int $ln, 207*f315390bSGreg Roach float $top, 208*f315390bSGreg Roach float $left, 209*f315390bSGreg Roach bool $fill, 21024f2a3afSGreg Roach int $stretch, 21124f2a3afSGreg Roach string $bocolor, 21224f2a3afSGreg Roach string $tcolor, 21324f2a3afSGreg Roach bool $reseth 21424f2a3afSGreg Roach ): ReportBaseCell; 215b6f35a76SGreg Roach 216b6f35a76SGreg Roach /** 217b6f35a76SGreg Roach * Create a new TextBox object. 218b6f35a76SGreg Roach * 219b6f35a76SGreg Roach * @param float $width Text box width 220b6f35a76SGreg Roach * @param float $height Text box height 221b6f35a76SGreg Roach * @param bool $border 222b6f35a76SGreg Roach * @param string $bgcolor Background color code in HTML 223b6f35a76SGreg Roach * @param bool $newline 224b6f35a76SGreg Roach * @param float $left 225b6f35a76SGreg Roach * @param float $top 226b6f35a76SGreg Roach * @param bool $pagecheck 227b6f35a76SGreg Roach * @param string $style 228b6f35a76SGreg Roach * @param bool $fill 229b6f35a76SGreg Roach * @param bool $padding 230b6f35a76SGreg Roach * @param bool $reseth 231b6f35a76SGreg Roach * 232b6f35a76SGreg Roach * @return ReportBaseTextbox 233b6f35a76SGreg Roach */ 234b6f35a76SGreg Roach abstract public function createTextBox( 235b6f35a76SGreg Roach float $width, 236b6f35a76SGreg Roach float $height, 237b6f35a76SGreg Roach bool $border, 238b6f35a76SGreg Roach string $bgcolor, 239b6f35a76SGreg Roach bool $newline, 240b6f35a76SGreg Roach float $left, 241b6f35a76SGreg Roach float $top, 242b6f35a76SGreg Roach bool $pagecheck, 243b6f35a76SGreg Roach string $style, 244b6f35a76SGreg Roach bool $fill, 245b6f35a76SGreg Roach bool $padding, 246b6f35a76SGreg Roach bool $reseth 247b6f35a76SGreg Roach ): ReportBaseTextbox; 248b6f35a76SGreg Roach 249b6f35a76SGreg Roach /** 250b6f35a76SGreg Roach * Create a text element. 251b6f35a76SGreg Roach * 252b6f35a76SGreg Roach * @param string $style 253b6f35a76SGreg Roach * @param string $color 254b6f35a76SGreg Roach * 255b6f35a76SGreg Roach * @return ReportBaseText 256b6f35a76SGreg Roach */ 257b6f35a76SGreg Roach abstract public function createText(string $style, string $color): ReportBaseText; 258b6f35a76SGreg Roach 259b6f35a76SGreg Roach /** 260b6f35a76SGreg Roach * Create a line. 261b6f35a76SGreg Roach * 262b6f35a76SGreg Roach * @param float $x1 263b6f35a76SGreg Roach * @param float $y1 264b6f35a76SGreg Roach * @param float $x2 265b6f35a76SGreg Roach * @param float $y2 266b6f35a76SGreg Roach * 267b6f35a76SGreg Roach * @return ReportBaseLine 268b6f35a76SGreg Roach */ 269b6f35a76SGreg Roach abstract public function createLine(float $x1, float $y1, float $x2, float $y2): ReportBaseLine; 270b6f35a76SGreg Roach 271b6f35a76SGreg Roach /** 272b6f35a76SGreg Roach * Create a new image object. 273b6f35a76SGreg Roach * 274b6f35a76SGreg Roach * @param string $file Filename 275b6f35a76SGreg Roach * @param float $x 276b6f35a76SGreg Roach * @param float $y 277b6f35a76SGreg Roach * @param float $w Image width 278b6f35a76SGreg Roach * @param float $h Image height 279b6f35a76SGreg Roach * @param string $align L:left, C:center, R:right or empty to use x/y 280b6f35a76SGreg Roach * @param string $ln T:same line, N:next line 281b6f35a76SGreg Roach * 282b6f35a76SGreg Roach * @return ReportBaseImage 283b6f35a76SGreg Roach */ 284b6f35a76SGreg Roach abstract public function createImage(string $file, float $x, float $y, float $w, float $h, string $align, string $ln): ReportBaseImage; 285b6f35a76SGreg Roach 286b6f35a76SGreg Roach /** 287b6f35a76SGreg Roach * Create a new image object from Media Object. 288b6f35a76SGreg Roach * 289b6f35a76SGreg Roach * @param MediaFile $media_file 290b6f35a76SGreg Roach * @param float $x 291b6f35a76SGreg Roach * @param float $y 292b6f35a76SGreg Roach * @param float $w Image width 293b6f35a76SGreg Roach * @param float $h Image height 294b6f35a76SGreg Roach * @param string $align L:left, C:center, R:right or empty to use x/y 295b6f35a76SGreg Roach * @param string $ln T:same line, N:next line 296b6f35a76SGreg Roach * 297b6f35a76SGreg Roach * @return ReportBaseImage 298b6f35a76SGreg Roach */ 299b6f35a76SGreg Roach abstract public function createImageFromObject( 300b6f35a76SGreg Roach MediaFile $media_file, 301b6f35a76SGreg Roach float $x, 302b6f35a76SGreg Roach float $y, 303b6f35a76SGreg Roach float $w, 304b6f35a76SGreg Roach float $h, 305b6f35a76SGreg Roach string $align, 3069458f20aSGreg Roach string $ln 307b6f35a76SGreg Roach ): ReportBaseImage; 308b6f35a76SGreg Roach 309b6f35a76SGreg Roach /** 310b6f35a76SGreg Roach * Create a new Footnote object. 311b6f35a76SGreg Roach * 312b6f35a76SGreg Roach * @param string $style Style name 313b6f35a76SGreg Roach * 314b6f35a76SGreg Roach * @return ReportBaseFootnote 315b6f35a76SGreg Roach */ 31624f2a3afSGreg Roach abstract public function createFootnote(string $style): ReportBaseFootnote; 317b6f35a76SGreg Roach 318b6f35a76SGreg Roach /** 319b6f35a76SGreg Roach * Initial Setup 320b6f35a76SGreg Roach * Setting up document wide defaults that will be inherited of the report modules 321b6f35a76SGreg Roach * As DEFAULT A4 and Portrait will be used if not set 322b6f35a76SGreg Roach * 323b6f35a76SGreg Roach * @return void 324b6f35a76SGreg Roach */ 325b6f35a76SGreg Roach public function setup(): void 326b6f35a76SGreg Roach { 327b6f35a76SGreg Roach $this->rtl = I18N::direction() === 'rtl'; 328b6f35a76SGreg Roach 329b6f35a76SGreg Roach $this->rkeywords = ''; 330b6f35a76SGreg Roach 331b6f35a76SGreg Roach // I18N: This is a report footer. %s is the name of the application. 332b6f35a76SGreg Roach $this->generated_by = I18N::translate('Generated by %s', Webtrees::NAME . ' ' . Webtrees::VERSION); 333b6f35a76SGreg Roach 334b6f35a76SGreg Roach // Paper size - defaults to A4 if the report fails to define a size. 335b6f35a76SGreg Roach [$this->page_width, $this->page_height] = self::PAPER_SIZES[$this->page_format] ?? self::PAPER_SIZES['A4']; 336b6f35a76SGreg Roach } 337b6f35a76SGreg Roach 338b6f35a76SGreg Roach /** 339a393a2a1SGreg Roach * Process the Header, Body or Footer 340b6f35a76SGreg Roach * 341a393a2a1SGreg Roach * @param string $p Header (H), Body (B) or Footer (F) 342b6f35a76SGreg Roach * 343b6f35a76SGreg Roach * @return void 344b6f35a76SGreg Roach */ 345b6f35a76SGreg Roach public function setProcessing(string $p): void 346b6f35a76SGreg Roach { 347b6f35a76SGreg Roach $this->processing = $p; 348b6f35a76SGreg Roach } 349b6f35a76SGreg Roach 350b6f35a76SGreg Roach /** 351b6f35a76SGreg Roach * Add the Title when raw character data is used in Title 352b6f35a76SGreg Roach * 353b6f35a76SGreg Roach * @param string $data 354b6f35a76SGreg Roach * 355b6f35a76SGreg Roach * @return void 356b6f35a76SGreg Roach */ 357b6f35a76SGreg Roach public function addTitle(string $data): void 358b6f35a76SGreg Roach { 359b6f35a76SGreg Roach $this->title .= $data; 360b6f35a76SGreg Roach } 361b6f35a76SGreg Roach 362b6f35a76SGreg Roach /** 363b6f35a76SGreg Roach * Add the Description when raw character data is used in Description 364b6f35a76SGreg Roach * 365b6f35a76SGreg Roach * @param string $data 366b6f35a76SGreg Roach * 367b6f35a76SGreg Roach * @return void 368b6f35a76SGreg Roach */ 369b6f35a76SGreg Roach public function addDescription(string $data): void 370b6f35a76SGreg Roach { 371b6f35a76SGreg Roach $this->rsubject .= $data; 372b6f35a76SGreg Roach } 373b6f35a76SGreg Roach 374b6f35a76SGreg Roach /** 375b6f35a76SGreg Roach * Add Style to Styles array 376b6f35a76SGreg Roach * 377*f315390bSGreg Roach * @param array{'name': string, 'font': string, 'style': string, 'size': float} $style 378b6f35a76SGreg Roach * 379b6f35a76SGreg Roach * @return void 380b6f35a76SGreg Roach */ 381b6f35a76SGreg Roach public function addStyle(array $style): void 382b6f35a76SGreg Roach { 383b6f35a76SGreg Roach $this->styles[$style['name']] = $style; 384b6f35a76SGreg Roach } 385b6f35a76SGreg Roach 386b6f35a76SGreg Roach /** 387b6f35a76SGreg Roach * Get a style from the Styles array 388b6f35a76SGreg Roach * 389b6f35a76SGreg Roach * @param string $s Style name 390b6f35a76SGreg Roach * 391*f315390bSGreg Roach * @return array{'name': string, 'font': string, 'style': string, 'size': float} 392b6f35a76SGreg Roach */ 393b6f35a76SGreg Roach public function getStyle(string $s): array 394b6f35a76SGreg Roach { 3954348fc02SGreg Roach return $this->styles[$s]; 396b6f35a76SGreg Roach } 397b6f35a76SGreg Roach} 398