Viewing file: Pdf.php (28.83 KB) -rwxr-x--- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
<?php /** * PhpMyAdmin\Plugins\Export\Helpers\Pdf class */
declare(strict_types=1);
namespace PhpMyAdmin\Plugins\Export\Helpers;
use PhpMyAdmin\ConfigStorage\Relation; use PhpMyAdmin\DatabaseInterface; use PhpMyAdmin\Dbal\ResultInterface; use PhpMyAdmin\FieldMetadata; use PhpMyAdmin\Pdf as PdfLib; use PhpMyAdmin\Transformations; use PhpMyAdmin\Util; use TCPDF_STATIC;
use function __; use function array_key_exists; use function count; use function ksort;
/** * Adapted from a LGPL script by Philip Clarke */ class Pdf extends PdfLib { /** @var array */ public $tablewidths;
/** @var array */ public $headerset;
/** @var int|float */ private $dataY;
/** @var int|float */ private $cellFontSize;
/** @var int */ private $titleFontSize;
/** @var string */ private $titleText;
/** @var string */ private $dbAlias;
/** @var string */ private $tableAlias;
/** @var string */ private $purpose;
/** @var array */ private $colTitles;
/** @var ResultInterface */ private $results;
/** @var array */ private $colAlign;
/** @var mixed */ private $titleWidth;
/** @var mixed */ private $colFits;
/** @var array */ private $displayColumn;
/** @var int */ private $numFields;
/** @var FieldMetadata[] */ private $fields;
/** @var int|float */ private $sColWidth;
/** @var string */ private $currentDb;
/** @var string */ private $currentTable;
/** @var array */ private $aliases;
/** @var Relation */ private $relation;
/** @var Transformations */ private $transformations;
/** * Constructs PDF and configures standard parameters. * * @param string $orientation page orientation * @param string $unit unit * @param string $format the format used for pages * @param bool $unicode true means that the input text is unicode * @param string $encoding charset encoding; default is UTF-8. * @param bool $diskcache DEPRECATED TCPDF FEATURE * @param false|int $pdfa If not false, set the document to PDF/A mode and the good version (1 or 3) */ public function __construct( $orientation = 'P', $unit = 'mm', $format = 'A4', $unicode = true, $encoding = 'UTF-8', $diskcache = false, $pdfa = false ) { global $dbi;
parent::__construct($orientation, $unit, $format, $unicode, $encoding, $diskcache, $pdfa); $this->relation = new Relation($dbi); $this->transformations = new Transformations(); }
/** * Add page if needed. * * @param float|int $h cell height. Default value: 0 * @param mixed $y starting y position, leave empty for current * position * @param bool $addpage if true add a page, otherwise only return * the true/false state */ public function checkPageBreak($h = 0, $y = '', $addpage = true): bool { if (TCPDF_STATIC::empty_string($y)) { $y = $this->y; }
$current_page = $this->page; // phpcs:ignore Squiz.NamingConventions.ValidVariableName.MemberNotCamelCaps if ($y + $h > $this->PageBreakTrigger && ! $this->InFooter && $this->AcceptPageBreak()) { if ($addpage) { //Automatic page break $x = $this->x; // phpcs:ignore Squiz.NamingConventions.ValidVariableName.MemberNotCamelCaps $this->AddPage($this->CurOrientation); $this->y = $this->dataY; $oldpage = $this->page - 1;
$this_page_orm = $this->pagedim[$this->page]['orm']; $old_page_orm = $this->pagedim[$oldpage]['orm']; $this_page_olm = $this->pagedim[$this->page]['olm']; $old_page_olm = $this->pagedim[$oldpage]['olm']; if ($this->rtl) { if ($this_page_orm != $old_page_orm) { $this->x = $x - ($this_page_orm - $old_page_orm); } else { $this->x = $x; } } else { if ($this_page_olm != $old_page_olm) { $this->x = $x + $this_page_olm - $old_page_olm; } else { $this->x = $x; } } }
return true; }
// account for columns mode return $current_page != $this->page; }
/** * This method is used to render the page header. */ // phpcs:ignore PSR1.Methods.CamelCapsMethodName.NotCamelCaps public function Header(): void { global $maxY; // We don't want automatic page breaks while generating header // as this can lead to infinite recursion as auto generated page // will want header as well causing another page break // FIXME: Better approach might be to try to compact the content $this->setAutoPageBreak(false); // Check if header for this page already exists // phpcs:disable Squiz.NamingConventions.ValidVariableName.MemberNotCamelCaps if (! isset($this->headerset[$this->page])) { $this->setY($this->tMargin - ($this->FontSizePt / $this->k) * 5); $this->cellFontSize = $this->FontSizePt; $this->setFont(PdfLib::PMA_PDF_FONT, '', ($this->titleFontSize ?: $this->FontSizePt)); $this->Cell(0, $this->FontSizePt, $this->titleText, 0, 1, 'C'); $this->setFont(PdfLib::PMA_PDF_FONT, '', $this->cellFontSize); $this->setY($this->tMargin - ($this->FontSizePt / $this->k) * 2.5); $this->Cell( 0, $this->FontSizePt, __('Database:') . ' ' . $this->dbAlias . ', ' . __('Table:') . ' ' . $this->tableAlias . ', ' . __('Purpose:') . ' ' . $this->purpose, 0, 1, 'L' ); $l = $this->lMargin; foreach ($this->colTitles as $col => $txt) { $this->setXY($l, $this->tMargin); $this->MultiCell($this->tablewidths[$col], $this->FontSizePt, $txt); $l += $this->tablewidths[$col]; $maxY = $maxY < $this->GetY() ? $this->GetY() : $maxY; }
$this->setXY($this->lMargin, $this->tMargin); $this->setFillColor(200, 200, 200); $l = $this->lMargin; foreach ($this->colTitles as $col => $txt) { $this->setXY($l, $this->tMargin); $this->Cell($this->tablewidths[$col], $maxY - $this->tMargin, '', 1, 0, 'L', true); $this->setXY($l, $this->tMargin); $this->MultiCell($this->tablewidths[$col], $this->FontSizePt, $txt, 0, 'C'); $l += $this->tablewidths[$col]; }
$this->setFillColor(255, 255, 255); // set headerset $this->headerset[$this->page] = 1; }
// phpcs:enable
$this->dataY = $maxY; $this->setAutoPageBreak(true); }
/** * Generate table * * @param int $lineheight Height of line */ public function morepagestable($lineheight = 8): void { // some things to set and 'remember' $l = $this->lMargin; $startheight = $h = $this->dataY; $startpage = $currpage = $this->page;
// calculate the whole width $fullwidth = 0; foreach ($this->tablewidths as $width) { $fullwidth += $width; }
// Now let's start to write the table $row = 0; $tmpheight = []; $maxpage = $this->page;
while ($data = $this->results->fetchRow()) { $this->page = $currpage; // write the horizontal borders $this->Line($l, $h, $fullwidth + $l, $h); // write the content and remember the height of the highest col foreach ($data as $col => $txt) { $this->page = $currpage; $this->setXY($l, $h); if ($this->tablewidths[$col] > 0) { $this->MultiCell($this->tablewidths[$col], $lineheight, $txt, 0, $this->colAlign[$col]); $l += $this->tablewidths[$col]; }
if (! isset($tmpheight[$row . '-' . $this->page])) { $tmpheight[$row . '-' . $this->page] = 0; }
if ($tmpheight[$row . '-' . $this->page] < $this->GetY()) { $tmpheight[$row . '-' . $this->page] = $this->GetY(); }
if ($this->page > $maxpage) { $maxpage = $this->page; }
unset($data[$col]); }
// get the height we were in the last used page $h = $tmpheight[$row . '-' . $maxpage]; // set the "pointer" to the left margin $l = $this->lMargin; // set the $currpage to the last page $currpage = $maxpage; unset($data[$row]); $row++; }
// draw the borders // we start adding a horizontal line on the last page $this->page = $maxpage; $this->Line($l, $h, $fullwidth + $l, $h); // now we start at the top of the document and walk down for ($i = $startpage; $i <= $maxpage; $i++) { $this->page = $i; $l = $this->lMargin; $t = $i == $startpage ? $startheight : $this->tMargin; $lh = $i == $maxpage ? $h : $this->h - $this->bMargin; $this->Line($l, $t, $l, $lh); foreach ($this->tablewidths as $width) { $l += $width; $this->Line($l, $t, $l, $lh); } }
// set it to the last page, if not it'll cause some problems $this->page = $maxpage; }
/** * Defines the top margin. * The method can be called before creating the first page. * * @param float $topMargin the margin */ public function setTopMargin($topMargin): void { $this->tMargin = $topMargin; }
/** * Prints triggers * * @param string $db database name * @param string $table table name */ public function getTriggers($db, $table): void { global $dbi;
$triggers = $dbi->getTriggers($db, $table); if ($triggers === []) { return; //prevents printing blank trigger list for any table }
unset( $this->tablewidths, $this->colTitles, $this->titleWidth, $this->colFits, $this->displayColumn, $this->colAlign );
/** * Making table heading * Keeping column width constant */ $this->colTitles[0] = __('Name'); $this->tablewidths[0] = 90; $this->colTitles[1] = __('Time'); $this->tablewidths[1] = 80; $this->colTitles[2] = __('Event'); $this->tablewidths[2] = 40; $this->colTitles[3] = __('Definition'); $this->tablewidths[3] = 240;
for ($columns_cnt = 0; $columns_cnt < 4; $columns_cnt++) { $this->colAlign[$columns_cnt] = 'L'; $this->displayColumn[$columns_cnt] = true; }
// Starting to fill table with required info
$this->setY($this->tMargin); $this->AddPage(); $this->setFont(PdfLib::PMA_PDF_FONT, '', 9);
$l = $this->lMargin; $startheight = $h = $this->dataY; $startpage = $currpage = $this->page;
// calculate the whole width $fullwidth = 0; foreach ($this->tablewidths as $width) { $fullwidth += $width; }
$row = 0; $tmpheight = []; $maxpage = $this->page; $data = [];
foreach ($triggers as $trigger) { $data[] = $trigger['name']; $data[] = $trigger['action_timing']; $data[] = $trigger['event_manipulation']; $data[] = $trigger['definition']; $this->page = $currpage; // write the horizontal borders $this->Line($l, $h, $fullwidth + $l, $h); // write the content and remember the height of the highest col foreach ($data as $col => $txt) { $this->page = $currpage; $this->setXY($l, $h); if ($this->tablewidths[$col] > 0) { $this->MultiCell( $this->tablewidths[$col], // phpcs:ignore Squiz.NamingConventions.ValidVariableName.MemberNotCamelCaps $this->FontSizePt, $txt, 0, $this->colAlign[$col] ); $l += $this->tablewidths[$col]; }
if (! isset($tmpheight[$row . '-' . $this->page])) { $tmpheight[$row . '-' . $this->page] = 0; }
if ($tmpheight[$row . '-' . $this->page] < $this->GetY()) { $tmpheight[$row . '-' . $this->page] = $this->GetY(); }
if ($this->page <= $maxpage) { continue; }
$maxpage = $this->page; }
// get the height we were in the last used page $h = $tmpheight[$row . '-' . $maxpage]; // set the "pointer" to the left margin $l = $this->lMargin; // set the $currpage to the last page $currpage = $maxpage; unset($data); $row++; }
// draw the borders // we start adding a horizontal line on the last page $this->page = $maxpage; $this->Line($l, $h, $fullwidth + $l, $h); // now we start at the top of the document and walk down for ($i = $startpage; $i <= $maxpage; $i++) { $this->page = $i; $l = $this->lMargin; $t = $i == $startpage ? $startheight : $this->tMargin; $lh = $i == $maxpage ? $h : $this->h - $this->bMargin; $this->Line($l, $t, $l, $lh); foreach ($this->tablewidths as $width) { $l += $width; $this->Line($l, $t, $l, $lh); } }
// set it to the last page, if not it'll cause some problems $this->page = $maxpage; }
/** * Print $table's CREATE definition * * @param string $db the database name * @param string $table the table name * @param bool $do_relation whether to include relation comments * @param bool $do_comments whether to include the pmadb-style column * comments as comments in the structure; * this is deprecated but the parameter is * left here because /export calls * PMA_exportStructure() also for other * export types which use this parameter * @param bool $do_mime whether to include mime comments * @param bool $view whether we're handling a view * @param array $aliases aliases of db/table/columns */ public function getTableDef( $db, $table, $do_relation, $do_comments, $do_mime, $view = false, array $aliases = [] ): void { global $dbi;
$relationParameters = $this->relation->getRelationParameters();
unset( $this->tablewidths, $this->colTitles, $this->titleWidth, $this->colFits, $this->displayColumn, $this->colAlign );
/** * Gets fields properties */ $dbi->selectDb($db);
/** * All these three checks do_relation, do_comment and do_mime is * not required. As presently all are set true by default. * But when, methods to take user input will be developed, * it will be of use */ // Check if we can use Relations if ($do_relation) { // Find which tables are related with the current one and write it in // an array $res_rel = $this->relation->getForeigners($db, $table); $have_rel = ! empty($res_rel); } else { $have_rel = false; }
//column count and table heading
$this->colTitles[0] = __('Column'); $this->tablewidths[0] = 90; $this->colTitles[1] = __('Type'); $this->tablewidths[1] = 80; $this->colTitles[2] = __('Null'); $this->tablewidths[2] = 40; $this->colTitles[3] = __('Default'); $this->tablewidths[3] = 120;
for ($columns_cnt = 0; $columns_cnt < 4; $columns_cnt++) { $this->colAlign[$columns_cnt] = 'L'; $this->displayColumn[$columns_cnt] = true; }
if ($do_relation && $have_rel) { $this->colTitles[$columns_cnt] = __('Links to'); $this->displayColumn[$columns_cnt] = true; $this->colAlign[$columns_cnt] = 'L'; $this->tablewidths[$columns_cnt] = 120; }
if ($do_comments) { $columns_cnt++; $this->colTitles[$columns_cnt] = __('Comments'); $this->displayColumn[$columns_cnt] = true; $this->colAlign[$columns_cnt] = 'L'; $this->tablewidths[$columns_cnt] = 120; }
if ($do_mime && $relationParameters->browserTransformationFeature !== null) { $columns_cnt++; $this->colTitles[$columns_cnt] = __('Media type'); $this->displayColumn[$columns_cnt] = true; $this->colAlign[$columns_cnt] = 'L'; $this->tablewidths[$columns_cnt] = 120; }
// Starting to fill table with required info
$this->setY($this->tMargin); $this->AddPage(); $this->setFont(PdfLib::PMA_PDF_FONT, '', 9);
// Now let's start to write the table structure
if ($do_comments) { $comments = $this->relation->getComments($db, $table); }
if ($do_mime && $relationParameters->browserTransformationFeature !== null) { $mime_map = $this->transformations->getMime($db, $table, true); }
$columns = $dbi->getColumns($db, $table);
// some things to set and 'remember' $l = $this->lMargin; $startheight = $h = $this->dataY; $startpage = $currpage = $this->page; // calculate the whole width $fullwidth = 0; foreach ($this->tablewidths as $width) { $fullwidth += $width; }
$row = 0; $tmpheight = []; $maxpage = $this->page; $data = [];
// fun begin foreach ($columns as $column) { $extracted_columnspec = Util::extractColumnSpec($column['Type']);
$type = $extracted_columnspec['print_type']; if (empty($type)) { $type = ' '; }
if (! isset($column['Default'])) { if ($column['Null'] !== 'NO') { $column['Default'] = 'NULL'; } }
$data[] = $column['Field']; $data[] = $type; $data[] = $column['Null'] == '' || $column['Null'] === 'NO' ? 'No' : 'Yes'; $data[] = $column['Default'] ?? '';
$field_name = $column['Field'];
if ($do_relation && $have_rel) { $data[] = isset($res_rel[$field_name]) ? $res_rel[$field_name]['foreign_table'] . ' (' . $res_rel[$field_name]['foreign_field'] . ')' : ''; }
if ($do_comments) { $data[] = $comments[$field_name] ?? ''; }
if ($do_mime) { $data[] = isset($mime_map[$field_name]) ? $mime_map[$field_name]['mimetype'] : ''; }
$this->page = $currpage; // write the horizontal borders $this->Line($l, $h, $fullwidth + $l, $h); // write the content and remember the height of the highest col foreach ($data as $col => $txt) { $this->page = $currpage; $this->setXY($l, $h); if ($this->tablewidths[$col] > 0) { $this->MultiCell( $this->tablewidths[$col], // phpcs:ignore Squiz.NamingConventions.ValidVariableName.MemberNotCamelCaps $this->FontSizePt, $txt, 0, $this->colAlign[$col] ); $l += $this->tablewidths[$col]; }
if (! isset($tmpheight[$row . '-' . $this->page])) { $tmpheight[$row . '-' . $this->page] = 0; }
if ($tmpheight[$row . '-' . $this->page] < $this->GetY()) { $tmpheight[$row . '-' . $this->page] = $this->GetY(); }
if ($this->page <= $maxpage) { continue; }
$maxpage = $this->page; }
// get the height we were in the last used page $h = $tmpheight[$row . '-' . $maxpage]; // set the "pointer" to the left margin $l = $this->lMargin; // set the $currpage to the last page $currpage = $maxpage; unset($data); $row++; }
// draw the borders // we start adding a horizontal line on the last page $this->page = $maxpage; $this->Line($l, $h, $fullwidth + $l, $h); // now we start at the top of the document and walk down for ($i = $startpage; $i <= $maxpage; $i++) { $this->page = $i; $l = $this->lMargin; $t = $i == $startpage ? $startheight : $this->tMargin; $lh = $i == $maxpage ? $h : $this->h - $this->bMargin; $this->Line($l, $t, $l, $lh); foreach ($this->tablewidths as $width) { $l += $width; $this->Line($l, $t, $l, $lh); } }
// set it to the last page, if not it'll cause some problems $this->page = $maxpage; }
/** * MySQL report * * @param string $query Query to execute */ public function mysqlReport($query): void { global $dbi;
unset( $this->tablewidths, $this->colTitles, $this->titleWidth, $this->colFits, $this->displayColumn, $this->colAlign );
/** * Pass 1 for column widths */ $this->results = $dbi->query($query, DatabaseInterface::CONNECT_USER, DatabaseInterface::QUERY_UNBUFFERED); $this->numFields = $this->results->numFields(); $this->fields = $dbi->getFieldsMeta($this->results);
// sColWidth = starting col width (an average size width) $availableWidth = $this->w - $this->lMargin - $this->rMargin; $this->sColWidth = $availableWidth / $this->numFields; $totalTitleWidth = 0;
// loop through results header and set initial // col widths/ titles/ alignment // if a col title is less than the starting col width, // reduce that column size $colFits = []; $titleWidth = []; for ($i = 0; $i < $this->numFields; $i++) { $col_as = $this->fields[$i]->name; $db = $this->currentDb; $table = $this->currentTable; if (! empty($this->aliases[$db]['tables'][$table]['columns'][$col_as])) { $col_as = $this->aliases[$db]['tables'][$table]['columns'][$col_as]; }
/** @var float $stringWidth */ $stringWidth = $this->GetStringWidth($col_as); $stringWidth += 6; // save the real title's width $titleWidth[$i] = $stringWidth; $totalTitleWidth += $stringWidth;
// set any column titles less than the start width to // the column title width if ($stringWidth < $this->sColWidth) { $colFits[$i] = $stringWidth; }
$this->colTitles[$i] = $col_as; $this->displayColumn[$i] = true;
$this->colAlign[$i] = 'L';
if ($this->fields[$i]->isType(FieldMetadata::TYPE_INT)) { $this->colAlign[$i] = 'R'; }
if (! $this->fields[$i]->isType(FieldMetadata::TYPE_BLOB)) { continue; }
/** * @todo do not deactivate completely the display * but show the field's name and [BLOB] */ if ($this->fields[$i]->isBinary()) { $this->displayColumn[$i] = false; unset($this->colTitles[$i]); }
$this->colAlign[$i] = 'L'; }
// title width verification if ($totalTitleWidth > $availableWidth) { $adjustingMode = true; } else { $adjustingMode = false; // we have enough space for all the titles at their // original width so use the true title's width foreach ($titleWidth as $key => $val) { $colFits[$key] = $val; } }
// loop through the data; any column whose contents // is greater than the column size is resized /** * @todo force here a LIMIT to avoid reading all rows */ while ($row = $this->results->fetchRow()) { foreach ($colFits as $key => $val) { /** @var float $stringWidth */ $stringWidth = $this->GetStringWidth($row[$key]); $stringWidth += 6; if ($adjustingMode && ($stringWidth > $this->sColWidth)) { // any column whose data's width is bigger than // the start width is now discarded unset($colFits[$key]); } else { // if data's width is bigger than the current column width, // enlarge the column (but avoid enlarging it if the // data's width is very big) if ($stringWidth > $val && $stringWidth < $this->sColWidth * 3) { $colFits[$key] = $stringWidth; } } } }
$totAlreadyFitted = 0; foreach ($colFits as $key => $val) { // set fitted columns to smallest size $this->tablewidths[$key] = $val; // to work out how much (if any) space has been freed up $totAlreadyFitted += $val; }
if ($adjustingMode) { $surplus = (count($colFits) * $this->sColWidth) - $totAlreadyFitted; $surplusToAdd = $surplus / ($this->numFields - count($colFits)); } else { $surplusToAdd = 0; }
for ($i = 0; $i < $this->numFields; $i++) { if (! array_key_exists($i, $colFits)) { $this->tablewidths[$i] = $this->sColWidth + $surplusToAdd; }
if ($this->displayColumn[$i] != false) { continue; }
$this->tablewidths[$i] = 0; }
ksort($this->tablewidths);
// Pass 2
$this->results = $dbi->query($query, DatabaseInterface::CONNECT_USER, DatabaseInterface::QUERY_UNBUFFERED); $this->setY($this->tMargin); $this->AddPage(); $this->setFont(PdfLib::PMA_PDF_FONT, '', 9); // phpcs:ignore Squiz.NamingConventions.ValidVariableName.MemberNotCamelCaps $this->morepagestable($this->FontSizePt); }
public function setTitleFontSize(int $titleFontSize): void { $this->titleFontSize = $titleFontSize; }
public function setTitleText(string $titleText): void { $this->titleText = $titleText; }
public function setCurrentDb(?string $currentDb): void { $this->currentDb = $currentDb ?? ''; }
public function setCurrentTable(?string $currentTable): void { $this->currentTable = $currentTable ?? ''; }
public function setDbAlias(?string $dbAlias): void { $this->dbAlias = $dbAlias ?? ''; }
public function setTableAlias(?string $tableAlias): void { $this->tableAlias = $tableAlias ?? ''; }
/** * @param array $aliases */ public function setAliases(array $aliases): void { $this->aliases = $aliases; }
public function setPurpose(string $purpose): void { $this->purpose = $purpose; } }
|