<?php

/*******************************************************************************
 * FPDF                                                                         *
 *                                                                              *
 * Version: 1.86                                                                *
 * Date:    2023-06-25                                                          *
 * Author:  Olivier PLATHEY                                                     *
 *******************************************************************************/

class FPDF
{
	const VERSION = '1.86';
	protected $page; // current page number
	protected $n; // current object number
	protected $offsets; // array of object offsets
	protected $buffer; // buffer holding in-memory PDF
	protected $pages; // array containing pages
	protected $state; // current document state
	protected $compress; // compression flag
	protected $iconv; // whether iconv is available1814
	protected $k; // scale factor (number of points in user unit)
	protected $DefOrientation; // default orientation
	protected $CurOrientation; // current orientation
	protected $StdPageSizes; // standard page sizes
	protected $DefPageSize; // default page size
	protected $CurPageSize; // current page size
	protected $CurRotation; // current page rotation
	protected $PageInfo; // page-related data
	protected $wPt, $hPt; // dimensions of current page in points
	protected $w, $h; // dimensions of current page in user unit
	protected $lMargin; // left margin
	protected $tMargin; // top margin
	protected $rMargin; // right margin
	protected $bMargin; // page break margin
	protected $cMargin; // cell margin
	protected $x, $y; // current position in user unit
	protected $lasth; // height of last printed cell
	protected $LineWidth; // line width in user unit
	protected $fontpath; // directory containing fonts
	protected $CoreFonts; // array of core font names
	protected $fonts; // array of used fonts
	protected $FontFiles; // array of font files
	protected $encodings; // array of encodings
	protected $cmaps; // array of ToUnicode CMaps
	protected $FontFamily; // current font family
	protected $FontStyle; // current font style
	protected $underline; // underlining flag
	protected $CurrentFont; // current font info
	protected $FontSizePt; // current font size in points
	protected $FontSize; // current font size in user unit
	protected $DrawColor; // commands for drawing color
	protected $FillColor; // commands for filling color
	protected $TextColor; // commands for text color
	protected $ColorFlag; // indicates whether fill and text colors are different
	protected $WithAlpha; // indicates whether alpha channel is used
	protected $ws; // word spacing
	protected $images; // array of used images
	protected $PageLinks; // array of links in pages
	protected $links; // array of internal links
	protected $AutoPageBreak; // automatic page breaking
	protected $PageBreakTrigger; // threshold used to trigger page breaks
	protected $InHeader; // flag set when processing header
	protected $InFooter; // flag set when processing footer
	protected $AliasNbPages; // alias for total number of pages
	protected $ZoomMode; // zoom display mode
	protected $LayoutMode; // layout display mode
	protected $metadata; // document properties
	protected $CreationDate; // document creation date
	protected $PDFVersion; // PDF version number

	/*******************************************************************************
	 *                               Public methods                                 *
	 *******************************************************************************/

	function __construct($orientation = 'P', $unit = 'mm', $size = 'A4')
	{
		// Initialization of properties
		$this->state = 0;
		$this->page = 0;
		$this->n = 2;
		$this->buffer = '';
		$this->pages = array();
		$this->PageInfo = array();
		$this->fonts = array();
		$this->FontFiles = array();
		$this->encodings = array();
		$this->cmaps = array();
		$this->images = array();
		$this->links = array();
		$this->InHeader = false;
		$this->InFooter = false;
		$this->lasth = 0;
		$this->FontFamily = '';
		$this->FontStyle = '';
		$this->FontSizePt = 12;
		$this->underline = false;
		$this->DrawColor = '0 G';
		$this->FillColor = '0 g';
		$this->TextColor = '0 g';
		$this->ColorFlag = false;
		$this->WithAlpha = false;
		$this->ws = 0;
		$this->iconv = function_exists('iconv');
		// Font path
		if (defined('FPDF_FONTPATH'))
			$this->fontpath = FPDF_FONTPATH;
		else
			$this->fontpath = dirname(__FILE__) . '/font/';
		// Core fonts
		$this->CoreFonts = array('courier', 'helvetica', 'times', 'symbol', 'zapfdingbats');
		// Scale factor
		if ($unit == 'pt')
			$this->k = 1;
		elseif ($unit == 'mm')
			$this->k = 72 / 25.4;
		elseif ($unit == 'cm')
			$this->k = 72 / 2.54;
		elseif ($unit == 'in')
			$this->k = 72;
		else
			$this->Error('Incorrect unit: ' . $unit);
		// Page sizes
		$this->StdPageSizes = array(
			'a3' => array(841.89, 1190.55),
			'a4' => array(595.28, 841.89),
			'a5' => array(420.94, 595.28),
			'letter' => array(612, 792),
			'legal' => array(612, 1008)
		);
		$size = $this->_getpagesize($size);
		$this->DefPageSize = $size;
		$this->CurPageSize = $size;
		// Page orientation
		$orientation = strtolower($orientation);
		if ($orientation == 'p' || $orientation == 'portrait') {
			$this->DefOrientation = 'P';
			$this->w = $size[0];
			$this->h = $size[1];
		} elseif ($orientation == 'l' || $orientation == 'landscape') {
			$this->DefOrientation = 'L';
			$this->w = $size[1];
			$this->h = $size[0];
		} else
			$this->Error('Incorrect orientation: ' . $orientation);
		$this->CurOrientation = $this->DefOrientation;
		$this->wPt = $this->w * $this->k;
		$this->hPt = $this->h * $this->k;
		// Page rotation
		$this->CurRotation = 0;
		// Page margins (1 cm)
		$margin = 28.35 / $this->k;
		$this->SetMargins($margin, $margin);
		// Interior cell margin (1 mm)
		$this->cMargin = $margin / 10;
		// Line width (0.2 mm)
		$this->LineWidth = .567 / $this->k;
		// Automatic page break
		$this->SetAutoPageBreak(true, 2 * $margin);
		// Default display mode
		$this->SetDisplayMode('default');
		// Enable compression
		$this->SetCompression(true);
		// Metadata
		$this->metadata = array('Producer' => 'FPDF ' . self::VERSION);
		// Set default PDF version number
		$this->PDFVersion = '1.3';
	}

	function SetMargins($left, $top, $right = null)
	{
		// Set left, top and right margins
		$this->lMargin = $left;
		$this->tMargin = $top;
		if ($right === null)
			$right = $left;
		$this->rMargin = $right;
	}

	function SetLeftMargin($margin)
	{
		// Set left margin
		$this->lMargin = $margin;
		if ($this->page > 0 && $this->x < $margin)
			$this->x = $margin;
	}

	function SetTopMargin($margin)
	{
		// Set top margin
		$this->tMargin = $margin;
	}

	function SetRightMargin($margin)
	{
		// Set right margin
		$this->rMargin = $margin;
	}

	function SetAutoPageBreak($auto, $margin = 0)
	{
		// Set auto page break mode and triggering margin
		$this->AutoPageBreak = $auto;
		$this->bMargin = $margin;
		$this->PageBreakTrigger = $this->h - $margin;
	}

	function SetDisplayMode($zoom, $layout = 'default')
	{
		// Set display mode in viewer
		if ($zoom == 'fullpage' || $zoom == 'fullwidth' || $zoom == 'real' || $zoom == 'default' || !is_string($zoom))
			$this->ZoomMode = $zoom;
		else
			$this->Error('Incorrect zoom display mode: ' . $zoom);
		if ($layout == 'single' || $layout == 'continuous' || $layout == 'two' || $layout == 'default')
			$this->LayoutMode = $layout;
		else
			$this->Error('Incorrect layout display mode: ' . $layout);
	}

	function SetCompression($compress)
	{
		// Set page compression
		if (function_exists('gzcompress'))
			$this->compress = $compress;
		else
			$this->compress = false;
	}

	function SetTitle($title, $isUTF8 = false)
	{
		// Title of document
		$this->metadata['Title'] = $isUTF8 ? $title : $this->_UTF8encode($title);
	}

	function SetAuthor($author, $isUTF8 = false)
	{
		// Author of document
		$this->metadata['Author'] = $isUTF8 ? $author : $this->_UTF8encode($author);
	}

	function SetSubject($subject, $isUTF8 = false)
	{
		// Subject of document
		$this->metadata['Subject'] = $isUTF8 ? $subject : $this->_UTF8encode($subject);
	}

	function SetKeywords($keywords, $isUTF8 = false)
	{
		// Keywords of document
		$this->metadata['Keywords'] = $isUTF8 ? $keywords : $this->_UTF8encode($keywords);
	}

	function SetCreator($creator, $isUTF8 = false)
	{
		// Creator of document
		$this->metadata['Creator'] = $isUTF8 ? $creator : $this->_UTF8encode($creator);
	}

	function AliasNbPages($alias = '{nb}')
	{
		// Define an alias for total number of pages
		$this->AliasNbPages = $alias;
	}

	function Error($msg)
	{
		// Fatal error
		throw new Exception('FPDF error: ' . $msg);
	}

	function Close()
	{
		// Terminate document
		if ($this->state == 3)
			return;
		if ($this->page == 0)
			$this->AddPage();
		// Page footer
		$this->InFooter = true;
		$this->Footer();
		$this->InFooter = false;
		// Close page
		$this->_endpage();
		// Close document
		$this->_enddoc();
	}

	function AddPage($orientation = '', $size = '', $rotation = 0)
	{
		// Start a new page
		if ($this->state == 3)
			$this->Error('The document is closed');
		$family = $this->FontFamily;
		$style = $this->FontStyle . ($this->underline ? 'U' : '');
		$fontsize = $this->FontSizePt;
		$lw = $this->LineWidth;
		$dc = $this->DrawColor;
		$fc = $this->FillColor;
		$tc = $this->TextColor;
		$cf = $this->ColorFlag;
		if ($this->page > 0) {
			// Page footer
			$this->InFooter = true;
			$this->Footer();
			$this->InFooter = false;
			// Close page
			$this->_endpage();
		}
		// Start new page
		$this->_beginpage($orientation, $size, $rotation);
		// Set line cap style to square
		$this->_out('2 J');
		// Set line width
		$this->LineWidth = $lw;
		$this->_out(sprintf('%.2F w', $lw * $this->k));
		// Set font
		if ($family)
			$this->SetFont($family, $style, $fontsize);
		// Set colors
		$this->DrawColor = $dc;
		if ($dc != '0 G')
			$this->_out($dc);
		$this->FillColor = $fc;
		if ($fc != '0 g')
			$this->_out($fc);
		$this->TextColor = $tc;
		$this->ColorFlag = $cf;
		// Page header
		$this->InHeader = true;
		$this->Header();
		$this->InHeader = false;
		// Restore line width
		if ($this->LineWidth != $lw) {
			$this->LineWidth = $lw;
			$this->_out(sprintf('%.2F w', $lw * $this->k));
		}
		// Restore font
		if ($family)
			$this->SetFont($family, $style, $fontsize);
		// Restore colors
		if ($this->DrawColor != $dc) {
			$this->DrawColor = $dc;
			$this->_out($dc);
		}
		if ($this->FillColor != $fc) {
			$this->FillColor = $fc;
			$this->_out($fc);
		}
		$this->TextColor = $tc;
		$this->ColorFlag = $cf;
	}

	function Header()
	{
		// To be implemented in your own inherited class
	}

	function Footer()
	{
		// To be implemented in your own inherited class
	}

	function PageNo()
	{
		// Get current page number
		return $this->page;
	}

	function SetDrawColor($r, $g = null, $b = null)
	{
		// Set color for all stroking operations
		if (($r == 0 && $g == 0 && $b == 0) || $g === null)
			$this->DrawColor = sprintf('%.3F G', $r / 255);
		else
			$this->DrawColor = sprintf('%.3F %.3F %.3F RG', $r / 255, $g / 255, $b / 255);
		if ($this->page > 0)
			$this->_out($this->DrawColor);
	}

	function SetFillColor($r, $g = null, $b = null)
	{
		// Set color for all filling operations
		if (($r == 0 && $g == 0 && $b == 0) || $g === null)
			$this->FillColor = sprintf('%.3F g', $r / 255);
		else
			$this->FillColor = sprintf('%.3F %.3F %.3F rg', $r / 255, $g / 255, $b / 255);
		$this->ColorFlag = ($this->FillColor != $this->TextColor);
		if ($this->page > 0)
			$this->_out($this->FillColor);
	}

	function SetTextColor($r, $g = null, $b = null)
	{
		// Set color for text
		if (($r == 0 && $g == 0 && $b == 0) || $g === null)
			$this->TextColor = sprintf('%.3F g', $r / 255);
		else
			$this->TextColor = sprintf('%.3F %.3F %.3F rg', $r / 255, $g / 255, $b / 255);
		$this->ColorFlag = ($this->FillColor != $this->TextColor);
	}

	function GetStringWidth($s)
	{
		// Get width of a string in the current font
		$cw = $this->CurrentFont['cw'];
		$w = 0;
		$s = (string) $s;
		$l = strlen($s);
		for ($i = 0; $i < $l; $i++)
			$w += $cw[$s[$i]];
		return $w * $this->FontSize / 1000;
	}

	function SetLineWidth($width)
	{
		// Set line width
		$this->LineWidth = $width;
		if ($this->page > 0)
			$this->_out(sprintf('%.2F w', $width * $this->k));
	}

	function Line($x1, $y1, $x2, $y2)
	{
		// Draw a line
		$this->_out(sprintf('%.2F %.2F m %.2F %.2F l S', $x1 * $this->k, ($this->h - $y1) * $this->k, $x2 * $this->k, ($this->h - $y2) * $this->k));
	}

	function Rect($x, $y, $w, $h, $style = '')
	{
		// Draw a rectangle
		if ($style == 'F')
			$op = 'f';
		elseif ($style == 'FD' || $style == 'DF')
			$op = 'B';
		else
			$op = 'S';
		$this->_out(sprintf('%.2F %.2F %.2F %.2F re %s', $x * $this->k, ($this->h - $y) * $this->k, $w * $this->k, -$h * $this->k, $op));
	}

	function AddFont($family, $style = '', $file = '', $dir = '')
	{
		// Add a TrueType, OpenType or Type1 font
		$family = strtolower($family);
		if ($file == '')
			$file = str_replace(' ', '', $family) . strtolower($style) . '.php';
		$style = strtoupper($style);
		if ($style == 'IB')
			$style = 'BI';
		$fontkey = $family . $style;
		if (isset($this->fonts[$fontkey]))
			return;
		if (strpos($file, '/') !== false || strpos($file, "\\") !== false)
			$this->Error('Incorrect font definition file name: ' . $file);
		if ($dir == '')
			$dir = $this->fontpath;
		if (substr($dir, -1) != '/' && substr($dir, -1) != '\\')
			$dir .= '/';
		$info = $this->_loadfont($dir . $file);
		$info['i'] = count($this->fonts) + 1;
		if (!empty($info['file'])) {
			// Embedded font
			$info['file'] = $dir . $info['file'];
			if ($info['type'] == 'TrueType')
				$this->FontFiles[$info['file']] = array('length1' => $info['originalsize']);
			else
				$this->FontFiles[$info['file']] = array('length1' => $info['size1'], 'length2' => $info['size2']);
		}
		$this->fonts[$fontkey] = $info;
	}

	function SetFont($family, $style = '', $size = 0)
	{
		// Select a font; size given in points
		if ($family == '')
			$family = $this->FontFamily;
		else
			$family = strtolower($family);
		$style = strtoupper($style);
		if (strpos($style, 'U') !== false) {
			$this->underline = true;
			$style = str_replace('U', '', $style);
		} else
			$this->underline = false;
		if ($style == 'IB')
			$style = 'BI';
		if ($size == 0)
			$size = $this->FontSizePt;
		// Test if font is already selected
		if ($this->FontFamily == $family && $this->FontStyle == $style && $this->FontSizePt == $size)
			return;
		// Test if font is already loaded
		$fontkey = $family . $style;
		if (!isset($this->fonts[$fontkey])) {
			// Test if one of the core fonts
			if ($family == 'arial')
				$family = 'helvetica';
			if (in_array($family, $this->CoreFonts)) {
				if ($family == 'symbol' || $family == 'zapfdingbats')
					$style = '';
				$fontkey = $family . $style;
				if (!isset($this->fonts[$fontkey]))
					$this->AddFont($family, $style);
			} else
				$this->Error('Undefined font: ' . $family . ' ' . $style);
		}
		// Select it
		$this->FontFamily = $family;
		$this->FontStyle = $style;
		$this->FontSizePt = $size;
		$this->FontSize = $size / $this->k;
		$this->CurrentFont = $this->fonts[$fontkey];
		if ($this->page > 0)
			$this->_out(sprintf('BT /F%d %.2F Tf ET', $this->CurrentFont['i'], $this->FontSizePt));
	}

	function SetFontSize($size)
	{
		// Set font size in points
		if ($this->FontSizePt == $size)
			return;
		$this->FontSizePt = $size;
		$this->FontSize = $size / $this->k;
		if ($this->page > 0 && isset($this->CurrentFont))
			$this->_out(sprintf('BT /F%d %.2F Tf ET', $this->CurrentFont['i'], $this->FontSizePt));
	}

	function AddLink()
	{
		// Create a new internal link
		$n = count($this->links) + 1;
		$this->links[$n] = array(0, 0);
		return $n;
	}

	function SetLink($link, $y = 0, $page = -1)
	{
		// Set destination of internal link
		if ($y == -1)
			$y = $this->y;
		if ($page == -1)
			$page = $this->page;
		$this->links[$link] = array($page, $y);
	}

	function Link($x, $y, $w, $h, $link)
	{
		// Put a link on the page
		$this->PageLinks[$this->page][] = array($x * $this->k, $this->hPt - $y * $this->k, $w * $this->k, $h * $this->k, $link);
	}

	function Text($x, $y, $txt)
	{
		// Output a string
		if (!isset($this->CurrentFont))
			$this->Error('No font has been set');
		$txt = (string) $txt;
		$s = sprintf('BT %.2F %.2F Td (%s) Tj ET', $x * $this->k, ($this->h - $y) * $this->k, $this->_escape($txt));
		if ($this->underline && $txt !== '')
			$s .= ' ' . $this->_dounderline($x, $y, $txt);
		if ($this->ColorFlag)
			$s = 'q ' . $this->TextColor . ' ' . $s . ' Q';
		$this->_out($s);
	}

	function AcceptPageBreak()
	{
		// Accept automatic page break or not
		return $this->AutoPageBreak;
	}

	function Cell($w, $h = 0, $txt = '', $border = 0, $ln = 0, $align = '', $fill = false, $link = '')
	{
		// Output a cell
		$k = $this->k;
		if ($this->y + $h > $this->PageBreakTrigger && !$this->InHeader && !$this->InFooter && $this->AcceptPageBreak()) {
			// Automatic page break
			$x = $this->x;
			$ws = $this->ws;
			if ($ws > 0) {
				$this->ws = 0;
				$this->_out('0 Tw');
			}
			$this->AddPage($this->CurOrientation, $this->CurPageSize, $this->CurRotation);
			$this->x = $x;
			if ($ws > 0) {
				$this->ws = $ws;
				$this->_out(sprintf('%.3F Tw', $ws * $k));
			}
		}
		if ($w == 0)
			$w = $this->w - $this->rMargin - $this->x;
		$s = '';
		if ($fill || $border == 1) {
			if ($fill)
				$op = ($border == 1) ? 'B' : 'f';
			else
				$op = 'S';
			$s = sprintf('%.2F %.2F %.2F %.2F re %s ', $this->x * $k, ($this->h - $this->y) * $k, $w * $k, -$h * $k, $op);
		}
		if (is_string($border)) {
			$x = $this->x;
			$y = $this->y;
			if (strpos($border, 'L') !== false)
				$s .= sprintf('%.2F %.2F m %.2F %.2F l S ', $x * $k, ($this->h - $y) * $k, $x * $k, ($this->h - ($y + $h)) * $k);
			if (strpos($border, 'T') !== false)
				$s .= sprintf('%.2F %.2F m %.2F %.2F l S ', $x * $k, ($this->h - $y) * $k, ($x + $w) * $k, ($this->h - $y) * $k);
			if (strpos($border, 'R') !== false)
				$s .= sprintf('%.2F %.2F m %.2F %.2F l S ', ($x + $w) * $k, ($this->h - $y) * $k, ($x + $w) * $k, ($this->h - ($y + $h)) * $k);
			if (strpos($border, 'B') !== false)
				$s .= sprintf('%.2F %.2F m %.2F %.2F l S ', $x * $k, ($this->h - ($y + $h)) * $k, ($x + $w) * $k, ($this->h - ($y + $h)) * $k);
		}
		$txt = (string) $txt;
		if ($txt !== '') {
			if (!isset($this->CurrentFont))
				$this->Error('No font has been set');
			if ($align == 'R')
				$dx = $w - $this->cMargin - $this->GetStringWidth($txt);
			elseif ($align == 'C')
				$dx = ($w - $this->GetStringWidth($txt)) / 2;
			else
				$dx = $this->cMargin;
			if ($this->ColorFlag)
				$s .= 'q ' . $this->TextColor . ' ';
			$s .= sprintf('BT %.2F %.2F Td (%s) Tj ET', ($this->x + $dx) * $k, ($this->h - ($this->y + .5 * $h + .3 * $this->FontSize)) * $k, $this->_escape($txt));
			if ($this->underline)
				$s .= ' ' . $this->_dounderline($this->x + $dx, $this->y + .5 * $h + .3 * $this->FontSize, $txt);
			if ($this->ColorFlag)
				$s .= ' Q';
			if ($link)
				$this->Link($this->x + $dx, $this->y + .5 * $h - .5 * $this->FontSize, $this->GetStringWidth($txt), $this->FontSize, $link);
		}
		if ($s)
			$this->_out($s);
		$this->lasth = $h;
		if ($ln > 0) {
			// Go to next line
			$this->y += $h;
			if ($ln == 1)
				$this->x = $this->lMargin;
		} else
			$this->x += $w;
	}

	function MultiCell($w, $h, $txt, $border = 0, $align = 'J', $fill = false)
	{
		// Output text with automatic or explicit line breaks
		if (!isset($this->CurrentFont))
			$this->Error('No font has been set');
		$cw = $this->CurrentFont['cw'];
		if ($w == 0)
			$w = $this->w - $this->rMargin - $this->x;
		$wmax = ($w - 2 * $this->cMargin) * 1000 / $this->FontSize;
		$s = str_replace("\r", '', (string) $txt);
		$nb = strlen($s);
		if ($nb > 0 && $s[$nb - 1] == "\n")
			$nb--;
		$b = 0;
		if ($border) {
			if ($border == 1) {
				$border = 'LTRB';
				$b = 'LRT';
				$b2 = 'LR';
			} else {
				$b2 = '';
				if (strpos($border, 'L') !== false)
					$b2 .= 'L';
				if (strpos($border, 'R') !== false)
					$b2 .= 'R';
				$b = (strpos($border, 'T') !== false) ? $b2 . 'T' : $b2;
			}
		}
		$sep = -1;
		$i = 0;
		$j = 0;
		$l = 0;
		$ns = 0;
		$nl = 1;
		while ($i < $nb) {
			// Get next character
			$c = $s[$i];
			if ($c == "\n") {
				// Explicit line break
				if ($this->ws > 0) {
					$this->ws = 0;
					$this->_out('0 Tw');
				}
				$this->Cell($w, $h, substr($s, $j, $i - $j), $b, 2, $align, $fill);
				$i++;
				$sep = -1;
				$j = $i;
				$l = 0;
				$ns = 0;
				$nl++;
				if ($border && $nl == 2)
					$b = $b2;
				continue;
			}
			if ($c == ' ') {
				$sep = $i;
				$ls = $l;
				$ns++;
			}
			$l += $cw[$c];
			if ($l > $wmax) {
				// Automatic line break
				if ($sep == -1) {
					if ($i == $j)
						$i++;
					if ($this->ws > 0) {
						$this->ws = 0;
						$this->_out('0 Tw');
					}
					$this->Cell($w, $h, substr($s, $j, $i - $j), $b, 2, $align, $fill);
				} else {
					if ($align == 'J') {
						$this->ws = ($ns > 1) ? ($wmax - $ls) / 1000 * $this->FontSize / ($ns - 1) : 0;
						$this->_out(sprintf('%.3F Tw', $this->ws * $this->k));
					}
					$this->Cell($w, $h, substr($s, $j, $sep - $j), $b, 2, $align, $fill);
					$i = $sep + 1;
				}
				$sep = -1;
				$j = $i;
				$l = 0;
				$ns = 0;
				$nl++;
				if ($border && $nl == 2)
					$b = $b2;
			} else
				$i++;
		}
		// Last chunk
		if ($this->ws > 0) {
			$this->ws = 0;
			$this->_out('0 Tw');
		}
		if ($border && strpos($border, 'B') !== false)
			$b .= 'B';
		$this->Cell($w, $h, substr($s, $j, $i - $j), $b, 2, $align, $fill);
		$this->x = $this->lMargin;
	}

	function Write($h, $txt, $link = '')
	{
		// Output text in flowing mode
		if (!isset($this->CurrentFont))
			$this->Error('No font has been set');
		$cw = $this->CurrentFont['cw'];
		$w = $this->w - $this->rMargin - $this->x;
		$wmax = ($w - 2 * $this->cMargin) * 1000 / $this->FontSize;
		$s = str_replace("\r", '', (string) $txt);
		$nb = strlen($s);
		$sep = -1;
		$i = 0;
		$j = 0;
		$l = 0;
		$nl = 1;
		while ($i < $nb) {
			// Get next character
			$c = $s[$i];
			if ($c == "\n") {
				// Explicit line break
				$this->Cell($w, $h, substr($s, $j, $i - $j), 0, 2, '', false, $link);
				$i++;
				$sep = -1;
				$j = $i;
				$l = 0;
				if ($nl == 1) {
					$this->x = $this->lMargin;
					$w = $this->w - $this->rMargin - $this->x;
					$wmax = ($w - 2 * $this->cMargin) * 1000 / $this->FontSize;
				}
				$nl++;
				continue;
			}
			if ($c == ' ')
				$sep = $i;
			$l += $cw[$c];
			if ($l > $wmax) {
				// Automatic line break
				if ($sep == -1) {
					if ($this->x > $this->lMargin) {
						// Move to next line
						$this->x = $this->lMargin;
						$this->y += $h;
						$w = $this->w - $this->rMargin - $this->x;
						$wmax = ($w - 2 * $this->cMargin) * 1000 / $this->FontSize;
						$i++;
						$nl++;
						continue;
					}
					if ($i == $j)
						$i++;
					$this->Cell($w, $h, substr($s, $j, $i - $j), 0, 2, '', false, $link);
				} else {
					$this->Cell($w, $h, substr($s, $j, $sep - $j), 0, 2, '', false, $link);
					$i = $sep + 1;
				}
				$sep = -1;
				$j = $i;
				$l = 0;
				if ($nl == 1) {
					$this->x = $this->lMargin;
					$w = $this->w - $this->rMargin - $this->x;
					$wmax = ($w - 2 * $this->cMargin) * 1000 / $this->FontSize;
				}
				$nl++;
			} else
				$i++;
		}
		// Last chunk
		if ($i != $j)
			$this->Cell($l / 1000 * $this->FontSize, $h, substr($s, $j), 0, 0, '', false, $link);
	}

	function Ln($h = null)
	{
		// Line feed; default value is the last cell height
		$this->x = $this->lMargin;
		if ($h === null)
			$this->y += $this->lasth;
		else
			$this->y += $h;
	}

	function Image($file, $x = null, $y = null, $w = 0, $h = 0, $type = '', $link = '')
	{
		// Put an image on the page
		if ($file == '')
			$this->Error('Image file name is empty');
		if (!isset($this->images[$file])) {
			// First use of this image, get info
			if ($type == '') {
				$pos = strrpos($file, '.');
				if (!$pos)
					$this->Error('Image file has no extension and no type was specified: ' . $file);
				$type = substr($file, $pos + 1);
			}
			$type = strtolower($type);
			if ($type == 'jpeg')
				$type = 'jpg';
			$mtd = '_parse' . $type;
			if (!method_exists($this, $mtd))
				$this->Error('Unsupported image type: ' . $type);
			$info = $this->$mtd($file);
			$info['i'] = count($this->images) + 1;
			$this->images[$file] = $info;
		} else
			$info = $this->images[$file];

		// Automatic width and height calculation if needed
		if ($w == 0 && $h == 0) {
			// Put image at 96 dpi
			$w = -96;
			$h = -96;
		}
		if ($w < 0)
			$w = -$info['w'] * 72 / $w / $this->k;
		if ($h < 0)
			$h = -$info['h'] * 72 / $h / $this->k;
		if ($w == 0)
			$w = $h * $info['w'] / $info['h'];
		if ($h == 0)
			$h = $w * $info['h'] / $info['w'];

		// Flowing mode
		if ($y === null) {
			if ($this->y + $h > $this->PageBreakTrigger && !$this->InHeader && !$this->InFooter && $this->AcceptPageBreak()) {
				// Automatic page break
				$x2 = $this->x;
				$this->AddPage($this->CurOrientation, $this->CurPageSize, $this->CurRotation);
				$this->x = $x2;
			}
			$y = $this->y;
			$this->y += $h;
		}

		if ($x === null)
			$x = $this->x;
		$this->_out(sprintf('q %.2F 0 0 %.2F %.2F %.2F cm /I%d Do Q', $w * $this->k, $h * $this->k, $x * $this->k, ($this->h - ($y + $h)) * $this->k, $info['i']));
		if ($link)
			$this->Link($x, $y, $w, $h, $link);
	}

	function GetPageWidth()
	{
		// Get current page width
		return $this->w;
	}

	function GetPageHeight()
	{
		// Get current page height
		return $this->h;
	}

	function GetX()
	{
		// Get x position
		return $this->x;
	}

	function SetX($x)
	{
		// Set x position
		if ($x >= 0)
			$this->x = $x;
		else
			$this->x = $this->w + $x;
	}

	function GetY()
	{
		// Get y position
		return $this->y;
	}

	function SetY($y, $resetX = true)
	{
		// Set y position and optionally reset x
		if ($y >= 0)
			$this->y = $y;
		else
			$this->y = $this->h + $y;
		if ($resetX)
			$this->x = $this->lMargin;
	}

	function SetXY($x, $y)
	{
		// Set x and y positions
		$this->SetX($x);
		$this->SetY($y, false);
	}

	function Output($dest = '', $name = '', $isUTF8 = false)
	{
		// Output PDF to some destination
		$this->Close();
		if (strlen($name) == 1 && strlen($dest) != 1) {
			// Fix parameter order
			$tmp = $dest;
			$dest = $name;
			$name = $tmp;
		}
		if ($dest == '')
			$dest = 'I';
		if ($name == '')
			$name = 'doc.pdf';
		switch (strtoupper($dest)) {
			case 'I':
				// Send to standard output
				$this->_checkoutput();
				if (PHP_SAPI != 'cli') {
					// We send to a browser
					header('Content-Type: application/pdf');
					header('Content-Disposition: inline; ' . $this->_httpencode('filename', $name, $isUTF8));
					header('Cache-Control: private, max-age=0, must-revalidate');
					header('Pragma: public');
				}
				echo $this->buffer;
				break;
			case 'D':
				// Download file
				$this->_checkoutput();
				header('Content-Type: application/pdf');
				header('Content-Disposition: attachment; ' . $this->_httpencode('filename', $name, $isUTF8));
				header('Cache-Control: private, max-age=0, must-revalidate');
				header('Pragma: public');
				echo $this->buffer;
				break;
			case 'F':
				// Save to local file
				if (!file_put_contents($name, $this->buffer))
					$this->Error('Unable to create output file: ' . $name);
				break;
			case 'S':
				// Return as a string
				return $this->buffer;
			default:
				$this->Error('Incorrect output destination: ' . $dest);
		}
		return '';
	}

	/*******************************************************************************
	 *                              Protected methods                               *
	 *******************************************************************************/

	protected function _checkoutput()
	{
		if (PHP_SAPI != 'cli') {
			if (headers_sent($file, $line))
				$this->Error("Some data has already been output, can't send PDF file (output started at $file:$line)");
		}
		if (ob_get_length()) {
			// The output buffer is not empty
			if (preg_match('/^(\xEF\xBB\xBF)?\s*$/', ob_get_contents())) {
				// It contains only a UTF-8 BOM and/or whitespace, let's clean it
				ob_clean();
			} else
				$this->Error("Some data has already been output, can't send PDF file");
		}
	}

	protected function _getpagesize($size)
	{
		if (is_string($size)) {
			$size = strtolower($size);
			if (!isset($this->StdPageSizes[$size]))
				$this->Error('Unknown page size: ' . $size);
			$a = $this->StdPageSizes[$size];
			return array($a[0] / $this->k, $a[1] / $this->k);
		} else {
			if ($size[0] > $size[1])
				return array($size[1], $size[0]);
			else
				return $size;
		}
	}

	protected function _beginpage($orientation, $size, $rotation)
	{
		$this->page++;
		$this->pages[$this->page] = '';
		$this->PageLinks[$this->page] = array();
		$this->state = 2;
		$this->x = $this->lMargin;
		$this->y = $this->tMargin;
		$this->FontFamily = '';
		// Check page size and orientation
		if ($orientation == '')
			$orientation = $this->DefOrientation;
		else
			$orientation = strtoupper($orientation[0]);
		if ($size == '')
			$size = $this->DefPageSize;
		else
			$size = $this->_getpagesize($size);
		if ($orientation != $this->CurOrientation || $size[0] != $this->CurPageSize[0] || $size[1] != $this->CurPageSize[1]) {
			// New size or orientation
			if ($orientation == 'P') {
				$this->w = $size[0];
				$this->h = $size[1];
			} else {
				$this->w = $size[1];
				$this->h = $size[0];
			}
			$this->wPt = $this->w * $this->k;
			$this->hPt = $this->h * $this->k;
			$this->PageBreakTrigger = $this->h - $this->bMargin;
			$this->CurOrientation = $orientation;
			$this->CurPageSize = $size;
		}
		if ($orientation != $this->DefOrientation || $size[0] != $this->DefPageSize[0] || $size[1] != $this->DefPageSize[1])
			$this->PageInfo[$this->page]['size'] = array($this->wPt, $this->hPt);
		if ($rotation != 0) {
			if ($rotation % 90 != 0)
				$this->Error('Incorrect rotation value: ' . $rotation);
			$this->PageInfo[$this->page]['rotation'] = $rotation;
		}
		$this->CurRotation = $rotation;
	}

	protected function _endpage()
	{
		$this->state = 1;
	}

	protected function _loadfont($path)
	{
		// Load a font definition file
		include($path);
		if (!isset($name))
			$this->Error('Could not include font definition file: ' . $path);
		if (isset($enc))
			$enc = strtolower($enc);
		if (!isset($subsetted))
			$subsetted = false;
		return get_defined_vars();
	}

	protected function _isascii($s)
	{
		// Test if string is ASCII
		$nb = strlen($s);
		for ($i = 0; $i < $nb; $i++) {
			if (ord($s[$i]) > 127)
				return false;
		}
		return true;
	}

	protected function _httpencode($param, $value, $isUTF8)
	{
		// Encode HTTP header field parameter
		if ($this->_isascii($value))
			return $param . '="' . $value . '"';
		if (!$isUTF8)
			$value = $this->_UTF8encode($value);
		return $param . "*=UTF-8''" . rawurlencode($value);
	}

	protected function _UTF8encode($s)
	{
		// Convert ISO-8859-1 to UTF-8
		if ($this->iconv)
			return iconv('ISO-8859-1', 'UTF-8', $s);
		$res = '';
		$nb = strlen($s);
		for ($i = 0; $i < $nb; $i++) {
			$c = $s[$i];
			$v = ord($c);
			if ($v >= 128) {
				$res .= chr(0xC0 | ($v >> 6));
				$res .= chr(0x80 | ($v & 0x3F));
			} else
				$res .= $c;
		}
		return $res;
	}

	protected function _UTF8toUTF16($s)
	{
		// Convert UTF-8 to UTF-16BE with BOM
		$res = "\xFE\xFF";
		if ($this->iconv)
			return $res . iconv('UTF-8', 'UTF-16BE', $s);
		$nb = strlen($s);
		$i = 0;
		while ($i < $nb) {
			$c1 = ord($s[$i++]);
			if ($c1 >= 224) {
				// 3-byte character
				$c2 = ord($s[$i++]);
				$c3 = ord($s[$i++]);
				$res .= chr((($c1 & 0x0F) << 4) + (($c2 & 0x3C) >> 2));
				$res .= chr((($c2 & 0x03) << 6) + ($c3 & 0x3F));
			} elseif ($c1 >= 192) {
				// 2-byte character
				$c2 = ord($s[$i++]);
				$res .= chr(($c1 & 0x1C) >> 2);
				$res .= chr((($c1 & 0x03) << 6) + ($c2 & 0x3F));
			} else {
				// Single-byte character
				$res .= "\0" . chr($c1);
			}
		}
		return $res;
	}

	protected function _escape($s)
	{
		// Escape special characters
		if (strpos($s, '(') !== false || strpos($s, ')') !== false || strpos($s, '\\') !== false || strpos($s, "\r") !== false)
			return str_replace(array('\\', '(', ')', "\r"), array('\\\\', '\\(', '\\)', '\\r'), $s);
		else
			return $s;
	}

	protected function _textstring($s)
	{
		// Format a text string
		if (!$this->_isascii($s))
			$s = $this->_UTF8toUTF16($s);
		return '(' . $this->_escape($s) . ')';
	}

	protected function _dounderline($x, $y, $txt)
	{
		// Underline text
		$up = $this->CurrentFont['up'];
		$ut = $this->CurrentFont['ut'];
		$w = $this->GetStringWidth($txt) + $this->ws * substr_count($txt, ' ');
		return sprintf('%.2F %.2F %.2F %.2F re f', $x * $this->k, ($this->h - ($y - $up / 1000 * $this->FontSize)) * $this->k, $w * $this->k, -$ut / 1000 * $this->FontSizePt);
	}

	protected function _parsejpg($file)
	{
		// Extract info from a JPEG file
		$a = getimagesize($file);
		if (!$a)
			$this->Error('Missing or incorrect image file: ' . $file);
		if ($a[2] != 2)
			$this->Error('Not a JPEG file: ' . $file);
		if (!isset($a['channels']) || $a['channels'] == 3)
			$colspace = 'DeviceRGB';
		elseif ($a['channels'] == 4)
			$colspace = 'DeviceCMYK';
		else
			$colspace = 'DeviceGray';
		$bpc = isset($a['bits']) ? $a['bits'] : 8;
		$data = file_get_contents($file);
		return array('w' => $a[0], 'h' => $a[1], 'cs' => $colspace, 'bpc' => $bpc, 'f' => 'DCTDecode', 'data' => $data);
	}

	protected function _parsepng($file)
	{
		// Extract info from a PNG file
		$f = fopen($file, 'rb');
		if (!$f)
			$this->Error('Can\'t open image file: ' . $file);
		$info = $this->_parsepngstream($f, $file);
		fclose($f);
		return $info;
	}

	protected function _parsepngstream($f, $file)
	{
		// Check signature
		if ($this->_readstream($f, 8) != chr(137) . 'PNG' . chr(13) . chr(10) . chr(26) . chr(10))
			$this->Error('Not a PNG file: ' . $file);

		// Read header chunk
		$this->_readstream($f, 4);
		if ($this->_readstream($f, 4) != 'IHDR')
			$this->Error('Incorrect PNG file: ' . $file);
		$w = $this->_readint($f);
		$h = $this->_readint($f);
		$bpc = ord($this->_readstream($f, 1));
		if ($bpc > 8)
			$this->Error('16-bit depth not supported: ' . $file);
		$ct = ord($this->_readstream($f, 1));
		if ($ct == 0 || $ct == 4)
			$colspace = 'DeviceGray';
		elseif ($ct == 2 || $ct == 6)
			$colspace = 'DeviceRGB';
		elseif ($ct == 3)
			$colspace = 'Indexed';
		else
			$this->Error('Unknown color type: ' . $file);
		if (ord($this->_readstream($f, 1)) != 0)
			$this->Error('Unknown compression method: ' . $file);
		if (ord($this->_readstream($f, 1)) != 0)
			$this->Error('Unknown filter method: ' . $file);
		if (ord($this->_readstream($f, 1)) != 0)
			$this->Error('Interlacing not supported: ' . $file);
		$this->_readstream($f, 4);
		$dp = '/Predictor 15 /Colors ' . ($colspace == 'DeviceRGB' ? 3 : 1) . ' /BitsPerComponent ' . $bpc . ' /Columns ' . $w;

		// Scan chunks looking for palette, transparency and image data
		$pal = '';
		$trns = '';
		$data = '';
		do {
			$n = $this->_readint($f);
			$type = $this->_readstream($f, 4);
			if ($type == 'PLTE') {
				// Read palette
				$pal = $this->_readstream($f, $n);
				$this->_readstream($f, 4);
			} elseif ($type == 'tRNS') {
				// Read transparency info
				$t = $this->_readstream($f, $n);
				if ($ct == 0)
					$trns = array(ord(substr($t, 1, 1)));
				elseif ($ct == 2)
					$trns = array(ord(substr($t, 1, 1)), ord(substr($t, 3, 1)), ord(substr($t, 5, 1)));
				else {
					$pos = strpos($t, chr(0));
					if ($pos !== false)
						$trns = array($pos);
				}
				$this->_readstream($f, 4);
			} elseif ($type == 'IDAT') {
				// Read image data block
				$data .= $this->_readstream($f, $n);
				$this->_readstream($f, 4);
			} elseif ($type == 'IEND')
				break;
			else
				$this->_readstream($f, $n + 4);
		} while ($n);

		if ($colspace == 'Indexed' && empty($pal))
			$this->Error('Missing palette in ' . $file);
		$info = array('w' => $w, 'h' => $h, 'cs' => $colspace, 'bpc' => $bpc, 'f' => 'FlateDecode', 'dp' => $dp, 'pal' => $pal, 'trns' => $trns);
		if ($ct >= 4) {
			// Extract alpha channel
			if (!function_exists('gzuncompress'))
				$this->Error('Zlib not available, can\'t handle alpha channel: ' . $file);
			$data = gzuncompress($data);
			$color = '';
			$alpha = '';
			if ($ct == 4) {
				// Gray image
				$len = 2 * $w;
				for ($i = 0; $i < $h; $i++) {
					$pos = (1 + $len) * $i;
					$color .= $data[$pos];
					$alpha .= $data[$pos];
					$line = substr($data, $pos + 1, $len);
					$color .= preg_replace('/(.)./s', '$1', $line);
					$alpha .= preg_replace('/.(.)/s', '$1', $line);
				}
			} else {
				// RGB image
				$len = 4 * $w;
				for ($i = 0; $i < $h; $i++) {
					$pos = (1 + $len) * $i;
					$color .= $data[$pos];
					$alpha .= $data[$pos];
					$line = substr($data, $pos + 1, $len);
					$color .= preg_replace('/(.{3})./s', '$1', $line);
					$alpha .= preg_replace('/.{3}(.)/s', '$1', $line);
				}
			}
			unset($data);
			$data = gzcompress($color);
			$info['smask'] = gzcompress($alpha);
			$this->WithAlpha = true;
			if ($this->PDFVersion < '1.4')
				$this->PDFVersion = '1.4';
		}
		$info['data'] = $data;
		return $info;
	}

	protected function _readstream($f, $n)
	{
		// Read n bytes from stream
		$res = '';
		while ($n > 0 && !feof($f)) {
			$s = fread($f, $n);
			if ($s === false)
				$this->Error('Error while reading stream');
			$n -= strlen($s);
			$res .= $s;
		}
		if ($n > 0)
			$this->Error('Unexpected end of stream');
		return $res;
	}

	protected function _readint($f)
	{
		// Read a 4-byte integer from stream
		$a = unpack('Ni', $this->_readstream($f, 4));
		return $a['i'];
	}

	protected function _parsegif($file)
	{
		// Extract info from a GIF file (via PNG conversion)
		if (!function_exists('imagepng'))
			$this->Error('GD extension is required for GIF support');
		if (!function_exists('imagecreatefromgif'))
			$this->Error('GD has no GIF read support');
		$im = imagecreatefromgif($file);
		if (!$im)
			$this->Error('Missing or incorrect image file: ' . $file);
		imageinterlace($im, 0);
		ob_start();
		imagepng($im);
		$data = ob_get_clean();
		imagedestroy($im);
		$f = fopen('php://temp', 'rb+');
		if (!$f)
			$this->Error('Unable to create memory stream');
		fwrite($f, $data);
		rewind($f);
		$info = $this->_parsepngstream($f, $file);
		fclose($f);
		return $info;
	}

	protected function _out($s)
	{
		// Add a line to the current page
		if ($this->state == 2)
			$this->pages[$this->page] .= $s . "\n";
		elseif ($this->state == 0)
			$this->Error('No page has been added yet');
		elseif ($this->state == 1)
			$this->Error('Invalid call');
		elseif ($this->state == 3)
			$this->Error('The document is closed');
	}

	protected function _put($s)
	{
		// Add a line to the document
		$this->buffer .= $s . "\n";
	}

	protected function _getoffset()
	{
		return strlen($this->buffer);
	}

	protected function _newobj($n = null)
	{
		// Begin a new object
		if ($n === null)
			$n = ++$this->n;
		$this->offsets[$n] = $this->_getoffset();
		$this->_put($n . ' 0 obj');
	}

	protected function _putstream($data)
	{
		$this->_put('stream');
		$this->_put($data);
		$this->_put('endstream');
	}

	protected function _putstreamobject($data)
	{
		if ($this->compress) {
			$entries = '/Filter /FlateDecode ';
			$data = gzcompress($data);
		} else
			$entries = '';
		$entries .= '/Length ' . strlen($data);
		$this->_newobj();
		$this->_put('<<' . $entries . '>>');
		$this->_putstream($data);
		$this->_put('endobj');
	}

	protected function _putlinks($n)
	{
		foreach ($this->PageLinks[$n] as $pl) {
			$this->_newobj();
			$rect = sprintf('%.2F %.2F %.2F %.2F', $pl[0], $pl[1], $pl[0] + $pl[2], $pl[1] - $pl[3]);
			$s = '<</Type /Annot /Subtype /Link /Rect [' . $rect . '] /Border [0 0 0] ';
			if (is_string($pl[4]))
				$s .= '/A <</S /URI /URI ' . $this->_textstring($pl[4]) . '>>>>';
			else {
				$l = $this->links[$pl[4]];
				if (isset($this->PageInfo[$l[0]]['size']))
					$h = $this->PageInfo[$l[0]]['size'][1];
				else
					$h = ($this->DefOrientation == 'P') ? $this->DefPageSize[1] * $this->k : $this->DefPageSize[0] * $this->k;
				$s .= sprintf('/Dest [%d 0 R /XYZ 0 %.2F null]>>', $this->PageInfo[$l[0]]['n'], $h - $l[1] * $this->k);
			}
			$this->_put($s);
			$this->_put('endobj');
		}
	}

	protected function _putpage($n)
	{
		$this->_newobj();
		$this->_put('<</Type /Page');
		$this->_put('/Parent 1 0 R');
		if (isset($this->PageInfo[$n]['size']))
			$this->_put(sprintf('/MediaBox [0 0 %.2F %.2F]', $this->PageInfo[$n]['size'][0], $this->PageInfo[$n]['size'][1]));
		if (isset($this->PageInfo[$n]['rotation']))
			$this->_put('/Rotate ' . $this->PageInfo[$n]['rotation']);
		$this->_put('/Resources 2 0 R');
		if (!empty($this->PageLinks[$n])) {
			$s = '/Annots [';
			foreach ($this->PageLinks[$n] as $pl)
				$s .= $pl[5] . ' 0 R ';
			$s .= ']';
			$this->_put($s);
		}
		if ($this->WithAlpha)
			$this->_put('/Group <</Type /Group /S /Transparency /CS /DeviceRGB>>');
		$this->_put('/Contents ' . ($this->n + 1) . ' 0 R>>');
		$this->_put('endobj');
		// Page content
		if (!empty($this->AliasNbPages))
			$this->pages[$n] = str_replace($this->AliasNbPages, $this->page, $this->pages[$n]);
		$this->_putstreamobject($this->pages[$n]);
		// Link annotations
		$this->_putlinks($n);
	}

	protected function _putpages()
	{
		$nb = $this->page;
		$n = $this->n;
		for ($i = 1; $i <= $nb; $i++) {
			$this->PageInfo[$i]['n'] = ++$n;
			$n++;
			foreach ($this->PageLinks[$i] as &$pl)
				$pl[5] = ++$n;
			unset($pl);
		}
		for ($i = 1; $i <= $nb; $i++)
			$this->_putpage($i);
		// Pages root
		$this->_newobj(1);
		$this->_put('<</Type /Pages');
		$kids = '/Kids [';
		for ($i = 1; $i <= $nb; $i++)
			$kids .= $this->PageInfo[$i]['n'] . ' 0 R ';
		$kids .= ']';
		$this->_put($kids);
		$this->_put('/Count ' . $nb);
		if ($this->DefOrientation == 'P') {
			$w = $this->DefPageSize[0];
			$h = $this->DefPageSize[1];
		} else {
			$w = $this->DefPageSize[1];
			$h = $this->DefPageSize[0];
		}
		$this->_put(sprintf('/MediaBox [0 0 %.2F %.2F]', $w * $this->k, $h * $this->k));
		$this->_put('>>');
		$this->_put('endobj');
	}

	protected function _putfonts()
	{
		foreach ($this->FontFiles as $file => $info) {
			// Font file embedding
			$this->_newobj();
			$this->FontFiles[$file]['n'] = $this->n;
			$font = file_get_contents($file);
			if (!$font)
				$this->Error('Font file not found: ' . $file);
			$compressed = (substr($file, -2) == '.z');
			if (!$compressed && isset($info['length2']))
				$font = substr($font, 6, $info['length1']) . substr($font, 6 + $info['length1'] + 6, $info['length2']);
			$this->_put('<</Length ' . strlen($font));
			if ($compressed)
				$this->_put('/Filter /FlateDecode');
			$this->_put('/Length1 ' . $info['length1']);
			if (isset($info['length2']))
				$this->_put('/Length2 ' . $info['length2'] . ' /Length3 0');
			$this->_put('>>');
			$this->_putstream($font);
			$this->_put('endobj');
		}
		foreach ($this->fonts as $k => $font) {
			// Encoding
			if (isset($font['diff'])) {
				if (!isset($this->encodings[$font['enc']])) {
					$this->_newobj();
					$this->_put('<</Type /Encoding /BaseEncoding /WinAnsiEncoding /Differences [' . $font['diff'] . ']>>');
					$this->_put('endobj');
					$this->encodings[$font['enc']] = $this->n;
				}
			}
			// ToUnicode CMap
			if (isset($font['uv'])) {
				if (isset($font['enc']))
					$cmapkey = $font['enc'];
				else
					$cmapkey = $font['name'];
				if (!isset($this->cmaps[$cmapkey])) {
					$cmap = $this->_tounicodecmap($font['uv']);
					$this->_putstreamobject($cmap);
					$this->cmaps[$cmapkey] = $this->n;
				}
			}
			// Font object
			$this->fonts[$k]['n'] = $this->n + 1;
			$type = $font['type'];
			$name = $font['name'];
			if ($font['subsetted'])
				$name = 'AAAAAA+' . $name;
			if ($type == 'Core') {
				// Core font
				$this->_newobj();
				$this->_put('<</Type /Font');
				$this->_put('/BaseFont /' . $name);
				$this->_put('/Subtype /Type1');
				if ($name != 'Symbol' && $name != 'ZapfDingbats')
					$this->_put('/Encoding /WinAnsiEncoding');
				if (isset($font['uv']))
					$this->_put('/ToUnicode ' . $this->cmaps[$cmapkey] . ' 0 R');
				$this->_put('>>');
				$this->_put('endobj');
			} elseif ($type == 'Type1' || $type == 'TrueType') {
				// Additional Type1 or TrueType/OpenType font
				$this->_newobj();
				$this->_put('<</Type /Font');
				$this->_put('/BaseFont /' . $name);
				$this->_put('/Subtype /' . $type);
				$this->_put('/FirstChar 32 /LastChar 255');
				$this->_put('/Widths ' . ($this->n + 1) . ' 0 R');
				$this->_put('/FontDescriptor ' . ($this->n + 2) . ' 0 R');
				if (isset($font['diff']))
					$this->_put('/Encoding ' . $this->encodings[$font['enc']] . ' 0 R');
				else
					$this->_put('/Encoding /WinAnsiEncoding');
				if (isset($font['uv']))
					$this->_put('/ToUnicode ' . $this->cmaps[$cmapkey] . ' 0 R');
				$this->_put('>>');
				$this->_put('endobj');
				// Widths
				$this->_newobj();
				$cw = $font['cw'];
				$s = '[';
				for ($i = 32; $i <= 255; $i++)
					$s .= $cw[chr($i)] . ' ';
				$this->_put($s . ']');
				$this->_put('endobj');
				// Descriptor
				$this->_newobj();
				$s = '<</Type /FontDescriptor /FontName /' . $name;
				foreach ($font['desc'] as $k => $v)
					$s .= ' /' . $k . ' ' . $v;
				if (!empty($font['file']))
					$s .= ' /FontFile' . ($type == 'Type1' ? '' : '2') . ' ' . $this->FontFiles[$font['file']]['n'] . ' 0 R';
				$this->_put($s . '>>');
				$this->_put('endobj');
			} else {
				// Allow for additional types
				$mtd = '_put' . strtolower($type);
				if (!method_exists($this, $mtd))
					$this->Error('Unsupported font type: ' . $type);
				$this->$mtd($font);
			}
		}
	}

	protected function _tounicodecmap($uv)
	{
		$ranges = '';
		$nbr = 0;
		$chars = '';
		$nbc = 0;
		foreach ($uv as $c => $v) {
			if (is_array($v)) {
				$ranges .= sprintf("<%02X> <%02X> <%04X>\n", $c, $c + $v[1] - 1, $v[0]);
				$nbr++;
			} else {
				$chars .= sprintf("<%02X> <%04X>\n", $c, $v);
				$nbc++;
			}
		}
		$s = "/CIDInit /ProcSet findresource begin\n";
		$s .= "12 dict begin\n";
		$s .= "begincmap\n";
		$s .= "/CIDSystemInfo\n";
		$s .= "<</Registry (Adobe)\n";
		$s .= "/Ordering (UCS)\n";
		$s .= "/Supplement 0\n";
		$s .= ">> def\n";
		$s .= "/CMapName /Adobe-Identity-UCS def\n";
		$s .= "/CMapType 2 def\n";
		$s .= "1 begincodespacerange\n";
		$s .= "<00> <FF>\n";
		$s .= "endcodespacerange\n";
		if ($nbr > 0) {
			$s .= "$nbr beginbfrange\n";
			$s .= $ranges;
			$s .= "endbfrange\n";
		}
		if ($nbc > 0) {
			$s .= "$nbc beginbfchar\n";
			$s .= $chars;
			$s .= "endbfchar\n";
		}
		$s .= "endcmap\n";
		$s .= "CMapName currentdict /CMap defineresource pop\n";
		$s .= "end\n";
		$s .= "end";
		return $s;
	}

	protected function _putimages()
	{
		foreach (array_keys($this->images) as $file) {
			$this->_putimage($this->images[$file]);
			unset($this->images[$file]['data']);
			unset($this->images[$file]['smask']);
		}
	}

	protected function _putimage(&$info)
	{
		$this->_newobj();
		$info['n'] = $this->n;
		$this->_put('<</Type /XObject');
		$this->_put('/Subtype /Image');
		$this->_put('/Width ' . $info['w']);
		$this->_put('/Height ' . $info['h']);
		if ($info['cs'] == 'Indexed')
			$this->_put('/ColorSpace [/Indexed /DeviceRGB ' . (strlen($info['pal']) / 3 - 1) . ' ' . ($this->n + 1) . ' 0 R]');
		else {
			$this->_put('/ColorSpace /' . $info['cs']);
			if ($info['cs'] == 'DeviceCMYK')
				$this->_put('/Decode [1 0 1 0 1 0 1 0]');
		}
		$this->_put('/BitsPerComponent ' . $info['bpc']);
		if (isset($info['f']))
			$this->_put('/Filter /' . $info['f']);
		if (isset($info['dp']))
			$this->_put('/DecodeParms <<' . $info['dp'] . '>>');
		if (isset($info['trns']) && is_array($info['trns'])) {
			$trns = '';
			for ($i = 0; $i < count($info['trns']); $i++)
				$trns .= $info['trns'][$i] . ' ' . $info['trns'][$i] . ' ';
			$this->_put('/Mask [' . $trns . ']');
		}
		if (isset($info['smask']))
			$this->_put('/SMask ' . ($this->n + 1) . ' 0 R');
		$this->_put('/Length ' . strlen($info['data']) . '>>');
		$this->_putstream($info['data']);
		$this->_put('endobj');
		// Soft mask
		if (isset($info['smask'])) {
			$dp = '/Predictor 15 /Colors 1 /BitsPerComponent 8 /Columns ' . $info['w'];
			$smask = array('w' => $info['w'], 'h' => $info['h'], 'cs' => 'DeviceGray', 'bpc' => 8, 'f' => $info['f'], 'dp' => $dp, 'data' => $info['smask']);
			$this->_putimage($smask);
		}
		// Palette
		if ($info['cs'] == 'Indexed')
			$this->_putstreamobject($info['pal']);
	}

	protected function _putxobjectdict()
	{
		foreach ($this->images as $image)
			$this->_put('/I' . $image['i'] . ' ' . $image['n'] . ' 0 R');
	}

	protected function _putresourcedict()
	{
		$this->_put('/ProcSet [/PDF /Text /ImageB /ImageC /ImageI]');
		$this->_put('/Font <<');
		foreach ($this->fonts as $font)
			$this->_put('/F' . $font['i'] . ' ' . $font['n'] . ' 0 R');
		$this->_put('>>');
		$this->_put('/XObject <<');
		$this->_putxobjectdict();
		$this->_put('>>');
	}

	protected function _putresources()
	{
		$this->_putfonts();
		$this->_putimages();
		// Resource dictionary
		$this->_newobj(2);
		$this->_put('<<');
		$this->_putresourcedict();
		$this->_put('>>');
		$this->_put('endobj');
	}

	protected function _putinfo()
	{
		$date = @date('YmdHisO', $this->CreationDate);
		$this->metadata['CreationDate'] = 'D:' . substr($date, 0, -2) . "'" . substr($date, -2) . "'";
		foreach ($this->metadata as $key => $value)
			$this->_put('/' . $key . ' ' . $this->_textstring($value));
	}

	protected function _putcatalog()
	{
		$n = $this->PageInfo[1]['n'];
		$this->_put('/Type /Catalog');
		$this->_put('/Pages 1 0 R');
		if ($this->ZoomMode == 'fullpage')
			$this->_put('/OpenAction [' . $n . ' 0 R /Fit]');
		elseif ($this->ZoomMode == 'fullwidth')
			$this->_put('/OpenAction [' . $n . ' 0 R /FitH null]');
		elseif ($this->ZoomMode == 'real')
			$this->_put('/OpenAction [' . $n . ' 0 R /XYZ null null 1]');
		elseif (!is_string($this->ZoomMode))
			$this->_put('/OpenAction [' . $n . ' 0 R /XYZ null null ' . sprintf('%.2F', $this->ZoomMode / 100) . ']');
		if ($this->LayoutMode == 'single')
			$this->_put('/PageLayout /SinglePage');
		elseif ($this->LayoutMode == 'continuous')
			$this->_put('/PageLayout /OneColumn');
		elseif ($this->LayoutMode == 'two')
			$this->_put('/PageLayout /TwoColumnLeft');
	}

	protected function _putheader()
	{
		$this->_put('%PDF-' . $this->PDFVersion);
	}

	protected function _puttrailer()
	{
		$this->_put('/Size ' . ($this->n + 1));
		$this->_put('/Root ' . $this->n . ' 0 R');
		$this->_put('/Info ' . ($this->n - 1) . ' 0 R');
	}

	protected function _enddoc()
	{
		$this->CreationDate = time();
		$this->_putheader();
		$this->_putpages();
		$this->_putresources();
		// Info
		$this->_newobj();
		$this->_put('<<');
		$this->_putinfo();
		$this->_put('>>');
		$this->_put('endobj');
		// Catalog
		$this->_newobj();
		$this->_put('<<');
		$this->_putcatalog();
		$this->_put('>>');
		$this->_put('endobj');
		// Cross-ref
		$offset = $this->_getoffset();
		$this->_put('xref');
		$this->_put('0 ' . ($this->n + 1));
		$this->_put('0000000000 65535 f ');
		for ($i = 1; $i <= $this->n; $i++)
			$this->_put(sprintf('%010d 00000 n ', $this->offsets[$i]));
		// Trailer
		$this->_put('trailer');
		$this->_put('<<');
		$this->_puttrailer();
		$this->_put('>>');
		$this->_put('startxref');
		$this->_put($offset);
		$this->_put('%%EOF');
		$this->state = 3;
	}

	function LineGraph($w, $h, $data, $options = '', $colors = null, $maxVal = 0, $nbDiv = 4)
	{
		/*******************************************
													 Explain the variables:
													 $w = the width of the diagram
													 $h = the height of the diagram
													 $data = the data for the diagram in the form of a multidimensional array
													 $options = the possible formatting options which include:
														 'V' = Print Vertical Divider lines
														 'H' = Print Horizontal Divider Lines
														 'kB' = Print bounding box around the Key (legend)
														 'vB' = Print bounding box around the values under the graph
														 'gB' = Print bounding box around the graph
														 'dB' = Print bounding box around the entire diagram
													 $colors = A multidimensional array containing RGB values
													 $maxVal = The Maximum Value for the graph vertically
													 $nbDiv = The number of vertical Divisions
		 *******************************************/
		$this->SetDrawColor(0, 0, 0);
		$this->SetLineWidth(0.2);
		$keys = array_keys($data);
		$ordinateWidth = 10;
		$w -= $ordinateWidth;
		$valX = $this->getX() + $ordinateWidth;
		$valY = $this->getY();
		$margin = 1;
		$titleH = 8;
		$titleW = $w;
		$lineh = 5;
		$keyH = count($data) * $lineh;
		$keyW = $w / 5;
		$graphValH = 5;
		$graphValW = $w - $keyW - 3 * $margin;
		$graphH = $h - (3 * $margin) - $graphValH;
		$graphW = $w - (2 * $margin) - ($keyW + $margin);
		// $graphW = $w;
		$graphX = $valX + $margin;
		$graphY = $valY + $margin;
		$graphValX = $valX + $margin;
		$graphValY = $valY + 2 * $margin + $graphH;
		$keyX = $valX + (2 * $margin) + $graphW;
		$keyY = $valY + $margin + .5 * ($h - (2 * $margin)) - .5 * ($keyH);
		//draw graph frame border
		if (strstr($options, 'gB')) {
			$this->Rect($valX, $valY, $w, $h);
		}
		//draw graph diagram border
		if (strstr($options, 'dB')) {
			$this->Rect($valX + $margin, $valY + $margin, $graphW, $graphH);
		}
		//draw key legend border
		if (strstr($options, 'kB')) {
			$this->Rect($keyX, $keyY, $keyW, $keyH);
		}
		//draw graph value box
		if (strstr($options, 'vB')) {
			$this->Rect($graphValX, $graphValY, $graphValW, $graphValH);
		}
		//define colors
		if ($colors === null) {
			$safeColors = array(0, 51, 102, 153, 204, 225);
			for ($i = 0; $i < count($data); $i++) {
				$colors[$keys[$i]] = array($safeColors[array_rand($safeColors)], $safeColors[array_rand($safeColors)], $safeColors[array_rand($safeColors)]);
			}
		}
		//form an array with all data values from the multi-demensional $data array
		$ValArray = array();
		foreach ($data as $key => $value) {
			foreach ($data[$key] as $val) {
				$ValArray[] = $val;
			}
		}
		//define max value
		if ($maxVal < ceil(max($ValArray))) {
			$maxVal = ceil(max($ValArray));
		}
		//draw horizontal lines
		$vertDivH = $graphH / $nbDiv;
		if (strstr($options, 'H')) {
			for ($i = 0; $i <= $nbDiv; $i++) {
				if ($i < $nbDiv) {
					$this->Line($graphX, $graphY + $i * $vertDivH, $graphX + $graphW, $graphY + $i * $vertDivH);
				} else {
					$this->Line($graphX, $graphY + $graphH, $graphX + $graphW, $graphY + $graphH);
				}
			}
		}
		//draw vertical lines
		// echo count($data[$keys[0]]);\
		// echo "<br>TEST FPDF <br>";
		// echo (count($data[$keys[0]]) ?? 2);

		$horiDivW = floor($graphW / ((count($data[$keys[0]]) == 1 ? 2 : count($data[$keys[0]])) - 1));
		if (strstr($options, 'V')) {
			$this->setDrawColor(230, 230, 230);
			$this->SetLineWidth(0.2);
			for ($i = 0; $i <= (count($data[$keys[0]]) - 1); $i++) {
				if ($i < (count($data[$keys[0]]) - 1)) {
					$this->Line($graphX + $i * $horiDivW, $graphY, $graphX + $i * $horiDivW, $graphY + $graphH);
				} else {
					$this->Line($graphX + $graphW, $graphY, $graphX + $graphW, $graphY + $graphH);
				}
			}
		}
		//draw graph lines
		foreach ($data as $key => $value) {
			$this->setDrawColor($colors[$key][0], $colors[$key][1], $colors[$key][2]);
			$this->SetLineWidth(0.2);
			$valueKeys = array_keys($value);
			for ($i = 0; $i < count($value); $i++) {
				if ($i == count($value) - 2) {
					$this->Line(
						$graphX + ($i * $horiDivW),
						$graphY + $graphH - ($value[$valueKeys[$i]] / $maxVal * $graphH),
						$graphX + $graphW,
						$graphY + $graphH - ($value[$valueKeys[$i + 1]] / $maxVal * $graphH)
					);
				} else if ($i < (count($value) - 1)) {
					$this->Line(
						$graphX + ($i * $horiDivW),
						$graphY + $graphH - ($value[$valueKeys[$i]] / $maxVal * $graphH),
						$graphX + ($i + 1) * $horiDivW,
						$graphY + $graphH - ($value[$valueKeys[$i + 1]] / $maxVal * $graphH)
					);
				}
			}
			//Set the Key (legend)
			$this->SetFont('Courier', 'B', 5);
			if (!isset($n))
				$n = 0;
			$this->Line($keyX + 1, $keyY + $lineh / 2 + $n * $lineh, $keyX + 8, $keyY + $lineh / 2 + $n * $lineh);
			$this->SetXY($keyX + 8, $keyY + $n * $lineh);
			$this->Cell($keyW, $lineh, $key, 0, 1, 'L');
			$n++;
		}
		//print the abscissa values
		$top_bottom_counter = 0;
		foreach ($valueKeys as $key => $value) {
			if ($top_bottom_counter == 0) {
				if ($key == 0) {

					$this->SetXY($graphValX - 2.5, $graphValY - 2);
					$this->Cell(30, $lineh, $value, 0, 0, 'L');
				} else if ($key == count($valueKeys) - 1) {

					$this->SetXY($graphValX + $graphValW - 25, $graphValY - 2);
					$this->Cell(30, $lineh, $value, 0, 0, 'R');
				} else {

					$this->SetXY($graphValX + $key * $horiDivW - 14.5, $graphValY - 2);
					$this->Cell(30, $lineh, $value, 0, 0, 'C');
				}
				$top_bottom_counter = 1;
			} else {
				if ($key == 0) {

					$this->SetXY($graphValX - 2.5, $graphValY);
					$this->Cell(30, $lineh, $value, 0, 0, 'L');
				} else if ($key == count($valueKeys) - 1) {

					$this->SetXY($graphValX + $graphValW - 25, $graphValY);
					$this->Cell(30, $lineh, $value, 0, 0, 'R');
				} else {

					$this->SetXY($graphValX + $key * $horiDivW - 14.5, $graphValY);
					$this->Cell(30, $lineh, $value, 0, 0, 'C');
				}
				$top_bottom_counter = 0;
			}
		}
		//print the ordinate values
		for ($i = 0; $i <= $nbDiv; $i++) {
			$this->SetXY($graphValX - 10, $graphY + ($nbDiv - $i) * $vertDivH - 3);
			$this->Cell(8, 6, number_format(sprintf('%.1f', $maxVal / $nbDiv * $i), 0), 0, 0, 'R');
		}
		$this->SetDrawColor(0, 0, 0);
		$this->SetLineWidth(0.2);
	}
	var $angle = 0;

	function Rotate($angle, $x = -1, $y = -1)
	{
		if ($x == -1)
			$x = $this->x;
		if ($y == -1)
			$y = $this->y;
		if ($this->angle != 0)
			$this->_out('Q');
		$this->angle = $angle;
		if ($angle != 0) {
			$angle *= M_PI / 180;
			$c = cos($angle);
			$s = sin($angle);
			$cx = $x * $this->k;
			$cy = ($this->h - $y) * $this->k;
			$this->_out(sprintf('q %.5F %.5F %.5F %.5F %.2F %.2F cm 1 0 0 1 %.2F %.2F cm', $c, $s, -$s, $c, $cx, $cy, -$cx, -$cy));
		}
	}

	function RotatedText($x, $y, $txt, $angle)
	{
		//Text rotated around its origin
		$this->Rotate($angle, $x, $y);
		$this->Text($x, $y, $txt);
		$this->Rotate(0);
	}

	function RotatedImage($file, $x, $y, $w, $h, $angle)
	{
		//Image rotated around its upper-left corner
		$this->Rotate($angle, $x, $y);
		$this->Image($file, $x, $y, $w, $h);
		$this->Rotate(0);
	}
}
// class PDF extends PDF_Rotate {
// 	function RotatedText($x, $y, $txt, $angle) {
// 		//Text rotated around its origin
// 		$this->Rotate($angle, $x, $y);
// 		$this->Text($x, $y, $txt);
// 		$this->Rotate(0);
// 	}

// 	function RotatedImage($file, $x, $y, $w, $h, $angle) {
// 		//Image rotated around its upper-left corner
// 		$this->Rotate($angle, $x, $y);
// 		$this->Image($file, $x, $y, $w, $h);
// 		$this->Rotate(0);
// 	}
// }

class PDF extends FPDF
{
	function RoundedRect($x, $y, $w, $h, $r, $corners = '1234', $style = '')
	{
		$k = $this->k;
		$hp = $this->h;
		if ($style == 'F')
			$op = 'f';
		elseif ($style == 'FD' || $style == 'DF')
			$op = 'B';
		else
			$op = 'S';
		$MyArc = 4 / 3 * (sqrt(2) - 1);
		$this->_out(sprintf('%.2F %.2F m', ($x + $r) * $k, ($hp - $y) * $k));

		$xc = $x + $w - $r;
		$yc = $y + $r;
		$this->_out(sprintf('%.2F %.2F l', $xc * $k, ($hp - $y) * $k));
		if (strpos($corners, '2') === false)
			$this->_out(sprintf('%.2F %.2F l', ($x + $w) * $k, ($hp - $y) * $k));
		else
			$this->_Arc($xc + $r * $MyArc, $yc - $r, $xc + $r, $yc - $r * $MyArc, $xc + $r, $yc);

		$xc = $x + $w - $r;
		$yc = $y + $h - $r;
		$this->_out(sprintf('%.2F %.2F l', ($x + $w) * $k, ($hp - $yc) * $k));
		if (strpos($corners, '3') === false)
			$this->_out(sprintf('%.2F %.2F l', ($x + $w) * $k, ($hp - ($y + $h)) * $k));
		else
			$this->_Arc($xc + $r, $yc + $r * $MyArc, $xc + $r * $MyArc, $yc + $r, $xc, $yc + $r);

		$xc = $x + $r;
		$yc = $y + $h - $r;
		$this->_out(sprintf('%.2F %.2F l', $xc * $k, ($hp - ($y + $h)) * $k));
		if (strpos($corners, '4') === false)
			$this->_out(sprintf('%.2F %.2F l', ($x) * $k, ($hp - ($y + $h)) * $k));
		else
			$this->_Arc($xc - $r * $MyArc, $yc + $r, $xc - $r, $yc + $r * $MyArc, $xc - $r, $yc);

		$xc = $x + $r;
		$yc = $y + $r;
		$this->_out(sprintf('%.2F %.2F l', ($x) * $k, ($hp - $yc) * $k));
		if (strpos($corners, '1') === false) {
			$this->_out(sprintf('%.2F %.2F l', ($x) * $k, ($hp - $y) * $k));
			$this->_out(sprintf('%.2F %.2F l', ($x + $r) * $k, ($hp - $y) * $k));
		} else
			$this->_Arc($xc - $r, $yc - $r * $MyArc, $xc - $r * $MyArc, $yc - $r, $xc, $yc - $r);
		$this->_out($op);
	}

	function _Arc($x1, $y1, $x2, $y2, $x3, $y3)
	{
		$h = $this->h;
		$this->_out(sprintf(
			'%.2F %.2F %.2F %.2F %.2F %.2F c ',
			$x1 * $this->k,
			($h - $y1) * $this->k,
			$x2 * $this->k,
			($h - $y2) * $this->k,
			$x3 * $this->k,
			($h - $y3) * $this->k
		));
	}
}