diff --git a/.php_cs.dist b/.php_cs.dist index 895ed80f56..77af33d16a 100644 --- a/.php_cs.dist +++ b/.php_cs.dist @@ -23,7 +23,7 @@ return PhpCsFixer\Config::create() 'combine_consecutive_unsets' => true, 'concat_space' => array('spacing' => 'one'), 'declare_equal_normalize' => true, - 'declare_strict_types' => false, // Too early to adopt strict types + 'declare_strict_types' => true, 'dir_constant' => true, 'elseif' => true, 'encoding' => true, @@ -71,6 +71,7 @@ return PhpCsFixer\Config::create() 'no_spaces_after_function_name' => true, 'no_spaces_around_offset' => true, 'no_spaces_inside_parenthesis' => true, + 'no_superfluous_phpdoc_tags' => true, 'no_trailing_comma_in_list_call' => true, 'no_trailing_comma_in_singleline_array' => true, 'no_trailing_whitespace' => true, @@ -109,10 +110,11 @@ return PhpCsFixer\Config::create() 'phpdoc_single_line_var_spacing' => true, 'phpdoc_summary' => false, 'phpdoc_to_comment' => true, + 'phpdoc_to_return_type' => false, 'phpdoc_trim' => true, 'phpdoc_types' => true, 'phpdoc_var_without_name' => true, - 'pow_to_exponentiation' => false, + 'pow_to_exponentiation' => true, 'pre_increment' => false, 'protected_to_private' => true, 'psr0' => true, @@ -137,7 +139,7 @@ return PhpCsFixer\Config::create() 'switch_case_semicolon_to_colon' => true, 'switch_case_space' => true, 'ternary_operator_spaces' => true, - 'ternary_to_null_coalescing' => false, // Cannot use that with PHP 5.6 + 'ternary_to_null_coalescing' => true, 'trailing_comma_in_multiline_array' => true, 'trim_array_spaces' => false, 'unary_operator_spaces' => true, diff --git a/.travis.yml b/.travis.yml index acdf95cc34..365bd7f244 100644 --- a/.travis.yml +++ b/.travis.yml @@ -3,10 +3,6 @@ language: php dist: xenial php: - - 5.3 - - 5.4 - - 5.5 - - 5.6 - 7.0 - 7.1 - 7.2 @@ -15,21 +11,11 @@ php: matrix: include: - - php: 5.3 - dist: precise - env: COMPOSER_MEMORY_LIMIT=3G - - php: 5.4 - dist: trusty - - php: 5.5 - dist: trusty - php: 7.0 env: COVERAGE=1 - php: 7.3 env: DEPENDENCIES="--ignore-platform-reqs" exclude: - - php: 5.3 - - php: 5.4 - - php: 5.5 - php: 7.0 - php: 7.3 allow_failures: diff --git a/CHANGELOG.md b/CHANGELOG.md index 66271bae00..546f210992 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,15 +5,33 @@ This project adheres to [Semantic Versioning](http://semver.org/). v0.17.0 (?? ??? 2019) ---------------------- +This release marked the addition of strict typing and return type declarations (PHP 7+). + ### Added - Add RightToLeft table presentation. @troosan #1550 - Set complex type in template @troosan #1565 - Add support for page vertical alignment. @troosan #672 #1569 +- Length validation and automatic unit conversion with `PhpOffice\PhpWord\Style\Lengths\{Absolute, Auto, Percent}` @0b10011 #1669 +- Color validation with `PhpOffice\PhpWord\Style\Colors\{BasicColor, SpecialColor}` @0b10011 #1669 +- BorderStyle validation with `PhpOffice\PhpWord\Style\BorderStyle` @0b10011 #1669 +- Support for additional `border-style` values (`hidden`, `groove`, `ridge`, `inset`, and `outset`) in HTML documents @0b10011 #1669 +- Support for all named colors from CSS in HTML documents @0b10011 #1669 +- Support for borders on `Cell` @0b10011 #1669 +- Support for writing percent and auto column widths on `Table` to ODT files @0b10011 #1669 +- Support for `space` and `shadow` on `Border` (and support for writing to Word2007) @0b10011 #1669 + +### Changed +- `float` and `int` are no longer supported for lengths. Only `PhpOffice\PhpWord\Style\Lengths\{Absolute, Auto, Percent}` are allowed. @0b10011 #1669 +- `string` is no longer supported for colors. Only `PhpOffice\PhpWord\Style\Colors\{BasicColor, SpecialColor}` is allowed. @0b10011 #1669 +- `string` is no longer supported for border styles. Only `PhpOffice\PhpWord\Style\BorderStyle` is allowed. @0b10011 #1669 ### Fixed - Fix HTML border-color parsing. @troosan #1551 #1570 +- Fixed specifying cell widths, background color, etc on `PhpOffice\PhpWord\Style\Cell` @0b10011 #1669 +- Escape arrays of replacements in `TemplateProcessor` @0b10011 #1669 ### Miscellaneous +- - Use embedded http server to test loading of remote images @troosan # v0.16.0 (30 dec 2018) diff --git a/README.md b/README.md index b15f83d74e..b326db7fb9 100644 --- a/README.md +++ b/README.md @@ -101,8 +101,13 @@ The following is a basic usage example of the PHPWord library. addText( '"Great achievement is usually born of great sacrifice, ' . 'and is never the result of selfishness." ' . '(Napoleon Hill)', - array('name' => 'Tahoma', 'size' => 10) + array('name' => 'Tahoma', 'size' => Absolute::from('pt', 10)) ); // Adding Text element with font customized using named font style... $fontStyleName = 'oneUserDefinedStyle'; $phpWord->addFontStyle( $fontStyleName, - array('name' => 'Tahoma', 'size' => 10, 'color' => '1B2232', 'bold' => true) + array('name' => 'Tahoma', 'size' => Absolute::from('pt', 10), 'color' => new Hex('1B2232'), 'bold' => true) ); $section->addText( '"The greatest accomplishment is not in never falling, ' @@ -144,10 +149,10 @@ $section->addText( ); // Adding Text element with font customized using explicitly created font style object... -$fontStyle = new \PhpOffice\PhpWord\Style\Font(); +$fontStyle = new Font(); $fontStyle->setBold(true); $fontStyle->setName('Tahoma'); -$fontStyle->setSize(13); +$fontStyle->setSize(Absolute::from('pt', 13)); $myTextElement = $section->addText('"Believe you can and you\'re halfway there." (Theodor Roosevelt)'); $myTextElement->setFontStyle($fontStyle); diff --git a/composer.json b/composer.json index bd57d6e3b1..4f00850f49 100644 --- a/composer.json +++ b/composer.json @@ -58,7 +58,7 @@ "fix": "Fixes issues found by PHP-CS" }, "require": { - "php": "^5.3.3 || ^7.0", + "php": "^7.0", "ext-xml": "*", "zendframework/zend-escaper": "^2.2", "phpoffice/common": "^0.2.9" @@ -66,11 +66,11 @@ "require-dev": { "ext-zip": "*", "ext-gd": "*", - "phpunit/phpunit": "^4.8.36 || ^7.0", - "squizlabs/php_codesniffer": "^2.9", - "friendsofphp/php-cs-fixer": "^2.2", - "phpmd/phpmd": "2.*", - "phploc/phploc": "2.* || 3.* || 4.*", + "phpunit/phpunit": "^6.0 || ^7.0", + "squizlabs/php_codesniffer": "*", + "friendsofphp/php-cs-fixer": "*", + "phpmd/phpmd": "*", + "phploc/phploc": "*", "dompdf/dompdf":"0.8.*", "tecnickcom/tcpdf": "6.*", "mpdf/mpdf": "5.7.4 || 6.* || 7.*", diff --git a/docs/containers.rst b/docs/containers.rst index 9ee58efcd4..ad7d2540cb 100644 --- a/docs/containers.rst +++ b/docs/containers.rst @@ -23,9 +23,11 @@ section. Example: .. code-block:: php + use PhpOffice\PhpWord\Style\Lengths\Absolute; + $sectionStyle = array( 'orientation' => 'landscape', - 'marginTop' => 600, + 'marginTop' => Absolute::from('twip', 600), 'colsNum' => 2, ); diff --git a/docs/elements.rst b/docs/elements.rst index 9d446b27d0..aaa6873aef 100644 --- a/docs/elements.rst +++ b/docs/elements.rst @@ -218,11 +218,11 @@ Table style can be defined with ``addTableStyle``: .. code-block:: php $tableStyle = array( - 'borderColor' => '006699', - 'borderSize' => 6, - 'cellMargin' => 50 + 'borderColor' => new Hex('006699'), + 'borderSize' => Absolute::from('twip', 6), + 'cellMargin' => Absolute::from('twip', 50) ); - $firstRowStyle = array('bgColor' => '66BBFF'); + $firstRowStyle = array('bgColor' => new Hex('66BBFF')); $phpWord->addTableStyle('myTable', $tableStyle, $firstRowStyle); $table = $section->addTable('myTable'); @@ -235,7 +235,7 @@ You can span a cell on multiple columns by using ``gridSpan`` or multiple rows b .. code-block:: php - $cell = $table->addCell(200); + $cell = $table->addCell(Absolute::from('twip', 200)); $cell->getStyle()->setGridSpan(5); See ``Sample_09_Tables.php`` for more code sample. @@ -260,10 +260,10 @@ Examples: $section->addImage( 'mars.jpg', array( - 'width' => 100, - 'height' => 100, - 'marginTop' => -1, - 'marginLeft' => -1, + 'width' => Absolute::from('twip', 100), + 'height' => Absolute::from('twip', 100), + 'marginTop' => Absolute::from('twip', -1), + 'marginLeft' => Absolute::from('twip', -1), 'wrappingStyle' => 'behind' ) ); @@ -285,7 +285,7 @@ header reference. After creating a header, you can use the $section = $phpWord->addSection(); $header = $section->addHeader(); - $header->addWatermark('resources/_earth.jpg', array('marginTop' => 200, 'marginLeft' => 55)); + $header->addWatermark('resources/_earth.jpg', array('marginTop' => Absolute::from('twip', 200), 'marginLeft' => Absolute::from('twip', 55))); Objects ------- @@ -316,7 +316,7 @@ Options for ``$tocStyle``: - ``tabLeader``. Fill type between the title text and the page number. Use the defined constants in ``\PhpOffice\PhpWord\Style\TOC``. - ``tabPos``. The position of the tab where the page number appears in *twip*. -- ``indent``. The indent factor of the titles in *twip*. +- ``indent``. The indent factor of the titles (``\PhpOffice\PhpWord\Style\Lengths\Absolute``). Footnotes & endnotes -------------------- @@ -437,13 +437,13 @@ Line elements can be added to sections by using ``addLine``. Available line style attributes: -- ``weight``. Line width in *twip*. -- ``color``. Defines the color of stroke. +- ``weight``. Line width (``\PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``color``. Defines the color of stroke (``\PhpOffice\PhpWord\Style\Colors\BasicColor``). - ``dash``. Line types: dash, rounddot, squaredot, dashdot, longdash, longdashdot, longdashdotdot. - ``beginArrow``. Start type of arrow: block, open, classic, diamond, oval. - ``endArrow``. End type of arrow: block, open, classic, diamond, oval. -- ``width``. Line-object width in *pt*. -- ``height``. Line-object height in *pt*. +- ``width``. Line-object width (``\PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``height``. Line-object height (``\PhpOffice\PhpWord\Style\Lengths\Absolute``). - ``flip``. Flip the line element: true, false. Chart diff --git a/docs/faq.rst b/docs/faq.rst index 19fca1057f..5f847b27af 100644 --- a/docs/faq.rst +++ b/docs/faq.rst @@ -18,7 +18,7 @@ the merrier, right? I’ve been running PHPWord from CodePlex flawlessly, but I can’t use the latest PHPWord from GitHub. Why? -------------------------------------------------------------------------------------------------------- -PHPWord requires PHP 5.3+ since 0.8, while PHPWord 0.6.3 from CodePlex +PHPWord requires PHP7.0+ since 0.17 (PHP 5.3+ since 0.8), while PHPWord 0.6.3 from CodePlex can run with PHP 5.2. There’s a lot of new features that we can get from PHP 5.3 and it’s been around since 2009! You should upgrade your PHP version to use PHPWord 0.8+. diff --git a/docs/general.rst b/docs/general.rst index f40a08c367..5ae35be401 100644 --- a/docs/general.rst +++ b/docs/general.rst @@ -15,8 +15,14 @@ folder `__. `__. '"Great achievement is usually born of great sacrifice, ' . 'and is never the result of selfishness." ' . '(Napoleon Hill)', - array('name' => 'Tahoma', 'size' => 10) + array('name' => 'Tahoma', 'size' => Absolute::from('pt', 10)) ); // Adding Text element with font customized using named font style... $fontStyleName = 'oneUserDefinedStyle'; $phpWord->addFontStyle( $fontStyleName, - array('name' => 'Tahoma', 'size' => 10, 'color' => '1B2232', 'bold' => true) + array('name' => 'Tahoma', 'size' => Absolute::from('pt', 10), 'color' => new Hex('1B2232'), 'bold' => true) ); $section->addText( '"The greatest accomplishment is not in never falling, ' @@ -58,23 +64,23 @@ folder `__. ); // Adding Text element with font customized using explicitly created font style object... - $fontStyle = new \PhpOffice\PhpWord\Style\Font(); + $fontStyle = new Font(); $fontStyle->setBold(true); $fontStyle->setName('Tahoma'); - $fontStyle->setSize(13); + $fontStyle->setSize(Absolute::from('pt', 13)); $myTextElement = $section->addText('"Believe you can and you\'re halfway there." (Theodor Roosevelt)'); $myTextElement->setFontStyle($fontStyle); // Saving the document as OOXML file... - $objWriter = \PhpOffice\PhpWord\IOFactory::createWriter($phpWord, 'Word2007'); + $objWriter = IOFactory::createWriter($phpWord, 'Word2007'); $objWriter->save('helloWorld.docx'); // Saving the document as ODF file... - $objWriter = \PhpOffice\PhpWord\IOFactory::createWriter($phpWord, 'ODText'); + $objWriter = IOFactory::createWriter($phpWord, 'ODText'); $objWriter->save('helloWorld.odt'); // Saving the document as HTML file... - $objWriter = \PhpOffice\PhpWord\IOFactory::createWriter($phpWord, 'HTML'); + $objWriter = IOFactory::createWriter($phpWord, 'HTML'); $objWriter->save('helloWorld.html'); /* Note: we skip RTF, because it's not XML-based and requires a different example. */ @@ -138,8 +144,10 @@ default font by using the following two functions: .. code-block:: php + use PhpOffice\PhpWord\Style\Lengths\Absolute; + $phpWord->setDefaultFontName('Times New Roman'); - $phpWord->setDefaultFontSize(12); + $phpWord->setDefaultFontSize(Absolute::from('pt', 12)); Document settings ----------------- @@ -251,25 +259,34 @@ name. Use the following functions: Measurement units ----------------- -The base length unit in Open Office XML is twip. Twip means "TWentieth -of an Inch Point", i.e. 1 twip = 1/1440 inch. +While Open Office XML uses twentieths of a point ("twips" or 1/1440 inch) +for most of its measurements, +all lengths in PHPWord use an instance of ``\PhpOffice\PhpWord\Style\Lengths\Length``. +Most measurements are limited to absolute units +(rather than percent or "auto" values), +so those use ``\PhpOffice\PhpWord\Style\Lengths\Absolute`` exclusively. +Methods are typehinted to ensure the right measurement is provided. -You can use PHPWord helper functions to convert inches, centimeters, or -points to twip. +You can use ``\PhpOffice\PhpWord\Style\Lengths\Absolute`` +to specify lengths in units you're comfortable with (eg, "pt"), +and PHPWord will automatically convert those lengths +to the appropriate ones for the format being exported to. .. code-block:: php + use PhpOffice\PhpWord\Style\Lengths\Absolute; + // Paragraph with 6 points space after $phpWord->addParagraphStyle('My Style', array( - 'spaceAfter' => \PhpOffice\PhpWord\Shared\Converter::pointToTwip(6)) - ); + 'spaceAfter' => Absolute::from('pt', 6) + )); $section = $phpWord->addSection(); $sectionStyle = $section->getStyle(); // half inch left margin - $sectionStyle->setMarginLeft(\PhpOffice\PhpWord\Shared\Converter::inchToTwip(.5)); + $sectionStyle->setMarginLeft(Absolute::from('in', .5)); // 2 cm right margin - $sectionStyle->setMarginRight(\PhpOffice\PhpWord\Shared\Converter::cmToTwip(2)); + $sectionStyle->setMarginRight(Absolute::from('cm', 2)); Document protection ------------------- @@ -317,12 +334,14 @@ There is no limit if the option is not set or the provided value is ``0``. Hyphenation Zone ~~~~~~~~~~~~~~~~ -The hyphenation zone (in *twip*) is the allowed amount of whitespace before hyphenation is applied. +The hyphenation zone is the allowed amount of whitespace before hyphenation is applied. The smaller the hyphenation zone the more words are hyphenated. Or in other words, the wider the hyphenation zone the less words are hyphenated. .. code-block:: php - $phpWord->getSettings()->setHyphenationZone(\PhpOffice\PhpWord\Shared\Converter::cmToTwip(1)); + use PhpOffice\PhpWord\Style\Lengths\Absolute; + + $phpWord->getSettings()->setHyphenationZone(Absolute::from('cm', 1)); Hyphenate Caps ~~~~~~~~~~~~~~ diff --git a/docs/installing.rst b/docs/installing.rst index 34353be8ae..286b95d4e7 100644 --- a/docs/installing.rst +++ b/docs/installing.rst @@ -8,7 +8,7 @@ Requirements Mandatory: -- PHP 5.3.3+ +- PHP 7+ - `XML Parser `__ extension - `Zend\\Escaper `__ component - Zend\\Stdlib component @@ -34,7 +34,7 @@ Example: { "require": { - "phpoffice/phpword": "v0.14.*" + "phpoffice/phpword": "v0.16.*" } } diff --git a/docs/recipes.rst b/docs/recipes.rst index 5042cdedc6..bb1d11173a 100644 --- a/docs/recipes.rst +++ b/docs/recipes.rst @@ -10,9 +10,11 @@ Use absolute positioning relative to margin horizontally and to line vertically. .. code-block:: php + use PhpOffice\PhpWord\Style\Lengths\Absolute; + $imageStyle = array( - 'width' => 40, - 'height' => 40, + 'width' => Absolute::from('twip', 40), + 'height' => Absolute::from('twip', 40), 'wrappingStyle' => 'square', 'positioning' => 'absolute', 'posHorizontalRel' => 'margin', @@ -72,10 +74,13 @@ Apply 'HeadingN' paragraph style to TextRun or Link. Sample code: .. code-block:: php + use PhpOffice\PhpWord\Style\Colors\Hex + use PhpOffice\PhpWord\Style\Lengths\Absolute; + $phpWord = new \PhpOffice\PhpWord\PhpWord(); - $phpWord->addTitleStyle(1, array('size' => 16, 'bold' => true)); - $phpWord->addTitleStyle(2, array('size' => 14, 'bold' => true)); - $phpWord->addFontStyle('Link', array('color' => '0000FF', 'underline' => 'single')); + $phpWord->addTitleStyle(1, array('size' => Absolute::from('pt', 16), 'bold' => true)); + $phpWord->addTitleStyle(2, array('size' => Absolute::from('pt', 14), 'bold' => true)); + $phpWord->addFontStyle('Link', array('color' => new Hex('0000FF'), 'underline' => 'single')); $section = $phpWord->addSection(); diff --git a/docs/styles.rst b/docs/styles.rst index 27f8ee66ce..08609dc909 100644 --- a/docs/styles.rst +++ b/docs/styles.rst @@ -10,28 +10,28 @@ Section Available Section style options: -- ``borderBottomColor``. Border bottom color. -- ``borderBottomSize``. Border bottom size in *twip*. -- ``borderLeftColor``. Border left color. -- ``borderLeftSize``. Border left size in *twip*. -- ``borderRightColor``. Border right color. -- ``borderRightSize``. Border right size in *twip*. -- ``borderTopColor``. Border top color. -- ``borderTopSize``. Border top size in *twip*. +- ``borderBottomColor``. Border bottom color (``PhpOffice\PhpWord\Style\Colors\BasicColor``). +- ``borderBottomSize``. Border bottom size (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``borderLeftColor``. Border left color (``PhpOffice\PhpWord\Style\Colors\BasicColor``). +- ``borderLeftSize``. Border left size (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``borderRightColor``. Border right color (``PhpOffice\PhpWord\Style\Colors\BasicColor``). +- ``borderRightSize``. Border right size (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``borderTopColor``. Border top color (``PhpOffice\PhpWord\Style\Colors\BasicColor``). +- ``borderTopSize``. Border top size (``PhpOffice\PhpWord\Style\Lengths\Absolute``). - ``breakType``. Section break type (nextPage, nextColumn, continuous, evenPage, oddPage). - ``colsNum``. Number of columns. -- ``colsSpace``. Spacing between columns. -- ``footerHeight``. Spacing to bottom of footer. -- ``gutter``. Page gutter spacing. -- ``headerHeight``. Spacing to top of header. -- ``marginTop``. Page margin top in *twip*. -- ``marginLeft``. Page margin left in *twip*. -- ``marginRight``. Page margin right in *twip*. -- ``marginBottom``. Page margin bottom in *twip*. +- ``colsSpace``. Spacing between columns (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``footerHeight``. Spacing to bottom of footer (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``gutter``. Page gutter spacing (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``headerHeight``. Spacing to top of header (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``marginTop``. Page margin top (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``marginLeft``. Page margin left (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``marginRight``. Page margin right (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``marginBottom``. Page margin bottom (``PhpOffice\PhpWord\Style\Lengths\Absolute``). - ``orientation``. Page orientation (``portrait``, which is default, or ``landscape``). See ``\PhpOffice\PhpWord\Style\Section::ORIENTATION_...`` class constants for possible values -- ``pageSizeH``. Page height in *twip*. Implicitly defined by ``orientation`` option. Any changes are discouraged. -- ``pageSizeW``. Page width in *twip*. Implicitly defined by ``orientation`` option. Any changes are discouraged. +- ``pageSizeH``. Page height (``PhpOffice\PhpWord\Style\Lengths\Absolute``). Implicitly defined by ``orientation`` option. Any changes are discouraged. +- ``pageSizeW``. Page width (``PhpOffice\PhpWord\Style\Lengths\Absolute``). Implicitly defined by ``orientation`` option. Any changes are discouraged. - ``vAlign``. Vertical Page Alignment See ``\PhpOffice\PhpWord\SimpleType\VerticalJc`` for possible values @@ -43,17 +43,16 @@ Font Available Font style options: - ``allCaps``. All caps, *true* or *false*. -- ``bgColor``. Font background color, e.g. *FF0000*. +- ``bgColor``. Font background color (``\PhpOffice\PhpWord\Style\Colors\BasicColor``). - ``bold``. Bold, *true* or *false*. -- ``color``. Font color, e.g. *FF0000*. +- ``color``. Font color (``\PhpOffice\PhpWord\Style\Colors\BasicColor``). - ``doubleStrikethrough``. Double strikethrough, *true* or *false*. -- ``fgColor``. Font highlight color, e.g. *yellow*, *green*, *blue*. - See ``\PhpOffice\PhpWord\Style\Font::FGCOLOR_...`` class constants for possible values +- ``fgColor``. Font highlight color (``\PhpOffice\PhpWord\Style\Colors\HighlightColor``). - ``hint``. Font content type, *default*, *eastAsia*, or *cs*. - ``italic``. Italic, *true* or *false*. - ``name``. Font name, e.g. *Arial*. - ``rtl``. Right to Left language, *true* or *false*. -- ``size``. Font size, e.g. *20*, *22*. +- ``size``. Font size (``\PhpOffice\PhpWord\Style\Lengths\Absolute``). - ``smallCaps``. Small caps, *true* or *false*. - ``strikethrough``. Strikethrough, *true* or *false*. - ``subScript``. Subscript, *true* or *false*. @@ -62,7 +61,7 @@ Available Font style options: See ``\PhpOffice\PhpWord\Style\Font::UNDERLINE_...`` class constants for possible values - ``lang``. Language, either a language code like *en-US*, *fr-BE*, etc. or an object (or as an array) if you need to set eastAsian or bidirectional languages See ``\PhpOffice\PhpWord\Style\Language`` class for some language codes. -- ``position``. The text position, raised or lowered, in half points +- ``position``. The text position, raised or lowered (``\PhpOffice\PhpWord\Style\Lengths\Absolute``) - ``hidden``. Hidden text, *true* or *false*. .. _paragraph-style: @@ -75,16 +74,16 @@ Available Paragraph style options: - ``alignment``. Supports all alignment modes since 1st Edition of ECMA-376 standard up till ISO/IEC 29500:2012. See ``\PhpOffice\PhpWord\SimpleType\Jc`` class constants for possible values. - ``basedOn``. Parent style. -- ``hanging``. Hanging in *twip*. -- ``indent``. Indent in *twip*. +- ``hanging``. Hanging (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``indent``. Indent (``PhpOffice\PhpWord\Style\Lengths\Absolute``). - ``keepLines``. Keep all lines on one page, *true* or *false*. - ``keepNext``. Keep paragraph with next paragraph, *true* or *false*. - ``lineHeight``. Text line height, e.g. *1.0*, *1.5*, etc. - ``next``. Style for next paragraph. - ``pageBreakBefore``. Start paragraph on next page, *true* or *false*. -- ``spaceBefore``. Space before paragraph in *twip*. -- ``spaceAfter``. Space after paragraph in *twip*. -- ``spacing``. Space between lines in *twip*. If spacingLineRule is auto, 240 (height of 1 line) will be added, so if you want a double line height, set this to 240. +- ``spaceBefore``. Space before paragraph (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``spaceAfter``. Space after paragraph (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``spacing``. Space between lines. If spacingLineRule is auto, 1 will be added, so if you want a double line height, set this to 2. - ``spacingLineRule``. Line Spacing Rule. *auto*, *exact*, *atLeast* See ``\PhpOffice\PhpWord\SimpleType\LineSpacingRule`` class constants for possible values. - ``suppressAutoHyphens``. Hyphenation for paragraph, *true* or *false*. @@ -105,24 +104,23 @@ Available Table style options: - ``alignment``. Supports all alignment modes since 1st Edition of ECMA-376 standard up till ISO/IEC 29500:2012. See ``\PhpOffice\PhpWord\SimpleType\JcTable`` and ``\PhpOffice\PhpWord\SimpleType\Jc`` class constants for possible values. -- ``bgColor``. Background color, e.g. '9966CC'. -- ``border(Top|Right|Bottom|Left)Color``. Border color, e.g. '9966CC'. -- ``border(Top|Right|Bottom|Left)Size``. Border size in *twip*. -- ``cellMargin(Top|Right|Bottom|Left)``. Cell margin in *twip*. -- ``indent``. Table indent from leading margin. Must be an instance of ``\PhpOffice\PhpWord\ComplexType\TblWidth``. -- ``width``. Table width in Fiftieths of a Percent or Twentieths of a Point. -- ``unit``. The unit to use for the width. One of ``\PhpOffice\PhpWord\SimpleType\TblWidth``. Defaults to *auto*. +- ``bgColor``. Background color (``PhpOffice\PhpWord\Style\Colors\BasicColor``). +- ``border(Top|Right|Bottom|Left)Color``. Border color (``PhpOffice\PhpWord\Style\Colors\BasicColor``). +- ``border(Top|Right|Bottom|Left)Size``. Border size (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``cellMargin(Top|Right|Bottom|Left)``. Cell margin (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``indent``. Table indent from leading margin (``PhpOffice\PhpWord\Style\Lengths\Length``). +- ``width``. Table width (``PhpOffice\PhpWord\Style\Lengths\Length``). - ``layout``. Table layout, either *fixed* or *autofit* See ``\PhpOffice\PhpWord\Style\Table`` for constants. -- ``cellSpacing`` Cell spacing in *twip* +- ``cellSpacing`` Cell spacing (``PhpOffice\PhpWord\Style\Lengths\Absolute``). - ``position`` Floating Table Positioning, see below for options - ``bidiVisual`` Present table as Right-To-Left Floating Table Positioning options: -- ``leftFromText`` Distance From Left of Table to Text in *twip* -- ``rightFromText`` Distance From Right of Table to Text in *twip* -- ``topFromText`` Distance From Top of Table to Text in *twip* -- ``bottomFromText`` Distance From Top of Table to Text in *twip* +- ``leftFromText`` Distance From Left of Table to Text (``PhpOffice\PhpWord\Style\Lengths\Absolute``) +- ``rightFromText`` Distance From Right of Table to Text (``PhpOffice\PhpWord\Style\Lengths\Absolute``) +- ``topFromText`` Distance From Top of Table to Text (``PhpOffice\PhpWord\Style\Lengths\Absolute``) +- ``bottomFromText`` Distance From Top of Table to Text (``PhpOffice\PhpWord\Style\Lengths\Absolute``) - ``vertAnchor`` Table Vertical Anchor, one of ``\PhpOffice\PhpWord\Style\TablePosition::VANCHOR_*`` - ``horzAnchor`` Table Horizontal Anchor, one of ``\PhpOffice\PhpWord\Style\TablePosition::HANCHOR_*`` - ``tblpXSpec`` Relative Horizontal Alignment From Anchor, one of ``\PhpOffice\PhpWord\Style\TablePosition::XALIGN_*`` @@ -138,15 +136,15 @@ Available Row style options: Available Cell style options: -- ``bgColor``. Background color, e.g. '9966CC'. -- ``border(Top|Right|Bottom|Left)Color``. Border color, e.g. '9966CC'. -- ``border(Top|Right|Bottom|Left)Size``. Border size in *twip*. +- ``bgColor``. Background color (``PhpOffice\PhpWord\Style\Colors\BasicColor``). +- ``border(Top|Right|Bottom|Left)Color``. Border color (``PhpOffice\PhpWord\Style\Colors\BasicColor``). +- ``border(Top|Right|Bottom|Left)Size``. Border size (``PhpOffice\PhpWord\Style\Lengths\Absolute``). - ``gridSpan``. Number of columns spanned. - ``textDirection(btLr|tbRl)``. Direction of text. You can use constants ``\PhpOffice\PhpWord\Style\Cell::TEXT_DIR_BTLR`` and ``\PhpOffice\PhpWord\Style\Cell::TEXT_DIR_TBRL`` - ``valign``. Vertical alignment, *top*, *center*, *both*, *bottom*. - ``vMerge``. *restart* or *continue*. -- ``width``. Cell width in *twip*. +- ``width``. Cell width (``PhpOffice\PhpWord\Style\Lengths\Length``). .. _image-style: @@ -157,14 +155,14 @@ Available Image style options: - ``alignment``. See ``\PhpOffice\PhpWord\SimpleType\Jc`` class for the details. - ``height``. Height in *pt*. -- ``marginLeft``. Left margin in inches, can be negative. -- ``marginTop``. Top margin in inches, can be negative. -- ``width``. Width in *pt*. +- ``marginLeft``. Left margin, can be negative (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``marginTop``. Top margin, can be negative (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``width``. Width (``PhpOffice\PhpWord\Style\Lengths\Absolute``). - ``wrappingStyle``. Wrapping style, *inline*, *square*, *tight*, *behind*, or *infront*. -- ``wrapDistanceTop``. Top text wrapping in pixels. -- ``wrapDistanceBottom``. Bottom text wrapping in pixels. -- ``wrapDistanceLeft``. Left text wrapping in pixels. -- ``wrapDistanceRight``. Right text wrapping in pixels. +- ``wrapDistanceTop``. Top text wrapping (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``wrapDistanceBottom``. Bottom text wrapping (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``wrapDistanceLeft``. Left text wrapping (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``wrapDistanceRight``. Right text wrapping (``PhpOffice\PhpWord\Style\Lengths\Absolute``). .. _numbering-level-style: @@ -193,8 +191,8 @@ Chart Available Chart style options: -- ``width``. Width (in EMU). -- ``height``. Height (in EMU). +- ``width``. Width (``PhpOffice\PhpWord\Style\Lengths\Absolute``). +- ``height``. Height (``PhpOffice\PhpWord\Style\Lengths\Absolute``). - ``3d``. Is 3D; applies to pie, bar, line, area, *true* or *false*. - ``colors``. A list of colors to use in the chart. - ``title``. The title for the chart. diff --git a/docs/templates-processing.rst b/docs/templates-processing.rst index 5b32aa18e0..af79460701 100644 --- a/docs/templates-processing.rst +++ b/docs/templates-processing.rst @@ -57,12 +57,14 @@ Example: .. code-block:: php + use PhpOffice\PhpWord\Style\Lengths\Absolute; + $templateProcessor = new TemplateProcessor('Template.docx'); $templateProcessor->setValue('Name', 'John Doe'); $templateProcessor->setValue(array('City', 'Street'), array('Detroit', '12th Street')); $templateProcessor->setImageValue('CompanyLogo', 'path/to/company/logo.png'); - $templateProcessor->setImageValue('UserLogo', array('path' => 'path/to/logo.png', 'width' => 100, 'height' => 100, 'ratio' => false)); + $templateProcessor->setImageValue('UserLogo', array('path' => 'path/to/logo.png', 'width' => Absolute::from('twip', 100), 'height' => Absolute::from('twip', 100), 'ratio' => false)); cloneBlock """""""""" @@ -89,10 +91,10 @@ The result will be Customer: ${customer_name#1} Address: ${customer_address#1} - + Customer: ${customer_name#2} Address: ${customer_address#2} - + Customer: ${customer_name#3} Address: ${customer_address#3} @@ -113,7 +115,7 @@ The result will then be Customer: Batman Address: Gotham City - + Customer: Superman Address: Metropolis @@ -234,13 +236,16 @@ See ``Sample_40_TemplateSetComplexValue.php`` for examples. .. code-block:: php - $table = new Table(array('borderSize' => 12, 'borderColor' => 'green', 'width' => 6000, 'unit' => TblWidth::TWIP)); + use PhpOffice\PhpWord\Style\Colors\HighlightColor; + use PhpOffice\PhpWord\Style\Lengths\Absolute; + + $table = new Table(array('borderSize' => Absolute::from('twip', 12), 'borderColor' => new HighlightColor('green'), 'width' => Absolute::from('twip', 6000))); $table->addRow(); - $table->addCell(150)->addText('Cell A1'); - $table->addCell(150)->addText('Cell A2'); - $table->addCell(150)->addText('Cell A3'); + $table->addCell(Absolute::from('twip', 150))->addText('Cell A1'); + $table->addCell(Absolute::from('twip', 150))->addText('Cell A2'); + $table->addCell(Absolute::from('twip', 150))->addText('Cell A3'); $table->addRow(); - $table->addCell(150)->addText('Cell B1'); - $table->addCell(150)->addText('Cell B2'); - $table->addCell(150)->addText('Cell B3'); + $table->addCell(Absolute::from('twip', 150))->addText('Cell B1'); + $table->addCell(Absolute::from('twip', 150))->addText('Cell B2'); + $table->addCell(Absolute::from('twip', 150))->addText('Cell B3'); $templateProcessor->setComplexBlock('table', $table); diff --git a/phpmd.xml.dist b/phpmd.xml.dist index 2077e02b04..0a19509c75 100644 --- a/phpmd.xml.dist +++ b/phpmd.xml.dist @@ -18,8 +18,9 @@ + - + @@ -30,4 +31,4 @@ - \ No newline at end of file + diff --git a/samples/Sample_01_SimpleText.php b/samples/Sample_01_SimpleText.php index 8af44d20aa..90dab41a4a 100644 --- a/samples/Sample_01_SimpleText.php +++ b/samples/Sample_01_SimpleText.php @@ -1,23 +1,29 @@ getSettings()->setThemeFontLang($languageEnGb); $fontStyleName = 'rStyle'; -$phpWord->addFontStyle($fontStyleName, array('bold' => true, 'italic' => true, 'size' => 16, 'allCaps' => true, 'doubleStrikethrough' => true)); +$phpWord->addFontStyle($fontStyleName, array('bold' => true, 'italic' => true, 'size' => Absolute::from('pt', 16), 'allCaps' => true, 'doubleStrikethrough' => true)); $paragraphStyleName = 'pStyle'; -$phpWord->addParagraphStyle($paragraphStyleName, array('alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER, 'spaceAfter' => 100)); +$phpWord->addParagraphStyle($paragraphStyleName, array('alignment' => Jc::CENTER, 'spaceAfter' => Absolute::from('pt', 100))); -$phpWord->addTitleStyle(1, array('bold' => true), array('spaceAfter' => 240)); +$phpWord->addTitleStyle(1, array('bold' => true), array('spaceAfter' => Absolute::from('pt', 240))); // New portrait section $section = $phpWord->addSection(); @@ -28,7 +34,7 @@ // $pStyle = new Font(); // $pStyle->setLang() -$section->addText('Ce texte-ci est en français.', array('lang' => \PhpOffice\PhpWord\Style\Language::FR_BE)); +$section->addText('Ce texte-ci est en français.', array('lang' => Language::FR_BE)); // Two text break $section->addTextBreak(2); @@ -42,12 +48,12 @@ // Inline font style $fontStyle['name'] = 'Times New Roman'; -$fontStyle['size'] = 20; +$fontStyle['size'] = Absolute::from('pt', 20); $textrun = $section->addTextRun(); $textrun->addText('I am inline styled ', $fontStyle); $textrun->addText('with '); -$textrun->addText('color', array('color' => '996699')); +$textrun->addText('color', array('color' => new Hex('996699'))); $textrun->addText(', '); $textrun->addText('bold', array('bold' => true)); $textrun->addText(', '); @@ -69,11 +75,11 @@ $textrun->addText(', '); $textrun->addText('fgColor', array('fgColor' => 'yellow')); $textrun->addText(', '); -$textrun->addText('scale', array('scale' => 200)); +$textrun->addText('scale', array('scale' => Absolute::from('twip', 200))); $textrun->addText(', '); -$textrun->addText('spacing', array('spacing' => 120)); +$textrun->addText('spacing', array('spacing' => Absolute::from('twip', 120))); $textrun->addText(', '); -$textrun->addText('kerning', array('kerning' => 10)); +$textrun->addText('kerning', array('kerning' => Absolute::from('twip', 10))); $textrun->addText('. '); // Link @@ -81,7 +87,7 @@ $section->addTextBreak(); // Image -$section->addImage('resources/_earth.jpg', array('width'=>18, 'height'=>18)); +$section->addImage('resources/_earth.jpg', array('width'=>Absolute::from('pt', 18), 'height'=>Absolute::from('pt', 18))); // Save file echo write($phpWord, basename(__FILE__, '.php'), $writers); diff --git a/samples/Sample_02_TabStops.php b/samples/Sample_02_TabStops.php index 4f588ee7b8..f94e922c78 100644 --- a/samples/Sample_02_TabStops.php +++ b/samples/Sample_02_TabStops.php @@ -1,9 +1,14 @@ array( - new \PhpOffice\PhpWord\Style\Tab('left', 1550), - new \PhpOffice\PhpWord\Style\Tab('center', 3200), - new \PhpOffice\PhpWord\Style\Tab('right', 5300), + new Tab('left', Absolute::from('twip', 1550)), + new Tab('center', Absolute::from('twip', 3200)), + new Tab('right', Absolute::from('twip', 5300)), ), ) ); $rightTabStyleName = 'rightTab'; -$phpWord->addParagraphStyle($rightTabStyleName, array('tabs' => array(new \PhpOffice\PhpWord\Style\Tab('right', 9090)))); +$phpWord->addParagraphStyle($rightTabStyleName, array('tabs' => array(new Tab('right', Absolute::from('twip', 9090))))); $leftTabStyleName = 'centerTab'; -$phpWord->addParagraphStyle($leftTabStyleName, array('tabs' => array(new \PhpOffice\PhpWord\Style\Tab('center', 4680)))); +$phpWord->addParagraphStyle($leftTabStyleName, array('tabs' => array(new Tab('center', Absolute::from('twip', 4680))))); // New portrait section $section = $phpWord->addSection(); diff --git a/samples/Sample_03_Sections.php b/samples/Sample_03_Sections.php index 5bb9ecc2e4..c9146b667c 100644 --- a/samples/Sample_03_Sections.php +++ b/samples/Sample_03_Sections.php @@ -1,14 +1,18 @@ addSection(array('borderColor' => '00FF00', 'borderSize' => 12)); +$section = $phpWord->addSection(array('borderColor' => new Hex('00FF00'), 'borderSize' => Absolute::from('twip', 12))); $section->addText('I am placed on a default section.'); // New landscape section @@ -19,7 +23,7 @@ // New portrait section $section = $phpWord->addSection( - array('paperSize' => 'Folio', 'marginLeft' => 600, 'marginRight' => 600, 'marginTop' => 600, 'marginBottom' => 600) + array('paperSize' => 'Folio', 'marginLeft' => Absolute::from('twip', 600), 'marginRight' => Absolute::from('twip', 600), 'marginTop' => Absolute::from('twip', 600), 'marginBottom' => Absolute::from('twip', 600)) ); $section->addText('This section uses other margins with folio papersize.'); @@ -32,12 +36,12 @@ // New portrait section with Header & Footer $section = $phpWord->addSection( array( - 'marginLeft' => 200, - 'marginRight' => 200, - 'marginTop' => 200, - 'marginBottom' => 200, - 'headerHeight' => 50, - 'footerHeight' => 50, + 'marginLeft' => Absolute::from('twip', 200), + 'marginRight' => Absolute::from('twip', 200), + 'marginTop' => Absolute::from('twip', 200), + 'marginBottom' => Absolute::from('twip', 200), + 'headerHeight' => Absolute::from('twip', 50), + 'footerHeight' => Absolute::from('twip', 50), ) ); $section->addText('This section and we play with header/footer height.'); diff --git a/samples/Sample_04_Textrun.php b/samples/Sample_04_Textrun.php index ecd0c88a9f..7224cc469a 100644 --- a/samples/Sample_04_Textrun.php +++ b/samples/Sample_04_Textrun.php @@ -1,22 +1,28 @@ addParagraphStyle($paragraphStyleName, array('spacing' => 100)); +$phpWord->addParagraphStyle($paragraphStyleName, array('spacing' => Absolute::from('twip', 100))); $boldFontStyleName = 'BoldText'; $phpWord->addFontStyle($boldFontStyleName, array('bold' => true)); $coloredFontStyleName = 'ColoredText'; -$phpWord->addFontStyle($coloredFontStyleName, array('color' => 'FF8080', 'bgColor' => 'FFFFCC')); +$phpWord->addFontStyle($coloredFontStyleName, array('color' => new Hex('FF8080'), 'bgColor' => new Hex('FFFFCC'))); $linkFontStyleName = 'NLink'; -$phpWord->addLinkStyle($linkFontStyleName, array('color' => '0000FF', 'underline' => \PhpOffice\PhpWord\Style\Font::UNDERLINE_SINGLE)); +$phpWord->addLinkStyle($linkFontStyleName, array('color' => new Hex('0000FF'), 'underline' => Font::UNDERLINE_SINGLE)); // New portrait section $section = $phpWord->addSection(); @@ -34,7 +40,7 @@ $textrun->addText(' Sample Link: '); $textrun->addLink('https://github.com/PHPOffice/PHPWord', 'PHPWord on GitHub', $linkFontStyleName); $textrun->addText(' Sample Image: '); -$textrun->addImage('resources/_earth.jpg', array('width' => 18, 'height' => 18)); +$textrun->addImage('resources/_earth.jpg', array('width' => Absolute::from('pt', 18), 'height' => Absolute::from('pt', 18))); $textrun->addText(' Sample Object: '); $textrun->addObject('resources/_sheet.xls'); $textrun->addText(' Here is some more text. '); diff --git a/samples/Sample_05_Multicolumn.php b/samples/Sample_05_Multicolumn.php index f7cefa58d3..370a7f255d 100644 --- a/samples/Sample_05_Multicolumn.php +++ b/samples/Sample_05_Multicolumn.php @@ -1,9 +1,13 @@ addSection( array( 'colsNum' => 2, - 'colsSpace' => 1440, + 'colsSpace' => Absolute::from('twip', 1440), 'breakType' => 'continuous', ) ); @@ -31,7 +35,7 @@ $section = $phpWord->addSection( array( 'colsNum' => 3, - 'colsSpace' => 720, + 'colsSpace' => Absolute::from('twip', 720), 'breakType' => 'continuous', ) ); diff --git a/samples/Sample_06_Footnote.php b/samples/Sample_06_Footnote.php index 19d6a52462..e2a062ee29 100644 --- a/samples/Sample_06_Footnote.php +++ b/samples/Sample_06_Footnote.php @@ -1,6 +1,10 @@ addParagraphStyle($paragraphStyleName, array('spacing' => 100)); +$phpWord->addParagraphStyle($paragraphStyleName, array('spacing' => Absolute::from('twip', 100))); $boldFontStyleName = 'BoldText'; $phpWord->addFontStyle($boldFontStyleName, array('bold' => true)); $coloredFontStyleName = 'ColoredText'; -$phpWord->addFontStyle($coloredFontStyleName, array('color' => 'FF8080', 'bgColor' => 'FFFFCC')); +$phpWord->addFontStyle($coloredFontStyleName, array('color' => new Hex('FF8080'), 'bgColor' => new Hex('FFFFCC'))); $linkFontStyleName = 'NLink'; -$phpWord->addLinkStyle($linkFontStyleName, array('color' => '0000FF', 'underline' => \PhpOffice\PhpWord\Style\Font::UNDERLINE_SINGLE)); +$phpWord->addLinkStyle($linkFontStyleName, array('color' => new Hex('0000FF'), 'underline' => Font::UNDERLINE_SINGLE)); // New portrait section $section = $phpWord->addSection(); @@ -38,7 +42,7 @@ $footnote->addText('links like '); $footnote->addLink('https://github.com/PHPOffice/PHPWord', 'PHPWord on GitHub', $linkFontStyleName); $footnote->addText(', image like '); -$footnote->addImage('resources/_earth.jpg', array('width' => 18, 'height' => 18)); +$footnote->addImage('resources/_earth.jpg', array('width' => Absolute::from('pt', 18), 'height' => Absolute::from('pt', 18))); $footnote->addText(', or object like '); $footnote->addObject('resources/_sheet.xls'); $footnote->addText('But you can only put footnote in section, not in header or footer.'); diff --git a/samples/Sample_07_TemplateCloneRow.php b/samples/Sample_07_TemplateCloneRow.php index 42c53269e8..8d3703aa1a 100644 --- a/samples/Sample_07_TemplateCloneRow.php +++ b/samples/Sample_07_TemplateCloneRow.php @@ -1,4 +1,5 @@ setDefaultParagraphStyle( array( - 'alignment' => \PhpOffice\PhpWord\SimpleType\Jc::BOTH, - 'spaceAfter' => \PhpOffice\PhpWord\Shared\Converter::pointToTwip(12), - 'spacing' => 120, + 'alignment' => Jc::BOTH, + 'spaceAfter' => Absolute::from('pt', 12), + 'spacing' => Absolute::from('twip', 120), ) ); @@ -20,7 +25,7 @@ . 'pagination. See "Line and Page Break" tab on paragraph properties ' . 'window to see the attribute set by these controls.', array('bold' => true), - array('space' => array('before' => 360, 'after' => 480)) + array('space' => array('before' => Absolute::from('twip', 360), 'after' => Absolute::from('twip', 480))) ); $section->addText( @@ -30,7 +35,7 @@ . 'itself at the bottom of a page. Set this option to "false" if you want ' . 'to disable this automatic control.', null, - array('widowControl' => false, 'indentation' => array('left' => 240, 'right' => 120)) + array('widowControl' => false, 'indentation' => array('left' => Absolute::from('twip', 240), 'right' => Absolute::from('twip', 120))) ); $section->addText( @@ -39,7 +44,7 @@ . 'breaks between paragraphs. Set this option to "true" if you do not want ' . 'your paragraph to be separated with the next paragraph.', null, - array('keepNext' => true, 'indentation' => array('firstLine' => 240)) + array('keepNext' => true, 'indentation' => array('firstLine' => Absolute::from('twip', 240))) ); $section->addText( @@ -48,7 +53,7 @@ . 'break within a paragraph. Set this option to "true" if you do not want ' . 'all lines of your paragraph to be in the same page.', null, - array('keepLines' => true, 'indentation' => array('left' => 240, 'hanging' => 240)) + array('keepLines' => true, 'indentation' => array('left' => Absolute::from('twip', 240), 'hanging' => Absolute::from('twip', 240))) ); $section->addText('Keep scrolling. More below.'); diff --git a/samples/Sample_09_Tables.php b/samples/Sample_09_Tables.php index 32d895732d..10f572d24f 100644 --- a/samples/Sample_09_Tables.php +++ b/samples/Sample_09_Tables.php @@ -1,5 +1,10 @@ addSection(); -$header = array('size' => 16, 'bold' => true); +$header = array('size' => Absolute::from('pt', 16), 'bold' => true); // 1. Basic table @@ -20,7 +25,7 @@ for ($r = 1; $r <= 8; $r++) { $table->addRow(); for ($c = 1; $c <= 5; $c++) { - $table->addCell(1750)->addText("Row {$r}, Cell {$c}"); + $table->addCell(Absolute::from('twip', 1750))->addText("Row {$r}, Cell {$c}"); } } @@ -30,27 +35,27 @@ $section->addText('Fancy table', $header); $fancyTableStyleName = 'Fancy Table'; -$fancyTableStyle = array('borderSize' => 6, 'borderColor' => '006699', 'cellMargin' => 80, 'alignment' => \PhpOffice\PhpWord\SimpleType\JcTable::CENTER, 'cellSpacing' => 50); -$fancyTableFirstRowStyle = array('borderBottomSize' => 18, 'borderBottomColor' => '0000FF', 'bgColor' => '66BBFF'); +$fancyTableStyle = array('borderSize' => Absolute::from('twip', 6), 'borderColor' => new Hex('006699'), 'cellMargin' => Absolute::from('twip', 80), 'alignment' => JcTable::CENTER, 'cellSpacing' => Absolute::from('twip', 50)); +$fancyTableFirstRowStyle = array('borderBottomSize' => Absolute::from('twip', 18), 'borderBottomColor' => new Hex('0000FF'), 'bgColor' => new Hex('66BBFF')); $fancyTableCellStyle = array('valign' => 'center'); -$fancyTableCellBtlrStyle = array('valign' => 'center', 'textDirection' => \PhpOffice\PhpWord\Style\Cell::TEXT_DIR_BTLR); +$fancyTableCellBtlrStyle = array('valign' => 'center', 'textDirection' => Cell::TEXT_DIR_BTLR); $fancyTableFontStyle = array('bold' => true); $phpWord->addTableStyle($fancyTableStyleName, $fancyTableStyle, $fancyTableFirstRowStyle); $table = $section->addTable($fancyTableStyleName); -$table->addRow(900); -$table->addCell(2000, $fancyTableCellStyle)->addText('Row 1', $fancyTableFontStyle); -$table->addCell(2000, $fancyTableCellStyle)->addText('Row 2', $fancyTableFontStyle); -$table->addCell(2000, $fancyTableCellStyle)->addText('Row 3', $fancyTableFontStyle); -$table->addCell(2000, $fancyTableCellStyle)->addText('Row 4', $fancyTableFontStyle); -$table->addCell(500, $fancyTableCellBtlrStyle)->addText('Row 5', $fancyTableFontStyle); +$table->addRow(Absolute::from('twip', Absolute::from('twip', 900))); +$table->addCell(Absolute::from('twip', 2000), $fancyTableCellStyle)->addText('Row 1', $fancyTableFontStyle); +$table->addCell(Absolute::from('twip', 2000), $fancyTableCellStyle)->addText('Row 2', $fancyTableFontStyle); +$table->addCell(Absolute::from('twip', 2000), $fancyTableCellStyle)->addText('Row 3', $fancyTableFontStyle); +$table->addCell(Absolute::from('twip', 2000), $fancyTableCellStyle)->addText('Row 4', $fancyTableFontStyle); +$table->addCell(Absolute::from('twip', 500), $fancyTableCellBtlrStyle)->addText('Row 5', $fancyTableFontStyle); for ($i = 1; $i <= 8; $i++) { $table->addRow(); - $table->addCell(2000)->addText("Cell {$i}"); - $table->addCell(2000)->addText("Cell {$i}"); - $table->addCell(2000)->addText("Cell {$i}"); - $table->addCell(2000)->addText("Cell {$i}"); + $table->addCell(Absolute::from('twip', 2000))->addText("Cell {$i}"); + $table->addCell(Absolute::from('twip', 2000))->addText("Cell {$i}"); + $table->addCell(Absolute::from('twip', 2000))->addText("Cell {$i}"); + $table->addCell(Absolute::from('twip', 2000))->addText("Cell {$i}"); $text = (0 == $i % 2) ? 'X' : ''; - $table->addCell(500)->addText($text); + $table->addCell(Absolute::from('twip', 500))->addText($text); } /* @@ -65,11 +70,11 @@ $section->addPageBreak(); $section->addText('Table with colspan and rowspan', $header); -$fancyTableStyle = array('borderSize' => 6, 'borderColor' => '999999'); -$cellRowSpan = array('vMerge' => 'restart', 'valign' => 'center', 'bgColor' => 'FFFF00'); +$fancyTableStyle = array('borderSize' => 6, 'borderColor' => new Hex('999999')); +$cellRowSpan = array('vMerge' => 'restart', 'valign' => 'center', 'bgColor' => new Hex('FFFF00')); $cellRowContinue = array('vMerge' => 'continue'); $cellColSpan = array('gridSpan' => 2, 'valign' => 'center'); -$cellHCentered = array('alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER); +$cellHCentered = array('alignment' => Jc::CENTER); $cellVCentered = array('valign' => 'center'); $spanTableStyleName = 'Colspan Rowspan'; @@ -78,22 +83,22 @@ $table->addRow(); -$cell1 = $table->addCell(2000, $cellRowSpan); +$cell1 = $table->addCell(Absolute::from('twip', 2000), $cellRowSpan); $textrun1 = $cell1->addTextRun($cellHCentered); $textrun1->addText('A'); $textrun1->addFootnote()->addText('Row span'); -$cell2 = $table->addCell(4000, $cellColSpan); +$cell2 = $table->addCell(Absolute::from('twip', 4000), $cellColSpan); $textrun2 = $cell2->addTextRun($cellHCentered); $textrun2->addText('B'); $textrun2->addFootnote()->addText('Column span'); -$table->addCell(2000, $cellRowSpan)->addText('E', null, $cellHCentered); +$table->addCell(Absolute::from('twip', 2000), $cellRowSpan)->addText('E', null, $cellHCentered); $table->addRow(); $table->addCell(null, $cellRowContinue); -$table->addCell(2000, $cellVCentered)->addText('C', null, $cellHCentered); -$table->addCell(2000, $cellVCentered)->addText('D', null, $cellHCentered); +$table->addCell(Absolute::from('twip', 2000), $cellVCentered)->addText('C', null, $cellHCentered); +$table->addCell(Absolute::from('twip', 2000), $cellVCentered)->addText('D', null, $cellHCentered); $table->addCell(null, $cellRowContinue); /* @@ -111,35 +116,35 @@ $section->addPageBreak(); $section->addText('Table with colspan and rowspan', $header); -$styleTable = array('borderSize' => 6, 'borderColor' => '999999'); +$styleTable = array('borderSize' => 6, 'borderColor' => new Hex('999999')); $phpWord->addTableStyle('Colspan Rowspan', $styleTable); $table = $section->addTable('Colspan Rowspan'); $row = $table->addRow(); -$row->addCell(1000, array('vMerge' => 'restart'))->addText('A'); -$row->addCell(1000, array('gridSpan' => 2, 'vMerge' => 'restart'))->addText('B'); -$row->addCell(1000)->addText('1'); +$row->addCell(Absolute::from('twip', 1000), array('vMerge' => 'restart'))->addText('A'); +$row->addCell(Absolute::from('twip', 1000), array('gridSpan' => 2, 'vMerge' => 'restart'))->addText('B'); +$row->addCell(Absolute::from('twip', 1000))->addText('1'); $row = $table->addRow(); -$row->addCell(1000, array('vMerge' => 'continue')); -$row->addCell(1000, array('vMerge' => 'continue', 'gridSpan' => 2)); -$row->addCell(1000)->addText('2'); +$row->addCell(Absolute::from('twip', 1000), array('vMerge' => 'continue')); +$row->addCell(Absolute::from('twip', 1000), array('vMerge' => 'continue', 'gridSpan' => 2)); +$row->addCell(Absolute::from('twip', 1000))->addText('2'); $row = $table->addRow(); -$row->addCell(1000, array('vMerge' => 'continue')); -$row->addCell(1000)->addText('C'); -$row->addCell(1000)->addText('D'); -$row->addCell(1000)->addText('3'); +$row->addCell(Absolute::from('twip', 1000), array('vMerge' => 'continue')); +$row->addCell(Absolute::from('twip', 1000))->addText('C'); +$row->addCell(Absolute::from('twip', 1000))->addText('D'); +$row->addCell(Absolute::from('twip', 1000))->addText('3'); // 5. Nested table $section->addTextBreak(2); $section->addText('Nested table in a centered and 50% width table.', $header); -$table = $section->addTable(array('width' => 50 * 50, 'unit' => 'pct', 'alignment' => \PhpOffice\PhpWord\SimpleType\JcTable::CENTER)); +$table = $section->addTable(array('width' => new Percent(50), 'alignment' => JcTable::CENTER)); $cell = $table->addRow()->addCell(); $cell->addText('This cell contains nested table.'); -$innerCell = $cell->addTable(array('alignment' => \PhpOffice\PhpWord\SimpleType\JcTable::CENTER))->addRow()->addCell(); +$innerCell = $cell->addTable(array('alignment' => JcTable::CENTER))->addRow()->addCell(); $innerCell->addText('Inside nested table'); // 6. Table with floating position @@ -147,7 +152,7 @@ $section->addTextBreak(2); $section->addText('Table with floating positioning.', $header); -$table = $section->addTable(array('borderSize' => 6, 'borderColor' => '999999', 'position' => array('vertAnchor' => TablePosition::VANCHOR_TEXT, 'bottomFromText' => Converter::cmToTwip(1)))); +$table = $section->addTable(array('borderSize' => Absolute::from('twip', 6), 'borderColor' => new Hex('999999'), 'position' => array('vertAnchor' => TablePosition::VANCHOR_TEXT, 'bottomFromText' => Absolute::from('cm', 1)))); $cell = $table->addRow()->addCell(); $cell->addText('This is a single cell.'); diff --git a/samples/Sample_10_EastAsianFontStyle.php b/samples/Sample_10_EastAsianFontStyle.php index 87345ae0e8..f5a0f5e2c1 100644 --- a/samples/Sample_10_EastAsianFontStyle.php +++ b/samples/Sample_10_EastAsianFontStyle.php @@ -1,13 +1,18 @@ addSection(); -$header = array('size' => 16, 'bold' => true); +$header = array('size' => Absolute::from('pt', 16), 'bold' => true); //1.Use EastAisa FontStyle -$section->addText('中文楷体样式测试', array('name' => '楷体', 'size' => 16, 'color' => '1B2232', 'lang' => array('latin' => 'en-US', 'eastAsia' => 'zh-CN'))); +$section->addText('中文楷体样式测试', array('name' => '楷体', 'size' => Absolute::from('pt', 16), 'color' => new Hex('1B2232'), 'lang' => array('latin' => 'en-US', 'eastAsia' => 'zh-CN'))); // Save file echo write($phpWord, basename(__FILE__, '.php'), $writers); diff --git a/samples/Sample_11_ReadWord2007.php b/samples/Sample_11_ReadWord2007.php index c0b54c7a49..6b1ccacb6f 100644 --- a/samples/Sample_11_ReadWord2007.php +++ b/samples/Sample_11_ReadWord2007.php @@ -1,4 +1,5 @@ firstPage(); $table = $header->addTable(); $table->addRow(); -$cell = $table->addCell(4500); +$cell = $table->addCell(Absolute::from('twip', 4500)); $textrun = $cell->addTextRun(); $textrun->addText('This is the header with '); $textrun->addLink('https://github.com/PHPOffice/PHPWord', 'PHPWord on GitHub'); -$table->addCell(4500)->addImage('resources/PhpWord.png', array('width' => 80, 'height' => 80, 'alignment' => \PhpOffice\PhpWord\SimpleType\Jc::END)); +$table->addCell(Absolute::from('twip', 4500))->addImage('resources/PhpWord.png', array('width' => Absolute::from('pt', 80), 'height' => Absolute::from('pt', 80), 'alignment' => Jc::END)); // Add header for all other pages $subsequent = $section->addHeader(); $subsequent->addText('Subsequent pages in Section 1 will Have this!'); -$subsequent->addImage('resources/_mars.jpg', array('width' => 80, 'height' => 80)); +$subsequent->addImage('resources/_mars.jpg', array('width' => Absolute::from('pt', 80), 'height' => Absolute::from('pt', 80))); // Add footer $footer = $section->addFooter(); -$footer->addPreserveText('Page {PAGE} of {NUMPAGES}.', null, array('alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER)); +$footer->addPreserveText('Page {PAGE} of {NUMPAGES}.', null, array('alignment' => Jc::CENTER)); $footer->addLink('https://github.com/PHPOffice/PHPWord', 'PHPWord on GitHub'); // Write some text diff --git a/samples/Sample_13_Images.php b/samples/Sample_13_Images.php index f7be3be969..3834ce0906 100644 --- a/samples/Sample_13_Images.php +++ b/samples/Sample_13_Images.php @@ -1,12 +1,16 @@ addSection(); @@ -15,7 +19,7 @@ printSeparator($section); $section->addText('Local image with styles:'); -$section->addImage('resources/_earth.jpg', array('width' => 210, 'height' => 210, 'alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER)); +$section->addImage('resources/_earth.jpg', array('width' => Absolute::from('pt', 210), 'height' => Absolute::from('pt', 210), 'alignment' => Jc::CENTER)); // Remote image printSeparator($section); @@ -40,13 +44,13 @@ 'resources/_earth.jpg', array( 'positioning' => 'relative', - 'marginTop' => -1, - 'marginLeft' => 1, - 'width' => 80, - 'height' => 80, + 'marginTop' => Absolute::from('cm', -1), + 'marginLeft' => Absolute::from('cm', 1), + 'width' => Absolute::from('pt', 80), + 'height' => Absolute::from('pt', 80), 'wrappingStyle' => $wrappingStyle, - 'wrapDistanceRight' => Converter::cmToPoint(1), - 'wrapDistanceBottom' => Converter::cmToPoint(1), + 'wrapDistanceRight' => Absolute::from('cm', 1), + 'wrapDistanceBottom' => Absolute::from('cm', 1), ) ); $section->addText($text); @@ -58,14 +62,14 @@ $section->addImage( 'resources/_mars.jpg', array( - 'width' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(3), - 'height' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(3), - 'positioning' => \PhpOffice\PhpWord\Style\Image::POSITION_ABSOLUTE, - 'posHorizontal' => \PhpOffice\PhpWord\Style\Image::POSITION_HORIZONTAL_RIGHT, - 'posHorizontalRel' => \PhpOffice\PhpWord\Style\Image::POSITION_RELATIVE_TO_PAGE, - 'posVerticalRel' => \PhpOffice\PhpWord\Style\Image::POSITION_RELATIVE_TO_PAGE, - 'marginLeft' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(15.5), - 'marginTop' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(1.55), + 'width' => Absolute::from('cm', 3), + 'height' => Absolute::from('cm', 3), + 'positioning' => Image::POSITION_ABSOLUTE, + 'posHorizontal' => Image::POSITION_HORIZONTAL_RIGHT, + 'posHorizontalRel' => Image::POSITION_RELATIVE_TO_PAGE, + 'posVerticalRel' => Image::POSITION_RELATIVE_TO_PAGE, + 'marginLeft' => Absolute::from('cm', 15.5), + 'marginTop' => Absolute::from('cm', 1.55), ) ); @@ -76,20 +80,20 @@ $section->addImage( 'resources/_mars.jpg', array( - 'width' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(3), - 'height' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(3), - 'positioning' => \PhpOffice\PhpWord\Style\Image::POSITION_RELATIVE, - 'posHorizontal' => \PhpOffice\PhpWord\Style\Image::POSITION_HORIZONTAL_CENTER, - 'posHorizontalRel' => \PhpOffice\PhpWord\Style\Image::POSITION_RELATIVE_TO_COLUMN, - 'posVertical' => \PhpOffice\PhpWord\Style\Image::POSITION_VERTICAL_TOP, - 'posVerticalRel' => \PhpOffice\PhpWord\Style\Image::POSITION_RELATIVE_TO_LINE, + 'width' => Absolute::from('cm', 3), + 'height' => Absolute::from('cm', 3), + 'positioning' => Image::POSITION_RELATIVE, + 'posHorizontal' => Image::POSITION_HORIZONTAL_CENTER, + 'posHorizontalRel' => Image::POSITION_RELATIVE_TO_COLUMN, + 'posVertical' => Image::POSITION_VERTICAL_TOP, + 'posVerticalRel' => Image::POSITION_RELATIVE_TO_LINE, ) ); function printSeparator(Section $section) { $section->addTextBreak(); - $lineStyle = array('weight' => 0.2, 'width' => 150, 'height' => 0, 'align' => 'center'); + $lineStyle = array('weight' => Absolute::from('twip', 0.2), 'width' => Absolute::from('twip', 150), 'height' => Absolute::from('twip', 0), 'align' => 'center'); $section->addLine($lineStyle); $section->addTextBreak(2); } diff --git a/samples/Sample_14_ListItem.php b/samples/Sample_14_ListItem.php index f40e9f6fb9..6a3af200a1 100644 --- a/samples/Sample_14_ListItem.php +++ b/samples/Sample_14_ListItem.php @@ -1,16 +1,21 @@ addFontStyle($fontStyleName, array('color' => 'FF0000')); +$phpWord->addFontStyle($fontStyleName, array('color' => new Hex('FF0000'))); $paragraphStyleName = 'P-Style'; -$phpWord->addParagraphStyle($paragraphStyleName, array('spaceAfter' => 95)); +$phpWord->addParagraphStyle($paragraphStyleName, array('spaceAfter' => Absolute::from('twip', 95))); $multilevelNumberingStyleName = 'multilevel'; $phpWord->addNumberingStyle( @@ -18,8 +23,8 @@ array( 'type' => 'multilevel', 'levels' => array( - array('format' => 'decimal', 'text' => '%1.', 'left' => 360, 'hanging' => 360, 'tabPos' => 360), - array('format' => 'upperLetter', 'text' => '%2.', 'left' => 720, 'hanging' => 360, 'tabPos' => 720), + array('format' => 'decimal', 'text' => '%1.', 'left' => Absolute::from('twip', 360), 'hanging' => Absolute::from('twip', 360), 'tabPos' => Absolute::from('twip', 360)), + array('format' => 'upperLetter', 'text' => '%2.', 'left' => Absolute::from('twip', 720), 'hanging' => Absolute::from('twip', 360), 'tabPos' => Absolute::from('twip', 720)), ), ) ); @@ -86,9 +91,9 @@ ), ) ); -$phpWord->addTitleStyle(1, array('size' => 16), array('numStyle' => $headingNumberingStyleName, 'numLevel' => 0)); -$phpWord->addTitleStyle(2, array('size' => 14), array('numStyle' => $headingNumberingStyleName, 'numLevel' => 1)); -$phpWord->addTitleStyle(3, array('size' => 12), array('numStyle' => $headingNumberingStyleName, 'numLevel' => 2)); +$phpWord->addTitleStyle(1, array('size' => Absolute::from('pt', 16)), array('numStyle' => $headingNumberingStyleName, 'numLevel' => 0)); +$phpWord->addTitleStyle(2, array('size' => Absolute::from('pt', 14)), array('numStyle' => $headingNumberingStyleName, 'numLevel' => 1)); +$phpWord->addTitleStyle(3, array('size' => Absolute::from('pt', 12)), array('numStyle' => $headingNumberingStyleName, 'numLevel' => 2)); $section->addTitle('Heading 1', 1); $section->addTitle('Heading 2', 2); diff --git a/samples/Sample_15_Link.php b/samples/Sample_15_Link.php index 86c7973bc1..5ced064fef 100644 --- a/samples/Sample_15_Link.php +++ b/samples/Sample_15_Link.php @@ -1,13 +1,18 @@ addLinkStyle($linkFontStyleName, array('bold' => true, 'color' => '808000')); +$phpWord->addLinkStyle($linkFontStyleName, array('bold' => true, 'color' => new Hex('808000'))); // New section $section = $phpWord->addSection(); @@ -16,7 +21,7 @@ $section->addLink( 'https://github.com/PHPOffice/PHPWord', 'PHPWord on GitHub', - array('color' => '0000FF', 'underline' => \PhpOffice\PhpWord\Style\Font::UNDERLINE_SINGLE) + array('color' => new Hex('0000FF'), 'underline' => Font::UNDERLINE_SINGLE) ); $section->addTextBreak(2); $section->addLink('http://www.bing.com', null, $linkFontStyleName); diff --git a/samples/Sample_16_Object.php b/samples/Sample_16_Object.php index c4db7f6106..6347b20317 100644 --- a/samples/Sample_16_Object.php +++ b/samples/Sample_16_Object.php @@ -1,4 +1,5 @@ getSettings()->setUpdateFields(true); // New section $section = $phpWord->addSection(); // Define styles -$fontStyle12 = array('spaceAfter' => 60, 'size' => 12); -$fontStyle10 = array('size' => 10); -$phpWord->addTitleStyle(null, array('size' => 22, 'bold' => true)); -$phpWord->addTitleStyle(1, array('size' => 20, 'color' => '333333', 'bold' => true)); -$phpWord->addTitleStyle(2, array('size' => 16, 'color' => '666666')); -$phpWord->addTitleStyle(3, array('size' => 14, 'italic' => true)); -$phpWord->addTitleStyle(4, array('size' => 12)); +$fontStyle12 = array('spaceAfter' => Absolute::from('twip', 60), 'size' => Absolute::from('pt', 12)); +$fontStyle10 = array('size' => Absolute::from('pt', 10)); +$phpWord->addTitleStyle(null, array('size' => Absolute::from('pt', 22), 'bold' => true)); +$phpWord->addTitleStyle(1, array('size' => Absolute::from('pt', 20), 'color' => new Hex('333333'), 'bold' => true)); +$phpWord->addTitleStyle(2, array('size' => Absolute::from('pt', 16), 'color' => new Hex('666666'))); +$phpWord->addTitleStyle(3, array('size' => Absolute::from('pt', 14), 'italic' => true)); +$phpWord->addTitleStyle(4, array('size' => Absolute::from('pt', 12))); // Add text elements $section->addTitle('Table of contents 1', 0); diff --git a/samples/Sample_18_Watermark.php b/samples/Sample_18_Watermark.php index dbab2b7f26..fd1e6ad87b 100644 --- a/samples/Sample_18_Watermark.php +++ b/samples/Sample_18_Watermark.php @@ -1,14 +1,18 @@ addSection(); $header = $section->addHeader(); -$header->addWatermark('resources/_earth.jpg', array('marginTop' => 200, 'marginLeft' => 55)); +$header->addWatermark('resources/_earth.jpg', array('marginTop' => Absolute::from('twip', 200), 'marginLeft' => Absolute::from('twip', 55))); $section->addText('The header reference to the current section includes a watermark image.'); // Save file diff --git a/samples/Sample_19_TextBreak.php b/samples/Sample_19_TextBreak.php index 17e2b1cbd7..b038788293 100644 --- a/samples/Sample_19_TextBreak.php +++ b/samples/Sample_19_TextBreak.php @@ -1,20 +1,24 @@ 24); +$fontStyle24 = array('size' => Absolute::from('pt', 24)); -$paragraphStyle24 = array('spacing' => 240, 'size' => 24); +$paragraphStyle24 = array('spacing' => Absolute::from('twip', 240), 'size' => Absolute::from('pt', 24)); $fontStyleName = 'fontStyle'; -$phpWord->addFontStyle($fontStyleName, array('size' => 9)); +$phpWord->addFontStyle($fontStyleName, array('size' => Absolute::from('pt', 9))); $paragraphStyleName = 'paragraphStyle'; -$phpWord->addParagraphStyle($paragraphStyleName, array('spacing' => 480)); +$phpWord->addParagraphStyle($paragraphStyleName, array('spacing' => Absolute::from('twip', 480))); // New section $section = $phpWord->addSection(); diff --git a/samples/Sample_20_BGColor.php b/samples/Sample_20_BGColor.php index 628ae8902e..3ab501cfe9 100644 --- a/samples/Sample_20_BGColor.php +++ b/samples/Sample_20_BGColor.php @@ -1,19 +1,24 @@ addSection(); $section->addText( 'This is some text highlighted using fgColor (limited to 15 colors)', - array('fgColor' => \PhpOffice\PhpWord\Style\Font::FGCOLOR_YELLOW) + array('fgColor' => new HighlightColor('yellow')) ); -$section->addText('This one uses bgColor and is using hex value (0xfbbb10)', array('bgColor' => 'fbbb10')); -$section->addText('Compatible with font colors', array('color' => '0000ff', 'bgColor' => 'fbbb10')); +$section->addText('This one uses bgColor and is using hex value (0xfbbb10)', array('bgColor' => new Hex('fbbb10'))); +$section->addText('Compatible with font colors', array('color' => new Hex('0000ff'), 'bgColor' => new Hex('fbbb10'))); // Save file echo write($phpWord, basename(__FILE__, '.php'), $writers); diff --git a/samples/Sample_21_TableRowRules.php b/samples/Sample_21_TableRowRules.php index ca33de3bcd..122555ca22 100644 --- a/samples/Sample_21_TableRowRules.php +++ b/samples/Sample_21_TableRowRules.php @@ -1,9 +1,14 @@ addSection(); @@ -15,29 +20,29 @@ . 'the textbreak is still there:' ); -$table1 = $section->addTable(array('cellMargin' => 0, 'cellMarginRight' => 0, 'cellMarginBottom' => 0, 'cellMarginLeft' => 0)); -$table1->addRow(3750); -$cell1 = $table1->addCell(null, array('valign' => 'top', 'borderSize' => 30, 'borderColor' => 'ff0000')); -$cell1->addImage('./resources/_earth.jpg', array('width' => 250, 'height' => 250, 'alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER)); +$table1 = $section->addTable(array('cellMargin' => Asolute::from('twip', 0), 'cellMarginRight' => Asolute::from('twip', 0), 'cellMarginBottom' => Asolute::from('twip', 0), 'cellMarginLeft' => Asolute::from('twip', 0))); +$table1->addRow(Absolute::from('twip', 3750)); +$cell1 = $table1->addCell(null, array('valign' => 'top', 'borderSize' => Asolute::from('twip', 30), 'borderColor' => new Hex('ff0000'))); +$cell1->addImage('./resources/_earth.jpg', array('width' => Asolute::from('pt', 250), 'height' => Asolute::from('pt', 250), 'alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER)); $section->addTextBreak(); $section->addText("But if we set the rowStyle 'exactHeight' to true, the real row height is used, removing the textbreak:"); $table2 = $section->addTable( array( - 'cellMargin' => 0, - 'cellMarginRight' => 0, - 'cellMarginBottom' => 0, - 'cellMarginLeft' => 0, + 'cellMargin' => Asolute::from('twip', 0), + 'cellMarginRight' => Asolute::from('twip', 0), + 'cellMarginBottom' => Asolute::from('twip', 0), + 'cellMarginLeft' => Asolute::from('twip', 0), ) ); -$table2->addRow(3750, array('exactHeight' => true)); -$cell2 = $table2->addCell(null, array('valign' => 'top', 'borderSize' => 30, 'borderColor' => '00ff00')); -$cell2->addImage('./resources/_earth.jpg', array('width' => 250, 'height' => 250, 'alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER)); +$table2->addRow(Absolute::from('twip', 3750), array('exactHeight' => true)); +$cell2 = $table2->addCell(null, array('valign' => 'top', 'borderSize' => Asolute::from('twip', 30), 'borderColor' => new Hex('00ff00'))); +$cell2->addImage('./resources/_earth.jpg', array('width' => Asolute::from('pt', 250), 'height' => Asolute::from('pt', 250), 'alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER)); $section->addTextBreak(); $section->addText('In this example, image is 250px height. Rows are calculated in twips, and 1px = 15twips.'); -$section->addText('So: $' . "table2->addRow(3750, array('exactHeight'=>true));"); +$section->addText('So: $' . "table2->addRow(Absolute::from('twip', 3750), array('exactHeight'=>true));"); // Save file echo write($phpWord, basename(__FILE__, '.php'), $writers); diff --git a/samples/Sample_22_CheckBox.php b/samples/Sample_22_CheckBox.php index 5a2ac3e2cc..a7fb37d8fa 100644 --- a/samples/Sample_22_CheckBox.php +++ b/samples/Sample_22_CheckBox.php @@ -1,4 +1,5 @@ addSection(); @@ -12,10 +17,10 @@ $textbox = $section->addTextBox( array( 'alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER, - 'width' => 400, - 'height' => 150, - 'borderSize' => 1, - 'borderColor' => '#FF0000', + 'width' => Absolute::from('twip', 400), + 'height' => Absolute::from('twip', 150), + 'borderSize' => Absolute::from('twip', 1), + 'borderColor' => new Hex('FF0000'), ) ); $textbox->addText('Text box content in section.'); @@ -25,20 +30,20 @@ // Inside table $section->addTextBreak(2); -$cell = $section->addTable()->addRow()->addCell(300); -$textbox = $cell->addTextBox(array('borderSize' => 1, 'borderColor' => '#0000FF', 'innerMargin' => 100)); +$cell = $section->addTable()->addRow()->addCell(Absolute::from('twip', 300)); +$textbox = $cell->addTextBox(array('borderSize' => Absolute::from('twip', 1), 'borderColor' => new Hex('0000FF'), 'innerMargin' => Absolute::from('twip', 100))); $textbox->addText('Textbox inside table'); // Inside header with textrun $header = $section->addHeader(); -$textbox = $header->addTextBox(array('width' => 600, 'borderSize' => 1, 'borderColor' => '#00FF00')); +$textbox = $header->addTextBox(array('width' => Absolute::from('twip', 600), 'borderSize' => Absolute::from('twip', 1), 'borderColor' => new Hex('00FF00'))); $textrun = $textbox->addTextRun(); $textrun->addText('TextBox in header. TextBox can contain a TextRun '); $textrun->addText('with bold text', array('bold' => true)); $textrun->addText(', '); $textrun->addLink('https://github.com/PHPOffice/PHPWord', 'PHPWord on GitHub'); $textrun->addText(', and image '); -$textrun->addImage('resources/_earth.jpg', array('width' => 18, 'height' => 18)); +$textrun->addImage('resources/_earth.jpg', array('width' => Absolute::from('pt', 18), 'height' => Absolute::from('pt', 18))); $textrun->addText('.'); // Save file diff --git a/samples/Sample_26_Html.php b/samples/Sample_26_Html.php index 6bd926fe5b..e828565904 100644 --- a/samples/Sample_26_Html.php +++ b/samples/Sample_26_Html.php @@ -1,4 +1,5 @@ 14)); +PhpOffice\PhpWord\Style::addTitleStyle(1, array('size' => Absolute::from('pt', 14))); // New section $section = $phpWord->addSection(); @@ -45,7 +48,7 @@ $section->addText('The actual index:'); $section->addField('INDEX', array(), array('\\e " "'), 'right click to update the index'); -$textrun = $section->addTextRun(array('alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER)); +$textrun = $section->addTextRun(array('alignment' => Jc::CENTER)); $textrun->addText('This is the date of lunar calendar '); $textrun->addField('DATE', array('dateformat' => 'd-M-yyyy H:mm:ss'), array('PreserveFormat', 'LunarCalendar')); $textrun->addText(' written in a textrun.'); diff --git a/samples/Sample_28_ReadRTF.php b/samples/Sample_28_ReadRTF.php index 76ac3d48bb..cab61a7346 100644 --- a/samples/Sample_28_ReadRTF.php +++ b/samples/Sample_28_ReadRTF.php @@ -1,4 +1,5 @@ addSection(); @@ -13,16 +18,16 @@ $section->addText('Horizontal Line (Inline style):'); $section->addLine( array( - 'width' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(4), - 'height' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(0), + 'width' => Absolute::from('cm', 4), + 'height' => Absolute::from('cm', 0), 'positioning' => 'absolute', ) ); $section->addText('Vertical Line (Inline style):'); $section->addLine( array( - 'width' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(0), - 'height' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(1), + 'width' => Absolute::from('cm', 0), + 'height' => Absolute::from('cm', 1), 'positioning' => 'absolute', ) ); @@ -32,23 +37,23 @@ $section->addText('Positioned Line (red):'); $section->addLine( array( - 'width' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(4), - 'height' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(1), + 'width' => Absolute::from('cm', 4), + 'height' => Absolute::from('cm', 1), 'positioning' => 'absolute', 'posHorizontalRel' => 'page', 'posVerticalRel' => 'page', - 'marginLeft' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(10), - 'marginTop' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(8), + 'marginLeft' => Absolute::from('cm', 10), + 'marginTop' => Absolute::from('cm', 8), 'wrappingStyle' => \PhpOffice\PhpWord\Style\Image::WRAPPING_STYLE_SQUARE, - 'color' => 'red', + 'color' => new HighlightColor('red'), ) ); $section->addText('Horizontal Formatted Line'); $section->addLine( array( - 'width' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(15), - 'height' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(0), + 'width' => Absolute::from('cm', 15), + 'height' => Absolute::from('cm', 0), 'positioning' => 'absolute', 'beginArrow' => \PhpOffice\PhpWord\Style\Line::ARROW_STYLE_BLOCK, 'endArrow' => \PhpOffice\PhpWord\Style\Line::ARROW_STYLE_OVAL, diff --git a/samples/Sample_30_ReadHTML.php b/samples/Sample_30_ReadHTML.php index 029f8c8cfd..f610149f89 100644 --- a/samples/Sample_30_ReadHTML.php +++ b/samples/Sample_30_ReadHTML.php @@ -1,4 +1,5 @@ addSection(); // Define styles -$phpWord->addTitleStyle(1, array('size' => 14, 'bold' => true)); +$phpWord->addTitleStyle(1, array('size' => Absolute::from('pt', 14), 'bold' => true)); // Arc $section->addTitle('Arc', 1); @@ -17,8 +22,8 @@ 'arc', array( 'points' => '-90 20', - 'frame' => array('width' => 120, 'height' => 120), - 'outline' => array('color' => '#333333', 'weight' => 2, 'startArrow' => 'oval', 'endArrow' => 'open'), + 'frame' => array('width' => Absolute::from('pt', 120), 'height' => Absolute::from('pt', 120)), + 'outline' => array('color' => new Hex('333333'), 'weight' => 2, 'startArrow' => 'oval', 'endArrow' => 'open'), ) ); @@ -30,7 +35,7 @@ 'points' => '1,100 200,1 1,50 200,50', 'connector' => 'elbow', 'outline' => array( - 'color' => '#66cc00', + 'color' => new Hex('66cc00'), 'weight' => 2, 'dash' => 'dash', 'startArrow' => 'diamond', @@ -46,7 +51,7 @@ array( 'points' => '1,1 150,30', 'outline' => array( - 'color' => '#cc00ff', + 'color' => new Hex('cc00ff'), 'line' => 'thickThin', 'weight' => 3, 'startArrow' => 'oval', @@ -61,7 +66,7 @@ 'polyline', array( 'points' => '1,30 20,10 55,20 75,10 100,40 115,50, 120,15 200,50', - 'outline' => array('color' => '#cc6666', 'weight' => 2, 'startArrow' => 'none', 'endArrow' => 'classic'), + 'outline' => array('color' => new Hex('cc6666'), 'weight' => 2, 'startArrow' => 'none', 'endArrow' => 'classic'), ) ); @@ -71,9 +76,9 @@ 'rect', array( 'roundness' => 0.2, - 'frame' => array('width' => 100, 'height' => 100, 'left' => 1, 'top' => 1), - 'fill' => array('color' => '#FFCC33'), - 'outline' => array('color' => '#990000', 'weight' => 1), + 'frame' => array('width' => Absolute::from('pt', 100), 'height' => Absolute::from('pt', 100), 'left' => Absolute::from('pt', 1), 'top' => Absolute::from('pt', 1)), + 'fill' => array('color' => new Hex('FFCC33')), + 'outline' => array('color' => new Hex('990000'), 'weight' => 1), 'shadow' => array(), ) ); @@ -83,9 +88,9 @@ $section->addShape( 'oval', array( - 'frame' => array('width' => 100, 'height' => 70, 'left' => 1, 'top' => 1), - 'fill' => array('color' => '#33CC99'), - 'outline' => array('color' => '#333333', 'weight' => 2), + 'frame' => array('width' => Absolute::from('pt', 100), 'height' => Absolute::from('pt', 70), 'left' => Absolute::from('pt', 1), 'top' => Absolute::from('pt', 1)), + 'fill' => array('color' => new Hex('33CC99')), + 'outline' => array('color' => new Hex('333333'), 'weight' => 2), 'extrusion' => array(), ) ); diff --git a/samples/Sample_32_Chart.php b/samples/Sample_32_Chart.php index c24a6f8e37..5b58d007f1 100644 --- a/samples/Sample_32_Chart.php +++ b/samples/Sample_32_Chart.php @@ -1,15 +1,17 @@ addTitleStyle(1, array('size' => 14, 'bold' => true), array('keepNext' => true, 'spaceBefore' => 240)); -$phpWord->addTitleStyle(2, array('size' => 14, 'bold' => true), array('keepNext' => true, 'spaceBefore' => 240)); +$phpWord->addTitleStyle(1, array('size' => Absolute::from('pt', 14), 'bold' => true), array('keepNext' => true, 'spaceBefore' => Absolute::from('twip', 240))); +$phpWord->addTitleStyle(2, array('size' => Absolute::from('pt', 14), 'bold' => true), array('keepNext' => true, 'spaceBefore' => Absolute::from('twip', 240))); // 2D charts $section = $phpWord->addSection(); @@ -29,7 +31,7 @@ foreach ($chartTypes as $chartType) { $section->addTitle(ucfirst($chartType), 2); $chart = $section->addChart($chartType, $categories, $series1); - $chart->getStyle()->setWidth(Converter::inchToEmu(2.5))->setHeight(Converter::inchToEmu(2)); + $chart->getStyle()->setWidth(Absolute::from('in', 2.5))->setHeight(Absolute::from('in', 2)); $chart->getStyle()->setShowGridX($showGridLines); $chart->getStyle()->setShowGridY($showGridLines); $chart->getStyle()->setShowAxisLabels($showAxisLabels); @@ -50,8 +52,8 @@ $chartTypes = array('pie', 'bar', 'column', 'line', 'area'); $multiSeries = array('bar', 'column', 'line', 'area'); $style = array( - 'width' => Converter::cmToEmu(5), - 'height' => Converter::cmToEmu(4), + 'width' => Absolute::from('cm', 5), + 'height' => Absolute::from('cm', 4), '3d' => true, 'showAxisLabels' => $showAxisLabels, 'showGridX' => $showGridLines, diff --git a/samples/Sample_33_FormField.php b/samples/Sample_33_FormField.php index a855d42a02..b50697b43a 100644 --- a/samples/Sample_33_FormField.php +++ b/samples/Sample_33_FormField.php @@ -1,4 +1,5 @@ addSection(); @@ -11,29 +18,29 @@ $textrun = $section->addTextRun(); $textrun->addText('This is a Left to Right paragraph.'); -$textrun = $section->addTextRun(array('alignment' => \PhpOffice\PhpWord\SimpleType\Jc::END)); +$textrun = $section->addTextRun(array('alignment' => Jc::END)); $textrun->addText('سلام این یک پاراگراف راست به چپ است', array('rtl' => true)); $section->addText('Table visually presented as RTL'); -$style = array('rtl' => true, 'size' => 12); -$tableStyle = array('borderSize' => 6, 'borderColor' => '000000', 'width' => 5000, 'unit' => \PhpOffice\PhpWord\SimpleType\TblWidth::PERCENT, 'bidiVisual' => true); +$style = array('rtl' => true, 'size' => Absolute::from('pt', 12)); +$tableStyle = array('borderSize' => Absolute::from('twip', 6), 'borderColor' => new Hex('000000'), 'width' => Absolute::from('twip', 5000), 'bidiVisual' => true); $table = $section->addTable($tableStyle); -$cellHCentered = array('alignment' => \PhpOffice\PhpWord\SimpleType\Jc::CENTER); -$cellHEnd = array('alignment' => \PhpOffice\PhpWord\SimpleType\Jc::END); -$cellVCentered = array('valign' => \PhpOffice\PhpWord\Style\Cell::VALIGN_CENTER); +$cellHCentered = array('alignment' => Jc::CENTER); +$cellHEnd = array('alignment' => Jc::END); +$cellVCentered = array('valign' => Cell::VALIGN_CENTER); //Vidually bidirectinal table $table->addRow(); -$cell = $table->addCell(500, $cellVCentered); +$cell = $table->addCell(Absolute::from('twip', 500), $cellVCentered); $textrun = $cell->addTextRun($cellHCentered); $textrun->addText('ردیف', $style); -$cell = $table->addCell(11000); +$cell = $table->addCell(Absolute::from('twip', 11000)); $textrun = $cell->addTextRun($cellHEnd); $textrun->addText('سوالات', $style); -$cell = $table->addCell(500, $cellVCentered); +$cell = $table->addCell(Absolute::from('twip', 500), $cellVCentered); $textrun = $cell->addTextRun($cellHCentered); $textrun->addText('بارم', $style); diff --git a/samples/Sample_37_Comments.php b/samples/Sample_37_Comments.php index 268739bc67..c6573bb951 100644 --- a/samples/Sample_37_Comments.php +++ b/samples/Sample_37_Comments.php @@ -1,4 +1,5 @@ addTextRun(); $imageComment->addText('Hey, Mars does look '); -$imageComment->addText('red', array('color' => 'FF0000')); +$imageComment->addText('red', array('color' => new Hex('FF0000'))); $phpWord->addComment($commentOnImage); $image = $section->addImage('resources/_mars.jpg'); $image->setCommentRangeStart($commentOnImage); diff --git a/samples/Sample_38_Protection.php b/samples/Sample_38_Protection.php index ee2b460b2f..071f3c9b4f 100644 --- a/samples/Sample_38_Protection.php +++ b/samples/Sample_38_Protection.php @@ -1,4 +1,5 @@ addText('This title has been set ', array('bold' => true, 'italic' => true, 'color' => 'blue')); -$title->addText('dynamically', array('bold' => true, 'italic' => true, 'color' => 'red', 'underline' => 'single')); +$title->addText('This title has been set ', array('bold' => true, 'italic' => true, 'color' => new HighlightColor('blue'))); +$title->addText('dynamically', array('bold' => true, 'italic' => true, 'color' => new HighlightColor('red'), 'underline' => 'single')); $templateProcessor->setComplexBlock('title', $title); $inline = new TextRun(); -$inline->addText('by a red italic text', array('italic' => true, 'color' => 'red')); +$inline->addText('by a red italic text', array('italic' => true, 'color' => new HighlightColor('red'))); $templateProcessor->setComplexValue('inline', $inline); -$table = new Table(array('borderSize' => 12, 'borderColor' => 'green', 'width' => 6000, 'unit' => TblWidth::TWIP)); +$table = new Table(array('borderSize' => Absolute::from('twip', 12), 'borderColor' => new HighlightColor('green'), 'width' => Absolute::from('twip', 6000))); $table->addRow(); -$table->addCell(150)->addText('Cell A1'); -$table->addCell(150)->addText('Cell A2'); -$table->addCell(150)->addText('Cell A3'); +$table->addCell(Absolute::from('twip', 150))->addText('Cell A1'); +$table->addCell(Absolute::from('twip', 150))->addText('Cell A2'); +$table->addCell(Absolute::from('twip', 150))->addText('Cell A3'); $table->addRow(); -$table->addCell(150)->addText('Cell B1'); -$table->addCell(150)->addText('Cell B2'); -$table->addCell(150)->addText('Cell B3'); +$table->addCell(Absolute::from('twip', 150))->addText('Cell B1'); +$table->addCell(Absolute::from('twip', 150))->addText('Cell B2'); +$table->addCell(Absolute::from('twip', 150))->addText('Cell B3'); $templateProcessor->setComplexBlock('table', $table); $field = new Field('DATE', array('dateformat' => 'dddd d MMMM yyyy H:mm:ss'), array('PreserveFormat')); diff --git a/samples/Sample_Footer.php b/samples/Sample_Footer.php index 2d89bfd253..e399344d84 100644 --- a/samples/Sample_Footer.php +++ b/samples/Sample_Footer.php @@ -1,4 +1,4 @@ -value = $value; - TblWidthSimpleType::validate($type); - $this->type = $type; - } - - /** - * @return string - */ - public function getType() - { - return $this->type; - } - - /** - * @return int - */ - public function getValue() - { - return $this->value; - } -} diff --git a/src/PhpWord/ComplexType/TrackChangesView.php b/src/PhpWord/ComplexType/TrackChangesView.php index 92ea05eab3..4f410bdfdc 100644 --- a/src/PhpWord/ComplexType/TrackChangesView.php +++ b/src/PhpWord/ComplexType/TrackChangesView.php @@ -1,4 +1,5 @@ elementId = substr(md5(rand()), 0, 6); + $this->elementId = substr(md5((string) rand()), 0, 6); } /** @@ -299,8 +300,6 @@ public function getCommentRangeStart() /** * Set comment start - * - * @param Comment $value */ public function setCommentRangeStart(Comment $value) { @@ -323,8 +322,6 @@ public function getCommentRangeEnd() /** * Set comment end - * - * @param Comment $value */ public function setCommentRangeEnd(Comment $value) { @@ -431,7 +428,6 @@ public function isInSection() * @param mixed $styleObject Style object * @param mixed $styleValue Style value * @param bool $returnObject Always return object - * @return mixed */ protected function setNewStyle($styleObject, $styleValue = null, $returnObject = false) { @@ -439,7 +435,8 @@ protected function setNewStyle($styleObject, $styleValue = null, $returnObject = $styleObject->setStyleByArray($styleValue); $style = $styleObject; } else { - $style = $returnObject ? $styleObject : $styleValue; + $class = get_class($styleObject); + $style = $returnObject && !($styleValue instanceof $class) ? $styleObject : $styleValue; } return $style; @@ -447,8 +444,6 @@ protected function setNewStyle($styleObject, $styleValue = null, $returnObject = /** * Sets the trackChange information - * - * @param TrackChange $trackChange */ public function setTrackChange(TrackChange $trackChange) { diff --git a/src/PhpWord/Element/Bookmark.php b/src/PhpWord/Element/Bookmark.php index 16b020d741..32a904f0b6 100644 --- a/src/PhpWord/Element/Bookmark.php +++ b/src/PhpWord/Element/Bookmark.php @@ -1,4 +1,5 @@ width = $width; + $this->width = $width ?? new Absolute(null); $this->style = $this->setNewStyle(new CellStyle(), $style, true); } @@ -67,10 +70,8 @@ public function getStyle() /** * Get cell width - * - * @return int */ - public function getWidth() + public function getWidth(): Length { return $this->width; } diff --git a/src/PhpWord/Element/Chart.php b/src/PhpWord/Element/Chart.php index 92152c877d..a4ad8d0a29 100644 --- a/src/PhpWord/Element/Chart.php +++ b/src/PhpWord/Element/Chart.php @@ -1,4 +1,5 @@ source = $source; $this->style = $this->setNewStyle(new ImageStyle(), $style, true); @@ -161,10 +163,8 @@ public function getStyle() /** * Get image source - * - * @return string */ - public function getSource() + public function getSource(): string { return $this->source; } @@ -419,6 +419,15 @@ private function checkImage() } list($actualWidth, $actualHeight, $imageType) = $imageData; + // Convert pixel dimensions to Absolute + // FIXME: This is likely not the best approach. + // It looks like DPI can vary in a word doc, + // but there's no obvious DPI information + // that I can find for these images right now. + $dpi = new Dpi(); + $actualWidth = Absolute::fromPixels($dpi, $actualWidth); + $actualHeight = Absolute::fromPixels($dpi, $actualHeight); + // Check image type support $supportedTypes = array(IMAGETYPE_JPEG, IMAGETYPE_GIF, IMAGETYPE_PNG); if ($this->sourceType != self::SOURCE_GD && $this->sourceType != self::SOURCE_STRING) { @@ -454,7 +463,7 @@ private function setSourceType() } else { $this->sourceType = self::SOURCE_GD; } - } elseif (@file_exists($this->source)) { + } elseif (strpos($this->source, "\0") === false && file_exists($this->source)) { $this->memoryImage = false; $this->sourceType = self::SOURCE_LOCAL; } else { @@ -556,23 +565,20 @@ private function setFunctions() /** * Set proportional width/height if one dimension not available. - * - * @param int $actualWidth - * @param int $actualHeight */ - private function setProportionalSize($actualWidth, $actualHeight) + private function setProportionalSize(Absolute $actualWidth, Absolute $actualHeight) { - $styleWidth = $this->style->getWidth(); - $styleHeight = $this->style->getHeight(); - if (!($styleWidth && $styleHeight)) { - if ($styleWidth == null && $styleHeight == null) { - $this->style->setWidth($actualWidth); - $this->style->setHeight($actualHeight); - } elseif ($styleWidth) { - $this->style->setHeight($actualHeight * ($styleWidth / $actualWidth)); - } else { - $this->style->setWidth($actualWidth * ($styleHeight / $actualHeight)); - } + $styleWidth = $this->style->getWidth()->toInt('twip'); + $styleHeight = $this->style->getHeight()->toInt('twip'); + if ($styleWidth === null && $styleHeight === null) { + $this->style->setWidth($actualWidth); + $this->style->setHeight($actualHeight); + } elseif ($styleHeight === null) { + $newHeight = Absolute::from('twip', $actualHeight->toInt('twip') * ($styleWidth / $actualWidth->toInt('twip'))); + $this->style->setHeight($newHeight); + } elseif ($styleWidth === null) { + $newWidth = Absolute::from('twip', $actualWidth->toInt('twip') * ($styleHeight / $actualHeight->toInt('twip'))); + $this->style->setWidth($newWidth); } } diff --git a/src/PhpWord/Element/Line.php b/src/PhpWord/Element/Line.php index 7e40b9402d..fe891f67df 100644 --- a/src/PhpWord/Element/Line.php +++ b/src/PhpWord/Element/Line.php @@ -1,4 +1,5 @@ paragraphStyle = $this->setNewStyle(new Paragraph(), $paragraphStyle); $this->text = CommonText::toUTF8($text); - $matches = preg_split('/({.*?})/', $this->text, null, PREG_SPLIT_DELIM_CAPTURE | PREG_SPLIT_NO_EMPTY); + $matches = preg_split('/({.*?})/', $this->text ?? '', -1, PREG_SPLIT_DELIM_CAPTURE | PREG_SPLIT_NO_EMPTY); if (isset($matches[0])) { $this->text = $matches; } diff --git a/src/PhpWord/Element/Row.php b/src/PhpWord/Element/Row.php index da4dfe5d73..5e1b941a19 100644 --- a/src/PhpWord/Element/Row.php +++ b/src/PhpWord/Element/Row.php @@ -1,4 +1,5 @@ height = $height; + $this->height = $height ?? new Absolute(null); $this->style = $this->setNewStyle(new RowStyle(), $style, true); } /** * Add a cell * - * @param int $width - * @param mixed $style + * @param Length $width + * @param null|mixed $style * @return \PhpOffice\PhpWord\Element\Cell */ - public function addCell($width = null, $style = null) + public function addCell(Length $width = null, $style = null) { $cell = new Cell($width, $style); $cell->setParentContainer($this); @@ -97,10 +100,8 @@ public function getStyle() /** * Get row height - * - * @return int */ - public function getHeight() + public function getHeight(): Absolute { return $this->height; } diff --git a/src/PhpWord/Element/SDT.php b/src/PhpWord/Element/SDT.php index 5548f768c7..4e92074d25 100644 --- a/src/PhpWord/Element/SDT.php +++ b/src/PhpWord/Element/SDT.php @@ -1,4 +1,5 @@ setParentContainer($this); @@ -74,11 +75,11 @@ public function addRow($height = null, $style = null) /** * Add a cell * - * @param int $width - * @param mixed $style + * @param Length $width + * @param null|mixed $style * @return \PhpOffice\PhpWord\Element\Cell */ - public function addCell($width = null, $style = null) + public function addCell(Length $width = null, $style = null) { $index = count($this->rows) - 1; $row = $this->rows[$index]; @@ -109,22 +110,26 @@ public function getStyle() /** * Get table width - * - * @return int */ - public function getWidth() + public function getWidth(): Length { + if ($this->width === null) { + $this->width = new Absolute(null); + } + return $this->width; } /** * Set table width. * - * @param int $width + * @param Table $width */ - public function setWidth($width) + public function setWidth(Length $width): self { $this->width = $width; + + return $this; } /** @@ -152,9 +157,9 @@ public function countColumns() /** * The first declared cell width for each column * - * @return int[] + * @return Absolute[] */ - public function findFirstDefinedCellWidths() + public function findFirstDefinedCellWidths(): array { $cellWidths = array(); diff --git a/src/PhpWord/Element/Text.php b/src/PhpWord/Element/Text.php index f4d7f08185..c92ad2ab81 100644 --- a/src/PhpWord/Element/Text.php +++ b/src/PhpWord/Element/Text.php @@ -1,4 +1,5 @@ isCustomPropertySet($propertyName)) { return $this->customProperties[$propertyName]['value']; @@ -440,7 +438,6 @@ public function getCustomPropertyType($propertyName) * Set a Custom Property * * @param string $propertyName - * @param mixed $propertyValue * @param string $propertyType * 'i': Integer * 'f': Floating Point @@ -487,7 +484,6 @@ public function setCustomProperty($propertyName, $propertyValue = '', $propertyT * * @param string $propertyValue * @param string $propertyType - * @return mixed */ public static function convertProperty($propertyValue, $propertyType) { @@ -539,10 +535,6 @@ public static function convertPropertyType($propertyType) /** * Set default for null and empty value - * - * @param mixed $value - * @param mixed $default - * @return mixed */ private function setValue($value, $default) { diff --git a/src/PhpWord/Metadata/Protection.php b/src/PhpWord/Metadata/Protection.php index 15aa3ff189..c9523de852 100644 --- a/src/PhpWord/Metadata/Protection.php +++ b/src/PhpWord/Metadata/Protection.php @@ -1,4 +1,5 @@ doNotTrackFormatting = $doNotTrackFormatting === null ? true : $doNotTrackFormatting; } - /** - * @return mixed - */ public function getZoom() { return $this->zoom; } - /** - * @param mixed $zoom - */ public function setZoom($zoom) { if (is_numeric($zoom)) { @@ -446,20 +442,20 @@ public function setConsecutiveHyphenLimit($consecutiveHyphenLimit) $this->consecutiveHyphenLimit = (int) $consecutiveHyphenLimit; } - /** - * @return float|null - */ - public function getHyphenationZone() + public function getHyphenationZone(): Absolute { + if ($this->hyphenationZone === null) { + $this->hyphenationZone = new Absolute(null); + } + return $this->hyphenationZone; } - /** - * @param float $hyphenationZone Measurement unit is twip - */ - public function setHyphenationZone($hyphenationZone) + public function setHyphenationZone(Absolute $hyphenationZone): self { $this->hyphenationZone = $hyphenationZone; + + return $this; } /** diff --git a/src/PhpWord/PhpWord.php b/src/PhpWord/PhpWord.php index a78df2c4e5..d1f91c094a 100644 --- a/src/PhpWord/PhpWord.php +++ b/src/PhpWord/PhpWord.php @@ -1,4 +1,5 @@ collections[$key]; - return $collectionObject->addItem(isset($args[0]) ? $args[0] : null); + return $collectionObject->addItem($args[0] ?? null); } // Run add style method @@ -288,20 +286,16 @@ public function setDefaultFontName($fontName) /** * Get default font size - * - * @return int */ - public function getDefaultFontSize() + public function getDefaultFontSize(): Absolute { return Settings::getDefaultFontSize(); } /** * Set default font size. - * - * @param int $fontSize */ - public function setDefaultFontSize($fontSize) + public function setDefaultFontSize(Absolute $fontSize) { Settings::setDefaultFontSize($fontSize); } diff --git a/src/PhpWord/Reader/AbstractReader.php b/src/PhpWord/Reader/AbstractReader.php index 7db285f758..5c5003566e 100644 --- a/src/PhpWord/Reader/AbstractReader.php +++ b/src/PhpWord/Reader/AbstractReader.php @@ -1,4 +1,5 @@ styleFont['color'] = '000000'; + $oStylePrl->styleFont['color'] = new Hex('000000'); break; case 0x02: - $oStylePrl->styleFont['color'] = '0000FF'; + $oStylePrl->styleFont['color'] = new Hex('0000FF'); break; case 0x03: - $oStylePrl->styleFont['color'] = '00FFFF'; + $oStylePrl->styleFont['color'] = new Hex('00FFFF'); break; case 0x04: - $oStylePrl->styleFont['color'] = '00FF00'; + $oStylePrl->styleFont['color'] = new Hex('00FF00'); break; case 0x05: - $oStylePrl->styleFont['color'] = 'FF00FF'; + $oStylePrl->styleFont['color'] = new Hex('FF00FF'); break; case 0x06: - $oStylePrl->styleFont['color'] = 'FF0000'; + $oStylePrl->styleFont['color'] = new Hex('FF0000'); break; case 0x07: - $oStylePrl->styleFont['color'] = 'FFFF00'; + $oStylePrl->styleFont['color'] = new Hex('FFFF00'); break; case 0x08: - $oStylePrl->styleFont['color'] = 'FFFFFF'; + $oStylePrl->styleFont['color'] = new Hex('FFFFFF'); break; case 0x09: - $oStylePrl->styleFont['color'] = '000080'; + $oStylePrl->styleFont['color'] = new Hex('000080'); break; case 0x0A: - $oStylePrl->styleFont['color'] = '008080'; + $oStylePrl->styleFont['color'] = new Hex('008080'); break; case 0x0B: - $oStylePrl->styleFont['color'] = '008000'; + $oStylePrl->styleFont['color'] = new Hex('008000'); break; case 0x0C: - $oStylePrl->styleFont['color'] = '800080'; + $oStylePrl->styleFont['color'] = new Hex('800080'); break; case 0x0D: - $oStylePrl->styleFont['color'] = '800080'; + $oStylePrl->styleFont['color'] = new Hex('800080'); break; case 0x0E: - $oStylePrl->styleFont['color'] = '808000'; + $oStylePrl->styleFont['color'] = new Hex('808000'); break; case 0x0F: - $oStylePrl->styleFont['color'] = '808080'; + $oStylePrl->styleFont['color'] = new Hex('808080'); break; case 0x10: - $oStylePrl->styleFont['color'] = 'C0C0C0'; + $oStylePrl->styleFont['color'] = new Hex('C0C0C0'); } break; // sprmCHps case 0x43: - $oStylePrl->styleFont['size'] = dechex($operand / 2); + if (is_numeric($operand)) { + $oStylePrl->styleFont['size'] = Absolute::from('pt', $operand); + } break; // sprmCIss case 0x48: @@ -1854,14 +1860,14 @@ private function readPrl($data, $pos, $cbNum) // sprmCCv //@see : http://msdn.microsoft.com/en-us/library/dd952824%28v=office.12%29.aspx case 0x70: - $red = str_pad(dechex(self::getInt1d($this->dataWorkDocument, $pos)), 2, '0', STR_PAD_LEFT); + $red = self::getInt1d($this->dataWorkDocument, $pos); $pos += 1; - $green = str_pad(dechex(self::getInt1d($this->dataWorkDocument, $pos)), 2, '0', STR_PAD_LEFT); + $green = self::getInt1d($this->dataWorkDocument, $pos); $pos += 1; - $blue = str_pad(dechex(self::getInt1d($this->dataWorkDocument, $pos)), 2, '0', STR_PAD_LEFT); + $blue = self::getInt1d($this->dataWorkDocument, $pos); $pos += 1; $pos += 1; - $oStylePrl->styleFont['color'] = $red . $green . $blue; + $oStylePrl->styleFont['color'] = new Rgb($red, $green, $blue); $cbNum -= 4; break; default: @@ -1884,27 +1890,39 @@ private function readPrl($data, $pos, $cbNum) break; // sprmSXaPage case 0x1F: - $oStylePrl->styleSection['pageSizeW'] = $operand; + if (is_numeric($operand)) { + $oStylePrl->styleSection['pageSizeW'] = Absolute::from('pt', $operand); + } break; // sprmSYaPage case 0x20: - $oStylePrl->styleSection['pageSizeH'] = $operand; + if (is_numeric($operand)) { + $oStylePrl->styleSection['pageSizeH'] = Absolute::from('pt', $operand); + } break; // sprmSDxaLeft case 0x21: - $oStylePrl->styleSection['marginLeft'] = $operand; + if (is_numeric($operand)) { + $oStylePrl->styleSection['marginLeft'] = Absolute::from('pt', $operand); + } break; // sprmSDxaRight case 0x22: - $oStylePrl->styleSection['marginRight'] = $operand; + if (is_numeric($operand)) { + $oStylePrl->styleSection['marginRight'] = Absolute::from('pt', $operand); + } break; // sprmSDyaTop case 0x23: - $oStylePrl->styleSection['marginTop'] = $operand; + if (is_numeric($operand)) { + $oStylePrl->styleSection['marginTop'] = Absolute::from('pt', $operand); + } break; // sprmSDyaBottom case 0x24: - $oStylePrl->styleSection['marginBottom'] = $operand; + if (is_numeric($operand)) { + $oStylePrl->styleSection['marginBottom'] = Absolute::from('pt', $operand); + } break; // sprmSFBiDi case 0x28: @@ -2283,7 +2301,7 @@ private function generatePhpWord() if (isset($oCharacters->style->image)) { $fileImage = tempnam(sys_get_temp_dir(), 'PHPWord_MsDoc') . '.' . $oCharacters->style->image['format']; file_put_contents($fileImage, $oCharacters->style->image['data']); - $oSection->addImage($fileImage, array('width' => $oCharacters->style->image['width'], 'height' => $oCharacters->style->image['height'])); + $oSection->addImage($fileImage, array('width' => Absolute::from('pt', $oCharacters->style->image['width']), 'height' => Absolute::from('pt', $oCharacters->style->image['height']))); // print_r('>addImage<'.$fileImage.'>'.EOL); } } diff --git a/src/PhpWord/Reader/ODText.php b/src/PhpWord/Reader/ODText.php index 0b58dc50e1..b105a0a4b4 100644 --- a/src/PhpWord/Reader/ODText.php +++ b/src/PhpWord/Reader/ODText.php @@ -1,4 +1,5 @@ flags['styles'][$style][$property] = $value; } diff --git a/src/PhpWord/Reader/ReaderInterface.php b/src/PhpWord/Reader/ReaderInterface.php index 4024cdb3d6..31c49e597e 100644 --- a/src/PhpWord/Reader/ReaderInterface.php +++ b/src/PhpWord/Reader/ReaderInterface.php @@ -1,4 +1,5 @@ nodeName, array('w:ins', 'w:del', 'w:smartTag', 'w:hyperlink'))) { $nodes = $xmlReader->getElements('*', $domNode); @@ -229,14 +240,11 @@ protected function readRun(XMLReader $xmlReader, \DOMElement $domNode, $parent, /** * Parses nodes under w:r * - * @param XMLReader $xmlReader - * @param \DOMElement $node - * @param AbstractContainer $parent * @param string $docPart - * @param mixed $paragraphStyle - * @param mixed $fontStyle + * @param null|mixed $paragraphStyle + * @param null|mixed $fontStyle */ - protected function readRunChild(XMLReader $xmlReader, \DOMElement $node, AbstractContainer $parent, $docPart, $paragraphStyle = null, $fontStyle = null) + protected function readRunChild(XMLReader $xmlReader, DOMElement $node, AbstractContainer $parent, $docPart, $paragraphStyle = null, $fontStyle = null) { $runParent = $node->parentNode->parentNode; if ($node->nodeName == 'w:footnoteReference') { @@ -316,12 +324,9 @@ protected function readRunChild(XMLReader $xmlReader, \DOMElement $node, Abstrac /** * Read w:tbl. * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $domNode - * @param mixed $parent * @param string $docPart */ - protected function readTable(XMLReader $xmlReader, \DOMElement $domNode, $parent, $docPart = 'document') + protected function readTable(XMLReader $xmlReader, DOMElement $domNode, $parent, $docPart = 'document') { // Table style $tblStyle = null; @@ -334,39 +339,69 @@ protected function readTable(XMLReader $xmlReader, \DOMElement $domNode, $parent $tblNodes = $xmlReader->getElements('*', $domNode); foreach ($tblNodes as $tblNode) { if ('w:tblGrid' == $tblNode->nodeName) { // Column - // @todo Do something with table columns + // Do something with this data } elseif ('w:tr' == $tblNode->nodeName) { // Row - $rowHeight = $xmlReader->getAttribute('w:val', $tblNode, 'w:trPr/w:trHeight'); - $rowHRule = $xmlReader->getAttribute('w:hRule', $tblNode, 'w:trPr/w:trHeight'); - $rowHRule = $rowHRule == 'exact'; - $rowStyle = array( - 'tblHeader' => $xmlReader->elementExists('w:trPr/w:tblHeader', $tblNode), - 'cantSplit' => $xmlReader->elementExists('w:trPr/w:cantSplit', $tblNode), - 'exactHeight' => $rowHRule, - ); - - $row = $table->addRow($rowHeight, $rowStyle); - $rowNodes = $xmlReader->getElements('*', $tblNode); - foreach ($rowNodes as $rowNode) { - if ('w:trPr' == $rowNode->nodeName) { // Row style - // @todo Do something with row style - } elseif ('w:tc' == $rowNode->nodeName) { // Cell - $cellWidth = $xmlReader->getAttribute('w:w', $rowNode, 'w:tcPr/w:tcW'); - $cellStyle = null; - $cellStyleNode = $xmlReader->getElement('w:tcPr', $rowNode); - if (!is_null($cellStyleNode)) { - $cellStyle = $this->readCellStyle($xmlReader, $cellStyleNode); - } - - $cell = $row->addCell($cellWidth, $cellStyle); - $cellNodes = $xmlReader->getElements('*', $rowNode); - foreach ($cellNodes as $cellNode) { - if ('w:p' == $cellNode->nodeName) { // Paragraph - $this->readParagraph($xmlReader, $cellNode, $cell, $docPart); - } - } - } - } + $this->readTableRow($xmlReader, $table, $tblNode, $docPart); + } + } + } + + /** + * Reads row properties and cells + * @todo Add tests + */ + protected function readTableRow(XmlReader $xmlReader, Table $table, DOMElement $tblNode, $docPart) + { + $rowHeight = Absolute::fromMixed('twip', $xmlReader->getAttribute('w:val', $tblNode, 'w:trPr/w:trHeight')); + $rowHRule = $xmlReader->getAttribute('w:hRule', $tblNode, 'w:trPr/w:trHeight'); + $rowHRule = $rowHRule == 'exact'; + $rowStyle = array( + 'tblHeader' => $xmlReader->elementExists('w:trPr/w:tblHeader', $tblNode), + 'cantSplit' => $xmlReader->elementExists('w:trPr/w:cantSplit', $tblNode), + 'exactHeight' => $rowHRule, + ); + + $row = $table->addRow($rowHeight, $rowStyle); + $rowNodes = $xmlReader->getElements('*', $tblNode); + foreach ($rowNodes as $rowNode) { + if ('w:trPr' == $rowNode->nodeName) { // Row style + // Do something with row style + } elseif ('w:tc' == $rowNode->nodeName) { // Cell + $this->readTableCell($xmlReader, $row, $rowNode, $docPart); + } + } + } + + /** + * Reads table cell + * @todo Add tests + */ + protected function readTableCell(XmlReader $xmlReader, Row $row, DOMElement $rowNode, $docPart) + { + $type = $xmlReader->getAttribute('w:type', $rowNode, 'w:tcPr/w:tcW'); + if ($type === 'auto') { + $cellWidth = new Auto(); + } elseif ($type === 'dxa') { + $cellWidth = Absolute::fromMixed('twip', $xmlReader->getAttribute('w:w', $rowNode, 'w:tcPr/w:tcW')); + } elseif ($type === 'pct') { + $cellWidth = Percent::fromMixed($xmlReader->getAttribute('w:w', $rowNode, 'w:tcPr/w:tcW')); + } elseif ($type === 'nil') { + $cellWidth = new Absolute(null); + } else { + trigger_error(sprintf('Cell width type `%s` is not supported', $type), E_USER_WARNING); + $cellWidth = new Absolute(null); + } + $cellStyle = null; + $cellStyleNode = $xmlReader->getElement('w:tcPr', $rowNode); + if (!is_null($cellStyleNode)) { + $cellStyle = $this->readCellStyle($xmlReader, $cellStyleNode); + } + + $cell = $row->addCell($cellWidth, $cellStyle); + $cellNodes = $xmlReader->getElements('*', $rowNode); + foreach ($cellNodes as $cellNode) { + if ('w:p' == $cellNode->nodeName) { // Paragraph + $this->readParagraph($xmlReader, $cellNode, $cell, $docPart); } } } @@ -374,11 +409,9 @@ protected function readTable(XMLReader $xmlReader, \DOMElement $domNode, $parent /** * Read w:pPr. * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $domNode * @return array|null */ - protected function readParagraphStyle(XMLReader $xmlReader, \DOMElement $domNode) + protected function readParagraphStyle(XMLReader $xmlReader, DOMElement $domNode) { if (!$xmlReader->elementExists('w:pPr', $domNode)) { return null; @@ -390,10 +423,10 @@ protected function readParagraphStyle(XMLReader $xmlReader, \DOMElement $domNode 'alignment' => array(self::READ_VALUE, 'w:jc'), 'basedOn' => array(self::READ_VALUE, 'w:basedOn'), 'next' => array(self::READ_VALUE, 'w:next'), - 'indent' => array(self::READ_VALUE, 'w:ind', 'w:left'), - 'hanging' => array(self::READ_VALUE, 'w:ind', 'w:hanging'), - 'spaceAfter' => array(self::READ_VALUE, 'w:spacing', 'w:after'), - 'spaceBefore' => array(self::READ_VALUE, 'w:spacing', 'w:before'), + 'indent' => array(self::READ_TWIP, 'w:ind', 'w:left'), + 'hanging' => array(self::READ_TWIP, 'w:ind', 'w:hanging'), + 'spaceAfter' => array(self::READ_TWIP, 'w:spacing', 'w:after'), + 'spaceBefore' => array(self::READ_TWIP, 'w:spacing', 'w:before'), 'widowControl' => array(self::READ_FALSE, 'w:widowControl'), 'keepNext' => array(self::READ_TRUE, 'w:keepNext'), 'keepLines' => array(self::READ_TRUE, 'w:keepLines'), @@ -409,11 +442,9 @@ protected function readParagraphStyle(XMLReader $xmlReader, \DOMElement $domNode /** * Read w:rPr * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $domNode * @return array|null */ - protected function readFontStyle(XMLReader $xmlReader, \DOMElement $domNode) + protected function readFontStyle(XMLReader $xmlReader, DOMElement $domNode) { if (is_null($domNode)) { return null; @@ -431,8 +462,8 @@ protected function readFontStyle(XMLReader $xmlReader, \DOMElement $domNode) 'styleName' => array(self::READ_VALUE, 'w:rStyle'), 'name' => array(self::READ_VALUE, 'w:rFonts', array('w:ascii', 'w:hAnsi', 'w:eastAsia', 'w:cs')), 'hint' => array(self::READ_VALUE, 'w:rFonts', 'w:hint'), - 'size' => array(self::READ_SIZE, array('w:sz', 'w:szCs')), - 'color' => array(self::READ_VALUE, 'w:color'), + 'size' => array(self::READ_HPT, array('w:sz', 'w:szCs')), + 'color' => array(self::READ_COLOR, 'w:color'), 'underline' => array(self::READ_VALUE, 'w:u'), 'bold' => array(self::READ_TRUE, 'w:b'), 'italic' => array(self::READ_TRUE, 'w:i'), @@ -442,10 +473,10 @@ protected function readFontStyle(XMLReader $xmlReader, \DOMElement $domNode) 'allCaps' => array(self::READ_TRUE, 'w:caps'), 'superScript' => array(self::READ_EQUAL, 'w:vertAlign', 'w:val', 'superscript'), 'subScript' => array(self::READ_EQUAL, 'w:vertAlign', 'w:val', 'subscript'), - 'fgColor' => array(self::READ_VALUE, 'w:highlight'), + 'fgColor' => array(self::READ_HIGHLIGHT, 'w:highlight'), 'rtl' => array(self::READ_TRUE, 'w:rtl'), 'lang' => array(self::READ_VALUE, 'w:lang'), - 'position' => array(self::READ_VALUE, 'w:position'), + 'position' => array(self::READ_TWIP, 'w:position'), 'hidden' => array(self::READ_TRUE, 'w:vanish'), ); @@ -455,12 +486,10 @@ protected function readFontStyle(XMLReader $xmlReader, \DOMElement $domNode) /** * Read w:tblPr * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $domNode * @return string|array|null * @todo Capture w:tblStylePr w:type="firstRow" */ - protected function readTableStyle(XMLReader $xmlReader, \DOMElement $domNode) + protected function readTableStyle(XMLReader $xmlReader, DOMElement $domNode) { $style = null; $margins = array('top', 'left', 'bottom', 'right'); @@ -474,17 +503,17 @@ protected function readTableStyle(XMLReader $xmlReader, \DOMElement $domNode) $styleDefs = array(); foreach ($margins as $side) { $ucfSide = ucfirst($side); - $styleDefs["cellMargin$ucfSide"] = array(self::READ_VALUE, "w:tblCellMar/w:$side", 'w:w'); + $styleDefs["cellMargin$ucfSide"] = array(self::READ_TWIP, "w:tblCellMar/w:$side", 'w:w'); } foreach ($borders as $side) { $ucfSide = ucfirst($side); - $styleDefs["border{$ucfSide}Size"] = array(self::READ_VALUE, "w:tblBorders/w:$side", 'w:sz'); - $styleDefs["border{$ucfSide}Color"] = array(self::READ_VALUE, "w:tblBorders/w:$side", 'w:color'); - $styleDefs["border{$ucfSide}Style"] = array(self::READ_VALUE, "w:tblBorders/w:$side", 'w:val'); + $styleDefs["border{$ucfSide}Size"] = array(self::READ_EOP, "w:tblBorders/w:$side", 'w:sz'); + $styleDefs["border{$ucfSide}Color"] = array(self::READ_COLOR, "w:tblBorders/w:$side", 'w:color'); + $styleDefs["border{$ucfSide}Style"] = array(self::READ_BORDER_STYLE, "w:tblBorders/w:$side", 'w:val'); } $styleDefs['layout'] = array(self::READ_VALUE, 'w:tblLayout', 'w:type'); $styleDefs['bidiVisual'] = array(self::READ_TRUE, 'w:bidiVisual'); - $styleDefs['cellSpacing'] = array(self::READ_VALUE, 'w:tblCellSpacing', 'w:w'); + $styleDefs['cellSpacing'] = array(self::READ_TWIP, 'w:tblCellSpacing', 'w:w'); $style = $this->readStyleDefs($xmlReader, $styleNode, $styleDefs); $tablePositionNode = $xmlReader->getElement('w:tblpPr', $styleNode); @@ -505,23 +534,21 @@ protected function readTableStyle(XMLReader $xmlReader, \DOMElement $domNode) /** * Read w:tblpPr * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $domNode * @return array */ - private function readTablePosition(XMLReader $xmlReader, \DOMElement $domNode) + private function readTablePosition(XMLReader $xmlReader, DOMElement $domNode) { $styleDefs = array( - 'leftFromText' => array(self::READ_VALUE, '.', 'w:leftFromText'), - 'rightFromText' => array(self::READ_VALUE, '.', 'w:rightFromText'), - 'topFromText' => array(self::READ_VALUE, '.', 'w:topFromText'), - 'bottomFromText' => array(self::READ_VALUE, '.', 'w:bottomFromText'), + 'leftFromText' => array(self::READ_TWIP, '.', 'w:leftFromText'), + 'rightFromText' => array(self::READ_TWIP, '.', 'w:rightFromText'), + 'topFromText' => array(self::READ_TWIP, '.', 'w:topFromText'), + 'bottomFromText' => array(self::READ_TWIP, '.', 'w:bottomFromText'), 'vertAnchor' => array(self::READ_VALUE, '.', 'w:vertAnchor'), 'horzAnchor' => array(self::READ_VALUE, '.', 'w:horzAnchor'), 'tblpXSpec' => array(self::READ_VALUE, '.', 'w:tblpXSpec'), - 'tblpX' => array(self::READ_VALUE, '.', 'w:tblpX'), + 'tblpX' => array(self::READ_TWIP, '.', 'w:tblpX'), 'tblpYSpec' => array(self::READ_VALUE, '.', 'w:tblpYSpec'), - 'tblpY' => array(self::READ_VALUE, '.', 'w:tblpY'), + 'tblpY' => array(self::READ_TWIP, '.', 'w:tblpY'), ); return $this->readStyleDefs($xmlReader, $domNode, $styleDefs); @@ -530,36 +557,43 @@ private function readTablePosition(XMLReader $xmlReader, \DOMElement $domNode) /** * Read w:tblInd * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $domNode - * @return TblWidthComplexType + * @see http://officeopenxml.com/WPtableIndent.php */ - private function readTableIndent(XMLReader $xmlReader, \DOMElement $domNode) + private function readTableIndent(XMLReader $xmlReader, DOMElement $domNode): Length { $styleDefs = array( 'value' => array(self::READ_VALUE, '.', 'w:w'), 'type' => array(self::READ_VALUE, '.', 'w:type'), ); - $styleDefs = $this->readStyleDefs($xmlReader, $domNode, $styleDefs); + $defs = $this->readStyleDefs($xmlReader, $domNode, $styleDefs); + if ($defs['type'] === 'dxa') { + return Absolute::from('twip', (float) $defs['value']); + } elseif ($defs['type'] === 'nil') { + return Absolute::from('twip', 0); + } elseif ($defs['type'] === 'pct') { + return new Absolute(null); + } elseif ($defs['type'] === 'auto') { + return new Absolute(null); + } - return new TblWidthComplexType((int) $styleDefs['value'], $styleDefs['type']); + trigger_error(sprintf('Table indent type `%s` is not supported', $defs['type']), E_USER_WARNING); + + return new Absolute(null); } /** * Read w:tcPr * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $domNode * @return array */ - private function readCellStyle(XMLReader $xmlReader, \DOMElement $domNode) + private function readCellStyle(XMLReader $xmlReader, DOMElement $domNode) { $styleDefs = array( 'valign' => array(self::READ_VALUE, 'w:vAlign'), 'textDirection' => array(self::READ_VALUE, 'w:textDirection'), 'gridSpan' => array(self::READ_VALUE, 'w:gridSpan'), 'vMerge' => array(self::READ_VALUE, 'w:vMerge'), - 'bgColor' => array(self::READ_VALUE, 'w:shd', 'w:fill'), + 'bgColor' => array(self::READ_COLOR, 'w:shd', 'w:fill'), ); return $this->readStyleDefs($xmlReader, $domNode, $styleDefs); @@ -568,36 +602,33 @@ private function readCellStyle(XMLReader $xmlReader, \DOMElement $domNode) /** * Returns the first child element found * - * @param XMLReader $xmlReader - * @param \DOMElement $parentNode + * @param DOMElement $parentNode * @param string|array $elements * @return string|null */ - private function findPossibleElement(XMLReader $xmlReader, \DOMElement $parentNode = null, $elements) + private function findPossibleElement(XMLReader $xmlReader, DOMElement $parentNode = null, $elements) { - if (is_array($elements)) { - //if element is an array, we take the first element that exists in the XML - foreach ($elements as $possibleElement) { - if ($xmlReader->elementExists($possibleElement, $parentNode)) { - return $possibleElement; - } - } - } else { + if (!is_array($elements)) { return $elements; } + //if element is an array, we take the first element that exists in the XML + foreach ($elements as $possibleElement) { + if ($xmlReader->elementExists($possibleElement, $parentNode)) { + return $possibleElement; + } + } + return null; } /** * Returns the first attribute found * - * @param XMLReader $xmlReader - * @param \DOMElement $node * @param string|array $attributes * @return string|null */ - private function findPossibleAttribute(XMLReader $xmlReader, \DOMElement $node, $attributes) + private function findPossibleAttribute(XMLReader $xmlReader, DOMElement $node, $attributes) { //if attribute is an array, we take the first attribute that exists in the XML if (is_array($attributes)) { @@ -616,13 +647,12 @@ private function findPossibleAttribute(XMLReader $xmlReader, \DOMElement $node, /** * Read style definition * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $parentNode + * @param DOMElement $parentNode * @param array $styleDefs * @ignoreScrutinizerPatch * @return array */ - protected function readStyleDefs(XMLReader $xmlReader, \DOMElement $parentNode = null, $styleDefs = array()) + protected function readStyleDefs(XMLReader $xmlReader, DOMElement $parentNode = null, $styleDefs = array()) { $styles = array(); @@ -659,21 +689,29 @@ protected function readStyleDefs(XMLReader $xmlReader, \DOMElement $parentNode = * @param string $method * @ignoreScrutinizerPatch * @param string|null $attributeValue - * @param mixed $expected - * @return mixed */ private function readStyleDef($method, $attributeValue, $expected) { $style = $attributeValue; - if (self::READ_SIZE == $method) { - $style = $attributeValue / 2; + if (self::READ_EOP == $method) { + $style = Absolute::fromMixed('eop', $attributeValue); + } elseif (self::READ_TWIP == $method) { + $style = Absolute::fromMixed('twip', $attributeValue); + } elseif (self::READ_HPT == $method) { + $style = Absolute::fromMixed('hpt', $attributeValue); + } elseif (self::READ_BORDER_STYLE == $method) { + $style = BorderStyle::fromMixed($attributeValue); } elseif (self::READ_TRUE == $method) { $style = $this->isOn($attributeValue); } elseif (self::READ_FALSE == $method) { $style = !$this->isOn($attributeValue); } elseif (self::READ_EQUAL == $method) { $style = $attributeValue == $expected; + } elseif (self::READ_COLOR == $method) { + $style = BasicColor::fromMixed($attributeValue); + } elseif (self::READ_HIGHLIGHT == $method) { + $style = new HighlightColor($attributeValue); } return $style; diff --git a/src/PhpWord/Reader/Word2007/DocPropsApp.php b/src/PhpWord/Reader/Word2007/DocPropsApp.php index decc510390..63f3347ba5 100644 --- a/src/PhpWord/Reader/Word2007/DocPropsApp.php +++ b/src/PhpWord/Reader/Word2007/DocPropsApp.php @@ -1,4 +1,5 @@ array(self::READ_VALUE, 'w:type'), 'vAlign' => array(self::READ_VALUE, 'w:vAlign'), - 'pageSizeW' => array(self::READ_VALUE, 'w:pgSz', 'w:w'), - 'pageSizeH' => array(self::READ_VALUE, 'w:pgSz', 'w:h'), + 'pageSizeW' => array(self::READ_TWIP, 'w:pgSz', 'w:w'), + 'pageSizeH' => array(self::READ_TWIP, 'w:pgSz', 'w:h'), 'orientation' => array(self::READ_VALUE, 'w:pgSz', 'w:orient'), 'colsNum' => array(self::READ_VALUE, 'w:cols', 'w:num'), - 'colsSpace' => array(self::READ_VALUE, 'w:cols', 'w:space'), - 'marginTop' => array(self::READ_VALUE, 'w:pgMar', 'w:top'), - 'marginLeft' => array(self::READ_VALUE, 'w:pgMar', 'w:left'), - 'marginBottom' => array(self::READ_VALUE, 'w:pgMar', 'w:bottom'), - 'marginRight' => array(self::READ_VALUE, 'w:pgMar', 'w:right'), - 'headerHeight' => array(self::READ_VALUE, 'w:pgMar', 'w:header'), - 'footerHeight' => array(self::READ_VALUE, 'w:pgMar', 'w:footer'), - 'gutter' => array(self::READ_VALUE, 'w:pgMar', 'w:gutter'), + 'colsSpace' => array(self::READ_TWIP, 'w:cols', 'w:space'), + 'marginTop' => array(self::READ_TWIP, 'w:pgMar', 'w:top'), + 'marginLeft' => array(self::READ_TWIP, 'w:pgMar', 'w:left'), + 'marginBottom' => array(self::READ_TWIP, 'w:pgMar', 'w:bottom'), + 'marginRight' => array(self::READ_TWIP, 'w:pgMar', 'w:right'), + 'headerHeight' => array(self::READ_TWIP, 'w:pgMar', 'w:header'), + 'footerHeight' => array(self::READ_TWIP, 'w:pgMar', 'w:footer'), + 'gutter' => array(self::READ_TWIP, 'w:pgMar', 'w:gutter'), ); $styles = $this->readStyleDefs($xmlReader, $domNode, $styleDefs); // Header and footer - // @todo Cleanup this part $nodes = $xmlReader->getElements('*', $domNode); foreach ($nodes as $node) { if ($node->nodeName == 'w:headerReference' || $node->nodeName == 'w:footerReference') { @@ -141,8 +137,6 @@ private function readSectionStyle(XMLReader $xmlReader, \DOMElement $domNode) /** * Read w:p node. * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $node * @param \PhpOffice\PhpWord\Element\Section &$section * * @todo @@ -170,8 +164,6 @@ private function readWPNode(XMLReader $xmlReader, \DOMElement $node, Section &$s /** * Read w:sectPr node. * - * @param \PhpOffice\Common\XMLReader $xmlReader - * @param \DOMElement $node * @param \PhpOffice\PhpWord\Element\Section &$section */ private function readWSectPrNode(XMLReader $xmlReader, \DOMElement $node, Section &$section) diff --git a/src/PhpWord/Reader/Word2007/Endnotes.php b/src/PhpWord/Reader/Word2007/Endnotes.php index aa8b65d750..0201f1462d 100644 --- a/src/PhpWord/Reader/Word2007/Endnotes.php +++ b/src/PhpWord/Reader/Word2007/Endnotes.php @@ -1,4 +1,5 @@ getAttribute('w:val', $subnode, 'w:suff'); $level['text'] = $xmlReader->getAttribute('w:val', $subnode, 'w:lvlText'); $level['alignment'] = $xmlReader->getAttribute('w:val', $subnode, 'w:lvlJc'); - $level['tab'] = $xmlReader->getAttribute('w:pos', $subnode, 'w:pPr/w:tabs/w:tab'); - $level['left'] = $xmlReader->getAttribute('w:left', $subnode, 'w:pPr/w:ind'); - $level['hanging'] = $xmlReader->getAttribute('w:hanging', $subnode, 'w:pPr/w:ind'); + $level['tab'] = (int) $xmlReader->getAttribute('w:pos', $subnode, 'w:pPr/w:tabs/w:tab'); + $level['left'] = Absolute::from('twip', (int) $xmlReader->getAttribute('w:left', $subnode, 'w:pPr/w:ind')); + $level['hanging'] = Absolute::from('twip', (int) $xmlReader->getAttribute('w:hanging', $subnode, 'w:pPr/w:ind')); $level['font'] = $xmlReader->getAttribute('w:ascii', $subnode, 'w:rPr/w:rFonts'); $level['hint'] = $xmlReader->getAttribute('w:hint', $subnode, 'w:rPr/w:rFonts'); diff --git a/src/PhpWord/Reader/Word2007/Settings.php b/src/PhpWord/Reader/Word2007/Settings.php index 3084943b37..7ecaa9ca6f 100644 --- a/src/PhpWord/Reader/Word2007/Settings.php +++ b/src/PhpWord/Reader/Word2007/Settings.php @@ -1,4 +1,5 @@ setLatin($val); - $themeFontLang->setEastAsia($eastAsia); - $themeFontLang->setBidirectional($bidi); + if ($eastAsia !== null) { + $themeFontLang->setEastAsia($eastAsia); + } + if ($bidi !== null) { + $themeFontLang->setBidirectional($bidi); + } $phpWord->getSettings()->setThemeFontLang($themeFontLang); } /** * Sets the document protection - * - * @param XMLReader $xmlReader - * @param PhpWord $phpWord - * @param \DOMElement $node */ protected function setDocumentProtection(XMLReader $xmlReader, PhpWord $phpWord, \DOMElement $node) { @@ -116,10 +112,6 @@ protected function setDocumentProtection(XMLReader $xmlReader, PhpWord $phpWord, /** * Sets the proof state - * - * @param XMLReader $xmlReader - * @param PhpWord $phpWord - * @param \DOMElement $node */ protected function setProofState(XMLReader $xmlReader, PhpWord $phpWord, \DOMElement $node) { @@ -138,10 +130,6 @@ protected function setProofState(XMLReader $xmlReader, PhpWord $phpWord, \DOMEle /** * Sets the proof state - * - * @param XMLReader $xmlReader - * @param PhpWord $phpWord - * @param \DOMElement $node */ protected function setZoom(XMLReader $xmlReader, PhpWord $phpWord, \DOMElement $node) { @@ -155,10 +143,6 @@ protected function setZoom(XMLReader $xmlReader, PhpWord $phpWord, \DOMElement $ /** * Set the Revision view - * - * @param XMLReader $xmlReader - * @param PhpWord $phpWord - * @param \DOMElement $node */ protected function setRevisionView(XMLReader $xmlReader, PhpWord $phpWord, \DOMElement $node) { @@ -171,11 +155,6 @@ protected function setRevisionView(XMLReader $xmlReader, PhpWord $phpWord, \DOME $phpWord->getSettings()->setRevisionView($revisionView); } - /** - * @param XMLReader $xmlReader - * @param PhpWord $phpWord - * @param \DOMElement $node - */ protected function setConsecutiveHyphenLimit(XMLReader $xmlReader, PhpWord $phpWord, \DOMElement $node) { $value = $xmlReader->getAttribute('w:val', $node); @@ -185,17 +164,12 @@ protected function setConsecutiveHyphenLimit(XMLReader $xmlReader, PhpWord $phpW } } - /** - * @param XMLReader $xmlReader - * @param PhpWord $phpWord - * @param \DOMElement $node - */ protected function setHyphenationZone(XMLReader $xmlReader, PhpWord $phpWord, \DOMElement $node) { $value = $xmlReader->getAttribute('w:val', $node); if ($value !== null) { - $phpWord->getSettings()->setHyphenationZone($value); + $phpWord->getSettings()->setHyphenationZone(Absolute::from('twip', (int) $value)); } } } diff --git a/src/PhpWord/Reader/Word2007/Styles.php b/src/PhpWord/Reader/Word2007/Styles.php index 97f29b4316..f5c14b332d 100644 --- a/src/PhpWord/Reader/Word2007/Styles.php +++ b/src/PhpWord/Reader/Word2007/Styles.php @@ -1,4 +1,5 @@ 0) { + if ($value->toFloat('twip') > 0) { self::$defaultFontSize = $value; return true; @@ -423,10 +435,14 @@ public static function loadConfig($filename = null) // Set config value foreach ($config as $key => $value) { - $method = "set{$key}"; - if (method_exists(__CLASS__, $method)) { - self::$method($value); + $method = "set{$key}FromConfig"; + if (!method_exists(__CLASS__, $method)) { + $method = "set{$key}"; + if (!method_exists(__CLASS__, $method)) { + throw new Exception("No method found for key `$key` to set config value"); + } } + self::$method($value); } return $config; diff --git a/src/PhpWord/Shared/AbstractEnum.php b/src/PhpWord/Shared/AbstractEnum.php index f2375d876e..92df8b043a 100644 --- a/src/PhpWord/Shared/AbstractEnum.php +++ b/src/PhpWord/Shared/AbstractEnum.php @@ -1,4 +1,5 @@ nodeType) { $attributes = $node->attributes; // get all the attributes(eg: id, class) @@ -98,7 +108,7 @@ protected static function parseInlineStyle($node, $styles = array()) foreach ($attributes as $attribute) { switch ($attribute->name) { case 'style': - $styles = self::parseStyle($attribute, $styles); + $styles = self::parseStyle($node, $attribute, $styles, $inherited); break; case 'align': $styles['alignment'] = self::mapAlign($attribute->value); @@ -228,6 +238,24 @@ protected static function parseChildNodes($node, $element, $styles, $data) protected static function parseParagraph($node, $element, &$styles) { $styles['paragraph'] = self::recursiveParseStylesInHierarchy($node, $styles['paragraph']); + + $fontAttributes = array( + 'letter-spacing', + 'hidden', + 'underline', + 'strikethrough', + 'color', + 'bgColor', + 'bold', + 'italic', + ); + foreach ($fontAttributes as $fontAttribute) { + if (isset($styles['paragraph'][$fontAttribute])) { + $styles['font'][$fontAttribute] = $styles['paragraph'][$fontAttribute]; + unset($styles['paragraph'][$fontAttribute]); + } + } + $newElement = $element->addTextRun($styles['paragraph']); return $newElement; @@ -263,9 +291,10 @@ protected static function parseText($node, $element, &$styles) { $styles['font'] = self::recursiveParseStylesInHierarchy($node, $styles['font']); - //alignment applies on paragraph, not on font. Let's copy it there + //alignment applies on paragraph, not on font. Let's move it there if (isset($styles['font']['alignment']) && is_array($styles['paragraph'])) { $styles['paragraph']['alignment'] = $styles['font']['alignment']; + unset($styles['font']['alignment']); } if (is_callable(array($element, 'addText'))) { @@ -364,7 +393,7 @@ protected static function parseCell($node, $element, &$styles) $cell = $element->addCell(null, $cellStyles); if (self::shouldAddTextRun($node)) { - return $cell->addTextRun(self::parseInlineStyle($node, $styles['paragraph'])); + return $cell->addTextRun(self::parseInlineStyle($node, $styles['paragraph'], true)); } return $cell; @@ -373,7 +402,6 @@ protected static function parseCell($node, $element, &$styles) /** * Checks if $node contains an HTML element that cannot be added to TextRun * - * @param \DOMNode $node * @return bool Returns true if the node contains an HTML element that cannot be added to TextRun */ protected static function shouldAddTextRun(\DOMNode $node) @@ -390,15 +418,14 @@ protected static function shouldAddTextRun(\DOMNode $node) * Recursively parses styles on parent nodes * TODO if too slow, add caching of parent nodes, !! everything is static here so watch out for concurrency !! * - * @param \DOMNode $node * @param array &$styles */ - protected static function recursiveParseStylesInHierarchy(\DOMNode $node, array $style) + protected static function recursiveParseStylesInHierarchy(\DOMNode $node, array $style, bool $inherited = false) { - $parentStyle = self::parseInlineStyle($node, array()); + $parentStyle = self::parseInlineStyle($node, array(), $inherited); $style = array_merge($parentStyle, $style); if ($node->parentNode != null && XML_ELEMENT_NODE == $node->parentNode->nodeType) { - $style = self::recursiveParseStylesInHierarchy($node->parentNode, $style); + $style = self::recursiveParseStylesInHierarchy($node->parentNode, $style, true); } return $style; @@ -437,15 +464,15 @@ protected static function getListStyle($isOrderedList) return array( 'type' => 'multilevel', 'levels' => array( - array('format' => NumberFormat::DECIMAL, 'text' => '%1.', 'alignment' => 'left', 'tabPos' => 720, 'left' => 720, 'hanging' => 360), - array('format' => NumberFormat::LOWER_LETTER, 'text' => '%2.', 'alignment' => 'left', 'tabPos' => 1440, 'left' => 1440, 'hanging' => 360), - array('format' => NumberFormat::LOWER_ROMAN, 'text' => '%3.', 'alignment' => 'right', 'tabPos' => 2160, 'left' => 2160, 'hanging' => 180), - array('format' => NumberFormat::DECIMAL, 'text' => '%4.', 'alignment' => 'left', 'tabPos' => 2880, 'left' => 2880, 'hanging' => 360), - array('format' => NumberFormat::LOWER_LETTER, 'text' => '%5.', 'alignment' => 'left', 'tabPos' => 3600, 'left' => 3600, 'hanging' => 360), - array('format' => NumberFormat::LOWER_ROMAN, 'text' => '%6.', 'alignment' => 'right', 'tabPos' => 4320, 'left' => 4320, 'hanging' => 180), - array('format' => NumberFormat::DECIMAL, 'text' => '%7.', 'alignment' => 'left', 'tabPos' => 5040, 'left' => 5040, 'hanging' => 360), - array('format' => NumberFormat::LOWER_LETTER, 'text' => '%8.', 'alignment' => 'left', 'tabPos' => 5760, 'left' => 5760, 'hanging' => 360), - array('format' => NumberFormat::LOWER_ROMAN, 'text' => '%9.', 'alignment' => 'right', 'tabPos' => 6480, 'left' => 6480, 'hanging' => 180), + array('format' => NumberFormat::DECIMAL, 'text' => '%1.', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 720), 'left' => Absolute::from('twip', 720), 'hanging' => Absolute::from('twip', 360)), + array('format' => NumberFormat::LOWER_LETTER, 'text' => '%2.', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 1440), 'left' => Absolute::from('twip', 1440), 'hanging' => Absolute::from('twip', 360)), + array('format' => NumberFormat::LOWER_ROMAN, 'text' => '%3.', 'alignment' => 'right', 'tabPos' => Absolute::from('twip', 2160), 'left' => Absolute::from('twip', 2160), 'hanging' => Absolute::from('twip', 180)), + array('format' => NumberFormat::DECIMAL, 'text' => '%4.', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 2880), 'left' => Absolute::from('twip', 2880), 'hanging' => Absolute::from('twip', 360)), + array('format' => NumberFormat::LOWER_LETTER, 'text' => '%5.', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 3600), 'left' => Absolute::from('twip', 3600), 'hanging' => Absolute::from('twip', 360)), + array('format' => NumberFormat::LOWER_ROMAN, 'text' => '%6.', 'alignment' => 'right', 'tabPos' => Absolute::from('twip', 4320), 'left' => Absolute::from('twip', 4320), 'hanging' => Absolute::from('twip', 180)), + array('format' => NumberFormat::DECIMAL, 'text' => '%7.', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 5040), 'left' => Absolute::from('twip', 5040), 'hanging' => Absolute::from('twip', 360)), + array('format' => NumberFormat::LOWER_LETTER, 'text' => '%8.', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 5760), 'left' => Absolute::from('twip', 5760), 'hanging' => Absolute::from('twip', 360)), + array('format' => NumberFormat::LOWER_ROMAN, 'text' => '%9.', 'alignment' => 'right', 'tabPos' => Absolute::from('twip', 6480), 'left' => Absolute::from('twip', 6480), 'hanging' => Absolute::from('twip', 180)), ), ); } @@ -453,15 +480,15 @@ protected static function getListStyle($isOrderedList) return array( 'type' => 'hybridMultilevel', 'levels' => array( - array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => 720, 'left' => 720, 'hanging' => 360, 'font' => 'Symbol', 'hint' => 'default'), - array('format' => NumberFormat::BULLET, 'text' => 'o', 'alignment' => 'left', 'tabPos' => 1440, 'left' => 1440, 'hanging' => 360, 'font' => 'Courier New', 'hint' => 'default'), - array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => 2160, 'left' => 2160, 'hanging' => 360, 'font' => 'Wingdings', 'hint' => 'default'), - array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => 2880, 'left' => 2880, 'hanging' => 360, 'font' => 'Symbol', 'hint' => 'default'), - array('format' => NumberFormat::BULLET, 'text' => 'o', 'alignment' => 'left', 'tabPos' => 3600, 'left' => 3600, 'hanging' => 360, 'font' => 'Courier New', 'hint' => 'default'), - array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => 4320, 'left' => 4320, 'hanging' => 360, 'font' => 'Wingdings', 'hint' => 'default'), - array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => 5040, 'left' => 5040, 'hanging' => 360, 'font' => 'Symbol', 'hint' => 'default'), - array('format' => NumberFormat::BULLET, 'text' => 'o', 'alignment' => 'left', 'tabPos' => 5760, 'left' => 5760, 'hanging' => 360, 'font' => 'Courier New', 'hint' => 'default'), - array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => 6480, 'left' => 6480, 'hanging' => 360, 'font' => 'Wingdings', 'hint' => 'default'), + array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 720), 'left' => Absolute::from('twip', 720), 'hanging' => Absolute::from('twip', 360), 'font' => 'Symbol', 'hint' => 'default'), + array('format' => NumberFormat::BULLET, 'text' => 'o', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 1440), 'left' => Absolute::from('twip', 1440), 'hanging' => Absolute::from('twip', 360), 'font' => 'Courier New', 'hint' => 'default'), + array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 2160), 'left' => Absolute::from('twip', 2160), 'hanging' => Absolute::from('twip', 360), 'font' => 'Wingdings', 'hint' => 'default'), + array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 2880), 'left' => Absolute::from('twip', 2880), 'hanging' => Absolute::from('twip', 360), 'font' => 'Symbol', 'hint' => 'default'), + array('format' => NumberFormat::BULLET, 'text' => 'o', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 3600), 'left' => Absolute::from('twip', 3600), 'hanging' => Absolute::from('twip', 360), 'font' => 'Courier New', 'hint' => 'default'), + array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 4320), 'left' => Absolute::from('twip', 4320), 'hanging' => Absolute::from('twip', 360), 'font' => 'Wingdings', 'hint' => 'default'), + array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 5040), 'left' => Absolute::from('twip', 5040), 'hanging' => Absolute::from('twip', 360), 'font' => 'Symbol', 'hint' => 'default'), + array('format' => NumberFormat::BULLET, 'text' => 'o', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 5760), 'left' => Absolute::from('twip', 5760), 'hanging' => Absolute::from('twip', 360), 'font' => 'Courier New', 'hint' => 'default'), + array('format' => NumberFormat::BULLET, 'text' => '', 'alignment' => 'left', 'tabPos' => Absolute::from('twip', 6480), 'left' => Absolute::from('twip', 6480), 'hanging' => Absolute::from('twip', 360), 'font' => 'Wingdings', 'hint' => 'default'), ), ); } @@ -495,126 +522,134 @@ protected static function parseListItem($node, $element, &$styles, $data) * @param array $styles * @return array */ - protected static function parseStyle($attribute, $styles) + protected static function parseStyle(DOMNode $node, $attribute, $styles, bool $inherited) { $properties = explode(';', trim($attribute->value, " \t\n\r\0\x0B;")); foreach ($properties as $property) { - list($cKey, $cValue) = array_pad(explode(':', $property, 2), 2, null); - $cValue = trim($cValue); - switch (trim($cKey)) { - case 'text-decoration': - switch ($cValue) { - case 'underline': - $styles['underline'] = 'single'; - break; - case 'line-through': - $styles['strikethrough'] = true; - break; - } - break; - case 'text-align': - $styles['alignment'] = self::mapAlign($cValue); - break; - case 'display': - $styles['hidden'] = $cValue === 'none' || $cValue === 'hidden'; - break; - case 'direction': - $styles['rtl'] = $cValue === 'rtl'; - break; - case 'font-size': - $styles['size'] = Converter::cssToPoint($cValue); - break; - case 'font-family': - $cValue = array_map('trim', explode(',', $cValue)); - $styles['name'] = ucwords($cValue[0]); - break; - case 'color': - $styles['color'] = trim($cValue, '#'); - break; - case 'background-color': - $styles['bgColor'] = trim($cValue, '#'); - break; - case 'line-height': - $matches = array(); - if (preg_match('/([0-9]+\.?[0-9]*[a-z]+)/', $cValue, $matches)) { - //matches number with a unit, e.g. 12px, 15pt, 20mm, ... - $spacingLineRule = \PhpOffice\PhpWord\SimpleType\LineSpacingRule::EXACT; - $spacing = Converter::cssToTwip($matches[1]); - } elseif (preg_match('/([0-9]+)%/', $cValue, $matches)) { - //matches percentages - $spacingLineRule = \PhpOffice\PhpWord\SimpleType\LineSpacingRule::AUTO; - //we are subtracting 1 line height because the Spacing writer is adding one line - $spacing = ((((int) $matches[1]) / 100) * Paragraph::LINE_HEIGHT) - Paragraph::LINE_HEIGHT; - } else { - //any other, wich is a multiplier. E.g. 1.2 - $spacingLineRule = \PhpOffice\PhpWord\SimpleType\LineSpacingRule::AUTO; - //we are subtracting 1 line height because the Spacing writer is adding one line - $spacing = ($cValue * Paragraph::LINE_HEIGHT) - Paragraph::LINE_HEIGHT; - } - $styles['spacingLineRule'] = $spacingLineRule; - $styles['line-spacing'] = $spacing; - break; - case 'letter-spacing': - $styles['letter-spacing'] = Converter::cssToTwip($cValue); - break; - case 'text-indent': - $styles['indentation']['firstLine'] = Converter::cssToTwip($cValue); - break; - case 'font-weight': - $tValue = false; - if (preg_match('#bold#', $cValue)) { - $tValue = true; // also match bolder - } - $styles['bold'] = $tValue; - break; - case 'font-style': - $tValue = false; - if (preg_match('#(?:italic|oblique)#', $cValue)) { - $tValue = true; - } - $styles['italic'] = $tValue; - break; - case 'margin-top': - $styles['spaceBefore'] = Converter::cssToPoint($cValue); - break; - case 'margin-bottom': - $styles['spaceAfter'] = Converter::cssToPoint($cValue); - break; - case 'border-color': - self::mapBorderColor($styles, $cValue); - break; - case 'border-width': - $styles['borderSize'] = Converter::cssToPoint($cValue); - break; - case 'border-style': - $styles['borderStyle'] = self::mapBorderStyle($cValue); - break; - case 'width': - if (preg_match('/([0-9]+[a-z]+)/', $cValue, $matches)) { - $styles['width'] = Converter::cssToTwip($matches[1]); - $styles['unit'] = \PhpOffice\PhpWord\SimpleType\TblWidth::TWIP; - } elseif (preg_match('/([0-9]+)%/', $cValue, $matches)) { - $styles['width'] = $matches[1] * 50; - $styles['unit'] = \PhpOffice\PhpWord\SimpleType\TblWidth::PERCENT; - } elseif (preg_match('/([0-9]+)/', $cValue, $matches)) { - $styles['width'] = $matches[1]; - $styles['unit'] = \PhpOffice\PhpWord\SimpleType\TblWidth::AUTO; - } - break; - case 'border': - if (preg_match('/([0-9]+[^0-9]*)\s+(\#[a-fA-F0-9]+)\s+([a-z]+)/', $cValue, $matches)) { - $styles['borderSize'] = Converter::cssToPoint($matches[1]); - $styles['borderColor'] = trim($matches[2], '#'); - $styles['borderStyle'] = self::mapBorderStyle($matches[3]); - } - break; - } + list($property, $value) = array_pad(explode(':', $property, 2), 2, ''); + self::mapStyleDeclaration($node, trim($property), trim($value), $styles, $inherited); } return $styles; } + private static function mapStyleDeclaration(DOMNode $node, string $property, string $value, array &$styles, bool $inherited) + { + switch ($property) { + case 'text-decoration': + switch ($value) { + case 'underline': + $styles['underline'] = 'single'; + break; + case 'line-through': + $styles['strikethrough'] = true; + break; + } + break; + case 'text-align': + $styles['alignment'] = self::mapAlign($value); + break; + case 'display': + // If `display: none` or `visibility: hidden` is set, + // element should be hidden + $styles['hidden'] = $value === 'none' || ($styles['hidden'] ?? false); + break; + case 'direction': + $styles['rtl'] = $value === 'rtl'; + break; + case 'font-size': + $styles['size'] = self::cssToAbsolute($value); + break; + case 'font-family': + $value = array_map('trim', explode(',', $value)); + $styles['name'] = ucwords($value[0]); + break; + case 'color': + $styles['color'] = BasicColor::fromMixed(trim($value, '#')); + break; + case 'background-color': + $styles['bgColor'] = BasicColor::fromMixed(trim($value, '#')); + break; + case 'line-height': + $matches = array(); + if (preg_match('/([0-9]+\.?[0-9]*[a-z]+)/', $value, $matches)) { + //matches number with a unit, e.g. 12px, 15pt, 20mm, ... + $styles['spacingLineRule'] = LineSpacingRule::EXACT; + $styles['line-spacing'] = array('line' => self::cssToAbsolute($matches[1])); + } elseif (preg_match('/([0-9]+\.?[0-9]*)%/', $value, $matches)) { + //matches percentages + $styles['line-height'] = new Percent((float) $matches[1]); + } else { + //any other, wich is a multiplier. E.g. 1.2 + $styles['line-height'] = new Percent($value * 100); + } + break; + case 'letter-spacing': + $styles['letter-spacing'] = self::cssToAbsolute($value); + break; + case 'text-indent': + $styles['indentation']['firstLine'] = self::cssToAbsolute($value); + break; + case 'font-weight': + $tValue = false; + if (preg_match('#bold#', $value)) { + $tValue = true; // also match bolder + } + $styles['bold'] = $tValue; + break; + case 'font-style': + $tValue = false; + if (preg_match('#(?:italic|oblique)#', $value)) { + $tValue = true; + } + $styles['italic'] = $tValue; + break; + case 'margin-top': + $styles['spaceBefore'] = self::cssToAbsolute($value); + break; + case 'margin-bottom': + $styles['spaceAfter'] = self::cssToAbsolute($value); + break; + case 'border-color': + if (!$inherited) { + self::mapBorderColor($node, $styles, $value); + } + break; + case 'border-width': + if (!$inherited) { + self::mapBorderWidth($node, $styles, $value); + } + break; + case 'border-style': + if (!$inherited) { + self::mapBorderStyle($node, $styles, $value); + } + break; + case 'width': + if (preg_match('/([0-9]+[a-z]+)/', $value, $matches)) { + $styles['width'] = self::cssToAbsolute($matches[1]); + } elseif (preg_match('/([0-9]+)%/', $value, $matches)) { + $styles['width'] = Percent::fromMixed($matches[1]); + } elseif (preg_match('/([0-9]+)/', $value, $matches)) { + $styles['width'] = Absolute::fromMixed('twip', $matches[1]); + } + break; + case 'border': + if (!$inherited && preg_match('/([0-9]+[^0-9]*)\s+(\#[a-fA-F0-9]+)\s+([a-z]+)/', $value, $matches)) { + self::mapBorderColor($node, $styles, $matches[2]); + self::mapBorderWidth($node, $styles, $matches[1]); + self::mapBorderStyle($node, $styles, $matches[3]); + } + break; + case 'visibility': + // If `display: none` or `visibility: hidden` is set, + // element should be hidden + $styles['hidden'] = $value === 'hidden' || ($styles['hidden'] ?? false); + break; + } + } + /** * Parse image node * @@ -633,14 +668,10 @@ protected static function parseImage($node, $element) $src = $attribute->value; break; case 'width': - $width = $attribute->value; - $style['width'] = $width; - $style['unit'] = \PhpOffice\PhpWord\Style\Image::UNIT_PX; + $style['width'] = Absolute::fromPixels(new Dpi(), (float) $attribute->value); break; case 'height': - $height = $attribute->value; - $style['height'] = $height; - $style['unit'] = \PhpOffice\PhpWord\Style\Image::UNIT_PX; + $style['height'] = Absolute::fromPixels(new Dpi(), (float) $attribute->value); break; case 'style': $styleattr = explode(';', $attribute->value); @@ -650,17 +681,17 @@ protected static function parseImage($node, $element) switch ($k) { case 'float': if (trim($v) == 'right') { - $style['hPos'] = \PhpOffice\PhpWord\Style\Image::POS_RIGHT; - $style['hPosRelTo'] = \PhpOffice\PhpWord\Style\Image::POS_RELTO_PAGE; - $style['pos'] = \PhpOffice\PhpWord\Style\Image::POS_RELATIVE; - $style['wrap'] = \PhpOffice\PhpWord\Style\Image::WRAP_TIGHT; + $style['hPos'] = Image::POS_RIGHT; + $style['hPosRelTo'] = Image::POS_RELTO_PAGE; + $style['pos'] = Image::POS_RELATIVE; + $style['wrap'] = Image::WRAP_TIGHT; $style['overlap'] = true; } if (trim($v) == 'left') { - $style['hPos'] = \PhpOffice\PhpWord\Style\Image::POS_LEFT; - $style['hPosRelTo'] = \PhpOffice\PhpWord\Style\Image::POS_RELTO_PAGE; - $style['pos'] = \PhpOffice\PhpWord\Style\Image::POS_RELATIVE; - $style['wrap'] = \PhpOffice\PhpWord\Style\Image::WRAP_TIGHT; + $style['hPos'] = Image::POS_LEFT; + $style['hPosRelTo'] = Image::POS_RELTO_PAGE; + $style['pos'] = Image::POS_RELATIVE; + $style['wrap'] = Image::WRAP_TIGHT; $style['overlap'] = true; } break; @@ -722,35 +753,252 @@ protected static function parseImage($node, $element) /** * Transforms a CSS border style into a word border style - * - * @param string $cssBorderStyle - * @return null|string + * @param mixed $styles */ - protected static function mapBorderStyle($cssBorderStyle) + protected static function mapBorderStyle(DOMNode $node, &$styles, string $cssStyles) { - switch ($cssBorderStyle) { - case 'none': - case 'dashed': - case 'dotted': - case 'double': - return $cssBorderStyle; - default: - return 'single'; + $cssStyles = self::expandBorderSides($node, $cssStyles); + + $mapping = array( + 'none' => 'none', + 'hidden' => 'none', + 'dotted' => 'dotted', + 'dashed' => 'dashed', + 'solid' => 'single', + 'double' => 'double', + 'groove' => 'threeDEngrave', + 'ridge' => 'threeDEmboss', + 'inset' => 'inset', + 'outset' => 'outset', + ); + + foreach ($cssStyles as $side => $cssStyle) { + $existingBorder = self::getBorderSide($styles, $side); + $existingBorder->setStyle(new BorderStyle($mapping[$cssStyle] ?? 'single')); } } - protected static function mapBorderColor(&$styles, $cssBorderColor) + /** + * @see https://developer.mozilla.org/en-US/docs/Web/CSS/color_value + */ + protected static $colorKeywords = array( + 'black' => '000000', + 'silver' => 'c0c0c0', + 'gray' => '808080', + 'white' => 'ffffff', + 'maroon' => '800000', + 'red' => 'ff0000', + 'purple' => '800080', + 'fuchsia' => 'ff00ff', + 'green' => '008000', + 'lime' => '00ff00', + 'olive' => '808000', + 'yellow' => 'ffff00', + 'navy' => '000080', + 'blue' => '0000ff', + 'teal' => '008080', + 'aqua' => '00ffff', + 'orange' => 'ffa500', + 'aliceblue' => 'f0f8ff', + 'antiquewhite' => 'faebd7', + 'aquamarine' => '7fffd4', + 'azure' => 'f0ffff', + 'beige' => 'f5f5dc', + 'bisque' => 'ffe4c4', + 'blanchedalmond' => 'ffebcd', + 'blueviolet' => '8a2be2', + 'brown' => 'a52a2a', + 'burlywood' => 'deb887', + 'cadetblue' => '5f9ea0', + 'chartreuse' => '7fff00', + 'chocolate' => 'd2691e', + 'coral' => 'ff7f50', + 'cornflowerblue' => '6495ed', + 'cornsilk' => 'fff8dc', + 'crimson' => 'dc143c', + 'cyan' => '00ffff', + 'darkblue' => '00008b', + 'darkcyan' => '008b8b', + 'darkgoldenrod' => 'b8860b', + 'darkgray' => 'a9a9a9', + 'darkgreen' => '006400', + 'darkgrey' => 'a9a9a9', + 'darkkhaki' => 'bdb76b', + 'darkmagenta' => '8b008b', + 'darkolivegreen' => '556b2f', + 'darkorange' => 'ff8c00', + 'darkorchid' => '9932cc', + 'darkred' => '8b0000', + 'darksalmon' => 'e9967a', + 'darkseagreen' => '8fbc8f', + 'darkslateblue' => '483d8b', + 'darkslategray' => '2f4f4f', + 'darkslategrey' => '2f4f4f', + 'darkturquoise' => '00ced1', + 'darkviolet' => '9400d3', + 'deeppink' => 'ff1493', + 'deepskyblue' => '00bfff', + 'dimgray' => '696969', + 'dimgrey' => '696969', + 'dodgerblue' => '1e90ff', + 'firebrick' => 'b22222', + 'floralwhite' => 'fffaf0', + 'forestgreen' => '228b22', + 'gainsboro' => 'dcdcdc', + 'ghostwhite' => 'f8f8ff', + 'gold' => 'ffd700', + 'goldenrod' => 'daa520', + 'greenyellow' => 'adff2f', + 'grey' => '808080', + 'honeydew' => 'f0fff0', + 'hotpink' => 'ff69b4', + 'indianred' => 'cd5c5c', + 'indigo' => '4b0082', + 'ivory' => 'fffff0', + 'khaki' => 'f0e68c', + 'lavender' => 'e6e6fa', + 'lavenderblush' => 'fff0f5', + 'lawngreen' => '7cfc00', + 'lemonchiffon' => 'fffacd', + 'lightblue' => 'add8e6', + 'lightcoral' => 'f08080', + 'lightcyan' => 'e0ffff', + 'lightgoldenrodyellow' => 'fafad2', + 'lightgray' => 'd3d3d3', + 'lightgreen' => '90ee90', + 'lightgrey' => 'd3d3d3', + 'lightpink' => 'ffb6c1', + 'lightsalmon' => 'ffa07a', + 'lightseagreen' => '20b2aa', + 'lightskyblue' => '87cefa', + 'lightslategray' => '778899', + 'lightslategrey' => '778899', + 'lightsteelblue' => 'b0c4de', + 'lightyellow' => 'ffffe0', + 'limegreen' => '32cd32', + 'linen' => 'faf0e6', + 'magenta' => 'ff00ff', + 'mediumaquamarine' => '66cdaa', + 'mediumblue' => '0000cd', + 'mediumorchid' => 'ba55d3', + 'mediumpurple' => '9370db', + 'mediumseagreen' => '3cb371', + 'mediumslateblue' => '7b68ee', + 'mediumspringgreen' => '00fa9a', + 'mediumturquoise' => '48d1cc', + 'mediumvioletred' => 'c71585', + 'midnightblue' => '191970', + 'mintcream' => 'f5fffa', + 'mistyrose' => 'ffe4e1', + 'moccasin' => 'ffe4b5', + 'navajowhite' => 'ffdead', + 'oldlace' => 'fdf5e6', + 'olivedrab' => '6b8e23', + 'orangered' => 'ff4500', + 'orchid' => 'da70d6', + 'palegoldenrod' => 'eee8aa', + 'palegreen' => '98fb98', + 'paleturquoise' => 'afeeee', + 'palevioletred' => 'db7093', + 'papayawhip' => 'ffefd5', + 'peachpuff' => 'ffdab9', + 'peru' => 'cd853f', + 'pink' => 'ffc0cb', + 'plum' => 'dda0dd', + 'powderblue' => 'b0e0e6', + 'rosybrown' => 'bc8f8f', + 'royalblue' => '4169e1', + 'saddlebrown' => '8b4513', + 'salmon' => 'fa8072', + 'sandybrown' => 'f4a460', + 'seagreen' => '2e8b57', + 'seashell' => 'fff5ee', + 'sienna' => 'a0522d', + 'skyblue' => '87ceeb', + 'slateblue' => '6a5acd', + 'slategray' => '708090', + 'slategrey' => '708090', + 'snow' => 'fffafa', + 'springgreen' => '00ff7f', + 'steelblue' => '4682b4', + 'tan' => 'd2b48c', + 'thistle' => 'd8bfd8', + 'tomato' => 'ff6347', + 'turquoise' => '40e0d0', + 'violet' => 'ee82ee', + 'wheat' => 'f5deb3', + 'whitesmoke' => 'f5f5f5', + 'yellowgreen' => '9acd32', + 'rebeccapurple' => '663399', + ); + + protected static function mapBorderColor(DOMNode $node, &$styles, string $cssColors) { - $numColors = substr_count($cssBorderColor, '#'); - if ($numColors === 1) { - $styles['borderColor'] = trim($cssBorderColor, '#'); - } elseif ($numColors > 1) { - $colors = explode(' ', $cssBorderColor); - $borders = array('borderTopColor', 'borderRightColor', 'borderBottomColor', 'borderLeftColor'); - for ($i = 0; $i < min(4, $numColors, count($colors)); $i++) { - $styles[$borders[$i]] = trim($colors[$i], '#'); + $cssColors = self::expandBorderSides($node, $cssColors); + + foreach ($cssColors as $side => $cssColor) { + $existingBorder = self::getBorderSide($styles, $side); + if (array_key_exists($cssColor, self::$colorKeywords)) { + $cssColor = self::$colorKeywords[$cssColor]; } + $existingBorder->setColor(BasicColor::fromMixed(ltrim($cssColor, '#'))); + } + } + + protected static function mapBorderWidth(DOMNode $node, &$styles, string $cssSizes) + { + $cssSizes = self::expandBorderSides($node, $cssSizes); + + foreach ($cssSizes as $side => $cssSize) { + $existingBorder = self::getBorderSide($styles, $side); + $existingBorder->setSize(self::cssToAbsolute($cssSize)); + } + } + + protected static function expandBorderSides(DOMNode $node, string $valuesString) + { + $sideMapping = array( + 'table' => array('top', 'end', 'bottom', 'start'), + 'th' => array('top', 'end', 'bottom', 'start'), + 'td' => array('top', 'end', 'bottom', 'start'), + 'p' => array('top', 'right', 'bottom', 'left'), + ); + if (!array_key_exists($node->nodeName, $sideMapping)) { + trigger_error(sprintf('Node `%s` does not support borders', $node->nodeName), E_USER_WARNING); + } + + $sides = $sideMapping[$node->nodeName]; + $values = explode(' ', $valuesString); + if (count($values) > count($sides)) { + trigger_error(sprintf('Provided `%s` style `%s` had more than %d values', $node->nodeName, $valuesString, count($sides)), E_USER_WARNING); + $values = array_slice($values, 0, count($sides)); + } + $values = array_combine(array_slice($sides, 0, count($values)), $values); + if ($values === false) { + throw new Exception(sprintf('Mismatch between number of items in `$sides` and `$values`. This should never happen. Provided values: `%s`, Sides: `%s`, Values: `%s`', $valuesString, serialize($sides), serialize($values))); + } + + if (count($values) === 1) { + $values[$sides[1]] = $values[$sides[0]]; + $values[$sides[2]] = $values[$sides[0]]; + $values[$sides[3]] = $values[$sides[0]]; + } elseif (count($values) === 2) { + $values[$sides[2]] = $values[$sides[0]]; + $values[$sides[3]] = $values[$sides[1]]; + } elseif (count($values) === 3) { + $values[$sides[3]] = $values[$sides[1]]; } + + return $values; + } + + protected static function getBorderSide(&$styles, string $side): BorderSide + { + if (($styles['bordersFromArray'][$side] ?? null) === null) { + $styles['bordersFromArray'][$side] = new BorderSide(); + } + + return $styles['bordersFromArray'][$side]; } /** @@ -808,4 +1056,56 @@ protected static function parseLink($node, $element, &$styles) return $element->addLink($target, $node->textContent, $styles['font'], $styles['paragraph']); } + + /** + * Transforms a size in CSS format (eg. 10px, 10cm, ...) to Length + */ + public static function cssToLength(string $value): Length + { + if ($value === '0') { + return new Absolute(0); + } + + $matches = array(); + if (preg_match('/^[+-]?([0-9]+\.?[0-9]*)?(px|em|ex|%|in|cm|mm|pt|pc)$/i', $value, $matches)) { + $size = (float) $matches[1]; + $unit = $matches[2]; + + switch ($unit) { + case 'pt': + return Absolute::from('pt', $size); + case 'px': + return Absolute::fromPixels(new Dpi(), $size); + case 'cm': + return Absolute::from('cm', $size); + case 'mm': + return Absolute::from('mm', $size); + case 'in': + return Absolute::from('in', $size); + case 'pc': + return Absolute::from('pc', $size); + case '%': + return new Percent($size); + case 'em': + return new Percent($size * 100); + } + } + + return new Absolute(null); + } + + /** + * Transforms a size in CSS format (eg. 10px, 10cm, ...) to Absolute + * + * @return Absolute + */ + public static function cssToAbsolute(string $value): Length + { + $size = self::cssToLength($value); + if ($size instanceof Absolute) { + return $size; + } + + return new Absolute(null); + } } diff --git a/src/PhpWord/Shared/HtmlDpi.php b/src/PhpWord/Shared/HtmlDpi.php new file mode 100644 index 0000000000..87a8a11236 --- /dev/null +++ b/src/PhpWord/Shared/HtmlDpi.php @@ -0,0 +1,17 @@ +filename = $filename; @@ -133,7 +130,11 @@ public function open($filename, $flags = null) if (!$this->usePclzip) { $zip = new \ZipArchive(); - $result = $zip->open($this->filename, $flags); + if ($flags === null) { + $result = $zip->open($this->filename); + } else { + $result = $zip->open($this->filename, $flags); + } // Scrutizer will report the property numFiles does not exist // See https://github.com/scrutinizer-ci/php-analyzer/issues/190 diff --git a/src/PhpWord/SimpleType/DocProtect.php b/src/PhpWord/SimpleType/DocProtect.php index e386913d53..39bedda53e 100644 --- a/src/PhpWord/SimpleType/DocProtect.php +++ b/src/PhpWord/SimpleType/DocProtect.php @@ -1,4 +1,5 @@ getBorderTopSize(), - $this->getBorderLeftSize(), - $this->getBorderRightSize(), - $this->getBorderBottomSize(), - ); - } - - /** - * Set border size - * - * @param int|float $value - * @return self - */ - public function setBorderSize($value = null) - { - $this->setBorderTopSize($value); - $this->setBorderLeftSize($value); - $this->setBorderRightSize($value); - $this->setBorderBottomSize($value); +use PhpOffice\PhpWord\Exception\Exception; - return $this; - } +trait Border +{ + protected $borders = array(); - /** - * Get border color - * - * @return string[] - */ - public function getBorderColor() + protected function getAllowedSides(): array { return array( - $this->getBorderTopColor(), - $this->getBorderLeftColor(), - $this->getBorderRightColor(), - $this->getBorderBottomColor(), + 'top', + 'bottom', + 'left', + 'right', ); } /** - * Set border color + * Get all borders * - * @param string $value - * @return self + * @return BorderSide[] */ - public function setBorderColor($value = null) + public function getBorders(): array { - $this->setBorderTopColor($value); - $this->setBorderLeftColor($value); - $this->setBorderRightColor($value); - $this->setBorderBottomColor($value); + if (empty($this->borders)) { + $this->setBorders(new BorderSide()); + } - return $this; + return $this->borders; } /** - * Get border style - * - * @return string[] + * Get specific border */ - public function getBorderStyle() + public function getBorder(string $side): BorderSide { - return array( - $this->getBorderTopStyle(), - $this->getBorderLeftStyle(), - $this->getBorderRightStyle(), - $this->getBorderBottomStyle(), - ); - } - - /** - * Set border style - * - * @param string $value - * @return self - */ - public function setBorderStyle($value = null) - { - $this->setBorderTopStyle($value); - $this->setBorderLeftStyle($value); - $this->setBorderRightStyle($value); - $this->setBorderBottomStyle($value); + if (!in_array($side, $this->getAllowedSides())) { + throw new Exception(sprintf('Invalid side `%s` provided', $side)); + } - return $this; + return $this->getBorders()[$side]; } /** - * Get border top size - * - * @return int|float + * Set same border for all sides */ - public function getBorderTopSize() + public function setBorders(BorderSide $borderSide): self { - return $this->borderTopSize; - } - - /** - * Set border top size - * - * @param int|float $value - * @return self - */ - public function setBorderTopSize($value = null) - { - $this->borderTopSize = $this->setNumericVal($value, $this->borderTopSize); - - return $this; - } - - /** - * Get border top color - * - * @return string - */ - public function getBorderTopColor() - { - return $this->borderTopColor; - } - - /** - * Set border top color - * - * @param string $value - * @return self - */ - public function setBorderTopColor($value = null) - { - $this->borderTopColor = $value; + foreach ($this->getAllowedSides() as $side) { + $this->borders[$side] = clone $borderSide; + } return $this; } /** - * Get border top style - * - * @return string - */ - public function getBorderTopStyle() - { - return $this->borderTopStyle; - } - - /** - * Set border top Style - * - * @param string $value - * @return self + * Set same border for all sides */ - public function setBorderTopStyle($value = null) + public function setBordersFromArray(array $borders): self { - $this->borderTopStyle = $value; + foreach ($borders as $side => $border) { + $this->setBorder($side, $border); + } return $this; } /** - * Get border left size - * - * @return int|float + * Set same border for all sides */ - public function getBorderLeftSize() + public function setBorder(string $side, BorderSide $borderSide): self { - return $this->borderLeftSize; - } + if (!in_array($side, $this->getAllowedSides())) { + throw new Exception(sprintf('Invalid side `%s` provided', $side)); + } elseif (empty($this->borders)) { + $this->setBorders(new BorderSide()); + } - /** - * Set border left size - * - * @param int|float $value - * @return self - */ - public function setBorderLeftSize($value = null) - { - $this->borderLeftSize = $this->setNumericVal($value, $this->borderLeftSize); + $this->borders[$side] = clone $borderSide; return $this; } /** - * Get border left color - * - * @return string - */ - public function getBorderLeftColor() - { - return $this->borderLeftColor; - } - - /** - * Set border left color - * - * @param string $value - * @return self - */ - public function setBorderLeftColor($value = null) - { - $this->borderLeftColor = $value; - - return $this; - } - - /** - * Get border left style - * - * @return string - */ - public function getBorderLeftStyle() - { - return $this->borderLeftStyle; - } - - /** - * Set border left style - * - * @param string $value - * @return self - */ - public function setBorderLeftStyle($value = null) - { - $this->borderLeftStyle = $value; - - return $this; - } - - /** - * Get border right size - * - * @return int|float - */ - public function getBorderRightSize() - { - return $this->borderRightSize; - } - - /** - * Set border right size - * - * @param int|float $value - * @return self - */ - public function setBorderRightSize($value = null) - { - $this->borderRightSize = $this->setNumericVal($value, $this->borderRightSize); - - return $this; - } - - /** - * Get border right color - * - * @return string - */ - public function getBorderRightColor() - { - return $this->borderRightColor; - } - - /** - * Set border right color - * - * @param string $value - * @return self - */ - public function setBorderRightColor($value = null) - { - $this->borderRightColor = $value; - - return $this; - } - - /** - * Get border right style - * - * @return string - */ - public function getBorderRightStyle() - { - return $this->borderRightStyle; - } - - /** - * Set border right style - * - * @param string $value - * @return self - */ - public function setBorderRightStyle($value = null) - { - $this->borderRightStyle = $value; - - return $this; - } - - /** - * Get border bottom size - * - * @return int|float - */ - public function getBorderBottomSize() - { - return $this->borderBottomSize; - } - - /** - * Set border bottom size - * - * @param int|float $value - * @return self - */ - public function setBorderBottomSize($value = null) - { - $this->borderBottomSize = $this->setNumericVal($value, $this->borderBottomSize); - - return $this; - } - - /** - * Get border bottom color - * - * @return string - */ - public function getBorderBottomColor() - { - return $this->borderBottomColor; - } - - /** - * Set border bottom color - * - * @param string $value - * @return self - */ - public function setBorderBottomColor($value = null) - { - $this->borderBottomColor = $value; - - return $this; - } - - /** - * Get border bottom style - * - * @return string - */ - public function getBorderBottomStyle() - { - return $this->borderBottomStyle; - } - - /** - * Set border bottom style - * - * @param string $value - * @return self - */ - public function setBorderBottomStyle($value = null) - { - $this->borderBottomStyle = $value; - - return $this; - } - - /** - * Check if any of the border is not null - * - * @return bool + * Check if any borders have been added */ - public function hasBorder() + public function hasBorder(): bool { - $borders = $this->getBorderSize(); + foreach ($this->borders as $border) { + if (($border->getSize()->toFloat('twip') ?? 0) !== 0) { + return true; + } + } - return $borders !== array_filter($borders, 'is_null'); + return false; } } diff --git a/src/PhpWord/Style/BorderSide.php b/src/PhpWord/Style/BorderSide.php new file mode 100644 index 0000000000..37d7565cf1 --- /dev/null +++ b/src/PhpWord/Style/BorderSide.php @@ -0,0 +1,162 @@ +setSize($size ?? new Absolute(0)); + $this->setColor($color ?? new Hex(null)); + $this->setStyle($style ?? new BorderStyle('single')); + $this->setSpace($space ?? new Absolute(0)); + $this->setShadow($shadow); + } + + /** + * Get border size + */ + public function getSize(): Absolute + { + return $this->size; + } + + /** + * Set border size + */ + public function setSize(Absolute $size): self + { + if (!$size->isSpecified()) { + throw new Exception('Size must be specified'); + } + + $this->size = $size; + + return $this; + } + + /** + * Get border color + */ + public function getColor(): BasicColor + { + return $this->color; + } + + /** + * Set border color + */ + public function setColor(BasicColor $color): self + { + $this->color = $color; + + return $this; + } + + /** + * Get border style + */ + public function getStyle(): BorderStyle + { + return $this->style; + } + + /** + * Set border style + */ + public function setStyle(BorderStyle $style): self + { + $this->style = $style; + + return $this; + } + + /** + * Get spacing offset + */ + public function getSpace(): Absolute + { + return $this->space; + } + + /** + * Set spacing offset + */ + public function setSpace(Absolute $space): self + { + if (!$space->isSpecified()) { + throw new Exception('Space must be specified'); + } + + $this->space = $space; + + return $this; + } + + /** + * Get whether or not the border should have a "shadow" + */ + public function getShadow(): bool + { + return $this->shadow; + } + + /** + * Set whether or not the border should have a "shadow" + */ + public function setShadow(bool $shadow): self + { + $this->shadow = $shadow; + + return $this; + } +} diff --git a/src/PhpWord/Style/BorderStyle.php b/src/PhpWord/Style/BorderStyle.php new file mode 100644 index 0000000000..d99d861887 --- /dev/null +++ b/src/PhpWord/Style/BorderStyle.php @@ -0,0 +1,75 @@ + true, + 'dashDotStroked' => true, + 'dashed' => true, + 'dashSmallGap' => true, + 'dotDash' => true, + 'dotDotDash' => true, + 'dotted' => true, + 'double' => true, + 'doubleWave' => true, + 'inset' => true, + 'nil' => true, + 'none' => true, + 'outset' => true, + 'thick' => true, + 'thickThinLargeGap' => true, + 'thickThinMediumGap' => true, + 'thickThinSmallGap' => true, + 'thinThickLargeGap' => true, + 'thinThickMediumGap' => true, + 'thinThickSmallGap' => true, + 'thinThickThinLargeGap' => true, + 'thinThickThinMediumGap' => true, + 'thinThickThinSmallGap' => true, + 'threeDEmboss' => true, + 'threeDEngrave' => true, + 'triple' => true, + 'wave' => true, + ); + + private $style; + + public function __construct(string $style) + { + if (!static::isValid($style)) { + throw new Exception(sprintf("Provided border style must be valid. '%s' provided. Allowed: '%s'", $style, implode('\', \'', array_keys(self::$allowedStyles)))); + } + + $this->style = $style; + } + + public function getStyle() + { + return $this->style; + } + + public static function isValid(string $style): bool + { + return array_key_exists($style, self::$allowedStyles); + } + + public static function fromMixed($style = null): self + { + if (!static::isValid($style)) { + trigger_error(sprintf('Border style `%s` is not a valid option', $style), E_USER_WARNING); + + return new self('single'); + } + + return new self($style); + } +} diff --git a/src/PhpWord/Style/Cell.php b/src/PhpWord/Style/Cell.php index 1276b5b59a..ba98a8fc26 100644 --- a/src/PhpWord/Style/Cell.php +++ b/src/PhpWord/Style/Cell.php @@ -1,4 +1,5 @@ shading !== null) { - return $this->shading->getFill(); + if ($this->shading === null) { + $this->setBgColor(new Hex(null)); } - return null; + return $this->shading->getFill(); } /** * Set background * * @param string $value - * @return self */ - public function setBgColor($value = null) + public function setBgColor(BasicColor $value): self { return $this->setShading(array('fill' => $value)); } @@ -270,11 +273,9 @@ public function getShading() /** * Set shading - * - * @param mixed $value - * @return self + * @param null|mixed $value */ - public function setShading($value = null) + public function setShading($value = null): self { $this->setObjectVal($value, 'Shading', $this->shading); @@ -283,58 +284,23 @@ public function setShading($value = null) /** * Get cell width - * - * @return int */ - public function getWidth() + public function getWidth(): Length { + if ($this->width === null) { + $this->setWidth(new Absolute(null)); + } + return $this->width; } /** * Set cell width - * - * @param int $value - * @return self */ - public function setWidth($value) + public function setWidth(Length $value): self { - $this->setIntVal($value); + $this->width = $value; return $this; } - - /** - * Get width unit - * - * @return string - */ - public function getUnit() - { - return $this->unit; - } - - /** - * Set width unit - * - * @param string $value - */ - public function setUnit($value) - { - $this->unit = $this->setEnumVal($value, array(TblWidth::AUTO, TblWidth::PERCENT, TblWidth::TWIP), TblWidth::TWIP); - - return $this; - } - - /** - * Get default border color - * - * @deprecated 0.10.0 - * - * @codeCoverageIgnore - */ - public function getDefaultBorderColor() - { - return self::DEFAULT_BORDER_COLOR; - } } diff --git a/src/PhpWord/Style/Chart.php b/src/PhpWord/Style/Chart.php index 06b4829c3c..1eaeef79c6 100644 --- a/src/PhpWord/Style/Chart.php +++ b/src/PhpWord/Style/Chart.php @@ -1,4 +1,5 @@ width)) { + $this->width = Absolute::from('emu', $this->width); + } + return $this->width; } /** * Set width - * - * @param int $value - * @return self */ - public function setWidth($value = null) + public function setWidth(Absolute $value): self { - $this->width = $this->setIntVal($value, $this->width); + $this->width = $value; return $this; } /** * Get height - * - * @return int */ - public function getHeight() + public function getHeight(): Absolute { + if (is_int($this->height)) { + $this->height = Absolute::from('emu', $this->height); + } + return $this->height; } /** * Set height - * - * @param int $value - * @return self */ - public function setHeight($value = null) + public function setHeight(Absolute $value): self { - $this->height = $this->setIntVal($value, $this->height); + $this->height = $value; return $this; } @@ -373,7 +374,6 @@ public function getCategoryLabelPosition() * "low" - labels on the left side of the graph * "high" - labels on the right side of the graph * - * @param mixed $labelPosition * @return self */ public function setCategoryLabelPosition($labelPosition) @@ -402,7 +402,6 @@ public function getValueLabelPosition() * "high" - sets labels above the graph * * @param string - * @param mixed $labelPosition */ public function setValueLabelPosition($labelPosition) { diff --git a/src/PhpWord/Style/Colors/BasicColor.php b/src/PhpWord/Style/Colors/BasicColor.php new file mode 100644 index 0000000000..25d06132f0 --- /dev/null +++ b/src/PhpWord/Style/Colors/BasicColor.php @@ -0,0 +1,30 @@ +hex = null; + + return; + } + + $hex = strtoupper($hex); + + if (!static::isValid($hex)) { + throw new Exception(sprintf('Hex value must match `([0-9a-f]{3}){1,2}`. `%s` provided', $hex)); + } + + if (strlen($hex) === 3) { + // If #abc format is provided, expand to #aabbcc + $this->hex = ''; + foreach (str_split($hex) as $ch) { + $this->hex .= strtoupper($ch . $ch); + } + } else { + $this->hex = strtoupper($hex); + } + } + + public function isSpecified(): bool + { + return $this->hex !== null; + } + + public function toRgb() + { + if ($this->hex === null) { + return null; + } + + $rgb = array(); + foreach (str_split($this->hex, 2) as $c) { + $rgb[] = hexdec($c); + } + + return $rgb; + } + + public function toHex(bool $includeHash = false) + { + if ($this->hex === null) { + return null; + } + + return ($includeHash ? '#' : '') . $this->hex; + } + + public static function isValid(string $hex): bool + { + return preg_match('/^(?:[0-9a-fA-F]{3}){1,2}$/', $hex) ? true : false; + } +} diff --git a/src/PhpWord/Style/Colors/HighlightColor.php b/src/PhpWord/Style/Colors/HighlightColor.php new file mode 100644 index 0000000000..91c7d19207 --- /dev/null +++ b/src/PhpWord/Style/Colors/HighlightColor.php @@ -0,0 +1,78 @@ + 'FF0000', + 'green' => '00FF00', + 'cyan' => '00FFFF', + 'magenta' => 'FF00FF', + 'blue' => '0000FF', + 'red' => 'FF0000', + 'darkBlue' => '000080', + 'darkCyan' => '008080', + 'darkGreen' => '008000', + 'darkMagenta' => '800080', + 'darkRed' => '800000', + 'darkYellow' => '808000', + 'darkGray' => '808080', + 'lightGray' => 'C0C0C0', + 'black' => '000000', + ); + + private $color; + + public function __construct(string $color = null) + { + if ($color !== null) { + if (!static::isValid($color)) { + throw new Exception(sprintf("Provided color must be a valid highlight color. '%s' provided. Allowed: %s", $color, implode(', ', array_keys(self::$allowedColors)))); + } + } + + $this->color = $color; + } + + public function isSpecified(): bool + { + return $this->color !== null; + } + + public function getName() + { + return $this->color; + } + + public function toRgb() + { + if ($this->color === null) { + return null; + } + + $rgb = array(); + foreach (str_split(self::$allowedColors[$this->color], 2) as $c) { + $rgb[] = hexdec($c); + } + + return $rgb; + } + + public function toHex(bool $includeHash = false) + { + if ($this->color === null) { + return null; + } + + return ($includeHash ? '#' : '') . self::$allowedColors[$this->color]; + } + + public static function isValid(string $color): bool + { + return array_key_exists($color, self::$allowedColors); + } +} diff --git a/src/PhpWord/Style/Colors/NamedColorInterface.php b/src/PhpWord/Style/Colors/NamedColorInterface.php new file mode 100644 index 0000000000..279c71da5e --- /dev/null +++ b/src/PhpWord/Style/Colors/NamedColorInterface.php @@ -0,0 +1,12 @@ + 255 || $green < 0 || $green > 255 || $blue < 0 || $blue > 255) { + throw new Exception(sprintf('Provided values must be 0–255. Provided `Rgb(%s, %s, %s)`', $red, $green, $blue)); + } + + $this->red = $red; + $this->green = $green; + $this->blue = $blue; + } + + public function isSpecified(): bool + { + return true; + } + + public function toRgb(): array + { + return array($this->red, $this->green, $this->blue); + } + + public function toHex(bool $includeHash = false): string + { + return sprintf('%s%02X%02X%02X', ($includeHash ? '#' : ''), $this->red, $this->green, $this->blue); + } +} diff --git a/src/PhpWord/Style/Colors/SpecialColor.php b/src/PhpWord/Style/Colors/SpecialColor.php new file mode 100644 index 0000000000..0692e96240 --- /dev/null +++ b/src/PhpWord/Style/Colors/SpecialColor.php @@ -0,0 +1,26 @@ +getName(); + } + if ($this instanceof StaticColorInterface) { + return $this->toHex($includeHash); + } + throw new Exception(sprintf('All colors must implement NamedColorInterface or StaticColorInterface. \'%s\' does not implement either.', get_class($this))); + } + + abstract public function isSpecified(): bool; +} diff --git a/src/PhpWord/Style/Colors/StaticColorInterface.php b/src/PhpWord/Style/Colors/StaticColorInterface.php new file mode 100644 index 0000000000..1ba0ccb0f6 --- /dev/null +++ b/src/PhpWord/Style/Colors/StaticColorInterface.php @@ -0,0 +1,11 @@ + true, + 'background' => true, + 'activeCaption' => true, + 'inactiveCaption' => true, + 'menu' => true, + 'window' => true, + 'windowFrame' => true, + 'menuText' => true, + 'windowText' => true, + 'captionText' => true, + 'activeBorder' => true, + 'inactiveBorder' => true, + 'appWorkspace' => true, + 'highlight' => true, + 'highlightText' => true, + 'btnFace' => true, + 'btnShadow' => true, + 'grayText' => true, + 'btnText' => true, + 'inactiveCaptionText' => true, + 'btnHighlight' => true, + '3dDkShadow' => true, + '3dLight' => true, + 'infoText' => true, + 'infoBk' => true, + 'hotLight' => true, + 'gradientActiveCaption' => true, + 'gradientInactiveCaption' => true, + 'menuHighlight' => true, + 'menuBar' => true, + ); + + private $name; + private $lastColor; + + public function __construct(string $name, StaticColorInterface $lastColor) + { + if (!static::isValid($name)) { + throw new Exception(sprintf("Provided system color must be a valid system color. '%s' provided. Allowed: %s", $name, implode(', ', array_keys(self::$allowedColors)))); + } + + $this->name = $name; + $this->lastColor = clone $lastColor; + } + + public function isSpecified(): bool + { + return true; + } + + public function getName(): string + { + return $this->name; + } + + public function getLastColor(): StaticColorInterface + { + return clone $this->lastColor; + } + + public static function isValid(string $color): bool + { + return array_key_exists($color, self::$allowedColors); + } +} diff --git a/src/PhpWord/Style/Colors/ThemeColor.php b/src/PhpWord/Style/Colors/ThemeColor.php new file mode 100644 index 0000000000..9f6222ff4a --- /dev/null +++ b/src/PhpWord/Style/Colors/ThemeColor.php @@ -0,0 +1,57 @@ + true, + 'dk2' => true, + 'lt1' => true, + 'lt2' => true, + 'accent1' => true, + 'accent2' => true, + 'accent3' => true, + 'accent4' => true, + 'accent5' => true, + 'accent6' => true, + 'hlink' => true, + 'folHlink' => true, + ); + + private $name; + + public function __construct(string $name) + { + if (!static::isValid($name)) { + throw new Exception(sprintf("Provided color must be a valid theme color. '%s' provided. Allowed: %s", $name, implode(', ', array_keys(self::$allowedColors)))); + } + + $this->name = $name; + } + + public function isSpecified(): bool + { + return true; + } + + public function getName(): string + { + return $this->name; + } + + public static function isValid(string $color): bool + { + return array_key_exists($color, self::$allowedColors); + } +} diff --git a/src/PhpWord/Style/Extrusion.php b/src/PhpWord/Style/Extrusion.php index 4c860bcd63..e27fa41afa 100644 --- a/src/PhpWord/Style/Extrusion.php +++ b/src/PhpWord/Style/Extrusion.php @@ -1,4 +1,5 @@ size === null) { + $this->size = new Absolute(null); + } + return $this->size; } /** * Set font size - * - * @param int|float $value - * @return self */ - public function setSize($value = null) + public function setSize(Absolute $value): self { - $this->size = $this->setNumericVal($value, $this->size); + $this->size = $value; return $this; } /** * Get font color - * - * @return string */ - public function getColor() + public function getColor(): BasicColor { + if ($this->color === null) { + $this->color = new Hex(null); + } + return $this->color; } /** * Set font color - * - * @param string $value - * @return self */ - public function setColor($value = null) + public function setColor(BasicColor $value): self { $this->color = $value; @@ -623,21 +610,22 @@ public function setAllCaps($value = true) /** * Get foreground/highlight color - * - * @return string */ - public function getFgColor() + public function getFgColor(): HighlightColor { + if ($this->fgColor === null) { + $this->fgColor = new HighlightColor(null); + } + return $this->fgColor; } /** * Set foreground/highlight color * - * @param string $value * @return self */ - public function setFgColor($value = null) + public function setFgColor(HighlightColor $value) { $this->fgColor = $value; @@ -651,85 +639,96 @@ public function setFgColor($value = null) */ public function getBgColor() { - return $this->getChildStyleValue($this->shading, 'fill'); + $value = $this->getChildStyleValue($this->shading, 'fill'); + + if ($value === null) { + $this->setShading(array()); + $value = $this->getChildStyleValue($this->shading, 'fill'); + } + + return $value; } /** * Set background * - * @param string $value - * @return \PhpOffice\PhpWord\Style\Table + * @param BasicColor $value */ - public function setBgColor($value = null) + public function setBgColor(BasicColor $value = null): self { $this->setShading(array('fill' => $value)); + + return $this; } /** - * Get scale - * - * @return int + * Get horizontal scaling of characters + * @see http://www.officeopenxml.com/WPtextSpacing.php */ - public function getScale() + public function getScale(): Percent { + if ($this->scale === null) { + $this->scale = new Percent(null); + } + return $this->scale; } /** - * Set scale - * - * @param int $value - * @return self + * Set horizontal scaling of characters + * @see http://www.officeopenxml.com/WPtextSpacing.php */ - public function setScale($value = null) + public function setScale(Percent $value): self { - $this->scale = $this->setIntVal($value, null); + $this->scale = $value; return $this; } /** * Get font spacing - * - * @return int|float */ - public function getSpacing() + public function getSpacing(): Absolute { + if ($this->spacing === null) { + $this->spacing = new Absolute(null); + } + return $this->spacing; } /** * Set font spacing - * - * @param int|float $value - * @return self */ - public function setSpacing($value = null) + public function setSpacing(Absolute $value): self { - $this->spacing = $this->setNumericVal($value, null); + $this->spacing = $value; return $this; } /** - * Get font kerning - * - * @return int|float + * Get the smallest font size + * which should have kerning automatically adjusted. + * @see http://www.officeopenxml.com/WPtextSpacing.php */ - public function getKerning() + public function getKerning(): Absolute { + if ($this->kerning === null) { + $this->kerning = new Absolute(null); + } + return $this->kerning; } /** - * Set font kerning - * - * @param int|float $value - * @return self + * Set the smallest font size + * which should have kerning automatically adjusted. + * @see http://www.officeopenxml.com/WPtextSpacing.php */ - public function setKerning($value = null) + public function setKerning(Absolute $value): self { - $this->kerning = $this->setNumericVal($value, null); + $this->kerning = $value; return $this; } @@ -759,21 +758,16 @@ public function setNoProof($value = false) /** * Get line height - * - * @return int|float */ - public function getLineHeight() + public function getLineHeight(): Percent { return $this->getParagraph()->getLineHeight(); } /** - * Set lineheight - * - * @param int|float|string $value - * @return self + * Set line height */ - public function setLineHeight($value) + public function setLineHeight(Percent $value): self { $this->setParagraph(array('lineHeight' => $value)); @@ -793,7 +787,7 @@ public function getParagraph() /** * Set Paragraph * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setParagraph($value = null) @@ -833,13 +827,17 @@ public function setRTL($value = true) */ public function getShading() { + if ($this->shading === null) { + $this->setShading(array()); + } + return $this->shading; } /** * Set shading * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setShading($value = null) @@ -862,7 +860,7 @@ public function getLang() /** * Set language * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setLang($value = null) @@ -972,23 +970,22 @@ public function setHidden($value = true) /** * Get position - * - * @return int */ - public function getPosition() + public function getPosition(): Absolute { + if ($this->position === null) { + $this->position = new Absolute(null); + } + return $this->position; } /** * Set position - * - * @param int $value - * @return self */ - public function setPosition($value = null) + public function setPosition(Absolute $value): self { - $this->position = $this->setIntVal($value, null); + $this->position = $value; return $this; } diff --git a/src/PhpWord/Style/Frame.php b/src/PhpWord/Style/Frame.php index e87b7a803a..6086061a25 100644 --- a/src/PhpWord/Style/Frame.php +++ b/src/PhpWord/Style/Frame.php @@ -1,4 +1,5 @@ setAlignment($value); } - /** - * Get unit - * - * @return string - */ - public function getUnit() - { - return $this->unit; - } - - /** - * Set unit - * - * @param string $value - * @return self - */ - public function setUnit($value) - { - $this->unit = $value; - - return $this; - } - /** * Get width - * - * @return int|float */ - public function getWidth() + public function getWidth(): Absolute { + if ($this->width === null) { + $this->width = new Absolute(null); + } + return $this->width; } /** * Set width - * - * @param int|float $value - * @return self */ - public function setWidth($value = null) + public function setWidth(Absolute $value): self { - $this->width = $this->setNumericVal($value, null); + $this->width = $value; return $this; } /** * Get height - * - * @return int|float */ - public function getHeight() + public function getHeight(): Absolute { + if ($this->height === null) { + $this->height = new Absolute(null); + } + return $this->height; } /** * Set height - * - * @param int|float $value - * @return self */ - public function setHeight($value = null) + public function setHeight(Absolute $value): self { - $this->height = $this->setNumericVal($value, null); + $this->height = $value; return $this; } /** * Get left - * - * @return int|float */ - public function getLeft() + public function getLeft(): Absolute { + if (!$this->left instanceof Absolute) { + $this->left = new Absolute($this->left); + } + return $this->left; } /** * Set left - * - * @param int|float $value - * @return self */ - public function setLeft($value = 0) + public function setLeft(Absolute $value): self { - $this->left = $this->setNumericVal($value, 0); + $this->left = $value; return $this; } /** * Get topmost position - * - * @return int|float */ - public function getTop() + public function getTop(): Absolute { + if (!$this->top instanceof Absolute) { + $this->top = new Absolute($this->top); + } + return $this->top; } /** * Set topmost position - * - * @param int|float $value - * @return self */ - public function setTop($value = 0) + public function setTop(Absolute $value): self { - $this->top = $this->setNumericVal($value, 0); + $this->top = $value; return $this; } @@ -577,115 +537,112 @@ public function setWrap($value) /** * Get top distance from text wrap - * - * @return float */ - public function getWrapDistanceTop() + public function getWrapDistanceTop(): Absolute { + if ($this->wrapDistanceTop === null) { + $this->wrapDistanceTop = new Absolute(null); + } + return $this->wrapDistanceTop; } /** * Set top distance from text wrap * - * @param int $value * @return self */ - public function setWrapDistanceTop($value = null) + public function setWrapDistanceTop(Absolute $value) { - $this->wrapDistanceTop = $this->setFloatVal($value, null); + $this->wrapDistanceTop = $value; return $this; } /** * Get bottom distance from text wrap - * - * @return float */ - public function getWrapDistanceBottom() + public function getWrapDistanceBottom(): Absolute { + if ($this->wrapDistanceBottom === null) { + $this->wrapDistanceBottom = new Absolute(null); + } + return $this->wrapDistanceBottom; } /** * Set bottom distance from text wrap - * - * @param float $value - * @return self */ - public function setWrapDistanceBottom($value = null) + public function setWrapDistanceBottom(Absolute $value): self { - $this->wrapDistanceBottom = $this->setFloatVal($value, null); + $this->wrapDistanceBottom = $value; return $this; } /** * Get left distance from text wrap - * - * @return float */ - public function getWrapDistanceLeft() + public function getWrapDistanceLeft(): Absolute { + if ($this->wrapDistanceLeft === null) { + $this->wrapDistanceLeft = new Absolute(null); + } + return $this->wrapDistanceLeft; } /** * Set left distance from text wrap - * - * @param float $value - * @return self */ - public function setWrapDistanceLeft($value = null) + public function setWrapDistanceLeft(Absolute $value): self { - $this->wrapDistanceLeft = $this->setFloatVal($value, null); + $this->wrapDistanceLeft = $value; return $this; } /** * Get right distance from text wrap - * - * @return float */ - public function getWrapDistanceRight() + public function getWrapDistanceRight(): Absolute { + if ($this->wrapDistanceRight === null) { + $this->wrapDistanceRight = new Absolute(null); + } + return $this->wrapDistanceRight; } /** * Set right distance from text wrap - * - * @param float $value - * @return self */ - public function setWrapDistanceRight($value = null) + public function setWrapDistanceRight(Absolute $value): self { - $this->wrapDistanceRight = $this->setFloatVal($value, null); + $this->wrapDistanceRight = $value; return $this; } /** * Get position - * - * @return int */ - public function getPosition() + public function getPosition(): Absolute { + if ($this->position === null) { + $this->position = new Absolute(null); + } + return $this->position; } /** * Set position - * - * @param int $value - * @return self */ - public function setPosition($value = null) + public function setPosition(Absolute $value): self { - $this->position = $this->setIntVal($value, null); + $this->position = $value; return $this; } diff --git a/src/PhpWord/Style/Image.php b/src/PhpWord/Style/Image.php index 70aafe12cd..3348cf15d5 100644 --- a/src/PhpWord/Style/Image.php +++ b/src/PhpWord/Style/Image.php @@ -1,4 +1,5 @@ setUnit(self::UNIT_PT); // Backward compatibility setting // @todo Remove on 1.0.0 @@ -74,22 +76,16 @@ public function __construct() /** * Get margin top - * - * @return int|float */ - public function getMarginTop() + public function getMarginTop(): Absolute { return $this->getTop(); } /** * Set margin top - * - * @ignoreScrutinizerPatch - * @param int|float $value - * @return self */ - public function setMarginTop($value = 0) + public function setMarginTop(Absolute $value): self { $this->setTop($value); @@ -98,22 +94,16 @@ public function setMarginTop($value = 0) /** * Get margin left - * - * @return int|float */ - public function getMarginLeft() + public function getMarginLeft(): Absolute { return $this->getLeft(); } /** * Set margin left - * - * @ignoreScrutinizerPatch - * @param int|float $value - * @return self */ - public function setMarginLeft($value = 0) + public function setMarginLeft(Absolute $value): self { $this->setLeft($value); diff --git a/src/PhpWord/Style/Indentation.php b/src/PhpWord/Style/Indentation.php index e422395c80..2593f32b91 100644 --- a/src/PhpWord/Style/Indentation.php +++ b/src/PhpWord/Style/Indentation.php @@ -1,4 +1,5 @@ left instanceof Absolute) { + $this->left = new Absolute($this->left); + } + return $this->left; } /** * Set left - * - * @param int|float $value - * @return self */ - public function setLeft($value = null) + public function setLeft(Absolute $value): self { - $this->left = $this->setNumericVal($value, $this->left); + $this->left = $value; return $this; } /** * Get right - * - * @return int|float */ - public function getRight() + public function getRight(): Absolute { + if (!$this->right instanceof Absolute) { + $this->right = new Absolute($this->right); + } + return $this->right; } /** * Set right - * - * @param int|float $value - * @return self */ - public function setRight($value = null) + public function setRight(Absolute $value): self { - $this->right = $this->setNumericVal($value, $this->right); + $this->right = $value; return $this; } /** * Get first line - * - * @return int|float */ - public function getFirstLine() + public function getFirstLine(): Absolute { + if ($this->firstLine === null) { + $this->firstLine = new Absolute(null); + } + return $this->firstLine; } /** * Set first line - * - * @param int|float $value - * @return self */ - public function setFirstLine($value = null) + public function setFirstLine(Absolute $value): self { - $this->firstLine = $this->setNumericVal($value, $this->firstLine); + $this->firstLine = $value; return $this; } /** * Get hanging - * - * @return int|float */ - public function getHanging() + public function getHanging(): Absolute { + if ($this->hanging === null) { + $this->hanging = new Absolute(null); + } + return $this->hanging; } /** * Set hanging - * - * @param int|float $value - * @return self */ - public function setHanging($value = null) + public function setHanging(Absolute $value): self { - $this->hanging = $this->setNumericVal($value, $this->hanging); + $this->hanging = $value; return $this; } diff --git a/src/PhpWord/Style/Language.php b/src/PhpWord/Style/Language.php index 18ef88975f..ab758d1f07 100644 --- a/src/PhpWord/Style/Language.php +++ b/src/PhpWord/Style/Language.php @@ -1,4 +1,5 @@ eastAsia = $this->validateLocale($eastAsia); @@ -223,10 +224,9 @@ public function getBidirectional() /** * Validates that the language passed is in the format xx-xx * - * @param string $locale * @return string */ - private function validateLocale($locale) + private function validateLocale(string $locale) { if (strlen($locale) === 2) { return strtolower($locale) . '-' . strtoupper($locale); diff --git a/src/PhpWord/Style/Lengths/Absolute.php b/src/PhpWord/Style/Lengths/Absolute.php new file mode 100644 index 0000000000..a38a4005dc --- /dev/null +++ b/src/PhpWord/Style/Lengths/Absolute.php @@ -0,0 +1,127 @@ + 1, + + // 20 twips in a point + // Eop = 1/8 point + 'eop' => 2.5, + + // 20 twips in a point + 'pt' => 20, + + // 20 twips in a point + // 2 hpt (half point) in a point + 'hpt' => 10, + + // 20 twips in a point + // 12 points in a pica + 'pc' => 240, + + // 1440 twips in an inch + 'in' => 1440, + + // 1440 twips in an inch + // 2.54 centimeters in an inch + 'cm' => 1440 / 2.54, + + // 1440 twips in an inch + // 25.4 millimeters in an inch + 'mm' => 1440 / 25.4, + + // 20 twips in a point + // 12700 emus in a point + 'emu' => 20 / 12700, + ); + + public function __construct(float $twips = null) + { + $this->twips = $twips; + } + + public function isSpecified(): bool + { + return $this->twips !== null; + } + + public static function from(string $unit, float $length = null): self + { + if ($length === null) { + return new static(null); + } + + return new static($length * self::getConversion($unit)); + } + + public static function fromMixed(string $unit, $value = null): self + { + if ($value instanceof self) { + return clone $value; + } + + if (is_string($value) && (preg_match('/[^\d\.\,]/', $value) == 0)) { + $float = (float) $value; + } else { + $float = $value; + } + if (!is_numeric($float)) { + $float = null; + } + + if ($float === null) { + if ($value !== null) { + trigger_error(sprintf('Border size `%s` could not be converted to a float', $value), E_USER_WARNING); + } + + return new self(null); + } + + return self::from($unit, $float); + } + + public function toInt(string $unit) + { + $float = $this->toFloat($unit); + + return $float === null ? null : (int) round($float); + } + + public function toFloat(string $unit) + { + return $this->twips === null + ? null + : $this->twips / self::getConversion($unit); + } + + public static function fromPixels(Dpi $dpi, float $pixels) + { + return self::from('in', $pixels / $dpi->getDpi()); + } + + public function toPixels(Dpi $dpi) + { + $inches = $this->toFloat('in'); + + return $inches === null + ? null + : round($inches * $dpi->getDpi()); + } + + private static function getConversion(string $unit): float + { + if (!array_key_exists($unit, self::$conversions)) { + throw new Exception(sprintf('Cannot convert from unit `%s`', $unit)); + } + + return self::$conversions[$unit]; + } +} diff --git a/src/PhpWord/Style/Lengths/Auto.php b/src/PhpWord/Style/Lengths/Auto.php new file mode 100644 index 0000000000..f000fec8f9 --- /dev/null +++ b/src/PhpWord/Style/Lengths/Auto.php @@ -0,0 +1,12 @@ +percent = $percent; + } + + public function isSpecified(): bool + { + return $this->percent !== null; + } + + public function toInt() + { + $float = $this->toFloat(); + + return $float === null ? null : (int) round($float); + } + + public function toFloat() + { + return $this->percent; + } + + public static function fromMixed($value = null): self + { + if ($value instanceof self) { + return clone $value; + } + + // The 2006 version of the OOXML standard specified that the value was to be a decimal. When type="pct", the value was interpretted as fifths of a percent, so 4975=99.5%, and no % symbol was included in the attribute. In the 2011 version the value can be either a decimal or a percent, so a % symbol should be included when type="pct". + // @see http://officeopenxml.com/WPtableCellProperties-Width.php + $divideBy50 = true; + + if (!is_string($value)) { + $float = $value; + } elseif (preg_match('/^([0-9]+\\.?|[0-9]*\\.[0-9]+)%$/', $value)) { + $divideBy50 = false; + $float = (float) substr($value, 0, -1); + } elseif (!preg_match('/[^\d\.\,]/', $value)) { + $float = (float) $value; + } else { + $float = $value; + } + if (!is_numeric($float)) { + $float = null; + } + + if ($float === null) { + if ($value !== null) { + trigger_error(sprintf('Percent length `%s` could not be converted to a float', $value), E_USER_WARNING); + } + + return new self(null); + } + + return new self($divideBy50 ? $float / 50 : $float); + } +} diff --git a/src/PhpWord/Style/Line.php b/src/PhpWord/Style/Line.php index a9952eec0b..2ce547f720 100644 --- a/src/PhpWord/Style/Line.php +++ b/src/PhpWord/Style/Line.php @@ -1,4 +1,5 @@ weight === null) { + $this->weight = new Absolute(null); + } + return $this->weight; } /** * Set weight - * - * @param int $value Weight in points - * @return self */ - public function setWeight($value = null) + public function setWeight(Absolute $value): self { - $this->weight = $this->setNumericVal($value, $this->weight); + $this->weight = $value; return $this; } diff --git a/src/PhpWord/Style/LineNumbering.php b/src/PhpWord/Style/LineNumbering.php index 451252d87a..b152924a51 100644 --- a/src/PhpWord/Style/LineNumbering.php +++ b/src/PhpWord/Style/LineNumbering.php @@ -1,4 +1,5 @@ distance === null) { + $this->distance = new Absolute(null); + } + return $this->distance; } /** * Set distance - * - * @param int|float $value - * @return self */ - public function setDistance($value = null) + public function setDistance(Absolute $value): self { - $this->distance = $this->setNumericVal($value, $this->distance); + $this->distance = $value; return $this; } diff --git a/src/PhpWord/Style/ListItem.php b/src/PhpWord/Style/ListItem.php index 4293940fd2..098c761137 100644 --- a/src/PhpWord/Style/ListItem.php +++ b/src/PhpWord/Style/ListItem.php @@ -1,4 +1,5 @@ left === null) { + $this->left = new Absolute(null); + } + return $this->left; } /** * Set left - * - * @param int $value - * @return self + * @see http://www.officeopenxml.com/WPindentation.php */ - public function setLeft($value) + public function setLeft(Absolute $value): self { - $this->left = $this->setIntVal($value, $this->left); + $this->left = $value; return $this; } /** * Get hanging - * - * @return int + * @see http://www.officeopenxml.com/WPindentation.php */ - public function getHanging() + public function getHanging(): Absolute { + if ($this->hanging === null) { + $this->hanging = new Absolute(null); + } + return $this->hanging; } /** * Set hanging - * - * @param int $value - * @return self + * @see http://www.officeopenxml.com/WPindentation.php */ - public function setHanging($value) + public function setHanging(Absolute $value): self { - $this->hanging = $this->setIntVal($value, $this->hanging); + $this->hanging = $value; return $this; } /** * Get tab - * - * @return int */ - public function getTabPos() + public function getTabPos(): Absolute { + if ($this->tabPos === null) { + $this->tabPos = new Absolute(null); + } + return $this->tabPos; } /** * Set tab - * - * @param int $value - * @return self */ - public function setTabPos($value) + public function setTabPos(Absolute $value): self { - $this->tabPos = $this->setIntVal($value, $this->tabPos); + $this->tabPos = $value; return $this; } diff --git a/src/PhpWord/Style/Outline.php b/src/PhpWord/Style/Outline.php index a04ad974c0..b04e21076c 100644 --- a/src/PhpWord/Style/Outline.php +++ b/src/PhpWord/Style/Outline.php @@ -1,4 +1,5 @@ setStyleByArray($style); } - /** - * Get unit - * - * @return string - */ - public function getUnit() - { - return $this->unit; - } - /** * Get weight - * - * @return int|float */ - public function getWeight() + public function getWeight(): Absolute { + if ($this->weight === null) { + $this->weight = new Absolute(null); + } + return $this->weight; } /** * Set weight - * - * @param int|float $value - * @return self */ - public function setWeight($value = null) + public function setWeight(Absolute $value): self { - $this->weight = $this->setNumericVal($value, null); + $this->weight = $value; return $this; } /** * Get color - * - * @return string */ - public function getColor() + public function getColor(): BasicColor { + if ($this->color === null) { + $this->color = new Hex(null); + } + return $this->color; } /** * Set color - * - * @param string $value - * @return self */ - public function setColor($value = null) + public function setColor(BasicColor $value): self { $this->color = $value; diff --git a/src/PhpWord/Style/Paper.php b/src/PhpWord/Style/Paper.php index 3c93ed8f2f..27fee8dc33 100644 --- a/src/PhpWord/Style/Paper.php +++ b/src/PhpWord/Style/Paper.php @@ -1,4 +1,5 @@ sizes[$this->size]; - if ($unit == 'mm') { - $this->width = Converter::cmToTwip($width / 10); - $this->height = Converter::cmToTwip($height / 10); - } else { - $this->width = Converter::inchToTwip($width); - $this->height = Converter::inchToTwip($height); - } + $this->width = Absolute::from($unit, $width); + $this->height = Absolute::from($unit, $height); return $this; } /** * Get width - * - * @return float */ - public function getWidth() + public function getWidth(): Absolute { return $this->width; } /** * Get height - * - * @return float */ - public function getHeight() + public function getHeight(): Absolute { return $this->height; } diff --git a/src/PhpWord/Style/Paragraph.php b/src/PhpWord/Style/Paragraph.php index 6e9aaf15dc..1f65d2af2a 100644 --- a/src/PhpWord/Style/Paragraph.php +++ b/src/PhpWord/Style/Paragraph.php @@ -1,4 +1,5 @@ 'lineHeight', 'line-spacing' => 'spacing'); + protected $aliases = array('line-height' => 'lineHeight', 'line-spacing' => 'space'); /** * Parent style @@ -99,7 +103,7 @@ class Paragraph extends Border /** * Text line height * - * @var int + * @var Percent */ private $lineHeight; @@ -191,15 +195,11 @@ class Paragraph extends Border * Set Style value * * @param string $key - * @param mixed $value * @return self */ public function setStyleValue($key, $value) { $key = Text::removeUnderscorePrefix($key); - if ('indent' == $key || 'hanging' == $key) { - $value = $value * 720; - } return parent::setStyleValue($key, $value); } @@ -296,6 +296,17 @@ public function setAlign($value = null) return $this->setAlignment($value); } + protected function getAllowedSides(): array + { + return array( + 'top', + 'bottom', + 'left', + 'right', + 'between', + ); + } + /** * Get parent style ID * @@ -355,7 +366,7 @@ public function getIndentation() /** * Set shading * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setIndentation($value = null) @@ -367,42 +378,34 @@ public function setIndentation($value = null) /** * Get indentation - * - * @return int */ - public function getIndent() + public function getIndent(): Absolute { - return $this->getChildStyleValue($this->indentation, 'left'); + return $this->getChildStyleValue($this->indentation, 'left') ?? new Absolute(null); } /** * Set indentation * - * @param int $value * @return self */ - public function setIndent($value = null) + public function setIndent(Absolute $value) { return $this->setIndentation(array('left' => $value)); } /** * Get hanging - * - * @return int */ - public function getHanging() + public function getHanging(): Absolute { - return $this->getChildStyleValue($this->indentation, 'hanging'); + return $this->getChildStyleValue($this->indentation, 'hanging') ?? new Absolute(null); } /** * Set hanging - * - * @param int $value - * @return self */ - public function setHanging($value = null) + public function setHanging(Absolute $value): self { return $this->setIndentation(array('hanging' => $value)); } @@ -421,9 +424,9 @@ public function getSpace() /** * Set spacing * - * @param mixed $value - * @return self * @todo Rename to setSpacing in 1.0 + * @param null|mixed $value + * @return self */ public function setSpace($value = null) { @@ -434,63 +437,50 @@ public function setSpace($value = null) /** * Get space before paragraph - * - * @return int */ - public function getSpaceBefore() + public function getSpaceBefore(): Absolute { - return $this->getChildStyleValue($this->spacing, 'before'); + return $this->getChildStyleValue($this->spacing, 'before') ?? new Absolute(null); } /** * Set space before paragraph - * - * @param int $value - * @return self */ - public function setSpaceBefore($value = null) + public function setSpaceBefore(Absolute $value): self { return $this->setSpace(array('before' => $value)); } /** * Get space after paragraph - * - * @return int */ - public function getSpaceAfter() + public function getSpaceAfter(): Absolute { - return $this->getChildStyleValue($this->spacing, 'after'); + return $this->getChildStyleValue($this->spacing, 'after') ?? new Absolute(null); } /** * Set space after paragraph * - * @param int $value * @return self */ - public function setSpaceAfter($value = null) + public function setSpaceAfter(Absolute $value) { return $this->setSpace(array('after' => $value)); } /** * Get spacing between lines - * - * @return int|float */ - public function getSpacing() + public function getSpacing(): Absolute { - return $this->getChildStyleValue($this->spacing, 'line'); + return $this->getChildStyleValue($this->spacing, 'line') ?? new Absolute(null); } /** * Set spacing between lines - * - * @param int|float $value - * @return self */ - public function setSpacing($value = null) + public function setSpacing(Absolute $value): self { return $this->setSpace(array('line' => $value)); } @@ -518,10 +508,8 @@ public function setSpacingLineRule($value) /** * Get line height - * - * @return int|float */ - public function getLineHeight() + public function getLineHeight(): Percent { return $this->lineHeight; } @@ -529,23 +517,13 @@ public function getLineHeight() /** * Set the line height * - * @param int|float|string $lineHeight - * - * @throws \PhpOffice\PhpWord\Exception\InvalidStyleException * @return self */ - public function setLineHeight($lineHeight) + public function setLineHeight(Percent $lineHeight) { - if (is_string($lineHeight)) { - $lineHeight = (float) (preg_replace('/[^0-9\.\,]/', '', $lineHeight)); - } - - if ((!is_int($lineHeight) && !is_float($lineHeight)) || !$lineHeight) { - throw new InvalidStyleException('Line height must be a valid number'); - } - $this->lineHeight = $lineHeight; - $this->setSpacing(($lineHeight - 1) * self::LINE_HEIGHT); + $lineHeight = $lineHeight->toFloat(); + $this->setSpacing(Absolute::from('twip', ($lineHeight - 100) / 100 * self::LINE_HEIGHT)); $this->setSpacingLineRule(\PhpOffice\PhpWord\SimpleType\LineSpacingRule::AUTO); return $this; @@ -775,7 +753,7 @@ public function getShading() /** * Set shading * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setShading($value = null) diff --git a/src/PhpWord/Style/Row.php b/src/PhpWord/Style/Row.php index ad801af6e2..a9c4004a7f 100644 --- a/src/PhpWord/Style/Row.php +++ b/src/PhpWord/Style/Row.php @@ -1,4 +1,5 @@ orientation = $this->setEnumVal($value, $enum, $this->orientation); - /** @var int|float $longSide Type hint */ - $longSide = $this->pageSizeW >= $this->pageSizeH ? $this->pageSizeW : $this->pageSizeH; - - /** @var int|float $shortSide Type hint */ - $shortSide = $this->pageSizeW < $this->pageSizeH ? $this->pageSizeW : $this->pageSizeH; + $isWide = $this->pageSizeW->toInt('twip') >= $this->pageSizeH->toInt('twip'); + $shouldBeWide = $this->orientation === self::ORIENTATION_LANDSCAPE; - if ($this->orientation == self::ORIENTATION_PORTRAIT) { - $this->pageSizeW = $shortSide; - $this->pageSizeH = $longSide; - } else { - $this->pageSizeW = $longSide; - $this->pageSizeH = $shortSide; + // If orientation doesn't match dimensions, + // swap the dimensions. + if ($isWide !== $shouldBeWide) { + $newWidth = $this->pageSizeW; + $this->pageSizeW = $this->pageSizeH; + $this->pageSizeH = $newWidth; } return $this; @@ -285,25 +285,22 @@ public function setLandscape() /** * Get Page Size Width * - * @return int|float|null * * @since 0.12.0 */ - public function getPageSizeW() + public function getPageSizeW(): Absolute { return $this->pageSizeW; } /** - * @param int|float|null $value - * * @return \PhpOffice\PhpWord\Style\Section * * @since 0.12.0 */ - public function setPageSizeW($value = null) + public function setPageSizeW(Absolute $value): self { - $this->pageSizeW = $this->setNumericVal($value, self::DEFAULT_WIDTH); + $this->pageSizeW = $value; return $this; } @@ -311,9 +308,9 @@ public function setPageSizeW($value = null) /** * Get Page Size Height * - * @return int|float|null + * @return Absolute * - * @since 0.12.0 + * @since 0.12.0: Absolute */ public function getPageSizeH() { @@ -321,176 +318,169 @@ public function getPageSizeH() } /** - * @param int|float|null $value - * * @return \PhpOffice\PhpWord\Style\Section * * @since 0.12.0 */ - public function setPageSizeH($value = null) + public function setPageSizeH(Absolute $value): self { - $this->pageSizeH = $this->setNumericVal($value, self::DEFAULT_HEIGHT); + $this->pageSizeH = $value; return $this; } /** * Get Margin Top - * - * @return int|float */ - public function getMarginTop() + public function getMarginTop(): Absolute { + if (!($this->marginTop instanceof Absolute)) { + $this->marginTop = Absolute::from('twip', $this->marginTop); + } + return $this->marginTop; } /** * Set Margin Top - * - * @param int|float $value - * @return self */ - public function setMarginTop($value = null) + public function setMarginTop(Absolute $value): self { - $this->marginTop = $this->setNumericVal($value, self::DEFAULT_MARGIN); + $this->marginTop = $value; return $this; } /** * Get Margin Left - * - * @return int|float */ - public function getMarginLeft() + public function getMarginLeft(): Absolute { + if (!($this->marginLeft instanceof Absolute)) { + $this->marginLeft = Absolute::from('twip', $this->marginLeft); + } + return $this->marginLeft; } /** * Set Margin Left - * - * @param int|float $value - * @return self */ - public function setMarginLeft($value = null) + public function setMarginLeft(Absolute $value): self { - $this->marginLeft = $this->setNumericVal($value, self::DEFAULT_MARGIN); + $this->marginLeft = $value; return $this; } /** * Get Margin Right - * - * @return int|float */ - public function getMarginRight() + public function getMarginRight(): Absolute { + if (!($this->marginRight instanceof Absolute)) { + $this->marginRight = Absolute::from('twip', $this->marginRight); + } + return $this->marginRight; } /** * Set Margin Right - * - * @param int|float $value - * @return self */ - public function setMarginRight($value = null) + public function setMarginRight(Absolute $value): self { - $this->marginRight = $this->setNumericVal($value, self::DEFAULT_MARGIN); + $this->marginRight = $value; return $this; } /** * Get Margin Bottom - * - * @return int|float */ - public function getMarginBottom() + public function getMarginBottom(): Absolute { + if (!($this->marginBottom instanceof Absolute)) { + $this->marginBottom = Absolute::from('twip', $this->marginBottom); + } + return $this->marginBottom; } /** * Set Margin Bottom - * - * @param int|float $value - * @return self */ - public function setMarginBottom($value = null) + public function setMarginBottom(Absolute $value): self { - $this->marginBottom = $this->setNumericVal($value, self::DEFAULT_MARGIN); + $this->marginBottom = $value; return $this; } /** * Get gutter - * - * @return int|float */ - public function getGutter() + public function getGutter(): Absolute { + if (!($this->gutter instanceof Absolute)) { + $this->gutter = Absolute::from('twip', $this->gutter); + } + return $this->gutter; } /** * Set gutter - * - * @param int|float $value - * @return self */ - public function setGutter($value = null) + public function setGutter(Absolute $value): self { - $this->gutter = $this->setNumericVal($value, self::DEFAULT_GUTTER); + $this->gutter = $value; return $this; } /** * Get Header Height - * - * @return int|float */ - public function getHeaderHeight() + public function getHeaderHeight(): Absolute { + if (!($this->headerHeight instanceof Absolute)) { + $this->headerHeight = Absolute::from('twip', $this->headerHeight); + } + return $this->headerHeight; } /** * Set Header Height - * - * @param int|float $value - * @return self */ - public function setHeaderHeight($value = null) + public function setHeaderHeight(Absolute $value): self { - $this->headerHeight = $this->setNumericVal($value, self::DEFAULT_HEADER_HEIGHT); + $this->headerHeight = $value; return $this; } /** * Get Footer Height - * - * @return int|float */ - public function getFooterHeight() + public function getFooterHeight(): Absolute { + if (!($this->footerHeight instanceof Absolute)) { + $this->footerHeight = Absolute::from('twip', $this->footerHeight); + } + return $this->footerHeight; } /** * Set Footer Height * - * @param int|float $value * @return self */ - public function setFooterHeight($value = null) + public function setFooterHeight(Absolute $value) { - $this->footerHeight = $this->setNumericVal($value, self::DEFAULT_FOOTER_HEIGHT); + $this->footerHeight = $value; return $this; } @@ -543,23 +533,22 @@ public function setColsNum($value = null) /** * Get Section Space Between Columns - * - * @return int|float */ - public function getColsSpace() + public function getColsSpace(): Absolute { + if (!($this->colsSpace instanceof Absolute)) { + $this->colsSpace = Absolute::from('twip', $this->colsSpace); + } + return $this->colsSpace; } /** * Set Section Space Between Columns - * - * @param int|float $value - * @return self */ - public function setColsSpace($value = null) + public function setColsSpace(Absolute $value): self { - $this->colsSpace = $this->setNumericVal($value, self::DEFAULT_COLUMN_SPACING); + $this->colsSpace = $value; return $this; } @@ -600,7 +589,7 @@ public function getLineNumbering() /** * Set line numbering * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setLineNumbering($value = null) diff --git a/src/PhpWord/Style/Shading.php b/src/PhpWord/Style/Shading.php index 154df26c7a..3a4b2b9cd4 100644 --- a/src/PhpWord/Style/Shading.php +++ b/src/PhpWord/Style/Shading.php @@ -1,4 +1,5 @@ color === null) { + $this->color = new Hex(null); + } + return $this->color; } /** * Set pattern - * - * @param string $value - * @return self */ - public function setColor($value = null) + public function setColor(BasicColor $value): self { $this->color = $value; @@ -123,21 +126,20 @@ public function setColor($value = null) /** * Get fill - * - * @return string */ - public function getFill() + public function getFill(): BasicColor { + if ($this->fill === null) { + $this->fill = new Hex(null); + } + return $this->fill; } /** * Set fill - * - * @param string $value - * @return self */ - public function setFill($value = null) + public function setFill(BasicColor $value): self { $this->fill = $value; diff --git a/src/PhpWord/Style/Shadow.php b/src/PhpWord/Style/Shadow.php index 1379a32096..cfc3fd60a2 100644 --- a/src/PhpWord/Style/Shadow.php +++ b/src/PhpWord/Style/Shadow.php @@ -1,4 +1,5 @@ roundness === null) { + $this->roundness = new Percent(0); + } + return $this->roundness; } /** * Set roundness - * - * @param int|float $value - * @return self */ - public function setRoundness($value = null) + public function setRoundness(Percent $value): self { - $this->roundness = $this->setNumericVal($value, null); + $percent = $value->toFloat(); + if ($percent > 100) { + throw new Exception(sprintf('Provided roundness %f%% must be no greater than 100%%', $percent)); + } elseif ($percent < 0) { + throw new Exception(sprintf('Provided roundness %f%% must be no less than 0%%', $percent)); + } + + $this->roundness = $value; return $this; } @@ -151,7 +161,7 @@ public function getFrame() /** * Set frame * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setFrame($value = null) @@ -174,7 +184,7 @@ public function getFill() /** * Set fill * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setFill($value = null) @@ -197,7 +207,7 @@ public function getOutline() /** * Set outline * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setOutline($value = null) @@ -220,7 +230,7 @@ public function getShadow() /** * Set shadow * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setShadow($value = null) @@ -243,7 +253,7 @@ public function getExtrusion() /** * Set 3D extrusion * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setExtrusion($value = null) diff --git a/src/PhpWord/Style/Spacing.php b/src/PhpWord/Style/Spacing.php index 9bfb22822b..4e344d0285 100644 --- a/src/PhpWord/Style/Spacing.php +++ b/src/PhpWord/Style/Spacing.php @@ -1,4 +1,5 @@ before === null) { + $this->before = new Absolute(null); + } + return $this->before; } /** * Set before - * - * @param int|float $value - * @return self */ - public function setBefore($value = null) + public function setBefore(Absolute $value): self { - $this->before = $this->setNumericVal($value, $this->before); + $this->before = $value; return $this; } /** * Get after - * - * @return int|float */ - public function getAfter() + public function getAfter(): Absolute { + if ($this->after === null) { + $this->after = new Absolute(null); + } + return $this->after; } /** * Set after - * - * @param int|float $value - * @return self */ - public function setAfter($value = null) + public function setAfter(Absolute $value): self { - $this->after = $this->setNumericVal($value, $this->after); + $this->after = $value; return $this; } /** - * Get line - * - * @return int|float + * Get vertical spacing between lines of text within paragraph. + * Spacing may vary for same value depending on value of line rule. + * See `getLineRule()``. */ - public function getLine() + public function getLine(): Absolute { + if ($this->line === null) { + $this->line = new Absolute(null); + } + return $this->line; } /** - * Set distance - * - * @param int|float $value - * @return self + * Set vertical spacing between lines of text within paragraph. + * Spacing may vary for same value depending on value of line rule. + * See `setLineRule()``. */ - public function setLine($value = null) + public function setLine(Absolute $value): self { - $this->line = $this->setNumericVal($value, $this->line); + $this->line = $value; return $this; } diff --git a/src/PhpWord/Style/TOC.php b/src/PhpWord/Style/TOC.php index 2efd54a4a4..57c357b7ff 100644 --- a/src/PhpWord/Style/TOC.php +++ b/src/PhpWord/Style/TOC.php @@ -1,4 +1,5 @@ indent = Absolute::from('twip', 200); } /** * Get Tab Position - * - * @return int|float */ - public function getTabPos() + public function getTabPos(): Absolute { return $this->getPosition(); } /** * Set Tab Position - * - * @param int|float $value - * @return self */ - public function setTabPos($value) + public function setTabPos(Absolute $value): self { return $this->setPosition($value); } @@ -93,23 +93,18 @@ public function setTabLeader($value = self::TAB_LEADER_DOT) /** * Get Indent - * - * @return int|float */ - public function getIndent() + public function getIndent(): Absolute { return $this->indent; } /** * Set Indent - * - * @param int|float $value - * @return self */ - public function setIndent($value) + public function setIndent(Absolute $value): self { - $this->indent = $this->setNumericVal($value, $this->indent); + $this->indent = $value; return $this; } diff --git a/src/PhpWord/Style/Tab.php b/src/PhpWord/Style/Tab.php index d3cf5bd7f4..ce828c5e77 100644 --- a/src/PhpWord/Style/Tab.php +++ b/src/PhpWord/Style/Tab.php @@ -1,4 +1,5 @@ type = $this->setEnumVal($type, $stopTypes, $this->type); - $this->position = $this->setNumericVal($position, $this->position); + $this->position = $position ?? Absolute::from('twip', 0); $this->leader = $this->setEnumVal($leader, $leaderTypes, $this->leader); } @@ -150,23 +153,18 @@ public function setLeader($value) /** * Get position - * - * @return int|float */ - public function getPosition() + public function getPosition(): Absolute { return $this->position; } /** * Set position - * - * @param int|float $value - * @return self */ - public function setPosition($value) + public function setPosition(Absolute $value): self { - $this->position = $this->setNumericVal($value, $this->position); + $this->position = $value; return $this; } diff --git a/src/PhpWord/Style/Table.php b/src/PhpWord/Style/Table.php index f777ac671f..7ee75bb68e 100644 --- a/src/PhpWord/Style/Table.php +++ b/src/PhpWord/Style/Table.php @@ -1,4 +1,5 @@ setIndent(new Auto()); + $this->setWidth(new Auto()); + $this->setCellSpacing(new Absolute(null)); + $this->setCellMargin(new Absolute(null)); + // Clone first row from table style, but with certain properties disabled if ($firstRowStyle !== null && is_array($firstRowStyle)) { $this->firstRowStyle = clone $this; $this->firstRowStyle->isFirstRow = true; - unset($this->firstRowStyle->firstRowStyle, $this->firstRowStyle->borderInsideHSize, $this->firstRowStyle->borderInsideHColor, $this->firstRowStyle->borderInsideVSize, $this->firstRowStyle->borderInsideVColor, $this->firstRowStyle->cellMarginTop, $this->firstRowStyle->cellMarginLeft, $this->firstRowStyle->cellMarginRight, $this->firstRowStyle->cellMarginBottom, $this->firstRowStyle->cellSpacing); + $this->firstRowStyle + ->setCellSpacing(new Absolute(null)) + ->setCellMargin(new Absolute(null)); + unset($this->firstRowStyle->firstRowStyle); $this->firstRowStyle->setStyleByArray($firstRowStyle); } @@ -199,18 +167,31 @@ public function __construct($tableStyle = null, $firstRowStyle = null) } } - /** - * @param float|int $cellSpacing - */ - public function setCellSpacing($cellSpacing = null) + protected function getAllowedSides(): array + { + $sides = array( + 'top', + 'bottom', + 'start', + 'end', + ); + + if (!$this->isFirstRow) { + $sides[] = 'insideH'; + $sides[] = 'insideV'; + } + + return $sides; + } + + public function setCellSpacing(Absolute $cellSpacing): self { $this->cellSpacing = $cellSpacing; + + return $this; } - /** - * @return float|int - */ - public function getCellSpacing() + public function getCellSpacing(): Absolute { return $this->cellSpacing; } @@ -227,291 +208,130 @@ public function getFirstRow() /** * Get background - * - * @return string */ - public function getBgColor() + public function getBgColor(): BasicColor { - if ($this->shading !== null) { - return $this->shading->getFill(); + if ($this->shading === null) { + $this->setBgColor(new Hex(null)); } - return null; + return $this->shading->getFill(); } /** * Set background * - * @param string $value * @return self */ - public function setBgColor($value = null) + public function setBgColor(BasicColor $value) { $this->setShading(array('fill' => $value)); return $this; } - /** - * Get TLRBHV Border Size - * - * @return int[] - */ - public function getBorderSize() - { - return array( - $this->getBorderTopSize(), - $this->getBorderLeftSize(), - $this->getBorderRightSize(), - $this->getBorderBottomSize(), - $this->getBorderInsideHSize(), - $this->getBorderInsideVSize(), - ); - } - - /** - * Set TLRBHV Border Size - * - * @param int $value Border size in eighths of a point (1/8 point) - * @return self - */ - public function setBorderSize($value = null) - { - $this->setBorderTopSize($value); - $this->setBorderLeftSize($value); - $this->setBorderRightSize($value); - $this->setBorderBottomSize($value); - $this->setBorderInsideHSize($value); - $this->setBorderInsideVSize($value); - - return $this; - } - - /** - * Get TLRBHV Border Color - * - * @return string[] - */ - public function getBorderColor() - { - return array( - $this->getBorderTopColor(), - $this->getBorderLeftColor(), - $this->getBorderRightColor(), - $this->getBorderBottomColor(), - $this->getBorderInsideHColor(), - $this->getBorderInsideVColor(), - ); - } - - /** - * Set TLRBHV Border Color - * - * @param string $value - * @return self - */ - public function setBorderColor($value = null) - { - $this->setBorderTopColor($value); - $this->setBorderLeftColor($value); - $this->setBorderRightColor($value); - $this->setBorderBottomColor($value); - $this->setBorderInsideHColor($value); - $this->setBorderInsideVColor($value); - - return $this; - } - - /** - * Get border size inside horizontal - * - * @return int - */ - public function getBorderInsideHSize() - { - return $this->getTableOnlyProperty('borderInsideHSize'); - } - - /** - * Set border size inside horizontal - * - * @param int $value - * @return self - */ - public function setBorderInsideHSize($value = null) - { - return $this->setTableOnlyProperty('borderInsideHSize', $value); - } - - /** - * Get border color inside horizontal - * - * @return string - */ - public function getBorderInsideHColor() - { - return $this->getTableOnlyProperty('borderInsideHColor'); - } - - /** - * Set border color inside horizontal - * - * @param string $value - * @return self - */ - public function setBorderInsideHColor($value = null) - { - return $this->setTableOnlyProperty('borderInsideHColor', $value, false); - } - - /** - * Get border size inside vertical - * - * @return int - */ - public function getBorderInsideVSize() - { - return $this->getTableOnlyProperty('borderInsideVSize'); - } - - /** - * Set border size inside vertical - * - * @param int $value - * @return self - */ - public function setBorderInsideVSize($value = null) - { - return $this->setTableOnlyProperty('borderInsideVSize', $value); - } - - /** - * Get border color inside vertical - * - * @return string - */ - public function getBorderInsideVColor() - { - return $this->getTableOnlyProperty('borderInsideVColor'); - } - - /** - * Set border color inside vertical - * - * @param string $value - * @return self - */ - public function setBorderInsideVColor($value = null) - { - return $this->setTableOnlyProperty('borderInsideVColor', $value, false); - } - /** * Get cell margin top - * - * @return int */ - public function getCellMarginTop() + public function getCellMarginTop(): Absolute { - return $this->getTableOnlyProperty('cellMarginTop'); + return $this->cellMarginTop; } /** * Set cell margin top - * - * @param int $value - * @return self */ - public function setCellMarginTop($value = null) + public function setCellMarginTop(Absolute $value): self { - return $this->setTableOnlyProperty('cellMarginTop', $value); + if (!$this->isFirstRow) { + $this->cellMarginTop = $value; + } + + return $this; } /** * Get cell margin left - * - * @return int */ - public function getCellMarginLeft() + public function getCellMarginLeft(): Absolute { - return $this->getTableOnlyProperty('cellMarginLeft'); + return $this->cellMarginLeft; } /** * Set cell margin left - * - * @param int $value - * @return self */ - public function setCellMarginLeft($value = null) + public function setCellMarginLeft(Absolute $value): self { - return $this->setTableOnlyProperty('cellMarginLeft', $value); + if (!$this->isFirstRow) { + $this->cellMarginLeft = $value; + } + + return $this; } /** * Get cell margin right - * - * @return int */ - public function getCellMarginRight() + public function getCellMarginRight(): Absolute { - return $this->getTableOnlyProperty('cellMarginRight'); + return $this->cellMarginRight; } /** * Set cell margin right - * - * @param int $value - * @return self */ - public function setCellMarginRight($value = null) + public function setCellMarginRight(Absolute $value): self { - return $this->setTableOnlyProperty('cellMarginRight', $value); + if (!$this->isFirstRow) { + $this->cellMarginRight = $value; + } + + return $this; } /** * Get cell margin bottom - * - * @return int */ - public function getCellMarginBottom() + public function getCellMarginBottom(): Absolute { - return $this->getTableOnlyProperty('cellMarginBottom'); + return $this->cellMarginBottom; } /** * Set cell margin bottom - * - * @param int $value - * @return self */ - public function setCellMarginBottom($value = null) + public function setCellMarginBottom(Absolute $value): self { - return $this->setTableOnlyProperty('cellMarginBottom', $value); + if (!$this->isFirstRow) { + $this->cellMarginBottom = $value; + } + + return $this; } /** * Get cell margin * - * @return int[] + * @return Absolute[] */ - public function getCellMargin() + public function getCellMargin(): array { return array( - $this->cellMarginTop, - $this->cellMarginLeft, - $this->cellMarginRight, - $this->cellMarginBottom, + $this->getCellMarginTop(), + $this->getCellMarginLeft(), + $this->getCellMarginRight(), + $this->getCellMarginBottom(), ); } /** * Set TLRB cell margin * - * @param int $value Margin in twips + * @param Absolute $value Margin * @return self */ - public function setCellMargin($value = null) + public function setCellMargin(Absolute $value) { $this->setCellMarginTop($value); $this->setCellMarginLeft($value); @@ -528,9 +348,13 @@ public function setCellMargin($value = null) */ public function hasMargin() { - $margins = $this->getCellMargin(); + foreach ($this->getCellMargin() as $margin) { + if ($margin->toInt('twip') !== null) { + return true; + } + } - return $margins !== array_filter($margins, 'is_null'); + return false; } /** @@ -546,7 +370,7 @@ public function getShading() /** * Set shading * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setShading($value = null) @@ -610,47 +434,18 @@ public function setAlign($value = null) /** * Get width - * - * @return int|float */ - public function getWidth() + public function getWidth(): Length { return $this->width; } /** * Set width - * - * @param int|float $value - * @return self - */ - public function setWidth($value = null) - { - $this->width = $this->setNumericVal($value, $this->width); - - return $this; - } - - /** - * Get width unit - * - * @return string */ - public function getUnit() + public function setWidth(Length $value): self { - return $this->unit; - } - - /** - * Set width unit - * - * @param string $value - * @return self - */ - public function setUnit($value = null) - { - TblWidth::validate($value); - $this->unit = $value; + $this->width = $value; return $this; } @@ -679,48 +474,6 @@ public function setLayout($value = null) return $this; } - /** - * Get table style only property by checking if it's a firstRow - * - * This is necessary since firstRow style is cloned from table style but - * without certain properties activated, e.g. margins - * - * @param string $property - * @return int|string|null - */ - private function getTableOnlyProperty($property) - { - if (false === $this->isFirstRow) { - return $this->$property; - } - - return null; - } - - /** - * Set table style only property by checking if it's a firstRow - * - * This is necessary since firstRow style is cloned from table style but - * without certain properties activated, e.g. margins - * - * @param string $property - * @param int|string $value - * @param bool $isNumeric - * @return self - */ - private function setTableOnlyProperty($property, $value, $isNumeric = true) - { - if (false === $this->isFirstRow) { - if (true === $isNumeric) { - $this->$property = $this->setNumericVal($value, $this->$property); - } else { - $this->$property = $value; - } - } - - return $this; - } - /** * Get position * @@ -734,7 +487,7 @@ public function getPosition() /** * Set position * - * @param mixed $value + * @param null|mixed $value * @return self */ public function setPosition($value = null) @@ -744,20 +497,16 @@ public function setPosition($value = null) return $this; } - /** - * @return TblWidthComplexType - */ - public function getIndent() + public function getIndent(): Length { return $this->indent; } /** - * @param TblWidthComplexType $indent * @return self * @see http://www.datypic.com/sc/ooxml/e-w_tblInd-1.html */ - public function setIndent(TblWidthComplexType $indent) + public function setIndent(Length $indent) { $this->indent = $indent; @@ -767,7 +516,7 @@ public function setIndent(TblWidthComplexType $indent) /** * Get the columnWidths * - * @return null|int[] + * @return null|Length[] */ public function getColumnWidths() { @@ -776,12 +525,16 @@ public function getColumnWidths() /** * The column widths - * - * @param int[] $value */ - public function setColumnWidths(array $value = null) + public function setColumnWidths(array $values = null) { - $this->columnWidths = $value; + foreach ($values as $value) { + if (!($value instanceof Length)) { + throw new Exception(sprintf('Column widths must be specified with `%s`', Length::class)); + } + } + + $this->columnWidths = $values; } /** diff --git a/src/PhpWord/Style/TablePosition.php b/src/PhpWord/Style/TablePosition.php index d4b7083102..62a00830a8 100644 --- a/src/PhpWord/Style/TablePosition.php +++ b/src/PhpWord/Style/TablePosition.php @@ -1,4 +1,5 @@ leftFromText === null) { + $this->leftFromText = new Absolute(null); + } + return $this->leftFromText; } /** * Set distance from left of table to text - * - * @param int $value - * @return self */ - public function setLeftFromText($value = null) + public function setLeftFromText(Absolute $value): self { - $this->leftFromText = $this->setNumericVal($value, $this->leftFromText); + $this->leftFromText = $value; return $this; } /** * Get distance from right of table to text - * - * @return int */ - public function getRightFromText() + public function getRightFromText(): Absolute { + if ($this->rightFromText === null) { + $this->rightFromText = new Absolute(null); + } + return $this->rightFromText; } /** * Set distance from right of table to text - * - * @param int $value - * @return self */ - public function setRightFromText($value = null) + public function setRightFromText(Absolute $value): self { - $this->rightFromText = $this->setNumericVal($value, $this->rightFromText); + $this->rightFromText = $value; return $this; } /** * Get distance from top of table to text - * - * @return int */ - public function getTopFromText() + public function getTopFromText(): Absolute { + if ($this->topFromText === null) { + $this->topFromText = new Absolute(null); + } + return $this->topFromText; } /** * Set distance from top of table to text - * - * @param int $value - * @return self */ - public function setTopFromText($value = null) + public function setTopFromText(Absolute $value): self { - $this->topFromText = $this->setNumericVal($value, $this->topFromText); + $this->topFromText = $value; return $this; } /** * Get distance from bottom of table to text - * - * @return int */ - public function getBottomFromText() + public function getBottomFromText(): Absolute { + if ($this->bottomFromText === null) { + $this->bottomFromText = new Absolute(null); + } + return $this->bottomFromText; } /** * Set distance from bottom of table to text - * - * @param int $value - * @return self */ - public function setBottomFromText($value = null) + public function setBottomFromText(Absolute $value): self { - $this->bottomFromText = $this->setNumericVal($value, $this->bottomFromText); + $this->bottomFromText = $value; return $this; } @@ -333,23 +333,22 @@ public function setTblpXSpec($value = null) /** * Get absolute horizontal distance from anchor - * - * @return int */ - public function getTblpX() + public function getTblpX(): Absolute { + if ($this->tblpX === null) { + $this->tblpX = new Absolute(null); + } + return $this->tblpX; } /** * Set absolute horizontal distance from anchor - * - * @param int $value - * @return self */ - public function setTblpX($value = null) + public function setTblpX(Absolute $value): self { - $this->tblpX = $this->setNumericVal($value, $this->tblpX); + $this->tblpX = $value; return $this; } @@ -387,23 +386,22 @@ public function setTblpYSpec($value = null) /** * Get absolute vertical distance from anchor - * - * @return int */ - public function getTblpY() + public function getTblpY(): Absolute { + if ($this->tblpY === null) { + $this->tblpY = new Absolute(null); + } + return $this->tblpY; } /** * Set absolute vertical distance from anchor - * - * @param int $value - * @return self */ - public function setTblpY($value = null) + public function setTblpY(Absolute $value): self { - $this->tblpY = $this->setNumericVal($value, $this->tblpY); + $this->tblpY = $value; return $this; } diff --git a/src/PhpWord/Style/TextBox.php b/src/PhpWord/Style/TextBox.php index e9c0f0c064..9806fc45a2 100644 --- a/src/PhpWord/Style/TextBox.php +++ b/src/PhpWord/Style/TextBox.php @@ -1,4 +1,5 @@ innerMarginTop = $value; + + return $this; } /** * Get margin top - * - * @return int */ - public function getInnerMarginTop() + public function getInnerMarginTop(): Absolute { + if ($this->innerMarginTop === null) { + $this->setInnerMarginTop(new Absolute(null)); + } + return $this->innerMarginTop; } /** * Set margin left. - * - * @param int $value */ - public function setInnerMarginLeft($value = null) + public function setInnerMarginLeft(Absolute $value): self { $this->innerMarginLeft = $value; + + return $this; } /** * Get margin left - * - * @return int */ - public function getInnerMarginLeft() + public function getInnerMarginLeft(): Absolute { + if ($this->innerMarginLeft === null) { + $this->setInnerMarginLeft(new Absolute(null)); + } + return $this->innerMarginLeft; } /** * Set margin right. - * - * @param int $value */ - public function setInnerMarginRight($value = null) + public function setInnerMarginRight(Absolute $value): self { $this->innerMarginRight = $value; + + return $this; } /** * Get margin right - * - * @return int */ - public function getInnerMarginRight() + public function getInnerMarginRight(): Absolute { + if ($this->innerMarginRight === null) { + $this->setInnerMarginRight(new Absolute(null)); + } + return $this->innerMarginRight; } /** * Set margin bottom. - * - * @param int $value */ - public function setInnerMarginBottom($value = null) + public function setInnerMarginBottom(Absolute $value): self { $this->innerMarginBottom = $value; + + return $this; } /** * Get margin bottom - * - * @return int */ - public function getInnerMarginBottom() + public function getInnerMarginBottom(): Absolute { + if ($this->innerMarginBottom === null) { + $this->setInnerMarginBottom(new Absolute(null)); + } + return $this->innerMarginBottom; } /** * Set TLRB cell margin. * - * @param int $value Margin in twips + * @param Absolute $value Margin */ - public function setInnerMargin($value = null) + public function setInnerMargin(Absolute $value): self { $this->setInnerMarginTop($value); $this->setInnerMarginLeft($value); $this->setInnerMarginRight($value); $this->setInnerMarginBottom($value); + + return $this; } /** * Get cell margin * - * @return int[] + * @return Absolute[] */ public function getInnerMargin() { - return array($this->innerMarginLeft, $this->innerMarginTop, $this->innerMarginRight, $this->innerMarginBottom); + return array($this->getInnerMarginLeft(), $this->getInnerMarginTop(), $this->getInnerMarginRight(), $this->getInnerMarginBottom()); } /** @@ -180,7 +195,7 @@ public function hasInnerMargins() $margins = $this->getInnerMargin(); $numMargins = count($margins); for ($i = 0; $i < $numMargins; $i++) { - if ($margins[$i] !== null) { + if ($margins[$i]->toInt('twip') !== null) { $hasInnerMargins = true; } } @@ -191,40 +206,46 @@ public function hasInnerMargins() /** * Set border size. * - * @param int $value Size in points + * @param Absolute $value Size */ - public function setBorderSize($value = null) + public function setBorderSize(Absolute $value): self { $this->borderSize = $value; + + return $this; } /** * Get border size - * - * @return int */ - public function getBorderSize() + public function getBorderSize(): Absolute { + if ($this->borderSize === null) { + $this->borderSize = new Absolute(null); + } + return $this->borderSize; } /** * Set border color. - * - * @param string $value */ - public function setBorderColor($value = null) + public function setBorderColor(BasicColor $value): self { $this->borderColor = $value; + + return $this; } /** * Get border color - * - * @return string */ - public function getBorderColor() + public function getBorderColor(): BasicColor { + if ($this->borderColor === null) { + $this->borderColor = new Hex(null); + } + return $this->borderColor; } } diff --git a/src/PhpWord/Style/Theme/BodyFonts.php b/src/PhpWord/Style/Theme/BodyFonts.php new file mode 100644 index 0000000000..a39dcba120 --- /dev/null +++ b/src/PhpWord/Style/Theme/BodyFonts.php @@ -0,0 +1,42 @@ + 'Calibri', + 'EastAsian' => '', + 'ComplexScript' => '', + 'Jpan' => 'MS 明朝', + 'Hang' => '맑은 고딕', + 'Hans' => '宋体', + 'Hant' => '新細明體', + 'Arab' => 'Arial', + 'Hebr' => 'Arial', + 'Thai' => 'Cordia New', + 'Ethi' => 'Nyala', + 'Beng' => 'Vrinda', + 'Gujr' => 'Shruti', + 'Khmr' => 'DaunPenh', + 'Knda' => 'Tunga', + 'Guru' => 'Raavi', + 'Cans' => 'Euphemia', + 'Cher' => 'Plantagenet Cherokee', + 'Yiii' => 'Microsoft Yi Baiti', + 'Tibt' => 'Microsoft Himalaya', + 'Thaa' => 'MV Boli', + 'Deva' => 'Mangal', + 'Telu' => 'Gautami', + 'Taml' => 'Latha', + 'Syrc' => 'Estrangelo Edessa', + 'Orya' => 'Kalinga', + 'Mlym' => 'Kartika', + 'Laoo' => 'DokChampa', + 'Sinh' => 'Iskoola Pota', + 'Mong' => 'Mongolian Baiti', + 'Viet' => 'Arial', + 'Uigh' => 'Microsoft Uighur', + ); +} diff --git a/src/PhpWord/Style/Theme/ColorScheme.php b/src/PhpWord/Style/Theme/ColorScheme.php new file mode 100644 index 0000000000..77d6280e25 --- /dev/null +++ b/src/PhpWord/Style/Theme/ColorScheme.php @@ -0,0 +1,69 @@ + null, + 'dk2' => null, + 'lt1' => null, + 'lt2' => null, + 'accent1' => null, + 'accent2' => null, + 'accent3' => null, + 'accent4' => null, + 'accent5' => null, + 'accent6' => null, + 'hlink' => null, + 'folHlink' => null, + ); + + public function __construct(array $colorScheme) + { + if (count($colorScheme) !== count($this->colorScheme)) { + throw new Exception(sprintf('%s colors expected, but %s colors provided', count($this->colorScheme), count($colorScheme))); + } + + foreach (array_keys($this->colorScheme) as $name) { + if (!array_key_exists($name, $colorScheme)) { + throw new Exception(sprintf("Missing '$name' from provided color scheme")); + } + + $color = $colorScheme[$name]; + if (!($color instanceof SpecialColor)) { + throw new Exception(sprintf("Provided color for '%s' must be an instance of '%s', '%s' provided", $name, SpecialColor::class, gettype($color))); + } + + $this->colorScheme[$name] = $color; + } + } + + public function getColors(): array + { + $colors = array(); + foreach ($this->colorScheme as $name => $color) { + $colors[$name] = clone $color; + } + + return $colors; + } + + public function getColor(string $name): SpecialColor + { + if (!array_key_exists($name, $this->colorScheme)) { + throw new Exception(sprintf("No color exists for '$name'")); + } + + return clone $this->colorScheme[$name]; + } +} diff --git a/src/PhpWord/Style/Theme/FontScheme.php b/src/PhpWord/Style/Theme/FontScheme.php new file mode 100644 index 0000000000..338712c375 --- /dev/null +++ b/src/PhpWord/Style/Theme/FontScheme.php @@ -0,0 +1,33 @@ +name = $name; + $this->headingFonts = $headingFonts ?? new HeadingFonts(); + $this->bodyFonts = $bodyFonts ?? new BodyFonts(); + } + + public function getName(): string + { + return $this->name; + } + + public function getHeadingFonts(): HeadingFonts + { + return clone $this->headingFonts; + } + + public function getBodyFonts(): BodyFonts + { + return clone $this->bodyFonts; + } +} diff --git a/src/PhpWord/Style/Theme/Fonts.php b/src/PhpWord/Style/Theme/Fonts.php new file mode 100644 index 0000000000..dc8c0a8456 --- /dev/null +++ b/src/PhpWord/Style/Theme/Fonts.php @@ -0,0 +1,113 @@ + null, + 'Hang' => null, + 'Hans' => null, + 'Hant' => null, + 'Arab' => null, + 'Hebr' => null, + 'Thai' => null, + 'Ethi' => null, + 'Beng' => null, + 'Gujr' => null, + 'Khmr' => null, + 'Knda' => null, + 'Guru' => null, + 'Cans' => null, + 'Cher' => null, + 'Yiii' => null, + 'Tibt' => null, + 'Thaa' => null, + 'Deva' => null, + 'Telu' => null, + 'Taml' => null, + 'Syrc' => null, + 'Orya' => null, + 'Mlym' => null, + 'Laoo' => null, + 'Sinh' => null, + 'Mong' => null, + 'Viet' => null, + 'Uigh' => null, + ); + protected static $defaultFonts = array(); + + public function __construct(array $fonts = array()) + { + $this->latin = $this->readFont('Latin', $fonts); + $this->eastAsian = $this->readFont('EastAsian', $fonts); + $this->complexScript = $this->readFont('ComplexScript', $fonts); + foreach ($this->fonts as $script => $null) { + $this->fonts[$script] = $this->readFont($script, $fonts); + } + + foreach ($fonts as $script => $null) { + if ($script === 'Latin' || $script === 'EastAsian' || $script === 'ComplexScript') { + continue; + } + + if (!array_key_exists($script, $this->fonts)) { + throw new Exception(sprintf("Invalid script '%s' provided", $script)); + } + } + } + + protected function readFont(string $script, array $fonts): string + { + $font = $fonts[$script] ?? $this->getDefaultFont($script); + if (!is_string($font)) { + throw new Exception(sprintf("Font name expected, '%s' provided", gettype($font))); + } + + return $font; + } + + public function getLatin(): string + { + return $this->latin; + } + + public function getEastAsian(): string + { + return $this->eastAsian; + } + + public function getComplexScript(): string + { + return $this->complexScript; + } + + public function getFonts(): array + { + return $this->fonts; + } + + public function getFont(string $script): string + { + if (!array_key_exists($script, $this->fonts)) { + throw new Exception(sprintf("No font found for script '%s' in color scheme '%s'", $script, get_class($this))); + } + + return $this->fonts[$script]; + } + + public static function getDefaultFont(string $script): string + { + if (!array_key_exists($script, static::$defaultFonts)) { + throw new Exception(sprintf("No font found for script '%s' in color scheme '%s'", $script, get_class())); + } + + return static::$defaultFonts[$script]; + } +} diff --git a/src/PhpWord/Style/Theme/HeadingFonts.php b/src/PhpWord/Style/Theme/HeadingFonts.php new file mode 100644 index 0000000000..536637e49a --- /dev/null +++ b/src/PhpWord/Style/Theme/HeadingFonts.php @@ -0,0 +1,42 @@ + 'Cambria', + 'EastAsian' => '', + 'ComplexScript' => '', + 'Jpan' => 'MS ゴシック', + 'Hang' => '맑은 고딕', + 'Hans' => '宋体', + 'Hant' => '新細明體', + 'Arab' => 'Times New Roman', + 'Hebr' => 'Times New Roman', + 'Thai' => 'Angsana New', + 'Ethi' => 'Nyala', + 'Beng' => 'Vrinda', + 'Gujr' => 'Shruti', + 'Khmr' => 'MoolBoran', + 'Knda' => 'Tunga', + 'Guru' => 'Raavi', + 'Cans' => 'Euphemia', + 'Cher' => 'Plantagenet Cherokee', + 'Yiii' => 'Microsoft Yi Baiti', + 'Tibt' => 'Microsoft Himalaya', + 'Thaa' => 'MV Boli', + 'Deva' => 'Mangal', + 'Telu' => 'Gautami', + 'Taml' => 'Latha', + 'Syrc' => 'Estrangelo Edessa', + 'Orya' => 'Kalinga', + 'Mlym' => 'Kartika', + 'Laoo' => 'DokChampa', + 'Sinh' => 'Iskoola Pota', + 'Mong' => 'Mongolian Baiti', + 'Viet' => 'Times New Roman', + 'Uigh' => 'Microsoft Uighur', + ); +} diff --git a/src/PhpWord/Style/Theme/Theme.php b/src/PhpWord/Style/Theme/Theme.php new file mode 100644 index 0000000000..0eb42dbedd --- /dev/null +++ b/src/PhpWord/Style/Theme/Theme.php @@ -0,0 +1,44 @@ +colorScheme)) { + $this->colorScheme = new ColorScheme(array( + 'dk1' => new SystemColor('windowText', new Hex('000')), + 'lt1' => new SystemColor('window', new Hex('fff')), + 'dk2' => new Hex('1F497D'), + 'lt2' => new Hex('EEECE1'), + 'accent1' => new Hex('4F81BD'), + 'accent2' => new Hex('C0504D'), + 'accent3' => new Hex('9BBB59'), + 'accent4' => new Hex('8064A2'), + 'accent5' => new Hex('4BACC6'), + 'accent6' => new Hex('F79646'), + 'hlink' => new Hex('0000FF'), + 'folHlink' => new Hex('800080'), + )); + } + + return clone $this->colorScheme; + } + + public function getFontScheme(): FontScheme + { + if (!isset($this->fontScheme)) { + $this->fontScheme = new FontScheme(); + } + + return clone $this->fontScheme; + } +} diff --git a/src/PhpWord/Template.php b/src/PhpWord/Template.php index c42696f08c..3dc096755b 100644 --- a/src/PhpWord/Template.php +++ b/src/PhpWord/Template.php @@ -1,4 +1,5 @@ replaceXmlBlock($search, $xmlWriter->getData(), 'w:p'); } - /** - * @param mixed $search - * @param mixed $replace - * @param int $limit - */ - public function setValue($search, $replace, $limit = self::MAXIMUM_REPLACEMENTS_DEFAULT) + public function setValue($search, $replace, int $limit = self::MAXIMUM_REPLACEMENTS_DEFAULT) { if (is_array($search)) { foreach ($search as &$item) { @@ -318,15 +307,20 @@ public function setValue($search, $replace, $limit = self::MAXIMUM_REPLACEMENTS_ if (is_array($replace)) { foreach ($replace as &$item) { $item = static::ensureUtf8Encoded($item); + + if (Settings::isOutputEscapingEnabled()) { + $xmlEscaper = new Xml(); + $item = $xmlEscaper->escape($item); + } } unset($item); } else { $replace = static::ensureUtf8Encoded($replace); - } - if (Settings::isOutputEscapingEnabled()) { - $xmlEscaper = new Xml(); - $replace = $xmlEscaper->escape($replace); + if (Settings::isOutputEscapingEnabled()) { + $xmlEscaper = new Xml(); + $replace = $xmlEscaper->escape($replace); + } } $this->tempDocumentHeaders = $this->setValueForPart($search, $replace, $this->tempDocumentHeaders, $limit); @@ -336,8 +330,6 @@ public function setValue($search, $replace, $limit = self::MAXIMUM_REPLACEMENTS_ /** * Set values from a one-dimensional array of "variable => value"-pairs. - * - * @param array $values */ public function setValues(array $values) { @@ -388,7 +380,7 @@ private function chooseImageDimension($baseValue, $inlineValue, $defaultValue) if (is_null($value) && isset($inlineValue)) { $value = $inlineValue; } - if (!preg_match('/^([0-9]*(cm|mm|in|pt|pc|px|%|em|ex|)|auto)$/i', $value)) { + if (!preg_match('/^([0-9]*(cm|mm|in|pt|pc|px|%|em|ex|)|auto)$/i', (string) $value)) { $value = null; } if (is_null($value)) { @@ -462,8 +454,8 @@ private function prepareImageAttrs($replaceImage, $varInlineArgs) $imgPath = $replaceImage; } - $width = $this->chooseImageDimension($width, isset($varInlineArgs['width']) ? $varInlineArgs['width'] : null, 115); - $height = $this->chooseImageDimension($height, isset($varInlineArgs['height']) ? $varInlineArgs['height'] : null, 70); + $width = $this->chooseImageDimension($width, $varInlineArgs['width'] ?? null, 115); + $height = $this->chooseImageDimension($height, $varInlineArgs['height'] ?? null, 70); $imageData = @getimagesize($imgPath); if (!is_array($imageData)) { @@ -539,7 +531,6 @@ private function addImageToRelations($partFileName, $rid, $imgPath, $imageMimeTy } /** - * @param mixed $search * @param mixed $replace Path to image, or array("path" => xx, "width" => yy, "height" => zz) * @param int $limit */ @@ -559,7 +550,7 @@ public function setImageValue($search, $replace, $limit = self::MAXIMUM_REPLACEM $searchReplace = array(); foreach ($search as $searchIdx => $searchString) { - $searchReplace[$searchString] = isset($replacesList[$searchIdx]) ? $replacesList[$searchIdx] : $replacesList[0]; + $searchReplace[$searchString] = $replacesList[$searchIdx] ?? $replacesList[0]; } // collect document parts @@ -679,8 +670,8 @@ public function cloneRow($search, $numberOfClones) $extraRowStart = $this->findRowStart($extraRowEnd + 1); $extraRowEnd = $this->findRowEnd($extraRowEnd + 1); - // If extraRowEnd is lower then 7, there was no next row found. - if ($extraRowEnd < 7) { + // If extraRowEnd is null, there was no next row found. + if ($extraRowEnd === null) { break; } @@ -907,8 +898,6 @@ function ($match) { /** * Find and replace macros in the given XML section. * - * @param mixed $search - * @param mixed $replace * @param string $documentPartXML * @param int $limit * @@ -1048,25 +1037,31 @@ protected function findRowStart($offset) * * @param int $offset * - * @return int + * @return int|null */ protected function findRowEnd($offset) { - return strpos($this->tempDocumentMainPart, '', $offset) + 7; + $position = strpos($this->tempDocumentMainPart, '', $offset); + + return $position === false ? null : $position + 7; } /** * Get a slice of a string. - * - * @param int $startPosition - * @param int $endPosition - * - * @return string */ - protected function getSlice($startPosition, $endPosition = 0) + protected function getSlice(int $startPosition, int $endPosition = 0): string { + $length = strlen($this->tempDocumentMainPart); if (!$endPosition) { - $endPosition = strlen($this->tempDocumentMainPart); + $endPosition = $length; + } + + if ($startPosition < 0) { + throw new Exception(sprintf('Start position must be at least 0. `%s` provided', $startPosition)); + } elseif ($endPosition < $startPosition) { + throw new Exception(sprintf('Provided end position `%s` is smaller than the start position `%s`', $endPosition, $startPosition)); + } elseif ($endPosition > $length) { + throw new Exception(sprintf('Provided end position `%s` is longer than the length `%s`', $endPosition, $length)); } return substr($this->tempDocumentMainPart, $startPosition, ($endPosition - $startPosition)); diff --git a/src/PhpWord/Writer/AbstractWriter.php b/src/PhpWord/Writer/AbstractWriter.php index 2c1ad29460..f3a17004db 100644 --- a/src/PhpWord/Writer/AbstractWriter.php +++ b/src/PhpWord/Writer/AbstractWriter.php @@ -1,4 +1,5 @@ getStyle(); $cellBgColor = $cellStyle->getBgColor(); $cellFgColor = null; - if ($cellBgColor) { - $red = hexdec(substr($cellBgColor, 0, 2)); - $green = hexdec(substr($cellBgColor, 2, 2)); - $blue = hexdec(substr($cellBgColor, 4, 2)); - $cellFgColor = (($red * 0.299 + $green * 0.587 + $blue * 0.114) > 186) ? null : 'ffffff'; + if ($cellBgColor instanceof StaticColorInterface) { + list($red, $green, $blue) = $cellBgColor->toRgb(); + if (($red * 0.299 + $green * 0.587 + $blue * 0.114) <= 186) { + $cellFgColor = 'FFFFFF'; + } } $cellColSpan = $cellStyle->getGridSpan(); $cellRowSpan = 1; @@ -82,7 +85,7 @@ public function write() $cellTag = $tblHeader ? 'th' : 'td'; $cellColSpanAttr = (is_numeric($cellColSpan) && ($cellColSpan > 1) ? " colspan=\"{$cellColSpan}\"" : ''); $cellRowSpanAttr = ($cellRowSpan > 1 ? " rowspan=\"{$cellRowSpan}\"" : ''); - $cellBgColorAttr = (is_null($cellBgColor) ? '' : " bgcolor=\"#{$cellBgColor}\""); + $cellBgColorAttr = (is_null($cellBgColor->toHex()) ? '' : " bgcolor=\"{$cellBgColor->toHex(true)}\""); $cellFgColorAttr = (is_null($cellFgColor) ? '' : " color=\"#{$cellFgColor}\""); $content .= "<{$cellTag}{$cellColSpanAttr}{$cellRowSpanAttr}{$cellBgColorAttr}{$cellFgColorAttr}>" . PHP_EOL; $writer = new Container($this->parentWriter, $rowCells[$j]); diff --git a/src/PhpWord/Writer/HTML/Element/Text.php b/src/PhpWord/Writer/HTML/Element/Text.php index 04d76a8327..f49a96e182 100644 --- a/src/PhpWord/Writer/HTML/Element/Text.php +++ b/src/PhpWord/Writer/HTML/Element/Text.php @@ -1,4 +1,5 @@ array( 'font-family' => Settings::getDefaultFontName(), - 'font-size' => Settings::getDefaultFontSize() . 'pt', + 'font-size' => Settings::getDefaultFontSize()->toInt('pt') . 'pt', ), 'a.NoteRef' => array( 'text-decoration' => 'none', diff --git a/src/PhpWord/Writer/HTML/Style/AbstractStyle.php b/src/PhpWord/Writer/HTML/Style/AbstractStyle.php index cfb54cb8d7..4292002cd7 100644 --- a/src/PhpWord/Writer/HTML/Style/AbstractStyle.php +++ b/src/PhpWord/Writer/HTML/Style/AbstractStyle.php @@ -1,4 +1,5 @@ getName(); - $size = $style->getSize(); - $color = $style->getColor(); - $fgColor = $style->getFgColor(); + $size = $style->getSize()->toInt('pt'); + $color = $style->getColor()->toHexOrName(true); + $fgColor = $style->getFgColor()->getName(); $underline = $style->getUnderline() != FontStyle::UNDERLINE_NONE; $lineThrough = $style->isStrikethrough() || $style->isDoubleStrikethrough(); $css['font-family'] = $this->getValueIf($font !== null, "'{$font}'"); $css['font-size'] = $this->getValueIf($size !== null, "{$size}pt"); - $css['color'] = $this->getValueIf($color !== null, "#{$color}"); + $css['color'] = $this->getValueIf($color !== null, $color); $css['background'] = $this->getValueIf($fgColor != '', $fgColor); $css['font-weight'] = $this->getValueIf($style->isBold(), 'bold'); $css['font-style'] = $this->getValueIf($style->isItalic(), 'italic'); @@ -62,7 +63,7 @@ public function write() $css['font-variant'] = $this->getValueIf($style->isSmallCaps(), 'small-caps'); $css['display'] = $this->getValueIf($style->isHidden(), 'none'); - $spacing = $style->getSpacing(); + $spacing = $style->getSpacing()->toInt('pt'); $css['letter-spacing'] = $this->getValueIf(!is_null($spacing), ($spacing / 20) . 'pt'); return $this->assembleCss($css); diff --git a/src/PhpWord/Writer/HTML/Style/Generic.php b/src/PhpWord/Writer/HTML/Style/Generic.php index ee5d089607..ba205fbc19 100644 --- a/src/PhpWord/Writer/HTML/Style/Generic.php +++ b/src/PhpWord/Writer/HTML/Style/Generic.php @@ -1,4 +1,5 @@ getWidth(); - $height = $style->getHeight(); + $dpi = new Dpi(); + $width = $style->getWidth()->toPixels($dpi); + $height = $style->getHeight()->toPixels($dpi); $css['width'] = $this->getValueIf(is_numeric($width), $width . 'px'); $css['height'] = $this->getValueIf(is_numeric($height), $height . 'px'); diff --git a/src/PhpWord/Writer/HTML/Style/Paragraph.php b/src/PhpWord/Writer/HTML/Style/Paragraph.php index 863ef93b14..d69bab363d 100644 --- a/src/PhpWord/Writer/HTML/Style/Paragraph.php +++ b/src/PhpWord/Writer/HTML/Style/Paragraph.php @@ -1,4 +1,5 @@ getSpace(); if (!is_null($spacing)) { - $before = $spacing->getBefore(); - $after = $spacing->getAfter(); - $css['margin-top'] = $this->getValueIf(!is_null($before), ($before / 20) . 'pt'); - $css['margin-bottom'] = $this->getValueIf(!is_null($after), ($after / 20) . 'pt'); + $before = $spacing->getBefore()->toInt('pt'); + $after = $spacing->getAfter()->toInt('pt'); + $css['margin-top'] = $this->getValueIf(!is_null($before), $before . 'pt'); + $css['margin-bottom'] = $this->getValueIf(!is_null($after), $after . 'pt'); } else { $css['margin-top'] = '0'; $css['margin-bottom'] = '0'; diff --git a/src/PhpWord/Writer/ODText.php b/src/PhpWord/Writer/ODText.php index efd0d6a923..4ba268e392 100644 --- a/src/PhpWord/Writer/ODText.php +++ b/src/PhpWord/Writer/ODText.php @@ -1,4 +1,5 @@ getMediaIndex(); $target = 'Pictures/' . $element->getTarget(); $style = $element->getStyle(); - $width = Converter::pixelToCm($style->getWidth()); - $height = Converter::pixelToCm($style->getHeight()); + $width = $style->getWidth()->toInt('cm'); + $height = $style->getHeight()->toInt('cm'); $xmlWriter->startElement('text:p'); $xmlWriter->writeAttribute('text:style-name', 'Standard'); diff --git a/src/PhpWord/Writer/ODText/Element/Link.php b/src/PhpWord/Writer/ODText/Element/Link.php index d6fec50777..9b09f14c94 100644 --- a/src/PhpWord/Writer/ODText/Element/Link.php +++ b/src/PhpWord/Writer/ODText/Element/Link.php @@ -1,4 +1,5 @@ startElement('style:text-properties'); $xmlWriter->writeAttribute('style:use-window-font-color', 'true'); $xmlWriter->writeAttribute('style:font-name', Settings::getDefaultFontName()); - $xmlWriter->writeAttribute('fo:font-size', Settings::getDefaultFontSize() . 'pt'); + $xmlWriter->writeAttribute('fo:font-size', Settings::getDefaultFontSize()->toInt('pt') . 'pt'); $xmlWriter->writeAttribute('fo:language', $latinLang[0]); $xmlWriter->writeAttribute('fo:country', $latinLang[1]); $xmlWriter->writeAttribute('style:letter-kerning', 'true'); $xmlWriter->writeAttribute('style:font-name-asian', Settings::getDefaultFontName() . '2'); - $xmlWriter->writeAttribute('style:font-size-asian', Settings::getDefaultFontSize() . 'pt'); + $xmlWriter->writeAttribute('style:font-size-asian', Settings::getDefaultFontSize()->toInt('pt') . 'pt'); $xmlWriter->writeAttribute('style:language-asian', $asianLang[0]); $xmlWriter->writeAttribute('style:country-asian', $asianLang[1]); $xmlWriter->writeAttribute('style:font-name-complex', Settings::getDefaultFontName() . '2'); - $xmlWriter->writeAttribute('style:font-size-complex', Settings::getDefaultFontSize() . 'pt'); + $xmlWriter->writeAttribute('style:font-size-complex', Settings::getDefaultFontSize()->toInt('pt') . 'pt'); $xmlWriter->writeAttribute('style:language-complex', $complexLang[0]); $xmlWriter->writeAttribute('style:country-complex', $complexLang[1]); $xmlWriter->writeAttribute('fo:hyphenate', 'false'); @@ -113,8 +112,6 @@ private function writeDefault(XMLWriter $xmlWriter) /** * Write named styles. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter */ private function writeNamed(XMLWriter $xmlWriter) { @@ -135,8 +132,6 @@ private function writeNamed(XMLWriter $xmlWriter) /** * Write page layout styles. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter */ private function writePageLayout(XMLWriter $xmlWriter) { @@ -186,8 +181,6 @@ private function writePageLayout(XMLWriter $xmlWriter) /** * Write master style. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter */ private function writeMaster(XMLWriter $xmlWriter) { diff --git a/src/PhpWord/Writer/ODText/Style/AbstractStyle.php b/src/PhpWord/Writer/ODText/Style/AbstractStyle.php index f7679ab22c..88b6aa0a7d 100644 --- a/src/PhpWord/Writer/ODText/Style/AbstractStyle.php +++ b/src/PhpWord/Writer/ODText/Style/AbstractStyle.php @@ -1,4 +1,5 @@ getName(); $xmlWriter->writeAttributeIf($font != '', 'style:font-name', $font); $xmlWriter->writeAttributeIf($font != '', 'style:font-name-complex', $font); - $size = $style->getSize(); // Size - $xmlWriter->writeAttributeIf(is_numeric($size), 'fo:font-size', $size . 'pt'); - $xmlWriter->writeAttributeIf(is_numeric($size), 'style:font-size-asian', $size . 'pt'); - $xmlWriter->writeAttributeIf(is_numeric($size), 'style:font-size-complex', $size . 'pt'); + $size = $style->getSize()->toInt('pt'); + $xmlWriter->writeAttributeIf($size !== null, 'fo:font-size', $size . 'pt'); + $xmlWriter->writeAttributeIf($size !== null, 'style:font-size-asian', $size . 'pt'); + $xmlWriter->writeAttributeIf($size !== null, 'style:font-size-complex', $size . 'pt'); // Color - $color = $style->getColor(); - $xmlWriter->writeAttributeIf($color != '', 'fo:color', '#' . $color); + $color = $style->getColor()->toHexOrName(true); + $xmlWriter->writeAttributeIf($color != '', 'fo:color', $color); // Bold & italic $xmlWriter->writeAttributeIf($style->isBold(), 'fo:font-weight', 'bold'); diff --git a/src/PhpWord/Writer/ODText/Style/Image.php b/src/PhpWord/Writer/ODText/Style/Image.php index 13005a7f03..97ea75cfa5 100644 --- a/src/PhpWord/Writer/ODText/Style/Image.php +++ b/src/PhpWord/Writer/ODText/Style/Image.php @@ -1,4 +1,5 @@ getXmlWriter(); - $marginTop = (is_null($style->getSpaceBefore()) || $style->getSpaceBefore() == 0) ? '0' : round(17.6 / $style->getSpaceBefore(), 2); - $marginBottom = (is_null($style->getSpaceAfter()) || $style->getSpaceAfter() == 0) ? '0' : round(17.6 / $style->getSpaceAfter(), 2); + $spaceBefore = $style->getSpaceBefore()->toInt('twip'); + $spaceAfter = $style->getSpaceAfter()->toInt('twip'); + $marginTop = (is_null($spaceBefore) || $spaceBefore == 0) ? '0' : round(17.6 / $spaceBefore, 2); + $marginBottom = (is_null($spaceAfter) || $spaceAfter == 0) ? '0' : round(17.6 / $spaceAfter, 2); $xmlWriter->startElement('style:style'); $xmlWriter->writeAttribute('style:name', $style->getStyleName()); diff --git a/src/PhpWord/Writer/ODText/Style/Section.php b/src/PhpWord/Writer/ODText/Style/Section.php index 92d8891170..eaf10cbda8 100644 --- a/src/PhpWord/Writer/ODText/Style/Section.php +++ b/src/PhpWord/Writer/ODText/Style/Section.php @@ -1,4 +1,5 @@ getStyle(); - if (!$style instanceof \PhpOffice\PhpWord\Style\Table) { + if ($style === null) { return; + } elseif (is_string($style)) { + throw new Exception(sprintf('Incorrect value provided for style. `%s` expected, `string(%s)` provided', TableStyle::class, $style)); + } elseif (!$style instanceof TableStyle) { + throw new Exception(sprintf('Incorrect value provided for style. %s expected, %s provided', TableStyle::class, get_class($style))); } $xmlWriter = $this->getXmlWriter(); @@ -52,11 +62,21 @@ public function write() for ($i = 0; $i < $countCellWidths; $i++) { $width = $cellWidths[$i]; + if ($width instanceof Percent) { + $width = number_format($width->toFloat(), 2) . '%'; + } elseif ($width instanceof Absolute) { + $width = $width->toFloat('cm') . 'cm'; + } elseif ($width instanceof Auto) { + $width = null; + } else { + throw new Exception('Unsupported width `' . get_class($width) . '` provided'); + } + $xmlWriter->startElement('style:style'); $xmlWriter->writeAttribute('style:name', $style->getStyleName() . '.' . $i); $xmlWriter->writeAttribute('style:family', 'table-column'); $xmlWriter->startElement('style:table-column-properties'); - $xmlWriter->writeAttribute('style:column-width', number_format($width * 0.0017638889, 2, '.', '') . 'cm'); + $xmlWriter->writeAttributeIf($width !== null, 'style:column-width', $width); $xmlWriter->endElement(); // style:table-column-properties $xmlWriter->endElement(); // style:style } diff --git a/src/PhpWord/Writer/PDF.php b/src/PhpWord/Writer/PDF.php index 64dcc7898d..6a2e071068 100644 --- a/src/PhpWord/Writer/PDF.php +++ b/src/PhpWord/Writer/PDF.php @@ -1,4 +1,5 @@ writeOpening(); $content .= '{\*\shppict {\pict'; $content .= '\pngblip\picscalex100\picscaley100'; - $content .= '\picwgoal' . round(Converter::pixelToTwip($style->getWidth())); - $content .= '\pichgoal' . round(Converter::pixelToTwip($style->getHeight())); + $content .= '\picwgoal' . $style->getWidth()->toInt('twip'); + $content .= '\pichgoal' . $style->getHeight()->toInt('twip'); $content .= PHP_EOL; $content .= $this->element->getImageStringData(); $content .= '}}'; diff --git a/src/PhpWord/Writer/RTF/Element/Link.php b/src/PhpWord/Writer/RTF/Element/Link.php index 25954ed878..3a16ac7311 100644 --- a/src/PhpWord/Writer/RTF/Element/Link.php +++ b/src/PhpWord/Writer/RTF/Element/Link.php @@ -1,4 +1,5 @@ getCells() as $cell) { - $width = $cell->getWidth(); - $vMerge = $this->getVMerge($cell->getStyle()->getVMerge()); - if ($width === null) { - $width = 720; // Arbitrary default width + // Arbitrary default width + $width = 720; + $cellWidth = $cell->getWidth(); + if ($cellWidth instanceof Absolute && $cellWidth->isSpecified()) { + $width = $cellWidth->toInt('twip'); } + unset($cellWidth); + + $vMerge = $this->getVMerge($cell->getStyle()->getVMerge()); $rightMargin += $width; $content .= "{$vMerge}\cellx{$rightMargin} "; } @@ -90,7 +96,6 @@ private function writeRowDef(RowElement $row) /** * Write row * - * @param \PhpOffice\PhpWord\Element\Row $row * @return string */ private function writeRow(RowElement $row) @@ -108,7 +113,6 @@ private function writeRow(RowElement $row) /** * Write cell * - * @param \PhpOffice\PhpWord\Element\Cell $cell * @return string */ private function writeCell(CellElement $cell) diff --git a/src/PhpWord/Writer/RTF/Element/Text.php b/src/PhpWord/Writer/RTF/Element/Text.php index b9e56e890b..682ece035f 100644 --- a/src/PhpWord/Writer/RTF/Element/Text.php +++ b/src/PhpWord/Writer/RTF/Element/Text.php @@ -1,4 +1,5 @@ escaper->escape($docProps->$method()); } else { @@ -104,7 +105,7 @@ private function writeFormatting() $content .= '\nowidctlpar'; // No widow/orphan control $content .= '\lang' . $langId; $content .= '\kerning1'; // Point size (in half-points) above which to kern character pairs - $content .= '\fs' . (Settings::getDefaultFontSize() * 2); // Set the font size in half-points + $content .= '\fs' . Settings::getDefaultFontSize()->toInt('hpt'); // Set the font size in half-points $content .= PHP_EOL; return $content; diff --git a/src/PhpWord/Writer/RTF/Part/Header.php b/src/PhpWord/Writer/RTF/Part/Header.php index 01439bc6ad..bfb9992bea 100644 --- a/src/PhpWord/Writer/RTF/Part/Header.php +++ b/src/PhpWord/Writer/RTF/Part/Header.php @@ -1,4 +1,5 @@ colorTable as $color) { - list($red, $green, $blue) = Converter::htmlToRgb($color); + list($red, $green, $blue) = $color->toRgb(); $content .= "\\red{$red}\\green{$green}\\blue{$blue};"; } $content .= '}'; @@ -209,17 +213,18 @@ private function registerFont() /** * Register border colors. - * - * @param \PhpOffice\PhpWord\Style\Border $style */ - private function registerBorderColor($style) + private function registerBorderColor(SectionStyle $style): self { - $colors = $style->getBorderColor(); - foreach ($colors as $color) { - if ($color !== null) { - $this->registerTableItem($this->colorTable, $color); + $borders = $style->getBorders(); + foreach ($borders as $border) { + $color = $border->getColor(); + if ($color->isSpecified()) { + $this->registerTableItem($this->colorTable, $color, new Hex('000000')); } } + + return $this; } /** @@ -230,7 +235,7 @@ private function registerBorderColor($style) private function registerFontItems($style) { $defaultFont = Settings::getDefaultFontName(); - $defaultColor = Settings::DEFAULT_FONT_COLOR; + $defaultColor = new Hex(Settings::DEFAULT_FONT_COLOR); if ($style instanceof Font) { $this->registerTableItem($this->fontTable, $style->getName(), $defaultFont); @@ -243,10 +248,10 @@ private function registerFontItems($style) * Register individual font and color. * * @param array &$table - * @param string $value - * @param string $default + * @param BasicColor|string $value + * @param BasicColor|string $default */ - private function registerTableItem(&$table, $value, $default = null) + private function registerTableItem(&$table, $value, $default) { if (in_array($value, $table) === false && $value !== null && $value != $default) { $table[] = $value; diff --git a/src/PhpWord/Writer/RTF/Style/AbstractStyle.php b/src/PhpWord/Writer/RTF/Style/AbstractStyle.php index 57aa6bb9f7..8cf957f25f 100644 --- a/src/PhpWord/Writer/RTF/Style/AbstractStyle.php +++ b/src/PhpWord/Writer/RTF/Style/AbstractStyle.php @@ -1,4 +1,5 @@ sizes); - // Page border measure // 8 = from text, infront off; 32 = from edge, infront on; 40 = from edge, infront off $content .= '\pgbrdropt32'; - for ($i = 0; $i < $sizeCount; $i++) { - if ($this->sizes[$i] !== null) { - $color = null; - if (isset($this->colors[$i])) { - $color = $this->colors[$i]; - } - $content .= $this->writeSide($sides[$i], $this->sizes[$i], $color); - } + foreach ($this->borders as $side => $border) { + $content .= $this->writeSide($side, $border); } return $content; @@ -69,20 +57,19 @@ public function write() /** * Write side - * - * @param string $side - * @param int $width - * @param string $color - * @return string */ - private function writeSide($side, $width, $color = '') + private function writeSide(string $side, BorderSide $border): string { + if (!in_array($side, array('top', 'bottom', 'left', 'right'))) { + throw new Exception(sprintf('Invalid side `%s` provided', $side)); + } + /** @var \PhpOffice\PhpWord\Writer\RTF $rtfWriter */ $rtfWriter = $this->getParentWriter(); $colorIndex = 0; if ($rtfWriter !== null) { $colorTable = $rtfWriter->getColorTable(); - $index = array_search($color, $colorTable); + $index = array_search($border->getColor()->toHexOrName(), $colorTable); if ($index !== false && $colorIndex !== null) { $colorIndex = $index + 1; } @@ -92,7 +79,7 @@ private function writeSide($side, $width, $color = '') $content .= '\pgbrdr' . substr($side, 0, 1); $content .= '\brdrs'; // Single-thickness border; @todo Get other type of border - $content .= '\brdrw' . round($width); // Width + $content .= '\brdrw' . $border->getSize()->toInt('twip'); // Width $content .= '\brdrcf' . $colorIndex; // Color $content .= '\brsp480'; // Space in twips between borders and the paragraph (24pt, following OOXML) $content .= ' '; @@ -101,22 +88,28 @@ private function writeSide($side, $width, $color = '') } /** - * Set sizes. - * - * @param int[] $value + * @param BorderSide[] $borders */ - public function setSizes($value) + public function setBorders(array $borders): self { - $this->sizes = $value; + foreach ($borders as $side => $border) { + $this->setBorder($side, $border); + } + + return $this; } /** - * Set colors. - * - * @param string[] $value + * Set sizes. */ - public function setColors($value) + public function setBorder(string $side, BorderSide $border): self { - $this->colors = $value; + if (!in_array($side, array('top', 'bottom', 'left', 'right'))) { + throw new Exception(sprintf('Invalid side `%s` provided', $side)); + } + + $this->borders[$side] = $border; + + return $this; } } diff --git a/src/PhpWord/Writer/RTF/Style/Font.php b/src/PhpWord/Writer/RTF/Style/Font.php index b9001ea0ea..6da332c3ad 100644 --- a/src/PhpWord/Writer/RTF/Style/Font.php +++ b/src/PhpWord/Writer/RTF/Style/Font.php @@ -1,4 +1,5 @@ colorIndex; $content .= '\f' . $this->nameIndex; - $size = $style->getSize(); - $content .= $this->getValueIf(is_numeric($size), '\fs' . round($size * 2)); + $size = $style->getSize()->toInt('pt'); + $content .= $this->getValueIf($size !== null, '\fs' . $size); $content .= $this->getValueIf($style->isBold(), '\b'); $content .= $this->getValueIf($style->isItalic(), '\i'); diff --git a/src/PhpWord/Writer/RTF/Style/Indentation.php b/src/PhpWord/Writer/RTF/Style/Indentation.php index 50e8ad9940..5d006e86a9 100644 --- a/src/PhpWord/Writer/RTF/Style/Indentation.php +++ b/src/PhpWord/Writer/RTF/Style/Indentation.php @@ -1,4 +1,5 @@ getFirstLine()); - $content .= '\li' . round($style->getLeft()); - $content .= '\ri' . round($style->getRight()); + $content = '\fi' . $style->getFirstLine()->toInt('twip'); + $content .= '\li' . $style->getLeft()->toInt('twip'); + $content .= '\ri' . $style->getRight()->toInt('twip'); return $content . ' '; } diff --git a/src/PhpWord/Writer/RTF/Style/Paragraph.php b/src/PhpWord/Writer/RTF/Style/Paragraph.php index 8ef3e146d9..7dee1a36ef 100644 --- a/src/PhpWord/Writer/RTF/Style/Paragraph.php +++ b/src/PhpWord/Writer/RTF/Style/Paragraph.php @@ -1,4 +1,5 @@ '\qj', ); - $spaceAfter = $style->getSpaceAfter(); - $spaceBefore = $style->getSpaceBefore(); - $content = ''; if ($this->nestedLevel == 0) { $content .= '\pard\nowidctlpar '; @@ -65,7 +63,9 @@ public function write() $content .= $alignments[$style->getAlignment()]; } $content .= $this->writeIndentation($style->getIndentation()); + $spaceBefore = $style->getSpaceBefore()->toInt('twip'); $content .= $this->getValueIf($spaceBefore !== null, '\sb' . round($spaceBefore)); + $spaceAfter = $style->getSpaceAfter()->toInt('twip'); $content .= $this->getValueIf($spaceAfter !== null, '\sa' . round($spaceAfter)); $styles = $style->getStyleValues(); diff --git a/src/PhpWord/Writer/RTF/Style/Section.php b/src/PhpWord/Writer/RTF/Style/Section.php index 190bb67036..c3b53c7840 100644 --- a/src/PhpWord/Writer/RTF/Style/Section.php +++ b/src/PhpWord/Writer/RTF/Style/Section.php @@ -1,4 +1,5 @@ getValueIf($style->getPageSizeW() !== null, '\pgwsxn' . round($style->getPageSizeW())); - $content .= $this->getValueIf($style->getPageSizeH() !== null, '\pghsxn' . round($style->getPageSizeH())); + $content .= $this->getValueIf($style->getPageSizeW()->toInt('twip') !== null, '\pgwsxn' . $style->getPageSizeW()->toInt('twip')); + $content .= $this->getValueIf($style->getPageSizeH()->toInt('twip') !== null, '\pghsxn' . $style->getPageSizeH()->toInt('twip')); $content .= ' '; - $content .= $this->getValueIf($style->getMarginTop() !== null, '\margtsxn' . round($style->getMarginTop())); - $content .= $this->getValueIf($style->getMarginRight() !== null, '\margrsxn' . round($style->getMarginRight())); - $content .= $this->getValueIf($style->getMarginBottom() !== null, '\margbsxn' . round($style->getMarginBottom())); - $content .= $this->getValueIf($style->getMarginLeft() !== null, '\marglsxn' . round($style->getMarginLeft())); - $content .= $this->getValueIf($style->getHeaderHeight() !== null, '\headery' . round($style->getHeaderHeight())); - $content .= $this->getValueIf($style->getFooterHeight() !== null, '\footery' . round($style->getFooterHeight())); - $content .= $this->getValueIf($style->getGutter() !== null, '\guttersxn' . round($style->getGutter())); + $content .= $this->getValueIf($style->getMarginTop()->toInt('twip') !== null, '\margtsxn' . $style->getMarginTop()->toInt('twip')); + $content .= $this->getValueIf($style->getMarginRight()->toInt('twip') !== null, '\margrsxn' . $style->getMarginRight()->toInt('twip')); + $content .= $this->getValueIf($style->getMarginBottom()->toInt('twip') !== null, '\margbsxn' . $style->getMarginBottom()->toInt('twip')); + $content .= $this->getValueIf($style->getMarginLeft()->toInt('twip') !== null, '\marglsxn' . $style->getMarginLeft()->toInt('twip')); + $content .= $this->getValueIf($style->getHeaderHeight()->toInt('twip') !== null, '\headery' . $style->getHeaderHeight()->toInt('twip')); + $content .= $this->getValueIf($style->getFooterHeight()->toInt('twip') !== null, '\footery' . $style->getFooterHeight()->toInt('twip')); + $content .= $this->getValueIf($style->getGutter()->toInt('twip') !== null, '\guttersxn' . $style->getGutter()->toInt('twip')); $content .= ' '; // Borders if ($style->hasBorder()) { $styleWriter = new Border($style); $styleWriter->setParentWriter($this->getParentWriter()); - $styleWriter->setSizes($style->getBorderSize()); - $styleWriter->setColors($style->getBorderColor()); + $styleWriter->setBorders($style->getBorders()); $content .= $styleWriter->write(); } diff --git a/src/PhpWord/Writer/RTF/Style/Tab.php b/src/PhpWord/Writer/RTF/Style/Tab.php index a21b13d39d..13910610a1 100644 --- a/src/PhpWord/Writer/RTF/Style/Tab.php +++ b/src/PhpWord/Writer/RTF/Style/Tab.php @@ -1,4 +1,5 @@ getType()])) { $content .= $tabs[$style->getType()]; } - $content .= '\tx' . round($style->getPosition()); + $content .= '\tx' . round($style->getPosition()->toInt('twip')); return $content; } diff --git a/src/PhpWord/Writer/Word2007.php b/src/PhpWord/Writer/Word2007.php index eee215bef0..56abfbeb3a 100644 --- a/src/PhpWord/Writer/Word2007.php +++ b/src/PhpWord/Writer/Word2007.php @@ -1,4 +1,5 @@ startElement('wp:inline'); // EMU - $xmlWriter->writeElementBlock('wp:extent', array('cx' => $style->getWidth(), 'cy' => $style->getHeight())); + $xmlWriter->writeElementBlock('wp:extent', array('cx' => $style->getWidth()->toInt('emu'), 'cy' => $style->getHeight()->toInt('emu'))); $xmlWriter->writeElementBlock('wp:docPr', array('id' => $rId, 'name' => "Chart{$rId}")); $xmlWriter->startElement('a:graphic'); diff --git a/src/PhpWord/Writer/Word2007/Element/CheckBox.php b/src/PhpWord/Writer/Word2007/Element/CheckBox.php index 05692a074d..55a81c06a3 100644 --- a/src/PhpWord/Writer/Word2007/Element/CheckBox.php +++ b/src/PhpWord/Writer/Word2007/Element/CheckBox.php @@ -1,4 +1,5 @@ getType(); - if ($type == 'rect' && $style->getRoundness() !== null) { + if ($type == 'rect' && $style->getRoundness()->isSpecified()) { $type = 'roundrect'; } $method = "write{$type}"; @@ -76,9 +77,6 @@ public function write() /** * Write arc. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Shape $style */ private function writeArc(XMLWriter $xmlWriter, ShapeStyle $style) { @@ -90,9 +88,6 @@ private function writeArc(XMLWriter $xmlWriter, ShapeStyle $style) /** * Write curve. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Shape $style */ private function writeCurve(XMLWriter $xmlWriter, ShapeStyle $style) { @@ -105,9 +100,6 @@ private function writeCurve(XMLWriter $xmlWriter, ShapeStyle $style) /** * Write line. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Shape $style */ private function writeLine(XMLWriter $xmlWriter, ShapeStyle $style) { @@ -119,9 +111,6 @@ private function writeLine(XMLWriter $xmlWriter, ShapeStyle $style) /** * Write polyline. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Shape $style */ private function writePolyline(XMLWriter $xmlWriter, ShapeStyle $style) { @@ -130,13 +119,10 @@ private function writePolyline(XMLWriter $xmlWriter, ShapeStyle $style) /** * Write rectangle. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Shape $style */ private function writeRoundRect(XMLWriter $xmlWriter, ShapeStyle $style) { - $xmlWriter->writeAttribute('arcsize', $style->getRoundness()); + $xmlWriter->writeAttribute('arcsize', $style->getRoundness()->toFloat() / 100); } /** diff --git a/src/PhpWord/Writer/Word2007/Element/TOC.php b/src/PhpWord/Writer/Word2007/Element/TOC.php index 94437cbf0d..8a5f737d20 100644 --- a/src/PhpWord/Writer/Word2007/Element/TOC.php +++ b/src/PhpWord/Writer/Word2007/Element/TOC.php @@ -1,4 +1,5 @@ getStyleFont(); $isObject = ($fontStyle instanceof Font) ? true : false; $rId = $title->getRelationId(); - $indent = ($title->getDepth() - 1) * $tocStyle->getIndent(); + $indent = Absolute::from('twip', ($title->getDepth() - 1) * $tocStyle->getIndent()->toInt('twip')); $xmlWriter->startElement('w:p'); @@ -131,12 +131,8 @@ private function writeTitle(XMLWriter $xmlWriter, TOCElement $element, $title, $ /** * Write style - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Element\TOC $element - * @param int $indent */ - private function writeStyle(XMLWriter $xmlWriter, TOCElement $element, $indent) + private function writeStyle(XMLWriter $xmlWriter, TOCElement $element, Absolute $indent) { $tocStyle = $element->getStyleTOC(); $fontStyle = $element->getStyleFont(); @@ -166,6 +162,7 @@ private function writeStyle(XMLWriter $xmlWriter, TOCElement $element, $indent) $xmlWriter->endElement(); // Indent + $indent = $indent->toInt('twip'); if ($indent > 0) { $xmlWriter->startElement('w:ind'); $xmlWriter->writeAttribute('w:left', $indent); @@ -177,9 +174,6 @@ private function writeStyle(XMLWriter $xmlWriter, TOCElement $element, $indent) /** * Write TOC Field. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Element\TOC $element */ private function writeFieldMark(XMLWriter $xmlWriter, TOCElement $element) { diff --git a/src/PhpWord/Writer/Word2007/Element/Table.php b/src/PhpWord/Writer/Word2007/Element/Table.php index c365b028ab..a3ff851f36 100644 --- a/src/PhpWord/Writer/Word2007/Element/Table.php +++ b/src/PhpWord/Writer/Word2007/Element/Table.php @@ -1,4 +1,5 @@ startElement('w:tblGrid'); foreach ($cellWidths as $width) { $xmlWriter->startElement('w:gridCol'); - if ($width !== null) { - $xmlWriter->writeAttribute('w:w', $width); - $xmlWriter->writeAttribute('w:type', 'dxa'); - } + $xmlWriter->writeAttribute('w:w', $width->toInt('twip')); $xmlWriter->endElement(); } $xmlWriter->endElement(); // w:tblGrid @@ -93,8 +92,7 @@ private function writeColumns(XMLWriter $xmlWriter, TableElement $element) /** * Write row. * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Element\Row $row + * @see http://officeopenxml.com/WPtableRowProperties.php */ private function writeRow(XMLWriter $xmlWriter, RowElement $row) { @@ -119,8 +117,7 @@ private function writeRow(XMLWriter $xmlWriter, RowElement $row) /** * Write cell. * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Element\Cell $cell + * @see http://officeopenxml.com/WPtableCellProperties-Width.php */ private function writeCell(XMLWriter $xmlWriter, CellElement $cell) { @@ -130,6 +127,7 @@ private function writeCell(XMLWriter $xmlWriter, CellElement $cell) $cellStyle = $cell->getStyle(); if ($cellStyle instanceof CellStyle) { $styleWriter = new CellStyleWriter($xmlWriter, $cellStyle); + // Override width set in style. $styleWriter->setWidth($cell->getWidth()); $styleWriter->write(); } diff --git a/src/PhpWord/Writer/Word2007/Element/TableAlignment.php b/src/PhpWord/Writer/Word2007/Element/TableAlignment.php index f44e9ebe42..0c04def485 100644 --- a/src/PhpWord/Writer/Word2007/Element/TableAlignment.php +++ b/src/PhpWord/Writer/Word2007/Element/TableAlignment.php @@ -1,4 +1,5 @@ startElement('c:pt'); $xmlWriter->writeAttribute('idx', $index); if (\PhpOffice\PhpWord\Settings::isOutputEscapingEnabled()) { - $xmlWriter->writeElement('c:v', $value); + $xmlWriter->writeElement('c:v', (string) $value); } else { $xmlWriter->startElement('c:v'); - $xmlWriter->writeRaw($value); + $xmlWriter->writeRaw((string) $value); $xmlWriter->endElement(); // c:v } $xmlWriter->endElement(); // c:pt @@ -335,7 +330,6 @@ private function writeSeriesItem(XMLWriter $xmlWriter, $type, $values) * Write axis * * @see http://www.datypic.com/sc/ooxml/t-draw-chart_CT_CatAx.html - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param string $type */ private function writeAxis(XMLWriter $xmlWriter, $type) @@ -400,7 +394,6 @@ private function writeAxis(XMLWriter $xmlWriter, $type) * Write shape * * @see http://www.datypic.com/sc/ooxml/t-a_CT_ShapeProperties.html - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param bool $line */ private function writeShape(XMLWriter $xmlWriter, $line = false) diff --git a/src/PhpWord/Writer/Word2007/Part/Comments.php b/src/PhpWord/Writer/Word2007/Part/Comments.php index 33c9f59e6a..e8c4dcb672 100644 --- a/src/PhpWord/Writer/Word2007/Part/Comments.php +++ b/src/PhpWord/Writer/Word2007/Part/Comments.php @@ -1,4 +1,5 @@ writeAttribute('name', $property); switch ($propertyType) { case 'i': - $xmlWriter->writeElement('vt:i4', $propertyValue); + $xmlWriter->writeElement('vt:i4', (string) $propertyValue); break; case 'f': - $xmlWriter->writeElement('vt:r8', $propertyValue); + $xmlWriter->writeElement('vt:r8', (string) $propertyValue); break; case 'b': $xmlWriter->writeElement('vt:bool', ($propertyValue) ? 'true' : 'false'); @@ -63,11 +64,11 @@ public function write() if ($propertyValue instanceof \DateTime) { $xmlWriter->writeElement('vt:filetime', $propertyValue->format($this->dateFormat)); } else { - $xmlWriter->writeElement('vt:filetime', date($this->dateFormat, $propertyValue)); + $xmlWriter->writeElement('vt:filetime', date($this->dateFormat, (int) $propertyValue)); } break; default: - $xmlWriter->writeElement('vt:lpwstr', $propertyValue); + $xmlWriter->writeElement('vt:lpwstr', (string) $propertyValue); break; } $xmlWriter->endElement(); // property diff --git a/src/PhpWord/Writer/Word2007/Part/Document.php b/src/PhpWord/Writer/Word2007/Part/Document.php index 986b498570..317920f349 100644 --- a/src/PhpWord/Writer/Word2007/Part/Document.php +++ b/src/PhpWord/Writer/Word2007/Part/Document.php @@ -1,4 +1,5 @@ startElement('w:tabs'); $xmlWriter->startElement('w:tab'); $xmlWriter->writeAttribute('w:val', 'num'); - $xmlWriter->writeAttributeIf($tabPos !== null, 'w:pos', $tabPos); + $xmlWriter->writeAttributeIf($tabPos->isSpecified(), 'w:pos', $tabPos->toInt('twip')); $xmlWriter->endElement(); // w:tab $xmlWriter->endElement(); // w:tabs $xmlWriter->startElement('w:ind'); - $xmlWriter->writeAttributeIf($left !== null, 'w:left', $left); - $xmlWriter->writeAttributeIf($hanging !== null, 'w:hanging', $hanging); + $xmlWriter->writeAttributeIf($left->isSpecified(), 'w:left', $left->toInt('twip')); + $xmlWriter->writeAttributeIf($hanging->isSpecified(), 'w:hanging', $hanging->toInt('twip')); $xmlWriter->endElement(); // w:ind $xmlWriter->endElement(); // w:pPr @@ -169,8 +165,6 @@ private function writeParagraph(XMLWriter $xmlWriter, NumberingLevel $level) * * @since 0.11.0 * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\NumberingLevel $level * @todo Use font style writer */ private function writeFont(XMLWriter $xmlWriter, NumberingLevel $level) @@ -196,6 +190,6 @@ private function writeFont(XMLWriter $xmlWriter, NumberingLevel $level) */ private function getRandomHexNumber($length = 8) { - return strtoupper(substr(md5(rand()), 0, $length)); + return strtoupper(substr(md5((string) rand()), 0, $length)); } } diff --git a/src/PhpWord/Writer/Word2007/Part/Rels.php b/src/PhpWord/Writer/Word2007/Part/Rels.php index 661a4fa862..0d6c2e6a89 100644 --- a/src/PhpWord/Writer/Word2007/Part/Rels.php +++ b/src/PhpWord/Writer/Word2007/Part/Rels.php @@ -1,4 +1,5 @@ 'media/', 'object' => 'embeddings/'); $mediaType = $mediaRel['type']; - $type = isset($typeMapping[$mediaType]) ? $typeMapping[$mediaType] : $mediaType; - $targetPrefix = isset($targetMapping[$mediaType]) ? $targetMapping[$mediaType] : ''; + $type = $typeMapping[$mediaType] ?? $mediaType; + $targetPrefix = $targetMapping[$mediaType] ?? ''; $target = $mediaRel['target']; $targetMode = ($type == 'hyperlink') ? 'External' : ''; @@ -101,7 +100,6 @@ private function writeMediaRel(XMLWriter $xmlWriter, $relId, $mediaRel) * Format: * * - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param int $relId Relationship ID * @param string $type Relationship type * @param string $target Relationship target @@ -109,12 +107,10 @@ private function writeMediaRel(XMLWriter $xmlWriter, $relId, $mediaRel) * * @throws \PhpOffice\PhpWord\Exception\Exception */ - private function writeRel(XMLWriter $xmlWriter, $relId, $type, $target, $targetMode = '') + private function writeRel(XMLWriter $xmlWriter, int $relId, string $type, string $target, string $targetMode = '') { if ($type != '' && $target != '') { - if (strpos($relId, 'rId') === false) { - $relId = 'rId' . $relId; - } + $relId = 'rId' . $relId; $xmlWriter->startElement('Relationship'); $xmlWriter->writeAttribute('Id', $relId); $xmlWriter->writeAttribute('Type', 'http://schemas.openxmlformats.org/' . $type); diff --git a/src/PhpWord/Writer/Word2007/Part/RelsDocument.php b/src/PhpWord/Writer/Word2007/Part/RelsDocument.php index 2a0c5e1117..b93ae66c59 100644 --- a/src/PhpWord/Writer/Word2007/Part/RelsDocument.php +++ b/src/PhpWord/Writer/Word2007/Part/RelsDocument.php @@ -1,4 +1,5 @@ isSpecified()) { return; } $this->settings['w:hyphenationZone'] = array( - '@attributes' => array('w:val' => $hyphenationZone), + '@attributes' => array('w:val' => $hyphenationZone->toInt('twip')), ); } diff --git a/src/PhpWord/Writer/Word2007/Part/Styles.php b/src/PhpWord/Writer/Word2007/Part/Styles.php index d05338c77d..4dfc736d92 100644 --- a/src/PhpWord/Writer/Word2007/Part/Styles.php +++ b/src/PhpWord/Writer/Word2007/Part/Styles.php @@ -1,4 +1,5 @@ getParentWriter()->getPhpWord(); $fontName = $phpWord->getDefaultFontName(); - $fontSize = $phpWord->getDefaultFontSize(); + $fontSize = $phpWord->getDefaultFontSize()->toInt('hpt'); $language = $phpWord->getSettings()->getThemeFontLang(); $latinLanguage = ($language == null || $language->getLatin() === null) ? 'en-US' : $language->getLatin(); @@ -98,10 +98,10 @@ private function writeDefaultStyles(XMLWriter $xmlWriter, $styles) $xmlWriter->writeAttribute('w:cs', $fontName); $xmlWriter->endElement(); // w:rFonts $xmlWriter->startElement('w:sz'); - $xmlWriter->writeAttribute('w:val', $fontSize * 2); + $xmlWriter->writeAttribute('w:val', $fontSize); $xmlWriter->endElement(); // w:sz $xmlWriter->startElement('w:szCs'); - $xmlWriter->writeAttribute('w:val', $fontSize * 2); + $xmlWriter->writeAttribute('w:val', $fontSize); $xmlWriter->endElement(); // w:szCs $xmlWriter->startElement('w:lang'); $xmlWriter->writeAttribute('w:val', $latinLanguage); @@ -161,9 +161,7 @@ private function writeDefaultStyles(XMLWriter $xmlWriter, $styles) /** * Write font style. * - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param string $styleName - * @param \PhpOffice\PhpWord\Style\Font $style */ private function writeFontStyle(XMLWriter $xmlWriter, $styleName, FontStyle $style) { @@ -229,9 +227,7 @@ private function writeFontStyle(XMLWriter $xmlWriter, $styleName, FontStyle $sty /** * Write paragraph style. * - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param string $styleName - * @param \PhpOffice\PhpWord\Style\Paragraph $style */ private function writeParagraphStyle(XMLWriter $xmlWriter, $styleName, ParagraphStyle $style) { @@ -261,9 +257,7 @@ private function writeParagraphStyle(XMLWriter $xmlWriter, $styleName, Paragraph /** * Write table style. * - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param string $styleName - * @param \PhpOffice\PhpWord\Style\Table $style */ private function writeTableStyle(XMLWriter $xmlWriter, $styleName, TableStyle $style) { diff --git a/src/PhpWord/Writer/Word2007/Part/Theme.php b/src/PhpWord/Writer/Word2007/Part/Theme.php index f4ef478ebf..e91b49c46f 100644 --- a/src/PhpWord/Writer/Word2007/Part/Theme.php +++ b/src/PhpWord/Writer/Word2007/Part/Theme.php @@ -1,4 +1,5 @@ theme = new ThemeStyle(); + } + /** * Write part * @@ -55,45 +66,23 @@ public function write() */ private function writeColorScheme() { - $str = ''; + $name = 'Office'; + $colors = $this->theme->getColorScheme()->getColors(); + + $str = ''; + + foreach ($colors as $tag => $color) { + $str .= ''; + + if ($color instanceof SystemColor) { + $str .= ''; + } else { + $str .= ''; + } + + $str .= ''; + } - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; $str .= ''; return $str; @@ -106,78 +95,30 @@ private function writeColorScheme() */ private function writeFontScheme() { + $fontScheme = $this->theme->getFontScheme(); + $str = ''; $str .= ''; + $headingFonts = $fontScheme->getHeadingFonts(); $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; + $str .= ''; + $str .= ''; + $str .= ''; + foreach ($headingFonts->getFonts() as $script => $font) { + $str .= ''; + } $str .= ''; + $bodyFonts = $fontScheme->getBodyFonts(); $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; - $str .= ''; + $str .= ''; + $str .= ''; + $str .= ''; + foreach ($bodyFonts->getFonts() as $script => $font) { + $str .= ''; + } $str .= ''; $str .= ''; diff --git a/src/PhpWord/Writer/Word2007/Part/WebSettings.php b/src/PhpWord/Writer/Word2007/Part/WebSettings.php index 46252e8701..9e2992f9f8 100644 --- a/src/PhpWord/Writer/Word2007/Part/WebSettings.php +++ b/src/PhpWord/Writer/Word2007/Part/WebSettings.php @@ -1,4 +1,5 @@ style; } - /** - * Convert twip value - * - * @param int|float $value - * @param int $default (int|float) - * @return int|float - */ - protected function convertTwip($value, $default = 0) - { - $factors = array( - Settings::UNIT_CM => 567, - Settings::UNIT_MM => 56.7, - Settings::UNIT_INCH => 1440, - Settings::UNIT_POINT => 20, - Settings::UNIT_PICA => 240, - ); - $unit = Settings::getMeasurementUnit(); - $factor = 1; - if (in_array($unit, $factors) && $value != $default) { - $factor = $factors[$unit]; - } - - return $value * $factor; - } - /** * Write child style. * - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param string $name - * @param mixed $value */ protected function writeChildStyle(XMLWriter $xmlWriter, $name, $value) { diff --git a/src/PhpWord/Writer/Word2007/Style/Border.php b/src/PhpWord/Writer/Word2007/Style/Border.php new file mode 100644 index 0000000000..d841053e2f --- /dev/null +++ b/src/PhpWord/Writer/Word2007/Style/Border.php @@ -0,0 +1,58 @@ +getSize()->toInt('eop') ?? 0) === 0) { + // Don't write 0 width borders + return; + } + + $xmlWriter->startElement('w:' . $side); + $xmlWriter->writeAttribute('w:val', $border->getStyle()->getStyle()); + $xmlWriter->writeAttribute('w:sz', max(2, min(96, $border->getSize()->toInt('eop')))); + $xmlWriter->writeAttribute('w:space', $border->getSpace()->toInt('pt')); + $xmlWriter->writeAttribute('w:color', $border->getColor()->toHexOrName() ?? 'auto'); + $xmlWriter->writeAttribute('w:shadow', $border->getShadow() ? 'true' : 'false'); + $xmlWriter->endElement(); + } +} diff --git a/src/PhpWord/Writer/Word2007/Style/Cell.php b/src/PhpWord/Writer/Word2007/Style/Cell.php index 733b7b434e..818ef3a54e 100644 --- a/src/PhpWord/Writer/Word2007/Style/Cell.php +++ b/src/PhpWord/Writer/Word2007/Style/Cell.php @@ -1,4 +1,5 @@ startElement('w:tcPr'); // Width - if (!is_null($this->width) || !is_null($style->getWidth())) { - $width = is_null($this->width) ? $style->getWidth() : $this->width; - - $xmlWriter->startElement('w:tcW'); - $xmlWriter->writeAttribute('w:w', $width); - $xmlWriter->writeAttribute('w:type', $style->getUnit()); - $xmlWriter->endElement(); // w:tcW - } + $this->writeWidth($xmlWriter, $style); // Text direction $textDir = $style->getTextDirection(); @@ -63,18 +65,7 @@ public function write() $xmlWriter->writeElementIf(!is_null($vAlign), 'w:vAlign', 'w:val', $vAlign); // Border - if ($style->hasBorder()) { - $xmlWriter->startElement('w:tcBorders'); - - $styleWriter = new MarginBorder($xmlWriter); - $styleWriter->setSizes($style->getBorderSize()); - $styleWriter->setColors($style->getBorderColor()); - $styleWriter->setStyles($style->getBorderStyle()); - $styleWriter->setAttributes(array('defaultColor' => CellStyle::DEFAULT_BORDER_COLOR)); - $styleWriter->write(); - - $xmlWriter->endElement(); - } + $this->writeBorders($xmlWriter, $style); // Shading $shading = $style->getShading(); @@ -92,13 +83,58 @@ public function write() $xmlWriter->endElement(); // w:tcPr } + protected function writeWidth(XmlWriter $xmlWriter, CellStyle $style) + { + $width = is_null($this->width) ? $style->getWidth() : $this->width; + + if ($width instanceof Absolute) { + $width = $width->toFloat('twip'); + $unit = 'dxa'; + } elseif ($width instanceof Percent) { + $width = $width->toFloat() . '%'; + $unit = 'pct'; + } elseif ($width instanceof Auto) { + $width = null; + $unit = 'auto'; + } else { + throw new Exception('Unsupported width `' . get_class($width) . '` provided'); + } + + if ($width === null && $unit !== 'auto') { + return; + } elseif ($width !== null && $width == 0) { + $width = null; + $unit = 'nil'; + } + + $xmlWriter->startElement('w:tcW'); + $xmlWriter->writeAttributeIf(null !== $width, 'w:w', $width); + $xmlWriter->writeAttribute('w:type', $unit); + $xmlWriter->endElement(); // w:tcW + } + + protected function writeBorders(XmlWriter $xmlWriter, CellStyle $style) + { + if (!$style->hasBorder()) { + return; + } + + $xmlWriter->startElement('w:tcBorders'); + + foreach ($style->getBorders() as $side => $border) { + $this->writeBorder($xmlWriter, $side, $border); + } + + $xmlWriter->endElement(); + } + /** - * Set width. - * - * @param int $value + * Override width set in style. */ - public function setWidth($value = null) + public function setWidth(Length $value): self { - $this->width = $value; + $this->width = $value->isSpecified() ? $value : null; + + return $this; } } diff --git a/src/PhpWord/Writer/Word2007/Style/Extrusion.php b/src/PhpWord/Writer/Word2007/Style/Extrusion.php index 193993485d..6aae5936e6 100644 --- a/src/PhpWord/Writer/Word2007/Style/Extrusion.php +++ b/src/PhpWord/Writer/Word2007/Style/Extrusion.php @@ -1,4 +1,5 @@ startElement('o:extrusion'); $xmlWriter->writeAttribute('on', 't'); $xmlWriter->writeAttributeIf($style->getType() !== null, 'type', $style->getType()); - $xmlWriter->writeAttributeIf($style->getColor() !== null, 'color', $style->getColor()); + $xmlWriter->writeAttributeIf($style->getColor()->isSpecified(), 'color', $style->getColor()->toHexOrName()); $xmlWriter->endElement(); } } diff --git a/src/PhpWord/Writer/Word2007/Style/Fill.php b/src/PhpWord/Writer/Word2007/Style/Fill.php index 53d039742b..8951440785 100644 --- a/src/PhpWord/Writer/Word2007/Style/Fill.php +++ b/src/PhpWord/Writer/Word2007/Style/Fill.php @@ -1,4 +1,5 @@ getXmlWriter(); $xmlWriter->writeAttribute('on', 't'); - $xmlWriter->writeAttributeIf($style->getColor() !== null, 'fillcolor', $style->getColor()); + $xmlWriter->writeAttributeIf($style->getColor()->isSpecified(), 'fillcolor', $style->getColor()->toHexOrName()); } } diff --git a/src/PhpWord/Writer/Word2007/Style/Font.php b/src/PhpWord/Writer/Word2007/Style/Font.php index dd4fac4f0d..efb62c8573 100644 --- a/src/PhpWord/Writer/Word2007/Style/Font.php +++ b/src/PhpWord/Writer/Word2007/Style/Font.php @@ -1,4 +1,5 @@ getColor(); + $color = $style->getColor()->toHexOrName(); $xmlWriter->writeElementIf($color !== null, 'w:color', 'w:val', $color); // Size - $size = $style->getSize(); - $xmlWriter->writeElementIf($size !== null, 'w:sz', 'w:val', $size * 2); - $xmlWriter->writeElementIf($size !== null, 'w:szCs', 'w:val', $size * 2); + $size = $style->getSize()->toInt('hpt'); + $xmlWriter->writeElementIf($size !== null, 'w:sz', 'w:val', $size); + $xmlWriter->writeElementIf($size !== null, 'w:szCs', 'w:val', $size); // Bold, italic $xmlWriter->writeElementIf($style->isBold() !== null, 'w:b', 'w:val', $this->writeOnOf($style->isBold())); @@ -127,22 +128,24 @@ private function writeStyle() $xmlWriter->writeElementIf($style->getUnderline() != 'none', 'w:u', 'w:val', $style->getUnderline()); // Foreground-Color - $xmlWriter->writeElementIf($style->getFgColor() !== null, 'w:highlight', 'w:val', $style->getFgColor()); + $fgColor = $style->getFgColor()->toHexOrName(); + $xmlWriter->writeElementIf($fgColor !== null, 'w:highlight', 'w:val', $fgColor); // Superscript/subscript $xmlWriter->writeElementIf($style->isSuperScript(), 'w:vertAlign', 'w:val', 'superscript'); $xmlWriter->writeElementIf($style->isSubScript(), 'w:vertAlign', 'w:val', 'subscript'); // Spacing - $xmlWriter->writeElementIf($style->getScale() !== null, 'w:w', 'w:val', $style->getScale()); - $xmlWriter->writeElementIf($style->getSpacing() !== null, 'w:spacing', 'w:val', $style->getSpacing()); - $xmlWriter->writeElementIf($style->getKerning() !== null, 'w:kern', 'w:val', $style->getKerning() * 2); + $xmlWriter->writeElementIf($style->getScale()->isSpecified(), 'w:w', 'w:val', $style->getScale()->toInt()); + $spacing = $style->getSpacing()->toInt('twip'); + $xmlWriter->writeElementIf($spacing !== null, 'w:spacing', 'w:val', $spacing); + $xmlWriter->writeElementIf($style->getKerning()->isSpecified(), 'w:kern', 'w:val', $style->getKerning()->toInt('hpt')); // noProof $xmlWriter->writeElementIf($style->isNoProof() !== null, 'w:noProof', $this->writeOnOf($style->isNoProof())); // Background-Color - $shading = $style->getShading(); + $shading = $style->getShading()->getColor()->toHexOrName(); if (!is_null($shading)) { $styleWriter = new Shading($xmlWriter, $shading); $styleWriter->write(); @@ -155,7 +158,7 @@ private function writeStyle() } // Position - $xmlWriter->writeElementIf($style->getPosition() !== null, 'w:position', 'w:val', $style->getPosition()); + $xmlWriter->writeElementIf($style->getPosition()->toInt('hpt') !== null, 'w:position', 'w:val', $style->getPosition()->toInt('hpt')); $xmlWriter->endElement(); } diff --git a/src/PhpWord/Writer/Word2007/Style/Frame.php b/src/PhpWord/Writer/Word2007/Style/Frame.php index 10e5b151f7..ecd6ec0778 100644 --- a/src/PhpWord/Writer/Word2007/Style/Frame.php +++ b/src/PhpWord/Writer/Word2007/Style/Frame.php @@ -1,4 +1,5 @@ 'mso-wrap-distance-left', 'wrapDistanceRight' => 'mso-wrap-distance-right', ); - $sizeStyles = $this->getStyles($style, $properties, $style->getUnit()); + $sizeStyles = $this->getSizeStyles($style, $properties); $properties = array( 'pos' => 'position', @@ -108,8 +110,6 @@ public function writeAlignment() /** * Write wrap. * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Frame $style * @param string $wrap */ private function writeWrap(XMLWriter $xmlWriter, FrameStyle $style, $wrap) @@ -149,12 +149,10 @@ private function writeWrap(XMLWriter $xmlWriter, FrameStyle $style, $wrap) /** * Get style values in associative array * - * @param \PhpOffice\PhpWord\Style\Frame $style * @param array $properties - * @param string $suffix * @return array */ - private function getStyles(FrameStyle $style, $properties, $suffix = '') + private function getStyles(FrameStyle $style, $properties) { $styles = array(); @@ -162,7 +160,29 @@ private function getStyles(FrameStyle $style, $properties, $suffix = '') $method = "get{$key}"; $value = $style->$method(); if ($value !== null) { - $styles[$property] = $style->$method() . $suffix; + $styles[$property] = $style->$method(); + } + } + + return $styles; + } + + /** + * Get style values in associative array + * + * @param array $properties + * @return array + */ + private function getSizeStyles(FrameStyle $style, $properties) + { + $styles = array(); + + $dpi = new Dpi(); + foreach ($properties as $key => $property) { + $method = "get{$key}"; + $value = $style->$method()->toPixels($dpi); + if ($value !== null) { + $styles[$property] = $value . 'px'; } } diff --git a/src/PhpWord/Writer/Word2007/Style/Image.php b/src/PhpWord/Writer/Word2007/Style/Image.php index ef23ed103c..b0bdcfa9fb 100644 --- a/src/PhpWord/Writer/Word2007/Style/Image.php +++ b/src/PhpWord/Writer/Word2007/Style/Image.php @@ -1,4 +1,5 @@ startElement('w:ind'); - $xmlWriter->writeAttribute('w:left', $this->convertTwip($style->getLeft())); - $xmlWriter->writeAttribute('w:right', $this->convertTwip($style->getRight())); + $xmlWriter->writeAttribute('w:left', $style->getLeft()->toInt('twip')); + $xmlWriter->writeAttribute('w:right', $style->getRight()->toInt('twip')); - $firstLine = $style->getFirstLine(); - $xmlWriter->writeAttributeIf(!is_null($firstLine), 'w:firstLine', $this->convertTwip($firstLine)); + $firstLine = $style->getFirstLine()->toInt('twip'); + $xmlWriter->writeAttributeIf(!is_null($firstLine), 'w:firstLine', $firstLine); - $hanging = $style->getHanging(); - $xmlWriter->writeAttributeIf(!is_null($hanging), 'w:hanging', $this->convertTwip($hanging)); + $hanging = $style->getHanging()->toInt('twip'); + $xmlWriter->writeAttributeIf(!is_null($hanging), 'w:hanging', $hanging); $xmlWriter->endElement(); } diff --git a/src/PhpWord/Writer/Word2007/Style/Line.php b/src/PhpWord/Writer/Word2007/Style/Line.php index 154a42c13d..425efe0524 100644 --- a/src/PhpWord/Writer/Word2007/Style/Line.php +++ b/src/PhpWord/Writer/Word2007/Style/Line.php @@ -1,4 +1,5 @@ startElement('v:stroke'); - $xmlWriter->writeAttributeIf($style->getWeight() !== null, 'weight', $style->getWeight() . 'pt'); + $xmlWriter->writeAttributeIf($style->getWeight()->isSpecified(), 'weight', $style->getWeight()->toInt('pt') . 'pt'); $xmlWriter->writeAttributeIf($style->getColor() !== null, 'color', $style->getColor()); $xmlWriter->writeAttributeIf($style->getBeginArrow() !== null, 'startarrow', $style->getBeginArrow()); $xmlWriter->writeAttributeIf($style->getEndArrow() !== null, 'endarrow', $style->getEndArrow()); diff --git a/src/PhpWord/Writer/Word2007/Style/LineNumbering.php b/src/PhpWord/Writer/Word2007/Style/LineNumbering.php index 4bf08b65ab..df99443e39 100644 --- a/src/PhpWord/Writer/Word2007/Style/LineNumbering.php +++ b/src/PhpWord/Writer/Word2007/Style/LineNumbering.php @@ -1,4 +1,5 @@ startElement('w:lnNumType'); $xmlWriter->writeAttribute('w:start', $style->getStart() - 1); $xmlWriter->writeAttribute('w:countBy', $style->getIncrement()); - $xmlWriter->writeAttribute('w:distance', $style->getDistance()); + $xmlWriter->writeAttribute('w:distance', $style->getDistance()->toInt('twip')); $xmlWriter->writeAttribute('w:restart', $style->getRestart()); $xmlWriter->endElement(); } diff --git a/src/PhpWord/Writer/Word2007/Style/MarginBorder.php b/src/PhpWord/Writer/Word2007/Style/MarginBorder.php deleted file mode 100644 index f5c4b0153b..0000000000 --- a/src/PhpWord/Writer/Word2007/Style/MarginBorder.php +++ /dev/null @@ -1,150 +0,0 @@ -getXmlWriter(); - - $sides = array('top', 'left', 'right', 'bottom', 'insideH', 'insideV'); - - foreach ($this->sizes as $i => $size) { - if ($size !== null) { - $color = null; - if (isset($this->colors[$i])) { - $color = $this->colors[$i]; - } - $style = isset($this->styles[$i]) ? $this->styles[$i] : 'single'; - $this->writeSide($xmlWriter, $sides[$i], $this->sizes[$i], $color, $style); - } - } - } - - /** - * Write side. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param string $side - * @param int $width - * @param string $color - * @param string $borderStyle - */ - private function writeSide(XMLWriter $xmlWriter, $side, $width, $color = null, $borderStyle = 'solid') - { - $xmlWriter->startElement('w:' . $side); - if (!empty($this->colors)) { - if ($color === null && !empty($this->attributes)) { - if (isset($this->attributes['defaultColor'])) { - $color = $this->attributes['defaultColor']; - } - } - $xmlWriter->writeAttribute('w:val', $borderStyle); - $xmlWriter->writeAttribute('w:sz', $width); - $xmlWriter->writeAttributeIf($color != null, 'w:color', $color); - if (!empty($this->attributes)) { - if (isset($this->attributes['space'])) { - $xmlWriter->writeAttribute('w:space', $this->attributes['space']); - } - } - } else { - $xmlWriter->writeAttribute('w:w', $width); - $xmlWriter->writeAttribute('w:type', 'dxa'); - } - $xmlWriter->endElement(); - } - - /** - * Set sizes. - * - * @param int[] $value - */ - public function setSizes($value) - { - $this->sizes = $value; - } - - /** - * Set colors. - * - * @param string[] $value - */ - public function setColors($value) - { - $this->colors = $value; - } - - /** - * Set border styles. - * - * @param string[] $value - */ - public function setStyles($value) - { - $this->styles = $value; - } - - /** - * Set attributes. - * - * @param array $value - */ - public function setAttributes($value) - { - $this->attributes = $value; - } -} diff --git a/src/PhpWord/Writer/Word2007/Style/Outline.php b/src/PhpWord/Writer/Word2007/Style/Outline.php index ae4c1da386..8c99c09bbb 100644 --- a/src/PhpWord/Writer/Word2007/Style/Outline.php +++ b/src/PhpWord/Writer/Word2007/Style/Outline.php @@ -1,4 +1,5 @@ startElement('v:stroke'); $xmlWriter->writeAttribute('on', 't'); - $xmlWriter->writeAttributeIf($style->getColor() !== null, 'color', $style->getColor()); - $xmlWriter->writeAttributeIf($style->getWeight() !== null, 'weight', $style->getWeight() . $style->getUnit()); + $xmlWriter->writeAttributeIf($style->getColor()->isSpecified(), 'color', $style->getColor()->toHexOrName()); + $xmlWriter->writeAttributeIf($style->getWeight()->isSpecified(), 'weight', $style->getWeight()->toInt('emu')); $xmlWriter->writeAttributeIf($style->getDash() !== null, 'dashstyle', $style->getDash()); $xmlWriter->writeAttributeIf($style->getLine() !== null, 'linestyle', $style->getLine()); $xmlWriter->writeAttributeIf($style->getEndCap() !== null, 'endcap', $style->getEndCap()); diff --git a/src/PhpWord/Writer/Word2007/Style/Paragraph.php b/src/PhpWord/Writer/Word2007/Style/Paragraph.php index 6761608608..0fd10a3ce5 100644 --- a/src/PhpWord/Writer/Word2007/Style/Paragraph.php +++ b/src/PhpWord/Writer/Word2007/Style/Paragraph.php @@ -1,4 +1,5 @@ writeNumbering($xmlWriter, $styles['numbering']); // Border - if ($style->hasBorder()) { - $xmlWriter->startElement('w:pBdr'); - - $styleWriter = new MarginBorder($xmlWriter); - $styleWriter->setSizes($style->getBorderSize()); - $styleWriter->setColors($style->getBorderColor()); - $styleWriter->write(); - - $xmlWriter->endElement(); - } + $this->writeBorders($xmlWriter, $style); if (!$this->withoutPPR) { $xmlWriter->endElement(); // w:pPr @@ -146,7 +140,6 @@ private function writeStyle() /** * Write tabs. * - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param \PhpOffice\PhpWord\Style\Tab[] $tabs */ private function writeTabs(XMLWriter $xmlWriter, $tabs) @@ -164,7 +157,6 @@ private function writeTabs(XMLWriter $xmlWriter, $tabs) /** * Write numbering. * - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param array $numbering */ private function writeNumbering(XMLWriter $xmlWriter, $numbering) @@ -190,6 +182,25 @@ private function writeNumbering(XMLWriter $xmlWriter, $numbering) } } + /** + * Writes paragraph borders + * @see http://officeopenxml.com/WPborders.php + */ + private function writeBorders(XMLWriter $xmlWriter, ParagraphStyle $style) + { + if (!$style->hasBorder()) { + return; + } + + $xmlWriter->startElement('w:pBdr'); + + foreach ($style->getBorders() as $side => $border) { + $this->writeBorder($xmlWriter, $side, $border); + } + + $xmlWriter->endElement(); + } + /** * Set without w:pPr. * diff --git a/src/PhpWord/Writer/Word2007/Style/Row.php b/src/PhpWord/Writer/Word2007/Style/Row.php index 82028d24c4..0beb562404 100644 --- a/src/PhpWord/Writer/Word2007/Style/Row.php +++ b/src/PhpWord/Writer/Word2007/Style/Row.php @@ -1,4 +1,5 @@ height !== null) { $xmlWriter->startElement('w:trHeight'); - $xmlWriter->writeAttribute('w:val', $this->height); + $xmlWriter->writeAttribute('w:val', $this->height->toInt('twip')); $xmlWriter->writeAttribute('w:hRule', ($style->isExactHeight() ? 'exact' : 'atLeast')); $xmlWriter->endElement(); } @@ -56,10 +59,8 @@ public function write() /** * Set height. - * - * @param int $value */ - public function setHeight($value = null) + public function setHeight(Absolute $value) { $this->height = $value; } diff --git a/src/PhpWord/Writer/Word2007/Style/Section.php b/src/PhpWord/Writer/Word2007/Style/Section.php index 1122b6ff76..36cb4883c3 100644 --- a/src/PhpWord/Writer/Word2007/Style/Section.php +++ b/src/PhpWord/Writer/Word2007/Style/Section.php @@ -1,4 +1,5 @@ startElement('w:pgSz'); $xmlWriter->writeAttribute('w:orient', $style->getOrientation()); - $xmlWriter->writeAttribute('w:w', $style->getPageSizeW()); - $xmlWriter->writeAttribute('w:h', $style->getPageSizeH()); + $xmlWriter->writeAttribute('w:w', $style->getPageSizeW()->toInt('twip')); + $xmlWriter->writeAttribute('w:h', $style->getPageSizeH()->toInt('twip')); $xmlWriter->endElement(); // w:pgSz // Vertical alignment @@ -65,29 +69,18 @@ public function write() $xmlWriter->startElement('w:pgMar'); foreach ($margins as $attribute => $value) { list($method, $default) = $value; - $xmlWriter->writeAttribute($attribute, $this->convertTwip($style->$method(), $default)); + $xmlWriter->writeAttribute($attribute, $style->$method()->toInt('twip') ?? $default); } $xmlWriter->endElement(); // Borders - if ($style->hasBorder()) { - $xmlWriter->startElement('w:pgBorders'); - $xmlWriter->writeAttribute('w:offsetFrom', 'page'); - - $styleWriter = new MarginBorder($xmlWriter); - $styleWriter->setSizes($style->getBorderSize()); - $styleWriter->setColors($style->getBorderColor()); - $styleWriter->setAttributes(array('space' => '24')); - $styleWriter->write(); - - $xmlWriter->endElement(); - } + $this->writeBorders($xmlWriter, $style); // Columns $colsSpace = $style->getColsSpace(); $xmlWriter->startElement('w:cols'); $xmlWriter->writeAttribute('w:num', $style->getColsNum()); - $xmlWriter->writeAttribute('w:space', $this->convertTwip($colsSpace, SectionStyle::DEFAULT_COLUMN_SPACING)); + $xmlWriter->writeAttribute('w:space', $colsSpace->toInt('twip') ?? SectionStyle::DEFAULT_COLUMN_SPACING); $xmlWriter->endElement(); // Page numbering start @@ -98,4 +91,29 @@ public function write() $styleWriter = new LineNumbering($xmlWriter, $style->getLineNumbering()); $styleWriter->write(); } + + /** + * Writes section borders + * @see http://www.officeopenxml.com/WPsectionBorders.php + */ + private function writeBorders(XMLWriter $xmlWriter, SectionStyle $style) + { + if (!$style->hasBorder()) { + return; + } + + $xmlWriter->startElement('w:pgBorders'); + // Border should be drawn on 'allPages', 'firstPage', or 'notFirstPage' + $xmlWriter->writeAttribute('w:display', 'allPages'); + // Distance should be from 'page' edge or 'text' margin + $xmlWriter->writeAttribute('w:offsetFrom', 'page'); + // Drawn in 'back' or 'front' of text + $xmlWriter->writeAttribute('w:zOrder', 'back'); + + foreach ($style->getBorders() as $side => $border) { + $this->writeBorder($xmlWriter, $side, $border); + } + + $xmlWriter->endElement(); + } } diff --git a/src/PhpWord/Writer/Word2007/Style/Shading.php b/src/PhpWord/Writer/Word2007/Style/Shading.php index 0f9d6ccc08..67d6ac474b 100644 --- a/src/PhpWord/Writer/Word2007/Style/Shading.php +++ b/src/PhpWord/Writer/Word2007/Style/Shading.php @@ -1,4 +1,5 @@ startElement('w:shd'); $xmlWriter->writeAttributeIf(!is_null($style->getPattern()), 'w:val', $style->getPattern()); - $xmlWriter->writeAttributeIf(!is_null($style->getColor()), 'w:color', $style->getColor()); - $xmlWriter->writeAttributeIf(!is_null($style->getFill()), 'w:fill', $style->getFill()); + $xmlWriter->writeAttributeIf(!is_null($style->getColor()->toHexOrName()), 'w:color', $style->getColor()->toHexOrName()); + $xmlWriter->writeAttributeIf(!is_null($style->getFill()), 'w:fill', $style->getFill()->toHexOrName()); $xmlWriter->endElement(); } } diff --git a/src/PhpWord/Writer/Word2007/Style/Shadow.php b/src/PhpWord/Writer/Word2007/Style/Shadow.php index 7fcb12a960..0dd98fb8ef 100644 --- a/src/PhpWord/Writer/Word2007/Style/Shadow.php +++ b/src/PhpWord/Writer/Word2007/Style/Shadow.php @@ -1,4 +1,5 @@ startElement('v:shadow'); $xmlWriter->writeAttribute('on', 't'); - $xmlWriter->writeAttributeIf($style->getColor() !== null, 'color', $style->getColor()); + $xmlWriter->writeAttributeIf($style->getColor()->isSpecified(), 'color', $style->getColor()->toHexOrName()); $xmlWriter->writeAttributeIf($style->getOffset() !== null, 'offset', $style->getOffset()); $xmlWriter->endElement(); } diff --git a/src/PhpWord/Writer/Word2007/Style/Shape.php b/src/PhpWord/Writer/Word2007/Style/Shape.php index 2def6842a9..9c2028923c 100644 --- a/src/PhpWord/Writer/Word2007/Style/Shape.php +++ b/src/PhpWord/Writer/Word2007/Style/Shape.php @@ -1,4 +1,5 @@ startElement('w:spacing'); - $before = $style->getBefore(); - $xmlWriter->writeAttributeIf(!is_null($before), 'w:before', $this->convertTwip($before)); + $before = $style->getBefore()->toInt('twip'); + $xmlWriter->writeAttributeIf(!is_null($before), 'w:before', $before); - $after = $style->getAfter(); - $xmlWriter->writeAttributeIf(!is_null($after), 'w:after', $this->convertTwip($after)); + $after = $style->getAfter()->toInt('twip'); + $xmlWriter->writeAttributeIf(!is_null($after), 'w:after', $after); - $line = $style->getLine(); + $line = $style->getLine()->toInt('twip'); + // @see http://www.datypic.com/sc/ooxml/a-w_line-1.html //if linerule is auto, the spacing is supposed to include the height of the line itself, which is 240 twips if (null !== $line && 'auto' === $style->getLineRule()) { $line += \PhpOffice\PhpWord\Style\Paragraph::LINE_HEIGHT; diff --git a/src/PhpWord/Writer/Word2007/Style/Tab.php b/src/PhpWord/Writer/Word2007/Style/Tab.php index b41653f6c4..d29f0e2f57 100644 --- a/src/PhpWord/Writer/Word2007/Style/Tab.php +++ b/src/PhpWord/Writer/Word2007/Style/Tab.php @@ -1,4 +1,5 @@ startElement('w:tab'); $xmlWriter->writeAttribute('w:val', $style->getType()); $xmlWriter->writeAttribute('w:leader', $style->getLeader()); - $xmlWriter->writeAttribute('w:pos', $this->convertTwip($style->getPosition())); + $xmlWriter->writeAttribute('w:pos', $style->getPosition()->toInt('twip')); $xmlWriter->endElement(); } } diff --git a/src/PhpWord/Writer/Word2007/Style/Table.php b/src/PhpWord/Writer/Word2007/Style/Table.php index 443d670582..c69d33f450 100644 --- a/src/PhpWord/Writer/Word2007/Style/Table.php +++ b/src/PhpWord/Writer/Word2007/Style/Table.php @@ -1,4 +1,5 @@ writeAttribute('w:val', $style); $xmlWriter->endElement(); if (null !== $this->width) { - $this->writeTblWidth($xmlWriter, 'w:tblW', TblWidth::PERCENT, $this->width); + $this->writeTblWidth($xmlWriter, 'w:tblW', $this->width); } $xmlWriter->endElement(); } @@ -58,9 +65,6 @@ public function write() /** * Write full style. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Table $style */ private function writeStyle(XMLWriter $xmlWriter, TableStyle $style) { @@ -77,8 +81,8 @@ private function writeStyle(XMLWriter $xmlWriter, TableStyle $style) $xmlWriter->endElement(); } - $this->writeTblWidth($xmlWriter, 'w:tblW', $style->getUnit(), $style->getWidth()); - $this->writeTblWidth($xmlWriter, 'w:tblCellSpacing', TblWidth::TWIP, $style->getCellSpacing()); + $this->writeTblWidth($xmlWriter, 'w:tblW', $style->getWidth()); + $this->writeTblWidth($xmlWriter, 'w:tblCellSpacing', $style->getCellSpacing()); $this->writeIndent($xmlWriter, $style); $this->writeLayout($xmlWriter, $style->getLayout()); @@ -90,7 +94,7 @@ private function writeStyle(XMLWriter $xmlWriter, TableStyle $style) $xmlWriter->writeElementIf($style->isBidiVisual() !== null, 'w:bidiVisual', 'w:val', $this->writeOnOf($style->isBidiVisual())); $this->writeMargin($xmlWriter, $style); - $this->writeBorder($xmlWriter, $style); + $this->writeBorders($xmlWriter, $style); $xmlWriter->endElement(); // w:tblPr @@ -106,7 +110,6 @@ private function writeStyle(XMLWriter $xmlWriter, TableStyle $style) /** * Enable/Disable automatic resizing of the table * - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param string $layout autofit / fixed */ private function writeLayout(XMLWriter $xmlWriter, $layout) @@ -118,56 +121,71 @@ private function writeLayout(XMLWriter $xmlWriter, $layout) /** * Write margin. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Table $style */ private function writeMargin(XMLWriter $xmlWriter, TableStyle $style) { if ($style->hasMargin()) { $xmlWriter->startElement('w:tblCellMar'); - $styleWriter = new MarginBorder($xmlWriter); - $styleWriter->setSizes($style->getCellMargin()); - $styleWriter->write(); + $sides = array('top', 'bottom', 'start', 'end'); + foreach ($style->getCellMargin() as $key => $width) { + $width = $width->toInt('twip') ?? 0; + + $xmlWriter->startElement('w:' . $sides[$key]); + $xmlWriter->writeAttributeIf($width !== 0, 'w:w', $width); + $xmlWriter->writeAttribute('w:type', $width === 0 ? 'nil' : 'dxa'); + $xmlWriter->endElement(); + } - $xmlWriter->endElement(); // w:tblCellMar + $xmlWriter->endElement(); } } /** * Write border. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Table $style */ - private function writeBorder(XMLWriter $xmlWriter, TableStyle $style) + private function writeBorders(XMLWriter $xmlWriter, TableStyle $style) { - if ($style->hasBorder()) { - $xmlWriter->startElement('w:tblBorders'); + if (!$style->hasBorder()) { + return; + } - $styleWriter = new MarginBorder($xmlWriter); - $styleWriter->setSizes($style->getBorderSize()); - $styleWriter->setColors($style->getBorderColor()); - $styleWriter->write(); + $xmlWriter->startElement('w:tblBorders'); - $xmlWriter->endElement(); // w:tblBorders + foreach ($style->getBorders() as $side => $border) { + $this->writeBorder($xmlWriter, $side, $border); } + + $xmlWriter->endElement(); // w:tblBorders } /** * Writes a table width * - * @param \PhpOffice\Common\XMLWriter $xmlWriter * @param string $elementName - * @param string $unit - * @param int|float $width */ - private function writeTblWidth(XMLWriter $xmlWriter, $elementName, $unit, $width = null) + private function writeTblWidth(XMLWriter $xmlWriter, $elementName, Length $width) { - if (null === $width) { + if ($width instanceof Absolute) { + $width = $width->toFloat('twip'); + $unit = 'dxa'; + } elseif ($width instanceof Percent) { + $width = $width->toFloat(); + $unit = 'pct'; + } elseif ($width instanceof Auto) { + $width = null; + $unit = 'auto'; + } else { + throw new Exception('Unsupported width `' . get_class($width) . '` provided'); + } + + if ($width === null && $unit !== 'auto') { return; + } elseif ($width !== null && $width == 0) { + $width = null; + $unit = 'nil'; } + $xmlWriter->startElement($elementName); $xmlWriter->writeAttributeIf(null !== $width, 'w:w', $width); $xmlWriter->writeAttribute('w:type', $unit); @@ -176,9 +194,6 @@ private function writeTblWidth(XMLWriter $xmlWriter, $elementName, $unit, $width /** * Write row style. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Table $style */ private function writeFirstRow(XMLWriter $xmlWriter, TableStyle $style) { @@ -186,7 +201,7 @@ private function writeFirstRow(XMLWriter $xmlWriter, TableStyle $style) $xmlWriter->writeAttribute('w:type', 'firstRow'); $xmlWriter->startElement('w:tcPr'); - $this->writeBorder($xmlWriter, $style); + $this->writeBorders($xmlWriter, $style); $this->writeShading($xmlWriter, $style); $xmlWriter->endElement(); // w:tcPr @@ -195,9 +210,6 @@ private function writeFirstRow(XMLWriter $xmlWriter, TableStyle $style) /** * Write shading. - * - * @param \PhpOffice\Common\XMLWriter $xmlWriter - * @param \PhpOffice\PhpWord\Style\Table $style */ private function writeShading(XMLWriter $xmlWriter, TableStyle $style) { @@ -213,26 +225,14 @@ private function writeShading(XMLWriter $xmlWriter, TableStyle $style) /** * Set width. - * - * @param int $value */ - public function setWidth($value = null) + public function setWidth(Length $value) { $this->width = $value; } - /** - * @param XMLWriter $xmlWriter - * @param TableStyle $style - */ private function writeIndent(XMLWriter $xmlWriter, TableStyle $style) { - $indent = $style->getIndent(); - - if ($indent === null) { - return; - } - - $this->writeTblWidth($xmlWriter, 'w:tblInd', $indent->getType(), $indent->getValue()); + $this->writeTblWidth($xmlWriter, 'w:tblInd', $style->getIndent()); } } diff --git a/src/PhpWord/Writer/Word2007/Style/TablePosition.php b/src/PhpWord/Writer/Word2007/Style/TablePosition.php index fa57b93c2f..b887d7c7f3 100644 --- a/src/PhpWord/Writer/Word2007/Style/TablePosition.php +++ b/src/PhpWord/Writer/Word2007/Style/TablePosition.php @@ -1,4 +1,5 @@ getXmlWriter(); $xmlWriter->startElement('w:tblpPr'); foreach ($values as $property => $value) { + if ($value instanceof Absolute) { + $value = $value->toInt('twip'); + } $xmlWriter->writeAttribute('w:' . $property, $value); } $xmlWriter->endElement(); diff --git a/src/PhpWord/Writer/Word2007/Style/TextBox.php b/src/PhpWord/Writer/Word2007/Style/TextBox.php index 627d0c86e5..f702aa016c 100644 --- a/src/PhpWord/Writer/Word2007/Style/TextBox.php +++ b/src/PhpWord/Writer/Word2007/Style/TextBox.php @@ -1,4 +1,5 @@ getXmlWriter(); - $margins = implode(', ', $style->getInnerMargin()); + $margins = implode(', ', array_map(function ($value) { + return $value->toInt('twip'); + }, $style->getInnerMargin())); $xmlWriter->writeAttribute('inset', $margins); } @@ -54,8 +57,9 @@ public function writeBorder() $xmlWriter = $this->getXmlWriter(); $xmlWriter->startElement('v:stroke'); - $xmlWriter->writeAttributeIf($style->getBorderSize() !== null, 'weight', $style->getBorderSize() . 'pt'); - $xmlWriter->writeAttributeIf($style->getBorderColor() !== null, 'color', $style->getBorderColor()); + $borderSize = $style->getBorderSize()->toInt('pt'); + $xmlWriter->writeAttributeIf($borderSize !== null, 'weight', $borderSize . 'pt'); + $xmlWriter->writeAttributeIf($style->getBorderColor()->isSpecified(), 'color', $style->getBorderColor()->toHexOrName()); $xmlWriter->endElement(); // v:stroke } } diff --git a/src/PhpWord/Writer/WriterInterface.php b/src/PhpWord/Writer/WriterInterface.php index 499cde3bb1..6783bc9e9e 100644 --- a/src/PhpWord/Writer/WriterInterface.php +++ b/src/PhpWord/Writer/WriterInterface.php @@ -1,4 +1,5 @@ assertInstanceOf('PhpOffice\\PhpWord\\Element\\Cell', $oCell); - $this->assertNull($oCell->getWidth()); + $this->assertNull($oCell->getWidth()->toInt('twip')); } /** @@ -45,7 +47,7 @@ public function testConstructWithStyleArray() $oCell = new Cell(null, array('valign' => 'center')); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Cell', $oCell->getStyle()); - $this->assertNull($oCell->getWidth()); + $this->assertNull($oCell->getWidth()->toInt('twip')); } /** @@ -141,7 +143,7 @@ public function testAddImageSection() public function testAddImageHeader() { $src = __DIR__ . '/../_files/images/earth.jpg'; - $oCell = new Cell('header', 1); + $oCell = new Cell(Absolute::from('twip', 1), 'header'); $element = $oCell->addImage($src); $this->assertCount(1, $oCell->getElements()); @@ -154,7 +156,7 @@ public function testAddImageHeader() public function testAddImageFooter() { $src = __DIR__ . '/../_files/images/earth.jpg'; - $oCell = new Cell('footer', 1); + $oCell = new Cell(Absolute::from('twip', 1), 'footer'); $element = $oCell->addImage($src); $this->assertCount(1, $oCell->getElements()); diff --git a/tests/PhpWord/Element/CheckBoxTest.php b/tests/PhpWord/Element/CheckBoxTest.php index f732407b06..e4ca1b952e 100644 --- a/tests/PhpWord/Element/CheckBoxTest.php +++ b/tests/PhpWord/Element/CheckBoxTest.php @@ -1,4 +1,5 @@ assertEquals('fontStyle', $oCheckBox->getFontStyle()); - $oCheckBox->setFontStyle(array('bold' => true, 'italic' => true, 'size' => 16)); + $oCheckBox->setFontStyle(array('bold' => true, 'italic' => true, 'size' => Absolute::from('pt', 16))); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Font', $oCheckBox->getFontStyle()); } @@ -81,7 +83,7 @@ public function testParagraph() $oCheckBox = new CheckBox('chkBox', 'CheckBox', 'fontStyle', 'paragraphStyle'); $this->assertEquals('paragraphStyle', $oCheckBox->getParagraphStyle()); - $oCheckBox->setParagraphStyle(array('alignment' => Jc::CENTER, 'spaceAfter' => 100)); + $oCheckBox->setParagraphStyle(array('alignment' => Jc::CENTER, 'spaceAfter' => Absolute::from('twip', 100))); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Paragraph', $oCheckBox->getParagraphStyle()); } } diff --git a/tests/PhpWord/Element/CommentTest.php b/tests/PhpWord/Element/CommentTest.php index b9c3dfce5a..3ee0f6e5a2 100644 --- a/tests/PhpWord/Element/CommentTest.php +++ b/tests/PhpWord/Element/CommentTest.php @@ -1,4 +1,5 @@ 100)); + $oFootnote = new Footnote(array('spacing' => Absolute::from('twip', 100))); $this->assertInstanceOf( 'PhpOffice\\PhpWord\\Style\\Paragraph', diff --git a/tests/PhpWord/Element/HeaderTest.php b/tests/PhpWord/Element/HeaderTest.php index 4bbf7b74f3..330a1d96d9 100644 --- a/tests/PhpWord/Element/HeaderTest.php +++ b/tests/PhpWord/Element/HeaderTest.php @@ -1,4 +1,5 @@ 210, - 'height' => 210, + 'width' => Absolute::from('twip', 210), + 'height' => Absolute::from('twip', 210), 'alignment' => Jc::CENTER, 'wrappingStyle' => \PhpOffice\PhpWord\Style\Image::WRAPPING_STYLE_BEHIND, ) @@ -98,7 +100,7 @@ public function testStyle() { $oImage = new Image( __DIR__ . '/../_files/images/earth.jpg', - array('height' => 210, 'alignment' => Jc::CENTER) + array('height' => Absolute::from('twip', 210), 'alignment' => Jc::CENTER) ); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Image', $oImage->getStyle()); @@ -149,7 +151,7 @@ public function testUnsupportedImage() */ public function testRelationID() { - $oImage = new Image(__DIR__ . '/../_files/images/earth.jpg', array('width' => 100)); + $oImage = new Image(__DIR__ . '/../_files/images/earth.jpg', array('width' => Absolute::from('twip', 100))); $iVal = rand(1, 1000); $oImage->setRelationId($iVal); $this->assertEquals($iVal, $oImage->getRelationId()); diff --git a/tests/PhpWord/Element/LineTest.php b/tests/PhpWord/Element/LineTest.php index 20eee74f38..12357ec152 100644 --- a/tests/PhpWord/Element/LineTest.php +++ b/tests/PhpWord/Element/LineTest.php @@ -1,4 +1,5 @@ \PhpOffice\PhpWord\Shared\Converter::cmToPixel(14), - 'height' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(4), + 'width' => Absolute::from('cm', 14), + 'height' => Absolute::from('cm', 4), 'positioning' => 'absolute', 'posHorizontalRel' => 'page', 'posVerticalRel' => 'page', 'flip' => true, - 'marginLeft' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(5), - 'marginTop' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(3), + 'marginLeft' => Absolute::from('cm', 5), + 'marginTop' => Absolute::from('cm', 3), 'wrappingStyle' => \PhpOffice\PhpWord\Style\Image::WRAPPING_STYLE_SQUARE, 'beginArrow' => \PhpOffice\PhpWord\Style\Line::ARROW_STYLE_BLOCK, 'endArrow' => \PhpOffice\PhpWord\Style\Line::ARROW_STYLE_OVAL, 'dash' => \PhpOffice\PhpWord\Style\Line::DASH_STYLE_LONG_DASH_DOT_DOT, - 'weight' => 10, + 'weight' => Absolute::from('pt', 10), ) ); diff --git a/tests/PhpWord/Element/LinkTest.php b/tests/PhpWord/Element/LinkTest.php index e1be7521a6..359519c771 100644 --- a/tests/PhpWord/Element/LinkTest.php +++ b/tests/PhpWord/Element/LinkTest.php @@ -1,4 +1,5 @@ '0000FF', 'underline' => Font::UNDERLINE_SINGLE), - array('marginLeft' => 600, 'marginRight' => 600, 'marginTop' => 600, 'marginBottom' => 600) + array('color' => new Hex('0000FF'), 'underline' => Font::UNDERLINE_SINGLE), + array('alignment' => 'center') ); $this->assertInstanceOf('PhpOffice\\PhpWord\\Element\\Link', $oLink); diff --git a/tests/PhpWord/Element/ListItemRunTest.php b/tests/PhpWord/Element/ListItemRunTest.php index 95eb17eb29..a7ee70ad24 100644 --- a/tests/PhpWord/Element/ListItemRunTest.php +++ b/tests/PhpWord/Element/ListItemRunTest.php @@ -1,4 +1,5 @@ 100)); + $oListItemRun = new ListItemRun(0, null, array('spacing' => Absolute::from('twip', 100))); $this->assertInstanceOf('PhpOffice\\PhpWord\\Element\\ListItemRun', $oListItemRun); $this->assertCount(0, $oListItemRun->getElements()); diff --git a/tests/PhpWord/Element/ListItemTest.php b/tests/PhpWord/Element/ListItemTest.php index e5c815ec69..c19e63678b 100644 --- a/tests/PhpWord/Element/ListItemTest.php +++ b/tests/PhpWord/Element/ListItemTest.php @@ -1,4 +1,5 @@ '230px')); + $oObject = new OLEObject($src, array('width' => Absolute::from('in', 230 / 96))); $this->assertInstanceOf('PhpOffice\\PhpWord\\Element\\OLEObject', $oObject); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Image', $oObject->getStyle()); diff --git a/tests/PhpWord/Element/PageBreakTest.php b/tests/PhpWord/Element/PageBreakTest.php index d4491fe1b3..15b08f143b 100644 --- a/tests/PhpWord/Element/PageBreakTest.php +++ b/tests/PhpWord/Element/PageBreakTest.php @@ -1,4 +1,5 @@ 16, 'color' => '1B2232'), array('alignment' => Jc::CENTER)); + $oPreserveText = new PreserveText('text', array('size' => Absolute::from('pt', 16), 'color' => new Hex('1B2232')), array('alignment' => Jc::CENTER)); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Font', $oPreserveText->getFontStyle()); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Paragraph', $oPreserveText->getParagraphStyle()); } diff --git a/tests/PhpWord/Element/RowTest.php b/tests/PhpWord/Element/RowTest.php index 3c53450294..bafc51af02 100644 --- a/tests/PhpWord/Element/RowTest.php +++ b/tests/PhpWord/Element/RowTest.php @@ -1,4 +1,5 @@ assertInstanceOf('PhpOffice\\PhpWord\\Element\\Row', $oRow); - $this->assertNull($oRow->getHeight()); + $this->assertNull($oRow->getHeight()->toInt('twip')); $this->assertInternalType('array', $oRow->getCells()); $this->assertCount(0, $oRow->getCells()); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Row', $oRow->getStyle()); @@ -45,9 +48,9 @@ public function testConstruct() public function testConstructWithParams() { $iVal = rand(1, 1000); - $oRow = new Row($iVal, array('borderBottomSize' => 18, 'borderBottomColor' => '0000FF', 'bgColor' => '66BBFF')); + $oRow = new Row(Absolute::from('twip', $iVal)); - $this->assertEquals($iVal, $oRow->getHeight()); + $this->assertEquals($iVal, $oRow->getHeight()->toInt('twip')); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Row', $oRow->getStyle()); } diff --git a/tests/PhpWord/Element/SDTTest.php b/tests/PhpWord/Element/SDTTest.php index 2328dd76be..d63439a0c6 100644 --- a/tests/PhpWord/Element/SDTTest.php +++ b/tests/PhpWord/Element/SDTTest.php @@ -1,4 +1,5 @@ setStyle(array('orientation' => $expected, 'foo' => null)); + $object->setStyle(array('orientation' => $expected)); $this->assertEquals($expected, $object->getStyle()->getOrientation()); } @@ -125,7 +127,7 @@ public function testAddObjectException() */ public function testAddTitleWithStyle() { - Style::addTitleStyle(1, array('size' => 14)); + Style::addTitleStyle(1, array('size' => Absolute::from('pt', 14))); $section = new Section(0); $section->setPhpWord(new PhpWord()); $section->addTitle('Test', 1); diff --git a/tests/PhpWord/Element/TOCTest.php b/tests/PhpWord/Element/TOCTest.php index 5f5f518f43..74fa591043 100644 --- a/tests/PhpWord/Element/TOCTest.php +++ b/tests/PhpWord/Element/TOCTest.php @@ -1,4 +1,5 @@ 9062, + 'position' => Absolute::from('twip', 9062), 'leader' => \PhpOffice\PhpWord\Style\Tab::TAB_LEADER_DOT, - 'indent' => 200, + 'indent' => Absolute::from('twip', 200), ); - $object = new TOC(array('size' => 11), array('position' => $expected['position'])); + $object = new TOC(array('size' => Absolute::from('pt', 11)), array('position' => $expected['position'])); $tocStyle = $object->getStyleTOC(); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\TOC', $tocStyle); diff --git a/tests/PhpWord/Element/TableTest.php b/tests/PhpWord/Element/TableTest.php index 8ae5306c82..b8b8c29a02 100644 --- a/tests/PhpWord/Element/TableTest.php +++ b/tests/PhpWord/Element/TableTest.php @@ -1,4 +1,5 @@ assertInstanceOf('PhpOffice\\PhpWord\\Element\\Table', $oTable); $this->assertNull($oTable->getStyle()); - $this->assertNull($oTable->getWidth()); + $this->assertNull($oTable->getWidth()->toInt('twip')); $this->assertEquals(array(), $oTable->getRows()); $this->assertCount(0, $oTable->getRows()); } @@ -54,7 +58,7 @@ public function testStyleText() */ public function testStyleArray() { - $oTable = new Table(array('borderSize' => 6, 'borderColor' => '006699', 'cellMargin' => 80)); + $oTable = new Table(array('borderSize' => Absolute::from('eop', 6), 'borderColor' => new Colors\Hex('006699'), 'cellMargin' => Absolute::from('eop', 80))); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Table', $oTable->getStyle()); } @@ -66,8 +70,8 @@ public function testWidth() { $oTable = new Table(); $iVal = rand(1, 1000); - $oTable->setWidth($iVal); - $this->assertEquals($iVal, $oTable->getWidth()); + $oTable->setWidth(Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oTable->getWidth()->toInt('twip')); } /** diff --git a/tests/PhpWord/Element/TextBoxTest.php b/tests/PhpWord/Element/TextBoxTest.php index cd50acd4a6..b23e95b0c7 100644 --- a/tests/PhpWord/Element/TextBoxTest.php +++ b/tests/PhpWord/Element/TextBoxTest.php @@ -1,4 +1,5 @@ \PhpOffice\PhpWord\Shared\Converter::cmToPixel(4.5), - 'height' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(17.5), + 'width' => Absolute::from('cm', 4.5), + 'height' => Absolute::from('cm', 17.5), 'positioning' => 'absolute', - 'marginLeft' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(15.4), - 'marginTop' => \PhpOffice\PhpWord\Shared\Converter::cmToPixel(9.9), - 'stroke' => 0, - 'innerMargin' => 0, - 'borderSize' => 1, - 'borderColor' => '', + 'marginLeft' => Absolute::from('cm', 15.4), + 'marginTop' => Absolute::from('cm', 9.9), + 'innerMargin' => Absolute::from('eop', 0), + 'borderSize' => Absolute::from('eop', 1), ) ); diff --git a/tests/PhpWord/Element/TextBreakTest.php b/tests/PhpWord/Element/TextBreakTest.php index 13084c679a..da29a15e6b 100644 --- a/tests/PhpWord/Element/TextBreakTest.php +++ b/tests/PhpWord/Element/TextBreakTest.php @@ -1,4 +1,5 @@ 12); - $pStyle = array('spacing' => 240); + $fStyle = array('size' => Absolute::from('pt', 12)); + $pStyle = array('spacing' => Absolute::from('eop', 240)); $object = new TextBreak($fStyle, $pStyle); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Font', $object->getFontStyle()); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Paragraph', $object->getParagraphStyle()); diff --git a/tests/PhpWord/Element/TextRunTest.php b/tests/PhpWord/Element/TextRunTest.php index 2168bcc49a..eef1068d9f 100644 --- a/tests/PhpWord/Element/TextRunTest.php +++ b/tests/PhpWord/Element/TextRunTest.php @@ -1,4 +1,5 @@ 100)); + $oTextRun = new TextRun(array('spacing' => Absolute::from('twip', 100))); $this->assertInstanceOf('PhpOffice\\PhpWord\\Element\\TextRun', $oTextRun); $this->assertCount(0, $oTextRun->getElements()); @@ -178,7 +180,7 @@ public function testParagraph() $oText = new TextRun('paragraphStyle'); $this->assertEquals('paragraphStyle', $oText->getParagraphStyle()); - $oText->setParagraphStyle(array('alignment' => Jc::CENTER, 'spaceAfter' => 100)); + $oText->setParagraphStyle(array('alignment' => Jc::CENTER, 'spaceAfter' => Absolute::from('twip', 100))); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Paragraph', $oText->getParagraphStyle()); } } diff --git a/tests/PhpWord/Element/TextTest.php b/tests/PhpWord/Element/TextTest.php index 97be7ae5b6..b814762f9c 100644 --- a/tests/PhpWord/Element/TextTest.php +++ b/tests/PhpWord/Element/TextTest.php @@ -1,4 +1,5 @@ assertEquals('fontStyle', $oText->getFontStyle()); - $oText->setFontStyle(array('bold' => true, 'italic' => true, 'size' => 16)); + $oText->setFontStyle(array('bold' => true, 'italic' => true, 'size' => Absolute::from('pt', 16))); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Font', $oText->getFontStyle()); } @@ -80,7 +82,7 @@ public function testParagraph() $oText = new Text('text', 'fontStyle', 'paragraphStyle'); $this->assertEquals('paragraphStyle', $oText->getParagraphStyle()); - $oText->setParagraphStyle(array('alignment' => Jc::CENTER, 'spaceAfter' => 100)); + $oText->setParagraphStyle(array('alignment' => Jc::CENTER, 'spaceAfter' => Absolute::from('twip', 100))); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Paragraph', $oText->getParagraphStyle()); } } diff --git a/tests/PhpWord/Element/TitleTest.php b/tests/PhpWord/Element/TitleTest.php index 6ef87c3e8e..0c824387e0 100644 --- a/tests/PhpWord/Element/TitleTest.php +++ b/tests/PhpWord/Element/TitleTest.php @@ -1,4 +1,5 @@ setHyphenationZone($hyphenationZoneInTwip); - $this->assertSame($hyphenationZoneInTwip, $oSettings->getHyphenationZone()); + $oSettings->setHyphenationZone($hyphenationZone); + $this->assertSame($hyphenationZone->toInt('twip'), $oSettings->getHyphenationZone()->toInt('twip')); } public function testDefaultHyphenationZone() { $oSettings = new Settings(); - $this->assertNull($oSettings->getHyphenationZone()); + $this->assertNull($oSettings->getHyphenationZone()->toInt('twip')); } public function testDoNotHyphenateCaps() diff --git a/tests/PhpWord/PhpWordTest.php b/tests/PhpWord/PhpWordTest.php index d818e0f8bb..4617db1d29 100644 --- a/tests/PhpWord/PhpWordTest.php +++ b/tests/PhpWord/PhpWordTest.php @@ -1,4 +1,5 @@ assertEquals(new DocInfo(), $phpWord->getDocInfo()); $this->assertEquals(Settings::DEFAULT_FONT_NAME, $phpWord->getDefaultFontName()); - $this->assertEquals(Settings::DEFAULT_FONT_SIZE, $phpWord->getDefaultFontSize()); + $this->assertEquals(Settings::DEFAULT_FONT_SIZE, $phpWord->getDefaultFontSize()->toInt('pt')); } /** @@ -66,9 +68,9 @@ public function testSetGetDefaultFontSize() { $phpWord = new PhpWord(); $fontSize = 16; - $this->assertEquals(Settings::DEFAULT_FONT_SIZE, $phpWord->getDefaultFontSize()); - $phpWord->setDefaultFontSize($fontSize); - $this->assertEquals($fontSize, $phpWord->getDefaultFontSize()); + $this->assertEquals(Settings::DEFAULT_FONT_SIZE, $phpWord->getDefaultFontSize()->toInt('pt')); + $phpWord->setDefaultFontSize(Absolute::from('pt', $fontSize)); + $this->assertEquals($fontSize, $phpWord->getDefaultFontSize()->toInt('pt')); } /** diff --git a/tests/PhpWord/Reader/HTMLTest.php b/tests/PhpWord/Reader/HTMLTest.php index 38588afc17..17e77f3ac7 100644 --- a/tests/PhpWord/Reader/HTMLTest.php +++ b/tests/PhpWord/Reader/HTMLTest.php @@ -1,4 +1,5 @@ assertInstanceOf('PhpOffice\PhpWord\Style\Table', $elements[0]->getStyle()); /** @var \PhpOffice\PhpWord\Style\Table $tableStyle */ $tableStyle = $elements[0]->getStyle(); - $this->assertEquals(TblWidth::AUTO, $tableStyle->getUnit()); - $this->assertEquals(10.5, $tableStyle->getCellSpacing()); + $this->assertEquals(10.5, $tableStyle->getCellSpacing()->toFloat('twip')); + } + + /** + * Test reading of cell spacing + */ + public function testReadCellWidth() + { + $documentXml = ' + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + '; + + $phpWord = $this->getDocumentFromString(array('document' => $documentXml)); + + $cells = $phpWord->getSection(0)->getElement(0)->getRows()[0]->getCells(); + + $this->assertEquals(Absolute::from('twip', 10.3), $cells[0]->getWidth()); + $this->assertEquals(new Absolute(0), $cells[1]->getWidth()); + $this->assertEquals(new Percent(50), $cells[2]->getWidth()); + $this->assertEquals(new Absolute(0), $cells[3]->getWidth()); + $this->assertEquals(new Auto(), $cells[4]->getWidth()); + } + + /** + * Test reading of cell spacing + * @depends testReadCellWidth + * @expectedException \PHPUnit\Framework\Error\Warning + * @expectedExceptionMessage Cell width type `bad` is not supported + */ + public function testReadCellWidthBadType() + { + $documentXml = ' + + + + + + + + + + + + + + + + + + + '; + + $this->getDocumentFromString(array('document' => $documentXml)); + } + + /** + * Test reading of cell spacing + * @depends testReadCellWidthBadType + */ + public function testReadCellWidthBadTypeReturn() + { + $documentXml = ' + + + + + + + + + + + + + + + + + + + '; + + $phpWord = @$this->getDocumentFromString(array('document' => $documentXml)); + + $cells = $phpWord->getSection(0)->getElement(0)->getRows()[0]->getCells(); + + $this->assertEquals(new Absolute(null), $cells[0]->getWidth()); } /** @@ -90,16 +219,16 @@ public function testReadTablePosition() $this->assertInstanceOf('PhpOffice\PhpWord\Style\TablePosition', $elements[0]->getStyle()->getPosition()); /** @var \PhpOffice\PhpWord\Style\TablePosition $tableStyle */ $tableStyle = $elements[0]->getStyle()->getPosition(); - $this->assertEquals(10, $tableStyle->getLeftFromText()); - $this->assertEquals(20, $tableStyle->getRightFromText()); - $this->assertEquals(30, $tableStyle->getTopFromText()); - $this->assertEquals(40, $tableStyle->getBottomFromText()); + $this->assertEquals(10, $tableStyle->getLeftFromText()->toInt('twip')); + $this->assertEquals(20, $tableStyle->getRightFromText()->toInt('twip')); + $this->assertEquals(30, $tableStyle->getTopFromText()->toInt('twip')); + $this->assertEquals(40, $tableStyle->getBottomFromText()->toInt('twip')); $this->assertEquals(TablePosition::VANCHOR_PAGE, $tableStyle->getVertAnchor()); $this->assertEquals(TablePosition::HANCHOR_MARGIN, $tableStyle->getHorzAnchor()); $this->assertEquals(TablePosition::XALIGN_CENTER, $tableStyle->getTblpXSpec()); - $this->assertEquals(50, $tableStyle->getTblpX()); + $this->assertEquals(50, $tableStyle->getTblpX()->toInt('twip')); $this->assertEquals(TablePosition::YALIGN_TOP, $tableStyle->getTblpYSpec()); - $this->assertEquals(60, $tableStyle->getTblpY()); + $this->assertEquals(60, $tableStyle->getTblpY()->toInt('twip')); } /** @@ -126,26 +255,70 @@ public function testReadPosition() $this->assertInstanceOf('PhpOffice\PhpWord\Style\Font', $textRun->getElement(0)->getFontStyle()); /** @var \PhpOffice\PhpWord\Style\Font $fontStyle */ $fontStyle = $textRun->getElement(0)->getFontStyle(); - $this->assertEquals(15, $fontStyle->getPosition()); + $this->assertEquals(15, $fontStyle->getPosition()->toInt('twip')); } public function testReadIndent() + { + $indents = array( + array(2160, 'dxa', Absolute::from('twip', 2160)), + array('', 'nil', new Absolute(0)), + array('', 'auto', new Absolute(null)), + array(50, 'pct', new Absolute(null)), + ); + foreach ($indents as $indent) { + $documentXml = ' + + + + '; + + $phpWord = $this->getDocumentFromString(array('document' => $documentXml)); + + $elements = $phpWord->getSection(0)->getElements(); + $this->assertInstanceOf('PhpOffice\PhpWord\Element\Table', $elements[0]); + $this->assertInstanceOf('PhpOffice\PhpWord\Style\Table', $elements[0]->getStyle()); + /** @var \PhpOffice\PhpWord\Style\Table $tableStyle */ + $tableStyle = $elements[0]->getStyle(); + $this->assertEquals($indent[2], $tableStyle->getIndent()); + } + } + + /** + * @depends testReadIndent + * @expectedException \PHPUnit\Framework\Error\Warning + * @expectedExceptionMessage Table indent type `bad` is not supported + */ + public function testReadIndentBadType() { $documentXml = ' - + '; - $phpWord = $this->getDocumentFromString(array('document' => $documentXml)); + $this->getDocumentFromString(array('document' => $documentXml)); + } + + /** + * @depends testReadIndent + */ + public function testReadIndentBadTypeReturn() + { + $documentXml = ' + + + + '; + + $phpWord = @$this->getDocumentFromString(array('document' => $documentXml)); $elements = $phpWord->getSection(0)->getElements(); $this->assertInstanceOf('PhpOffice\PhpWord\Element\Table', $elements[0]); $this->assertInstanceOf('PhpOffice\PhpWord\Style\Table', $elements[0]->getStyle()); /** @var \PhpOffice\PhpWord\Style\Table $tableStyle */ $tableStyle = $elements[0]->getStyle(); - $this->assertSame(TblWidth::TWIP, $tableStyle->getIndent()->getType()); - $this->assertSame(2160, $tableStyle->getIndent()->getValue()); + $this->assertEquals(new Absolute(null), $tableStyle->getIndent()); } public function testReadTableRTL() diff --git a/tests/PhpWord/Reader/Word2007Test.php b/tests/PhpWord/Reader/Word2007Test.php index 6d660ce5ef..cc460c824d 100644 --- a/tests/PhpWord/Reader/Word2007Test.php +++ b/tests/PhpWord/Reader/Word2007Test.php @@ -1,4 +1,5 @@ assertEquals(Settings::DEFAULT_FONT_SIZE, Settings::getDefaultFontSize()); - $this->assertTrue(Settings::setDefaultFontSize(12)); - $this->assertFalse(Settings::setDefaultFontSize(null)); + $this->assertEquals(Settings::DEFAULT_FONT_SIZE, Settings::getDefaultFontSize()->toInt('pt')); + $this->assertTrue(Settings::setDefaultFontSize(Absolute::from('pt', 12))); + $this->assertFalse(Settings::setDefaultFontSize(Absolute::from('pt', null))); } /** @@ -137,4 +140,14 @@ public function testLoadConfig() // Test with invalid file $this->assertEmpty(Settings::loadConfig(__DIR__ . '/../../phpunit.xml.dist')); } + + /** + * Test loading bad config + * @expectedException \Exception + * @expectedExceptionMessage No method found for key `defaultFontClr` to set config value + */ + public function testLoadBadConfig() + { + Settings::loadConfig(__DIR__ . '/_files/broken-phpword.ini'); + } } diff --git a/tests/PhpWord/Shared/ConverterTest.php b/tests/PhpWord/Shared/ConverterTest.php deleted file mode 100644 index 15be8ec12a..0000000000 --- a/tests/PhpWord/Shared/ConverterTest.php +++ /dev/null @@ -1,139 +0,0 @@ -assertEquals($value / 2.54 * 1440, $result); - - $result = Converter::cmToInch($value); - $this->assertEquals($value / 2.54, $result); - - $result = Converter::cmToPixel($value); - $this->assertEquals($value / 2.54 * 96, $result); - - $result = Converter::cmToPoint($value); - $this->assertEquals($value / 2.54 * 72, $result); - - $result = Converter::cmToEmu($value); - $this->assertEquals(round($value / 2.54 * 96 * 9525), $result); - - $result = Converter::inchToTwip($value); - $this->assertEquals($value * 1440, $result); - - $result = Converter::inchToCm($value); - $this->assertEquals($value * 2.54, $result); - - $result = Converter::inchToPixel($value); - $this->assertEquals($value * 96, $result); - - $result = Converter::inchToPoint($value); - $this->assertEquals($value * 72, $result); - - $result = Converter::inchToEmu($value); - $this->assertEquals(round($value * 96 * 9525), $result); - - $result = Converter::pixelToTwip($value); - $this->assertEquals($value / 96 * 1440, $result); - - $result = Converter::pixelToCm($value); - $this->assertEquals($value / 96 * 2.54, $result); - - $result = Converter::pixelToPoint($value); - $this->assertEquals($value / 96 * 72, $result); - - $result = Converter::pixelToEmu($value); - $this->assertEquals(round($value * 9525), $result); - - $result = Converter::pointToTwip($value); - $this->assertEquals($value * 20, $result); - - $result = Converter::pointToCm($value); - $this->assertEquals($value * 0.035277778, $result, '', 0.00001); - - $result = Converter::pointToPixel($value); - $this->assertEquals($value / 72 * 96, $result); - - $result = Converter::pointToEmu($value); - $this->assertEquals(round($value / 72 * 96 * 9525), $result); - - $result = Converter::emuToPixel($value); - $this->assertEquals(round($value / 9525), $result); - - $result = Converter::picaToPoint($value); - $this->assertEquals($value / 6 * 72, $result, '', 0.00001); - - $result = Converter::degreeToAngle($value); - $this->assertEquals((int) round($value * 60000), $result); - - $result = Converter::angleToDegree($value); - $this->assertEquals(round($value / 60000), $result); - } - } - - /** - * Test htmlToRGB() - */ - public function testHtmlToRGB() - { - // Prepare test values [ original, expected ] - $values = array(); - $values[] = array('#FF99DD', array(255, 153, 221)); // With # - $values[] = array('FF99DD', array(255, 153, 221)); // 6 characters - $values[] = array('F9D', array(255, 153, 221)); // 3 characters - $values[] = array('0F9D', false); // 4 characters - // Conduct test - foreach ($values as $value) { - $result = Converter::htmlToRgb($value[0]); - $this->assertEquals($value[1], $result); - } - } - - /** - * Test css size to point - */ - public function testCssSizeParser() - { - $this->assertNull(Converter::cssToPoint('10em')); - $this->assertEquals(0, Converter::cssToPoint('0')); - $this->assertEquals(10, Converter::cssToPoint('10pt')); - $this->assertEquals(7.5, Converter::cssToPoint('10px')); - $this->assertEquals(720, Converter::cssToPoint('10in')); - $this->assertEquals(7.2, Converter::cssToPoint('0.1in')); - $this->assertEquals(120, Converter::cssToPoint('10pc')); - $this->assertEquals(28.346457, Converter::cssToPoint('10mm'), '', 0.000001); - $this->assertEquals(283.464567, Converter::cssToPoint('10cm'), '', 0.000001); - } -} diff --git a/tests/PhpWord/Shared/HtmlTest.php b/tests/PhpWord/Shared/HtmlTest.php index 5bc9e2411a..63ad727302 100644 --- a/tests/PhpWord/Shared/HtmlTest.php +++ b/tests/PhpWord/Shared/HtmlTest.php @@ -1,4 +1,5 @@ addSection(); - Html::addHtml($section, '

test

'); - Html::addHtml($section, '

test

'); - Html::addHtml($section, '

test

'); - Html::addHtml($section, '

test

'); + + $heights = array( + array('1', Paragraph::LINE_HEIGHT, LineSpacingRule::AUTO), + array('1.5', Paragraph::LINE_HEIGHT * 1.5, LineSpacingRule::AUTO), + array('2', Paragraph::LINE_HEIGHT * 2, LineSpacingRule::AUTO), + array('15pt', 300, LineSpacingRule::EXACT), + array('15.5pt', 310, LineSpacingRule::EXACT), + array('120%', Paragraph::LINE_HEIGHT * 1.2, LineSpacingRule::AUTO), + array('120.5%', round(Paragraph::LINE_HEIGHT * 1.205), LineSpacingRule::AUTO), + array('0.17in', 245, LineSpacingRule::EXACT), + array('1in', 1440, LineSpacingRule::EXACT), + ); + + foreach ($heights as $info) { + list($height, $expected, $rule) = $info; + Html::addHtml($section, '

test

'); + } $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); - $this->assertTrue($doc->elementExists('/w:document/w:body/w:p[1]/w:pPr/w:spacing')); - $this->assertEquals(Paragraph::LINE_HEIGHT * 1.5, $doc->getElementAttribute('/w:document/w:body/w:p[1]/w:pPr/w:spacing', 'w:line')); - $this->assertEquals(LineSpacingRule::AUTO, $doc->getElementAttribute('/w:document/w:body/w:p[1]/w:pPr/w:spacing', 'w:lineRule')); - $this->assertTrue($doc->elementExists('/w:document/w:body/w:p[2]/w:pPr/w:spacing')); - $this->assertEquals(300, $doc->getElementAttribute('/w:document/w:body/w:p[2]/w:pPr/w:spacing', 'w:line')); - $this->assertEquals(LineSpacingRule::EXACT, $doc->getElementAttribute('/w:document/w:body/w:p[2]/w:pPr/w:spacing', 'w:lineRule')); + $elem = 0; + foreach ($heights as $info) { + list($height, $expected, $rule) = $info; + $elem += 1; + $this->assertTrue($doc->elementExists('/w:document/w:body/w:p[' . $elem . ']/w:pPr/w:spacing')); + $this->assertEquals($expected, $doc->getElementAttribute('/w:document/w:body/w:p[' . $elem . ']/w:pPr/w:spacing', 'w:line')); + $this->assertEquals($rule, $doc->getElementAttribute('/w:document/w:body/w:p[' . $elem . ']/w:pPr/w:spacing', 'w:lineRule')); + } + } + + /** + * Test text-indent style + */ + public function testParagraphBorderWidths() + { + $dpi = new HtmlDpi(); + $borders = array( + 'border-width: 1px' => array( + 'top' => new BorderSide(Absolute::fromPixels($dpi, 1)), + 'right' => new BorderSide(Absolute::fromPixels($dpi, 1)), + 'bottom' => new BorderSide(Absolute::fromPixels($dpi, 1)), + 'left' => new BorderSide(Absolute::fromPixels($dpi, 1)), + ), + 'border-width: 1px 2px' => array( + 'top' => new BorderSide(Absolute::fromPixels($dpi, 1)), + 'right' => new BorderSide(Absolute::fromPixels($dpi, 2)), + 'bottom' => new BorderSide(Absolute::fromPixels($dpi, 1)), + 'left' => new BorderSide(Absolute::fromPixels($dpi, 2)), + ), + 'border-width: 1px 2px 3px' => array( + 'top' => new BorderSide(Absolute::fromPixels($dpi, 1)), + 'right' => new BorderSide(Absolute::fromPixels($dpi, 2)), + 'bottom' => new BorderSide(Absolute::fromPixels($dpi, 3)), + 'left' => new BorderSide(Absolute::fromPixels($dpi, 2)), + ), + 'border-width: 1px 2px 3px 4px' => array( + 'top' => new BorderSide(Absolute::fromPixels($dpi, 1)), + 'right' => new BorderSide(Absolute::fromPixels($dpi, 2)), + 'bottom' => new BorderSide(Absolute::fromPixels($dpi, 3)), + 'left' => new BorderSide(Absolute::fromPixels($dpi, 4)), + ), + ); + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $key = -1; + foreach ($borders as $css => $borderSides) { + $key += 1; + Html::addHtml($section, '

test

'); + foreach ($borderSides as $side => $border) { + $this->assertEquals($border, $section->getElement($key)->getParagraphStyle()->getBorder($side), "$side border width should match expectation for css `$css`"); + } + } + } - $this->assertTrue($doc->elementExists('/w:document/w:body/w:p[3]/w:pPr/w:spacing')); - $this->assertEquals(Paragraph::LINE_HEIGHT * 1.2, $doc->getElementAttribute('/w:document/w:body/w:p[3]/w:pPr/w:spacing', 'w:line')); - $this->assertEquals(LineSpacingRule::AUTO, $doc->getElementAttribute('/w:document/w:body/w:p[3]/w:pPr/w:spacing', 'w:lineRule')); + /** + * Test text-indent style + */ + public function testParagraphBorderColors() + { + $borders = array( + 'border-color: #f00' => array( + 'top' => new BorderSide(null, new Hex('f00')), + 'right' => new BorderSide(null, new Hex('f00')), + 'bottom' => new BorderSide(null, new Hex('f00')), + 'left' => new BorderSide(null, new Hex('f00')), + ), + 'border-color: #f00 blue' => array( + 'top' => new BorderSide(null, new Hex('f00')), + 'right' => new BorderSide(null, new Hex('00f')), + 'bottom' => new BorderSide(null, new Hex('f00')), + 'left' => new BorderSide(null, new Hex('00f')), + ), + 'border-color: #f00 blue lime' => array( + 'top' => new BorderSide(null, new Hex('f00')), + 'right' => new BorderSide(null, new Hex('00f')), + 'bottom' => new BorderSide(null, new Hex('0f0')), + 'left' => new BorderSide(null, new Hex('00f')), + ), + 'border-color: #f00 blue lime #ff0' => array( + 'top' => new BorderSide(null, new Hex('f00')), + 'right' => new BorderSide(null, new Hex('00f')), + 'bottom' => new BorderSide(null, new Hex('0f0')), + 'left' => new BorderSide(null, new Hex('ff0')), + ), + ); + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $key = -1; + foreach ($borders as $css => $borderSides) { + $key += 1; + Html::addHtml($section, '

test

'); + foreach ($borderSides as $side => $border) { + $this->assertEquals($border, $section->getElement($key)->getParagraphStyle()->getBorder($side), "$side border width should match expectation for css `$css`"); + } + } + } - $this->assertTrue($doc->elementExists('/w:document/w:body/w:p[4]/w:pPr/w:spacing')); - $this->assertEquals(244.8, $doc->getElementAttribute('/w:document/w:body/w:p[4]/w:pPr/w:spacing', 'w:line')); - $this->assertEquals(LineSpacingRule::EXACT, $doc->getElementAttribute('/w:document/w:body/w:p[4]/w:pPr/w:spacing', 'w:lineRule')); + /** + * Test text-indent style + */ + public function testParagraphBorderStyles() + { + $borders = array( + 'border-style: solid' => array( + 'top' => new BorderSide(null, null, new BorderStyle('single')), + 'right' => new BorderSide(null, null, new BorderStyle('single')), + 'bottom' => new BorderSide(null, null, new BorderStyle('single')), + 'left' => new BorderSide(null, null, new BorderStyle('single')), + ), + 'border-style: solid double' => array( + 'top' => new BorderSide(null, null, new BorderStyle('single')), + 'right' => new BorderSide(null, null, new BorderStyle('double')), + 'bottom' => new BorderSide(null, null, new BorderStyle('single')), + 'left' => new BorderSide(null, null, new BorderStyle('double')), + ), + 'border-style: solid double dotted' => array( + 'top' => new BorderSide(null, null, new BorderStyle('single')), + 'right' => new BorderSide(null, null, new BorderStyle('double')), + 'bottom' => new BorderSide(null, null, new BorderStyle('dotted')), + 'left' => new BorderSide(null, null, new BorderStyle('double')), + ), + 'border-style: solid double dotted dashed' => array( + 'top' => new BorderSide(null, null, new BorderStyle('single')), + 'right' => new BorderSide(null, null, new BorderStyle('double')), + 'bottom' => new BorderSide(null, null, new BorderStyle('dotted')), + 'left' => new BorderSide(null, null, new BorderStyle('dashed')), + ), + ); + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $key = -1; + foreach ($borders as $css => $borderSides) { + $key += 1; + Html::addHtml($section, '

test

'); + foreach ($borderSides as $side => $border) { + $this->assertEquals($border, $section->getElement($key)->getParagraphStyle()->getBorder($side), "$side border width should match expectation for css `$css`"); + } + } } /** @@ -297,7 +439,7 @@ public function testParseTable() header a - header b + header b header c @@ -316,14 +458,14 @@ public function testParseTable() //check border colors $this->assertEquals('00EE00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[2]/w:tcPr/w:tcBorders/w:top', 'w:color')); - $this->assertEquals('00EE00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[2]/w:tcPr/w:tcBorders/w:right', 'w:color')); + $this->assertEquals('00EE00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[2]/w:tcPr/w:tcBorders/w:end', 'w:color')); $this->assertEquals('00EE00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[2]/w:tcPr/w:tcBorders/w:bottom', 'w:color')); - $this->assertEquals('00EE00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[2]/w:tcPr/w:tcBorders/w:left', 'w:color')); + $this->assertEquals('00EE00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[2]/w:tcPr/w:tcBorders/w:start', 'w:color')); $this->assertEquals('00AA00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[3]/w:tcPr/w:tcBorders/w:top', 'w:color')); - $this->assertEquals('00BB00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[3]/w:tcPr/w:tcBorders/w:right', 'w:color')); + $this->assertEquals('00BB00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[3]/w:tcPr/w:tcBorders/w:end', 'w:color')); $this->assertEquals('00CC00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[3]/w:tcPr/w:tcBorders/w:bottom', 'w:color')); - $this->assertEquals('00DD00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[3]/w:tcPr/w:tcBorders/w:left', 'w:color')); + $this->assertEquals('00DD00', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr[1]/w:tc[3]/w:tcPr/w:tcBorders/w:start', 'w:color')); } /** @@ -605,11 +747,11 @@ public function testParseMalformedStyleIsIgnored() /** * Tests parsing hidden text */ - public function testParseHiddenText() + public function testParseDisplayNone() { $phpWord = new \PhpOffice\PhpWord\PhpWord(); $section = $phpWord->addSection(); - $html = '

This is some hidden text.

'; + $html = '

This is some hidden text.

'; Html::addHtml($section, $html); $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); @@ -617,6 +759,51 @@ public function testParseHiddenText() $this->assertTrue($doc->elementExists('/w:document/w:body/w:p/w:r/w:rPr/w:vanish')); } + /** + * Tests parsing hidden text + */ + public function testParseDisplayFoo() + { + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $html = '

This is some hidden text.

'; + Html::addHtml($section, $html); + + $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); + + $this->assertFalse($doc->elementExists('/w:document/w:body/w:p/w:r/w:rPr/w:vanish')); + } + + /** + * Tests parsing hidden text + */ + public function testParseVisibilityHidden() + { + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $html = '

This is some hidden text.

'; + Html::addHtml($section, $html); + + $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); + + $this->assertTrue($doc->elementExists('/w:document/w:body/w:p/w:r/w:rPr/w:vanish')); + } + + /** + * Tests parsing unrecognized visibility values + */ + public function testParseVisibilityFoo() + { + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $html = '

This is some hidden text.

'; + Html::addHtml($section, $html); + + $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); + + $this->assertFalse($doc->elementExists('/w:document/w:body/w:p/w:r/w:rPr/w:vanish')); + } + /** * Tests parsing letter spacing */ @@ -632,4 +819,45 @@ public function testParseLetterSpacing() $this->assertTrue($doc->elementExists('/w:document/w:body/w:p/w:r/w:rPr/w:spacing')); $this->assertEquals(150 * 15, $doc->getElement('/w:document/w:body/w:p/w:r/w:rPr/w:spacing')->getAttribute('w:val')); } + + /** + * Test css size to point + * @covers \PhpOffice\PhpWord\Shared\Html::cssToLength() + */ + public function testCssToLength() + { + $this->assertEquals(900, Html::cssToLength('9em')->toFloat()); + $this->assertEquals(0, Html::cssToLength('0')->toFloat('pt')); + $this->assertEquals(10, Html::cssToLength('10pt')->toFloat('pt')); + $this->assertEquals(7.5, Html::cssToLength('10px')->toFloat('pt')); + $this->assertEquals(720, Html::cssToLength('10in')->toFloat('pt')); + $this->assertEquals(7.2, Html::cssToLength('0.1in')->toFloat('pt')); + $this->assertEquals(120, Html::cssToLength('10pc')->toFloat('pt')); + $this->assertEquals(28.346457, Html::cssToLength('10mm')->toFloat('pt'), '', .00001); + $this->assertEquals(283.464567, Html::cssToLength('10cm')->toFloat('pt'), '', .0000001); + $this->assertEquals('10', Html::cssToLength('10%')->toFloat(), '', .0000001); + // Invalid length + $this->assertNull(Html::cssToLength('54mb')->toFloat('pt'), '', .0000001); + } + + /** + * Test css size to point + * @covers \PhpOffice\PhpWord\Shared\Html::cssToAbsolute() + */ + public function testCssToAbsolute() + { + $this->assertNull(Html::cssToAbsolute('10em')->toFloat('pt')); + $this->assertEquals(0, Html::cssToAbsolute('0')->toFloat('pt')); + $this->assertEquals(10, Html::cssToAbsolute('10pt')->toFloat('pt')); + $this->assertEquals(7.5, Html::cssToAbsolute('10px')->toFloat('pt')); + $this->assertEquals(720, Html::cssToAbsolute('10in')->toFloat('pt')); + $this->assertEquals(7.2, Html::cssToAbsolute('0.1in')->toFloat('pt')); + $this->assertEquals(120, Html::cssToAbsolute('10pc')->toFloat('pt')); + $this->assertEquals(28.346457, Html::cssToAbsolute('10mm')->toFloat('pt'), '', .00001); + $this->assertEquals(283.464567, Html::cssToAbsolute('10cm')->toFloat('pt'), '', .0000001); + // Percents only work for length + $this->assertNull(Html::cssToAbsolute('10%')->toFloat('pt'), '', .0000001); + // Invalid length + $this->assertNull(Html::cssToAbsolute('10mb')->toFloat('pt'), '', .0000001); + } } diff --git a/tests/PhpWord/Shared/ZipArchiveTest.php b/tests/PhpWord/Shared/ZipArchiveTest.php index ecd0961e24..399c456cc0 100644 --- a/tests/PhpWord/Shared/ZipArchiveTest.php +++ b/tests/PhpWord/Shared/ZipArchiveTest.php @@ -1,4 +1,5 @@ assertTrue(self::callProtectedMethod($stub, 'setBoolVal', array(true, false))); $this->assertEquals(12, self::callProtectedMethod($stub, 'setIntVal', array(12, 200))); - $this->assertEquals(871.1, self::callProtectedMethod($stub, 'setFloatVal', array(871.1, 2.1))); - $this->assertEquals(871.1, self::callProtectedMethod($stub, 'setFloatVal', array('871.1', 2.1))); $this->assertEquals('a', self::callProtectedMethod($stub, 'setEnumVal', array('a', array('a', 'b'), 'b'))); } /** - * Test setBoolVal, setIntVal, setFloatVal, setEnumVal with default value + * Test setBoolVal, setIntVal, setEnumVal with default value */ public function testSetValDefault() { @@ -58,7 +57,6 @@ public function testSetValDefault() $this->assertNotTrue(self::callProtectedMethod($stub, 'setBoolVal', array('a', false))); $this->assertEquals(200, self::callProtectedMethod($stub, 'setIntVal', array('foo', 200))); - $this->assertEquals(2.1, self::callProtectedMethod($stub, 'setFloatVal', array('foo', 2.1))); $this->assertEquals('b', self::callProtectedMethod($stub, 'setEnumVal', array(null, array('a', 'b'), 'b'))); } @@ -77,9 +75,7 @@ public function testSetValEnumException() /** * Helper function to call protected method * - * @param mixed $object * @param string $method - * @param array $args */ public static function callProtectedMethod($object, $method, array $args = array()) { diff --git a/tests/PhpWord/Style/BorderStyleTest.php b/tests/PhpWord/Style/BorderStyleTest.php new file mode 100644 index 0000000000..18a87126dd --- /dev/null +++ b/tests/PhpWord/Style/BorderStyleTest.php @@ -0,0 +1,110 @@ +assertEquals($style, $borderStyle->getStyle()); + + $borderStyle = BorderStyle::fromMixed($style); + $this->assertEquals($style, $borderStyle->getStyle()); + } + } + + /** + * Test setting style with invalid value + * @expectedException \Exception + * @expectedExceptionMessage Provided border style must be valid. 'badstyle' provided. Allowed: 'single', 'dashDotStroked', 'dashed', 'dashSmallGap', 'dotDash', 'dotDotDash', 'dotted', 'double', 'doubleWave', 'inset', 'nil', 'none', 'outset', 'thick', 'thickThinLargeGap', 'thickThinMediumGap', 'thickThinSmallGap', 'thinThickLargeGap', 'thinThickMediumGap', 'thinThickSmallGap', 'thinThickThinLargeGap', 'thinThickThinMediumGap', 'thinThickThinSmallGap', 'threeDEmboss', 'threeDEngrave', 'triple', 'wave' + */ + public function testSetGetBad() + { + new BorderStyle('badstyle'); + } + + /** + * Test setting style with wrong type + * @expectedException \TypeError + * @expectedExceptionMessage Argument 1 passed to PhpOffice\PhpWord\Style\BorderStyle::__construct() must be of the type string + */ + public function testSetGetWrongType() + { + new BorderStyle(54); + } + + /** + * Test getting style from bad value + * @expectedException \PHPUnit\Framework\Error\Warning + * @expectedExceptionMessage Border style `badstyle` is not a valid option + */ + public function testFromMixedBadValue() + { + BorderStyle::fromMixed('badstyle'); + } + + /** + * Test getting style from bad value + */ + public function testValueFromMixedBadValue() + { + $borderStyle = @BorderStyle::fromMixed('badstyle'); + $this->assertEquals('single', $borderStyle->getStyle()); + } +} diff --git a/tests/PhpWord/Style/CellTest.php b/tests/PhpWord/Style/CellTest.php index 3c31a45794..b885924adb 100644 --- a/tests/PhpWord/Style/CellTest.php +++ b/tests/PhpWord/Style/CellTest.php @@ -1,4 +1,5 @@ VerticalJc::TOP, 'textDirection' => Cell::TEXT_DIR_BTLR, - 'bgColor' => 'FFFF00', - 'borderTopSize' => 120, - 'borderTopColor' => 'FFFF00', - 'borderLeftSize' => 120, - 'borderLeftColor' => 'FFFF00', - 'borderRightSize' => 120, - 'borderRightColor' => 'FFFF00', - 'borderBottomSize' => 120, - 'borderBottomColor' => 'FFFF00', + 'bgColor' => new Hex('FFFF00'), 'gridSpan' => 2, 'vMerge' => Cell::VMERGE_RESTART, ); foreach ($attributes as $key => $value) { - $set = "set{$key}"; - $get = "get{$key}"; + $get = "get$key"; + $result = $cell->$get(); + if ($result instanceof BasicColor) { + $result = $result->toHex(); + } elseif ($result instanceof Absolute) { + $result = $result->toInt('eop'); + } + + $this->assertNull($result); - $this->assertNull($object->$get()); // Init with null value + $set = "set{$key}"; + $cell->$set($value); - $object->$set($value); + $get = "get$key"; + $result = $cell->$get(); + if ($result instanceof BasicColor) { + $result = $result->toHex(); + $value = $value->toHex(); + } - $this->assertEquals($value, $object->$get()); + $this->assertEquals($value, $result); } } /** - * Test border color + * Test borders */ - public function testBorderColor() + public function testBorders() { - $object = new Cell(); + $cell = new Cell(); + + $this->assertFalse($cell->hasBorder()); + $borders = array('top', 'end', 'bottom', 'start'); + $borderSides = array( + array(Absolute::from('pt', rand(1, 20)), new Hex('f93de1'), new BorderStyle('double'), Absolute::from('pt', rand(1, 20)), true), + array(Absolute::from('twip', rand(1, 400)), new Hex('000000'), new BorderStyle('outset'), Absolute::from('twip', rand(1, 400)), false), + array(Absolute::from('eop', rand(1, 160)), new Rgb(255, 0, 100), new BorderStyle('dotted'), Absolute::from('eop', rand(1, 160)), true), + ); + $lastBorderSide = array(new Absolute(0), new Hex(null), new BorderStyle('single'), new Absolute(0), false); + foreach ($borderSides as $key => $borderSide) { + $newBorder = new BorderSide(...$borderSide); + + foreach ($borders as $side) { + $currentBorder = $cell->getBorder($side); + $this->assertEquals($lastBorderSide[0], $currentBorder->getSize(), "Size for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[1], $currentBorder->getColor(), "Color for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[2], $currentBorder->getStyle(), "Style for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[3], $currentBorder->getSpace(), "Space for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[4], $currentBorder->getShadow(), "Shadow for border side #$key for side $side should match last border side still"); - $value = 'FF0000'; + $cell->setBorder($side, $newBorder); + $updatedBorder = $cell->getBorder($side); + $this->assertEquals($borderSide[0], $updatedBorder->getSize(), "Size for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[1], $updatedBorder->getColor(), "Color for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[2], $updatedBorder->getStyle(), "Style for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[3], $updatedBorder->getSpace(), "Space for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[4], $updatedBorder->getShadow(), "Shadow for border side #$key for side $side should match new border"); + } - $object->setStyleValue('borderColor', $value); - $expected = array($value, $value, $value, $value); - $this->assertEquals($expected, $object->getBorderColor()); + $lastBorderSide = $borderSide; + } } /** - * Test border size + * Test width */ - public function testBorderSize() + public function testWidth() { - $object = new Cell(); + $cell = new Cell(); + + // Undefined + $width = $cell->getWidth(); + $this->assertInstanceOf(Length::class, $width); + $this->assertNull($width->toInt('twip')); + + // Null + $cell->setWidth(new Absolute(null)); + $width = $cell->getWidth(); + $this->assertInstanceOf(Absolute::class, $width); + $this->assertNull($width->toInt('twip')); + + // Absolute + $cell->setWidth(Absolute::from('twip', 204)); + $width = $cell->getWidth(); + $this->assertInstanceOf(Length::class, $width); + $this->assertEquals(204, $width->toInt('twip')); + + // Percent + $cell->setWidth(new Percent(50)); + $width = $cell->getWidth(); + $this->assertInstanceOf(Percent::class, $width); + $this->assertEquals(50, $width->toInt()); - $value = 120; - $expected = array($value, $value, $value, $value); - $object->setStyleValue('borderSize', $value); - $this->assertEquals($expected, $object->getBorderSize()); + // Auto + $cell->setWidth(new Auto()); + $width = $cell->getWidth(); + $this->assertInstanceOf(Auto::class, $width); } } diff --git a/tests/PhpWord/Style/ChartTest.php b/tests/PhpWord/Style/ChartTest.php index 9929a8f5a7..91398a96bc 100644 --- a/tests/PhpWord/Style/ChartTest.php +++ b/tests/PhpWord/Style/ChartTest.php @@ -1,4 +1,5 @@ assertEquals($chart->getWidth(), 1000000); + $this->assertEquals($chart->getWidth()->toInt('emu'), 1000000); - $chart->setWidth(200); + $chart->setWidth(Absolute::from('emu', 200)); - $this->assertEquals($chart->getWidth(), 200); + $this->assertEquals($chart->getWidth()->toInt('emu'), 200); } /** @@ -46,11 +49,11 @@ public function testSetGetHeight() { $chart = new Chart(); - $this->assertEquals($chart->getHeight(), 1000000); + $this->assertEquals($chart->getHeight()->toInt('emu'), 1000000); - $chart->setHeight(200); + $chart->setHeight(Absolute::from('emu', 200)); - $this->assertEquals($chart->getHeight(), 200); + $this->assertEquals($chart->getHeight()->toInt('emu'), 200); } /** diff --git a/tests/PhpWord/Style/Colors/ColorTest.php b/tests/PhpWord/Style/Colors/ColorTest.php new file mode 100644 index 0000000000..d32dd9b960 --- /dev/null +++ b/tests/PhpWord/Style/Colors/ColorTest.php @@ -0,0 +1,97 @@ +assertInstanceOf($value[1], $color, $message); + $this->assertEquals($value[2], $color->toHexOrName(), $message); + } + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage All colors must implement NamedColorInterface or StaticColorInterface. 'class@anonymous + */ + public function testBadClass() + { + $color = BasicColor::fromMixed(new class() extends BasicColor { + public function isSpecified(): bool + { + return false; + } + }); + $color->toHexOrName(); + } + + /** + * @expectedException \PHPUnit\Framework\Error\Warning + * @expectedExceptionMessage Color `fakeColor` is not a valid color + */ + public function testInvalidColor() + { + BasicColor::fromMixed('fakeColor'); + } + + public function testValueInvalidColor() + { + $color = @BasicColor::fromMixed('fakeColor'); + $this->assertInstanceOf(BasicColor::class, $color); + $this->assertNull($color->toHexOrName()); + } + + /** + * @expectedException \Error + * @expectedExceptionMessage Cannot instantiate abstract class PhpOffice\PhpWord\Style\Colors\BasicColor + */ + public function testBasicInstantiation() + { + new BasicColor(); + } + + /** + * @expectedException \Error + * @expectedExceptionMessage Cannot instantiate abstract class PhpOffice\PhpWord\Style\Colors\SpecialColor + */ + public function testSpecialInstantiation() + { + new SpecialColor(); + } +} diff --git a/tests/PhpWord/Style/Colors/HexTest.php b/tests/PhpWord/Style/Colors/HexTest.php new file mode 100644 index 0000000000..f44f4d8856 --- /dev/null +++ b/tests/PhpWord/Style/Colors/HexTest.php @@ -0,0 +1,64 @@ +assertEquals($value[1], $result->toHex(), $message); + $this->assertEquals($value[1] === null ? null : '#' . $value[1], $result->toHex(true), $message); + $this->assertEquals($value[1], $result->toHexOrName(), $message); + $this->assertEquals($value[1] === null ? null : '#' . $value[1], $result->toHexOrName(true), $message); + $this->assertEquals($value[2], $result->toRgb(), $message); + } + } +} diff --git a/tests/PhpWord/Style/Colors/HighlightColorTest.php b/tests/PhpWord/Style/Colors/HighlightColorTest.php new file mode 100644 index 0000000000..fb69a2c85f --- /dev/null +++ b/tests/PhpWord/Style/Colors/HighlightColorTest.php @@ -0,0 +1,70 @@ +assertEquals($value[2], $result->toHex(), $message); + $this->assertEquals($value[2] === null ? null : '#' . $value[2], $result->toHex(true), $message); + $this->assertEquals($value[0], $result->toHexOrName(), $message); + $this->assertEquals($value[0], $result->toHexOrName(true), $message); + $this->assertEquals($value[1], $result->isSpecified(), $message); + $this->assertEquals($value[3], $result->toRgb(), $message); + $this->assertEquals($value[0], $result->getName(), $message); + } + } +} diff --git a/tests/PhpWord/Style/Colors/RgbTest.php b/tests/PhpWord/Style/Colors/RgbTest.php new file mode 100644 index 0000000000..fae2dc6e6d --- /dev/null +++ b/tests/PhpWord/Style/Colors/RgbTest.php @@ -0,0 +1,120 @@ +assertEquals($value[2], $result->toHex(), $message); + $this->assertEquals($value[2] === null ? null : '#' . $value[2], $result->toHex(true), $message); + $this->assertEquals($value[2], $result->toHexOrName(), $message); + $this->assertEquals($value[2] === null ? null : '#' . $value[2], $result->toHexOrName(true), $message); + $this->assertEquals($value[1], $result->isSpecified(), $message); + $this->assertEquals($value[0], $result->toRgb(), $message); + } + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Provided values must be 0–255. Provided `Rgb(-1, 131, 253)` + */ + public function testLowRed() + { + new Rgb(-1, 131, 253); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Provided values must be 0–255. Provided `Rgb(131, -1, 253)` + */ + public function testLowGreen() + { + new Rgb(131, -1, 253); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Provided values must be 0–255. Provided `Rgb(131, 253, -1)` + */ + public function testLowBlue() + { + new Rgb(131, 253, -1); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Provided values must be 0–255. Provided `Rgb(256, 131, 253)` + */ + public function testHighRed() + { + new Rgb(256, 131, 253); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Provided values must be 0–255. Provided `Rgb(131, 256, 253)` + */ + public function testHighGreen() + { + new Rgb(131, 256, 253); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Provided values must be 0–255. Provided `Rgb(131, 253, 256)` + */ + public function testHighBlue() + { + new Rgb(131, 253, 256); + } + + /** + * @expectedException \TypeError + * @expectedExceptionMessageRegEx (Argument 3 passed to PhpOffice\\PhpWord\\Style\\Colors\\Rgb::\_\_construct\(\) must be of the type integer, none given|Too few arguments to function PhpOffice\\PhpWord\\Style\\Colors\\Rgb::\_\_construct\(\), 2 passed) + */ + public function testTooFewArgs() + { + new Rgb(131, 253); + } +} diff --git a/tests/PhpWord/Style/Colors/SystemColorTest.php b/tests/PhpWord/Style/Colors/SystemColorTest.php new file mode 100644 index 0000000000..0c5bf3aa73 --- /dev/null +++ b/tests/PhpWord/Style/Colors/SystemColorTest.php @@ -0,0 +1,81 @@ +assertEquals($value, $result->getName(), $message); + $this->assertEquals($value, $result->toHexOrName(), $message); + $this->assertTrue($result->isSpecified(), $message); + + // Last color + $this->assertEquals('#000000', $result->getLastColor()->toHex(true)); + $this->assertEquals('000000', $result->getLastColor()->toHex()); + $this->assertEquals(array(0, 0, 0), $result->getLastColor()->toRgb()); + } + } +} diff --git a/tests/PhpWord/Style/Colors/ThemeColorTest.php b/tests/PhpWord/Style/Colors/ThemeColorTest.php new file mode 100644 index 0000000000..26eaccb9c4 --- /dev/null +++ b/tests/PhpWord/Style/Colors/ThemeColorTest.php @@ -0,0 +1,60 @@ +assertInstanceOf(BasicColor::class, $result); + $this->assertInstanceOf(NamedColorInterface::class, $result); + $this->assertEquals($value, $result->getName(), $message); + $this->assertEquals($value, $result->toHexOrName(), $message); + $this->assertTrue($result->isSpecified()); + } + } +} diff --git a/tests/PhpWord/Style/FontTest.php b/tests/PhpWord/Style/FontTest.php index 84916fc277..ac295dd942 100644 --- a/tests/PhpWord/Style/FontTest.php +++ b/tests/PhpWord/Style/FontTest.php @@ -1,4 +1,5 @@ null, - 'size' => null, + 'size' => new Absolute(null), 'hint' => null, - 'color' => null, + 'color' => new Hex(null), 'bold' => false, 'italic' => false, 'underline' => Font::UNDERLINE_NONE, @@ -70,21 +76,23 @@ public function testSetStyleValueWithNullOrEmpty() 'smallCaps' => false, 'allCaps' => false, 'rtl' => false, - 'fgColor' => null, - 'bgColor' => null, - 'scale' => null, - 'spacing' => null, - 'kerning' => null, + 'fgColor' => new HighlightColor(null), + 'bgColor' => new Hex(null), + 'scale' => new Percent(null), + 'spacing' => new Absolute(null), + 'kerning' => new Absolute(null), 'lang' => null, 'hidden' => false, ); foreach ($attributes as $key => $default) { $get = is_bool($default) ? "is{$key}" : "get{$key}"; - $this->assertEquals($default, $object->$get()); - $object->setStyleValue($key, null); - $this->assertEquals($default, $object->$get()); - $object->setStyleValue($key, ''); - $this->assertEquals($default, $object->$get()); + $new = $default; + if ($key === 'underline') { + $new = null; + } + $this->assertEquals($default, $object->$get(), "Attribute `$key` should start at default"); + $object->setStyleValue($key, $new); + $this->assertEquals($default, $object->$get(), "Attribute `$key` should remain at default if set to `null`"); } } @@ -97,8 +105,8 @@ public function testSetStyleValueNormal() $attributes = array( 'name' => 'Times New Roman', - 'size' => 9, - 'color' => '999999', + 'size' => Absolute::from('pt', 9), + 'color' => new Hex('999999'), 'hint' => 'eastAsia', 'bold' => true, 'italic' => true, @@ -109,12 +117,12 @@ public function testSetStyleValueNormal() 'doubleStrikethrough' => false, 'smallCaps' => true, 'allCaps' => false, - 'fgColor' => Font::FGCOLOR_YELLOW, - 'bgColor' => 'FFFF00', - 'lineHeight' => 2, - 'scale' => 150, - 'spacing' => 240, - 'kerning' => 10, + 'fgColor' => new HighlightColor('yellow'), + 'bgColor' => new Hex('FFFF00'), + 'lineHeight' => new Percent(200), + 'scale' => new Percent(150), + 'spacing' => Absolute::from('twip', 240), + 'kerning' => Absolute::from('hpt', 10), 'rtl' => true, 'noProof' => true, 'lang' => new Language(Language::EN_US), @@ -123,7 +131,15 @@ public function testSetStyleValueNormal() $object->setStyleByArray($attributes); foreach ($attributes as $key => $value) { $get = is_bool($value) ? "is{$key}" : "get{$key}"; - $this->assertEquals($value, $object->$get()); + $result = $object->$get(); + if ($result instanceof BasicColor) { + $result = $result->toHexOrName(); + $value = $value->toHexOrName(); + } elseif ($result instanceof Absolute) { + $result = $result->toInt('hpt'); + $value = $value->toInt('hpt'); + } + $this->assertEquals($value, $result); } } @@ -136,7 +152,7 @@ public function testLineHeight() $section = $phpWord->addSection(); // Test style array - $text = $section->addText('This is a test', array('line-height' => 2.0)); + $text = $section->addText('This is a test', array('line-height' => new Percent(200))); $doc = TestHelperDOCX::getDocument($phpWord); $element = $doc->getElement('/w:document/w:body/w:p/w:pPr/w:spacing'); @@ -148,7 +164,7 @@ public function testLineHeight() $this->assertEquals('auto', $lineRule); // Test setter - $text->getFontStyle()->setLineHeight(3.0); + $text->getFontStyle()->setLineHeight(new Percent(300)); $doc = TestHelperDOCX::getDocument($phpWord); $element = $doc->getElement('/w:document/w:body/w:p/w:pPr/w:spacing'); @@ -165,14 +181,14 @@ public function testLineHeight() public function testLineHeightFloatval() { $object = new Font(null, array('alignment' => Jc::CENTER)); - $object->setLineHeight('1.5pt'); - $this->assertEquals(1.5, $object->getLineHeight()); + $object->setLineHeight(new Percent(1.5)); + $this->assertEquals(1.5, $object->getLineHeight()->toFloat()); } /** * Test line height exception by using nonnumeric value * - * @expectedException \PhpOffice\PhpWord\Exception\InvalidStyleException + * @expectedException \TypeError */ public function testLineHeightException() { diff --git a/tests/PhpWord/Style/ImageTest.php b/tests/PhpWord/Style/ImageTest.php index 1d43d92152..4b2eb42dab 100644 --- a/tests/PhpWord/Style/ImageTest.php +++ b/tests/PhpWord/Style/ImageTest.php @@ -1,4 +1,5 @@ 200, - 'height' => 200, + 'width' => Absolute::from('twip', 200), + 'height' => Absolute::from('twip', 200), 'alignment' => Jc::START, - 'marginTop' => 240, - 'marginLeft' => 240, + 'marginTop' => Absolute::from('twip', 240), + 'marginLeft' => Absolute::from('twip', 240), 'wrappingStyle' => 'inline', - 'wrapDistanceLeft' => 10, - 'wrapDistanceRight' => 20, - 'wrapDistanceTop' => 30, - 'wrapDistanceBottom' => 40, + 'wrapDistanceLeft' => Absolute::from('twip', 10), + 'wrapDistanceRight' => Absolute::from('twip', 20), + 'wrapDistanceTop' => Absolute::from('twip', 30), + 'wrapDistanceBottom' => Absolute::from('twip', 40), ); foreach ($properties as $key => $value) { $set = "set{$key}"; $get = "get{$key}"; $object->$set($value); - $this->assertEquals($value, $object->$get()); + $result = $object->$get(); + if ($result instanceof Absolute) { + $result = $result->toInt('twip'); + $value = $value->toInt('twip'); + } + $this->assertEquals($value, $result); } } @@ -62,26 +69,31 @@ public function testSetStyleValue() $object = new Image(); $properties = array( - 'width' => 200, - 'height' => 200, + 'width' => Absolute::from('twip', 200), + 'height' => Absolute::from('twip', 200), 'alignment' => Jc::START, - 'marginTop' => 240, - 'marginLeft' => 240, - 'position' => 10, + 'marginTop' => Absolute::from('twip', 240), + 'marginLeft' => Absolute::from('twip', 240), + 'position' => Absolute::from('twip', 10), 'positioning' => \PhpOffice\PhpWord\Style\Image::POSITION_ABSOLUTE, 'posHorizontal' => \PhpOffice\PhpWord\Style\Image::POSITION_HORIZONTAL_CENTER, 'posVertical' => \PhpOffice\PhpWord\Style\Image::POSITION_VERTICAL_TOP, 'posHorizontalRel' => \PhpOffice\PhpWord\Style\Image::POSITION_RELATIVE_TO_COLUMN, 'posVerticalRel' => \PhpOffice\PhpWord\Style\Image::POSITION_RELATIVE_TO_IMARGIN, - 'wrapDistanceLeft' => 10, - 'wrapDistanceRight' => 20, - 'wrapDistanceTop' => 30, - 'wrapDistanceBottom' => 40, + 'wrapDistanceLeft' => Absolute::from('twip', 10), + 'wrapDistanceRight' => Absolute::from('twip', 20), + 'wrapDistanceTop' => Absolute::from('twip', 30), + 'wrapDistanceBottom' => Absolute::from('twip', 40), ); foreach ($properties as $key => $value) { $get = "get{$key}"; $object->setStyleValue("{$key}", $value); - $this->assertEquals($value, $object->$get()); + $result = $object->$get(); + if ($result instanceof Absolute) { + $result = $result->toInt('twip'); + $value = $value->toInt('twip'); + } + $this->assertEquals($value, $result); } } diff --git a/tests/PhpWord/Style/IndentationTest.php b/tests/PhpWord/Style/IndentationTest.php index b39a4d771d..a284e75c37 100644 --- a/tests/PhpWord/Style/IndentationTest.php +++ b/tests/PhpWord/Style/IndentationTest.php @@ -1,4 +1,5 @@ array(0, 10), - 'right' => array(0, 10), - 'firstLine' => array(null, 20), - 'hanging' => array(null, 20), + 'left' => array(0, Absolute::from('twip', 10)), + 'right' => array(0, Absolute::from('twip', 10)), + 'firstLine' => array(null, Absolute::from('twip', 20)), + 'hanging' => array(null, Absolute::from('twip', 20)), ); foreach ($properties as $property => $value) { list($default, $expected) = $value; $get = "get{$property}"; $set = "set{$property}"; - $this->assertEquals($default, $object->$get()); // Default value + $result = $object->$get(); + if ($expected instanceof Absolute) { + $result = $result->toInt('twip'); + } + $this->assertEquals($default, $result); // Default value $object->$set($expected); - $this->assertEquals($expected, $object->$get()); // New value + $result = $object->$get(); + if ($expected instanceof Absolute) { + $expected = $expected->toInt('twip'); + $result = $result->toInt('twip'); + } + $this->assertEquals($expected, $result); // New value } } } diff --git a/tests/PhpWord/Style/LanguageTest.php b/tests/PhpWord/Style/LanguageTest.php index 3bf516f8ae..ee072e14dd 100644 --- a/tests/PhpWord/Style/LanguageTest.php +++ b/tests/PhpWord/Style/LanguageTest.php @@ -1,4 +1,5 @@ assertEquals($value / 2.54 * 1440, $result->toFloat('twip')); + + $result = Absolute::from('cm', $value); + $this->assertEquals($value / 2.54, $result->toFloat('in')); + + $result = Absolute::from('cm', $value); + $this->assertEquals(round($value / 2.54 * 96), $result->toPixels(new DpiHelper(96))); + + $result = Absolute::from('cm', $value); + $this->assertEquals($value / 2.54 * 72, $result->toFloat('pt')); + + $result = Absolute::from('cm', $value); + $this->assertEquals($value / 2.54 * 72 * 8, $result->toFloat('eop')); + + $result = Absolute::from('cm', $value); + $this->assertEquals($value / 2.54 * 72 * 12700, $result->toFloat('emu'), '', 0.00000001); + + $result = Absolute::from('in', $value); + $this->assertEquals($value * 1440, $result->toFloat('twip')); + + $result = Absolute::from('in', $value); + $this->assertEquals($value * 2.54, $result->toFloat('cm')); + + $result = Absolute::from('in', $value); + $this->assertEquals(round($value * 96), $result->toPixels(new DpiHelper(96))); + + $result = Absolute::from('in', $value); + $this->assertEquals($value * 72, $result->toFloat('pt')); + + $result = Absolute::from('in', $value); + $this->assertEquals($value * 1440 / 2.5, $result->toFloat('eop')); + + $result = Absolute::from('in', $value); + $this->assertEquals($value * 72 * 12700, $result->toFloat('emu'), '', .000000001); + + $result = Absolute::fromPixels(new DpiHelper(96), $value); + $this->assertEquals($value / 96 * 1440, $result->toFloat('twip')); + + $result = Absolute::fromPixels(new DpiHelper(96), $value); + $this->assertEquals($value / 96 * 2.54, $result->toFloat('cm')); + + $result = Absolute::fromPixels(new DpiHelper(96), $value); + $this->assertEquals($value / 96 * 72, $result->toFloat('pt')); + + $result = Absolute::fromPixels(new DpiHelper(96), $value); + $this->assertEquals($value / 96 * 1440 / 2.5, $result->toFloat('eop')); + + $result = Absolute::fromPixels(new DpiHelper(96), $value); + $this->assertEquals($value / 96 * 72 * 12700, $result->toFloat('emu')); + + $result = Absolute::from('pt', $value); + $this->assertEquals($value * 20, $result->toFloat('twip')); + + $result = Absolute::from('pt', $value); + $this->assertEquals($value * 0.035277778, $result->toFloat('cm'), '', 0.00001); + + $result = Absolute::from('pt', $value); + $this->assertEquals(round($value / 72 * 96), $result->toPixels(new DpiHelper(96))); + + $result = Absolute::from('pt', $value); + $this->assertEquals($value * 20 / 2.5, $result->toFloat('eop')); + + $result = Absolute::from('pt', $value); + $this->assertEquals($value * 12700, $result->toFloat('emu'), '', 0.00000000001); + + $result = Absolute::from('eop', $value); + $this->assertEquals(round($value * 2.5 / 1440 * 96), $result->toPixels(new DpiHelper(96))); + + $result = Absolute::from('pc', $value); + $this->assertEquals($value, $result->toFloat('pc'), '', 0.00001); + } + } + + public function testTwips() + { + $this->assertEquals( + Absolute::from('twip', 5), + new Absolute(5), + 'Constructor should use twips' + ); + } + + public function testFromAbsolute() + { + $original = Absolute::from('twip', 5); + $new = Absolute::fromMixed('twip', $original); + $this->assertNotSame($original, $new, 'Lengths should be cloned to avoid accidental manipulation'); + $this->assertEquals($original->toInt('twip'), $new->toInt('twip')); + $this->assertEquals($original->toFloat('twip'), $new->toFloat('twip')); + } + + public function testFromNumeric() + { + $numbers = array( + '0', + '5143', + '5143.03', + ); + foreach ($numbers as $number) { + $length = Absolute::fromMixed('twip', $number); + $this->assertEquals((int) $number, $length->toInt('twip')); + $this->assertEquals((float) $number, $length->toFloat('twip')); + } + } + + public function testFromFloat() + { + $numbers = array( + 0, + 5143, + 5143.03, + ); + foreach ($numbers as $number) { + $length = Absolute::fromMixed('twip', $number); + $this->assertEquals((int) $number, $length->toInt('twip')); + $this->assertEquals((float) $number, $length->toFloat('twip')); + } + } + + public function testFromNull() + { + $length = Absolute::fromMixed('twip', null); + $this->assertNull($length->toInt('twip')); + $this->assertNull($length->toFloat('twip')); + } + + /** + * @expectedException \PHPUnit\Framework\Error\Warning + * @expectedExceptionMessage Border size `not a number` could not be converted to a float + */ + public function testFromInvalid() + { + Absolute::fromMixed('twip', 'not a number'); + } + + public function testValueFromInvalid() + { + $length = @Absolute::fromMixed('twip', 'not a number'); + $this->assertNull($length->toInt('twip'), 'Invalid values should be converted to null'); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Cannot convert from unit `badunit` + */ + public function testInvalidUnit() + { + Absolute::from('badunit', 5); + } + + public function testToPixels() + { + $dpis = array( + 72, + 96, + 254.3, + 400, + ); + $floats = array( + 0, + -2, + -.5, + .5, + 2, + 4, + 42, + ); + foreach ($dpis as $dpi) { + $dpi = new DpiHelper($dpi); + foreach ($floats as $float) { + $length = Absolute::from('in', $float); + $this->assertEquals(round($float * $dpi->getDpi()), $length->toPixels($dpi)); + } + } + } + + public function testNullToPixels() + { + $length = Absolute::from('in', null); + $this->assertNull($length->toPixels(new DpiHelper(96))); + } +} diff --git a/src/PhpWord/SimpleType/TblWidth.php b/tests/PhpWord/Style/Lengths/AutoTest.php similarity index 60% rename from src/PhpWord/SimpleType/TblWidth.php rename to tests/PhpWord/Style/Lengths/AutoTest.php index 7fd753deb0..0d06f17de5 100644 --- a/src/PhpWord/SimpleType/TblWidth.php +++ b/tests/PhpWord/Style/Lengths/AutoTest.php @@ -1,4 +1,5 @@ assertInstanceOf(Auto::class, new Auto()); + $this->assertInstanceOf(Auto::class, new Auto()); + $this->assertTrue((new Auto())->isSpecified()); + } } diff --git a/tests/PhpWord/Style/Lengths/DpiHelper.php b/tests/PhpWord/Style/Lengths/DpiHelper.php new file mode 100644 index 0000000000..41dd8949aa --- /dev/null +++ b/tests/PhpWord/Style/Lengths/DpiHelper.php @@ -0,0 +1,19 @@ +dpi = $dpi; + } + + public function getDpi(): float + { + return $this->dpi; + } +} diff --git a/tests/PhpWord/Style/Lengths/PercentTest.php b/tests/PhpWord/Style/Lengths/PercentTest.php new file mode 100644 index 0000000000..bee4d6616d --- /dev/null +++ b/tests/PhpWord/Style/Lengths/PercentTest.php @@ -0,0 +1,92 @@ +assertEquals(round($float), $length->toInt()); + $this->assertEquals((float) $float, $length->toFloat()); + } + } + + /** + * Test initialization + */ + public function testFromMixed() + { + $values = array( + '0' => 0, + '2.5' => 0.05, + '50' => 1, + '100' => 2, + '250' => 5, + '0%' => 0, + '2.5%' => 2.5, + '50%' => 50, + '100%' => 100, + '250%' => 250, + ); + foreach ($values as $input => $expected) { + $length = Percent::fromMixed($input); + $this->assertEquals(round($expected), $length->toInt(), sprintf('Value \'%s\' should convert to \'%s\'', $input, round($expected))); + $this->assertEquals($expected, $length->toFloat(), sprintf('Value \'%s\' should convert to \'%s\'', $input, $expected)); + } + } + + public function testFromPercent() + { + $original = new Percent(5); + $new = Percent::fromMixed($original); + $this->assertNotSame($original, $new, 'Lengths should be cloned to avoid accidental manipulation'); + $this->assertEquals($original->toInt(), $new->toInt()); + $this->assertEquals($original->toFloat(), $new->toFloat()); + } + + /** + * @expectedException \PHPUnit\Framework\Error\Warning + * @expectedExceptionMessage Percent length `not a number` could not be converted to a float + */ + public function testInvalidUnit() + { + Percent::fromMixed('not a number'); + } + + public function testValueInvalidUnit() + { + $length = @Percent::fromMixed('not a number'); + $this->assertNull($length->toInt()); + } +} diff --git a/tests/PhpWord/Style/LineNumberingTest.php b/tests/PhpWord/Style/LineNumberingTest.php index 0d3f4e0563..945acd695f 100644 --- a/tests/PhpWord/Style/LineNumberingTest.php +++ b/tests/PhpWord/Style/LineNumberingTest.php @@ -1,4 +1,5 @@ array(1, 2), 'increment' => array(1, 10), - 'distance' => array(null, 10), + 'distance' => array(null, Absolute::from('twip', 10)), 'restart' => array(null, 'continuous'), ); foreach ($properties as $property => $value) { @@ -41,11 +44,20 @@ public function testGetSetProperties() $get = "get{$property}"; $set = "set{$property}"; - $this->assertEquals($default, $object->$get()); // Default value + $result = $object->$get(); + if ($expected instanceof Absolute) { + $result = $result->toInt('twip'); + } + $this->assertEquals($default, $result); // Default value $object->$set($expected); - $this->assertEquals($expected, $object->$get()); // New value + $result = $object->$get(); + if ($expected instanceof Absolute) { + $expected = $expected->toInt('twip'); + $result = $result->toInt('twip'); + } + $this->assertEquals($expected, $result); // New value } } } diff --git a/tests/PhpWord/Style/LineTest.php b/tests/PhpWord/Style/LineTest.php index fba09f708d..c80c023d5c 100644 --- a/tests/PhpWord/Style/LineTest.php +++ b/tests/PhpWord/Style/LineTest.php @@ -1,4 +1,5 @@ \PhpOffice\PhpWord\Style\Line::ARROW_STYLE_BLOCK, 'endArrow' => \PhpOffice\PhpWord\Style\Line::ARROW_STYLE_OVAL, 'dash' => \PhpOffice\PhpWord\Style\Line::DASH_STYLE_LONG_DASH_DOT_DOT, - 'weight' => 10, + 'weight' => Absolute::from('pt', 10), 'color' => 'red', ); foreach ($properties as $key => $value) { $set = "set{$key}"; $get = "get{$key}"; $object->$set($value); - $this->assertEquals($value, $object->$get()); + $result = $object->$get(); + if ($value instanceof Absolute) { + $value = $value->toInt('pt'); + $result = $result->toInt('pt'); + } + $this->assertEquals($value, $result); } } @@ -60,13 +68,18 @@ public function testSetStyleValue() 'beginArrow' => \PhpOffice\PhpWord\Style\Line::ARROW_STYLE_BLOCK, 'endArrow' => \PhpOffice\PhpWord\Style\Line::ARROW_STYLE_OVAL, 'dash' => \PhpOffice\PhpWord\Style\Line::DASH_STYLE_LONG_DASH_DOT_DOT, - 'weight' => 10, + 'weight' => Absolute::from('pt', 10), 'color' => 'red', ); foreach ($properties as $key => $value) { $get = "get{$key}"; $object->setStyleValue("{$key}", $value); - $this->assertEquals($value, $object->$get()); + $result = $object->$get(); + if ($value instanceof Absolute) { + $value = $value->toInt('pt'); + $result = $result->toInt('pt'); + } + $this->assertEquals($value, $result); } } @@ -97,10 +110,10 @@ public function testSetGetConnectorType() */ public function testSetGetWeight() { - $expected = 10; + $expected = Absolute::from('pt', 10); $object = new Line(); $object->setWeight($expected); - $this->assertEquals($expected, $object->getWeight()); + $this->assertEquals($expected->toInt('pt'), $object->getWeight()->toInt('pt')); } /** diff --git a/tests/PhpWord/Style/ListItemTest.php b/tests/PhpWord/Style/ListItemTest.php index 71598e8030..1fd9fc8bc1 100644 --- a/tests/PhpWord/Style/ListItemTest.php +++ b/tests/PhpWord/Style/ListItemTest.php @@ -1,4 +1,5 @@ 'space', 'text' => '%1.', 'alignment' => Jc::START, - 'left' => 360, - 'hanging' => 360, - 'tabPos' => 360, + 'left' => Absolute::from('twip', 360), + 'hanging' => Absolute::from('twip', 360), + 'tabPos' => Absolute::from('twip', 360), 'font' => 'Arial', 'hint' => 'default', ); @@ -55,4 +57,70 @@ public function testSetGetNormal() $this->assertEquals($value, $object->$get()); } } + + /** + * @covers \PhpOffice\PhpWord\Style\NumberingLevel::getLeft + */ + public function testGetLeft() + { + $level = new NumberingLevel(); + $this->assertEquals(new Absolute(null), $level->getLeft()); + } + + /** + * @covers \PhpOffice\PhpWord\Style\NumberingLevel::setLeft + * @depends testGetLeft + */ + public function testSetLeft() + { + $level = new NumberingLevel(); + $this->assertEquals(new Absolute(null), $level->getLeft()); + $level->setLeft(Absolute::from('pt', 5)); + $this->assertNotEquals(new Absolute(null), $level->getLeft()); + $this->assertEquals(Absolute::from('pt', 5), $level->getLeft()); + } + + /** + * @covers \PhpOffice\PhpWord\Style\NumberingLevel::getHanging + */ + public function testGetHanging() + { + $level = new NumberingLevel(); + $this->assertEquals(new Absolute(null), $level->getHanging()); + } + + /** + * @covers \PhpOffice\PhpWord\Style\NumberingLevel::setHanging + * @depends testGetHanging + */ + public function testSetHanging() + { + $level = new NumberingLevel(); + $this->assertEquals(new Absolute(null), $level->getHanging()); + $level->setHanging(Absolute::from('pt', 5)); + $this->assertNotEquals(new Absolute(null), $level->getHanging()); + $this->assertEquals(Absolute::from('pt', 5), $level->getHanging()); + } + + /** + * @covers \PhpOffice\PhpWord\Style\NumberingLevel::getTabPos + */ + public function testGetTabPos() + { + $level = new NumberingLevel(); + $this->assertEquals(new Absolute(null), $level->getTabPos()); + } + + /** + * @covers \PhpOffice\PhpWord\Style\NumberingLevel::setTabPos + * @depends testGetTabPos + */ + public function testSetTabPos() + { + $level = new NumberingLevel(); + $this->assertEquals(new Absolute(null), $level->getTabPos()); + $level->setTabPos(Absolute::from('pt', 5)); + $this->assertNotEquals(new Absolute(null), $level->getTabPos()); + $this->assertEquals(Absolute::from('pt', 5), $level->getTabPos()); + } } diff --git a/tests/PhpWord/Style/NumberingTest.php b/tests/PhpWord/Style/NumberingTest.php index 2090f9f6fc..0fc4f3b795 100644 --- a/tests/PhpWord/Style/NumberingTest.php +++ b/tests/PhpWord/Style/NumberingTest.php @@ -1,4 +1,5 @@ assertEquals(new Absolute(null), $outline->getWeight()); + } + + /** + * covers PhpOffice\PhpWord\Style\Outline::setWeight + * @depends testGetWeight + */ + public function testSetWeight() + { + $outline = new Outline(); + $this->assertEquals(new Absolute(null), $outline->getWeight()); + $outline->setWeight(Absolute::from('cm', 1)); + $this->assertEquals(Absolute::from('cm', 1), $outline->getWeight()); + } + + /** + * covers PhpOffice\PhpWord\Style\Outline::getColor + */ + public function testGetColor() + { + $outline = new Outline(); + $this->assertEquals(new Hex(null), $outline->getColor()); + } + + /** + * covers PhpOffice\PhpWord\Style\Outline::setColor + * @depends testGetColor + */ + public function testSetColor() + { + $outline = new Outline(); + $this->assertEquals(new Hex(null), $outline->getColor()); + $outline->setColor(new Hex('123456')); + $this->assertNotEquals(new Hex(null), $outline->getColor()); + $this->assertEquals(new Hex('123456'), $outline->getColor()); + } +} diff --git a/tests/PhpWord/Style/PaperTest.php b/tests/PhpWord/Style/PaperTest.php index f8f0070180..9a6518a869 100644 --- a/tests/PhpWord/Style/PaperTest.php +++ b/tests/PhpWord/Style/PaperTest.php @@ -1,4 +1,5 @@ assertEquals('B5', $object->getSize()); - $this->assertEquals(9977.9527559055, $object->getWidth(), '', 0.000000001); - $this->assertEquals(14173.228346457, $object->getHeight(), '', 0.000000001); + $this->assertEquals(9977.9527559055, $object->getWidth()->toFloat('twip'), '', 0.000000001); + $this->assertEquals(14173.228346457, $object->getHeight()->toFloat('twip'), '', 0.000000001); } /** @@ -65,7 +66,7 @@ public function testFolioSize() $object->setSize('Folio'); $this->assertEquals('Folio', $object->getSize()); - $this->assertEquals(12240, $object->getWidth(), '', 0.1); - $this->assertEquals(18720, $object->getHeight(), '', 0.1); + $this->assertEquals(12240, $object->getWidth()->toInt('twip'), '', 0.1); + $this->assertEquals(18720, $object->getHeight()->toInt('twip'), '', 0.1); } } diff --git a/tests/PhpWord/Style/ParagraphTest.php b/tests/PhpWord/Style/ParagraphTest.php index 4fa0ef5a70..14bf7fff81 100644 --- a/tests/PhpWord/Style/ParagraphTest.php +++ b/tests/PhpWord/Style/ParagraphTest.php @@ -1,4 +1,5 @@ 240, - 'spaceBefore' => 240, - 'indent' => 1, - 'hanging' => 1, - 'spacing' => 120, + 'spaceAfter' => Absolute::from('twip', 240), + 'spaceBefore' => Absolute::from('twip', 240), + 'indent' => Absolute::from('twip', 1), + 'hanging' => Absolute::from('twip', 1), + 'spacing' => Absolute::from('twip', 120), 'spacingLineRule' => LineSpacingRule::AT_LEAST, 'basedOn' => 'Normal', 'next' => 'Normal', @@ -88,11 +91,13 @@ public function testSetStyleValueNormal() ); foreach ($attributes as $key => $value) { $get = $this->findGetter($key, $value, $object); - $object->setStyleValue("$key", $value); - if ('indent' == $key || 'hanging' == $key) { - $value = $value * 720; + $object->setStyleValue($key, $value); + $result = $object->$get(); + if ($value instanceof Absolute) { + $result = $result->toInt('twip'); + $value = $value->toInt('twip'); } - $this->assertEquals($value, $object->$get()); + $this->assertEquals($value, $result); } } @@ -119,7 +124,11 @@ public function testGetNullStyleValue() $attributes = array('spacing', 'indent', 'hanging', 'spaceBefore', 'spaceAfter', 'textAlignment'); foreach ($attributes as $key) { $get = $this->findGetter($key, null, $object); - $this->assertNull($object->$get()); + $result = $object->$get(); + if ($result instanceof Absolute) { + $result = $result->toInt('twip'); + } + $this->assertNull($result); } } @@ -129,7 +138,7 @@ public function testGetNullStyleValue() public function testTabs() { $object = new Paragraph(); - $object->setTabs(array(new Tab('left', 1550), new Tab('right', 5300))); + $object->setTabs(array(new Tab('left', Absolute::from('twip', 1550)), new Tab('right', Absolute::from('twip', 5300)))); $this->assertCount(2, $object->getTabs()); } @@ -142,7 +151,7 @@ public function testLineHeight() $section = $phpWord->addSection(); // Test style array - $text = $section->addText('This is a test', array(), array('line-height' => 2.0)); + $text = $section->addText('This is a test', array(), array('line-height' => new Percent(200))); $doc = TestHelperDOCX::getDocument($phpWord); $element = $doc->getElement('/w:document/w:body/w:p/w:pPr/w:spacing'); @@ -154,7 +163,7 @@ public function testLineHeight() $this->assertEquals('auto', $lineRule); // Test setter - $text->getParagraphStyle()->setLineHeight(3.0); + $text->getParagraphStyle()->setLineHeight(new Percent(300)); $doc = TestHelperDOCX::getDocument($phpWord); $element = $doc->getElement('/w:document/w:body/w:p/w:pPr/w:spacing'); @@ -171,14 +180,15 @@ public function testLineHeight() public function testLineHeightValidation() { $object = new Paragraph(); - $object->setLineHeight('12.5pt'); - $this->assertEquals(12.5, $object->getLineHeight()); + $object->setLineHeight(new Percent(12.5)); + $this->assertEquals(12.5, $object->getLineHeight()->toFloat()); + $this->assertEquals(13, $object->getLineHeight()->toInt()); } /** * Test line height exception by using nonnumeric value * - * @expectedException \PhpOffice\PhpWord\Exception\InvalidStyleException + * @expectedException \TypeError */ public function testLineHeightException() { diff --git a/tests/PhpWord/Style/RowTest.php b/tests/PhpWord/Style/RowTest.php index 534815b1b4..2524a678a6 100644 --- a/tests/PhpWord/Style/RowTest.php +++ b/tests/PhpWord/Style/RowTest.php @@ -1,4 +1,5 @@ assertEquals('portrait', $oSettings->getOrientation()); - $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeW(), '', 0.000000001); - $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeH(), '', 0.000000001); + $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeW()->toFloat('twip'), '', .00000001); + $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeH()->toFloat('twip'), '', .00000001); $this->assertEquals('A4', $oSettings->getPaperSize()); $oSettings->setSettingValue('orientation', 'landscape'); $this->assertEquals('landscape', $oSettings->getOrientation()); - $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeW(), '', 0.000000001); - $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeH(), '', 0.000000001); + $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeW()->toFloat('twip'), '', .00000001); + $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeH()->toFloat('twip'), '', .00000001); $iVal = rand(1, 1000); - $oSettings->setSettingValue('borderSize', $iVal); - $this->assertEquals(array($iVal, $iVal, $iVal, $iVal), $oSettings->getBorderSize()); - $this->assertEquals($iVal, $oSettings->getBorderBottomSize()); - $this->assertEquals($iVal, $oSettings->getBorderLeftSize()); - $this->assertEquals($iVal, $oSettings->getBorderRightSize()); - $this->assertEquals($iVal, $oSettings->getBorderTopSize()); - - $oSettings->setSettingValue('borderColor', 'FF00AA'); - $this->assertEquals(array('FF00AA', 'FF00AA', 'FF00AA', 'FF00AA'), $oSettings->getBorderColor()); - $this->assertEquals('FF00AA', $oSettings->getBorderBottomColor()); - $this->assertEquals('FF00AA', $oSettings->getBorderLeftColor()); - $this->assertEquals('FF00AA', $oSettings->getBorderRightColor()); - $this->assertEquals('FF00AA', $oSettings->getBorderTopColor()); - - $iVal = rand(1, 1000); - $oSettings->setSettingValue('headerHeight', $iVal); - $this->assertEquals($iVal, $oSettings->getHeaderHeight()); + $oSettings->setSettingValue('headerHeight', Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oSettings->getHeaderHeight()->toInt('twip')); $oSettings->setSettingValue('lineNumbering', array()); $oSettings->setSettingValue( @@ -69,7 +58,7 @@ public function testSettingValue() array( 'start' => 1, 'increment' => 1, - 'distance' => 240, + 'distance' => Absolute::from('twip', 240), 'restart' => 'newPage', ) ); @@ -88,20 +77,20 @@ public function testMargin() $oSettings = new Section(); $iVal = rand(1, 1000); - $oSettings->setMarginTop($iVal); - $this->assertEquals($iVal, $oSettings->getMarginTop()); + $oSettings->setMarginTop(Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oSettings->getMarginTop()->toInt('twip')); $iVal = rand(1, 1000); - $oSettings->setMarginBottom($iVal); - $this->assertEquals($iVal, $oSettings->getMarginBottom()); + $oSettings->setMarginBottom(Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oSettings->getMarginBottom()->toInt('twip')); $iVal = rand(1, 1000); - $oSettings->setMarginLeft($iVal); - $this->assertEquals($iVal, $oSettings->getMarginLeft()); + $oSettings->setMarginLeft(Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oSettings->getMarginLeft()->toInt('twip')); $iVal = rand(1, 1000); - $oSettings->setMarginRight($iVal); - $this->assertEquals($iVal, $oSettings->getMarginRight()); + $oSettings->setMarginRight(Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oSettings->getMarginRight()->toInt('twip')); } /** @@ -112,10 +101,10 @@ public function testPageWidth() // Section Settings $oSettings = new Section(); - $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeW(), '', 0.000000001); + $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeW()->toFloat('twip'), '', 0.000000001); $iVal = rand(1, 1000); - $oSettings->setSettingValue('pageSizeW', $iVal); - $this->assertEquals($iVal, $oSettings->getPageSizeW()); + $oSettings->setSettingValue('pageSizeW', Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oSettings->getPageSizeW()->toFloat('twip')); } /** @@ -126,10 +115,10 @@ public function testPageHeight() // Section Settings $oSettings = new Section(); - $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeH(), '', 0.000000001); + $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeH()->toFloat('twip'), '', 0.000000001); $iVal = rand(1, 1000); - $oSettings->setSettingValue('pageSizeH', $iVal); - $this->assertEquals($iVal, $oSettings->getPageSizeH()); + $oSettings->setSettingValue('pageSizeH', Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oSettings->getPageSizeH()->toInt('twip')); } /** @@ -142,8 +131,8 @@ public function testOrientationLandscape() $oSettings->setLandscape(); $this->assertEquals('landscape', $oSettings->getOrientation()); - $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeW(), '', 0.000000001); - $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeH(), '', 0.000000001); + $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeW()->toFloat('twip'), '', 0.000000001); + $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeH()->toFloat('twip'), '', 0.000000001); } /** @@ -156,69 +145,47 @@ public function testOrientationPortrait() $oSettings->setPortrait(); $this->assertEquals('portrait', $oSettings->getOrientation()); - $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeW(), '', 0.000000001); - $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeH(), '', 0.000000001); + $this->assertEquals(Section::DEFAULT_WIDTH, $oSettings->getPageSizeW()->toFloat('twip'), '', 0.000000001); + $this->assertEquals(Section::DEFAULT_HEIGHT, $oSettings->getPageSizeH()->toFloat('twip'), '', 0.000000001); } /** - * Set/get border size + * Test border color */ - public function testBorderSize() + public function testBorders() { - // Section Settings - $oSettings = new Section(); - - $iVal = rand(1, 1000); - $oSettings->setBorderSize($iVal); - $this->assertEquals(array($iVal, $iVal, $iVal, $iVal), $oSettings->getBorderSize()); - $this->assertEquals($iVal, $oSettings->getBorderBottomSize()); - $this->assertEquals($iVal, $oSettings->getBorderLeftSize()); - $this->assertEquals($iVal, $oSettings->getBorderRightSize()); - $this->assertEquals($iVal, $oSettings->getBorderTopSize()); - - $iVal = rand(1, 1000); - $oSettings->setBorderBottomSize($iVal); - $this->assertEquals($iVal, $oSettings->getBorderBottomSize()); - - $iVal = rand(1, 1000); - $oSettings->setBorderLeftSize($iVal); - $this->assertEquals($iVal, $oSettings->getBorderLeftSize()); - - $iVal = rand(1, 1000); - $oSettings->setBorderRightSize($iVal); - $this->assertEquals($iVal, $oSettings->getBorderRightSize()); - - $iVal = rand(1, 1000); - $oSettings->setBorderTopSize($iVal); - $this->assertEquals($iVal, $oSettings->getBorderTopSize()); - } - - /** - * Set/get border color - */ - public function testBorderColor() - { - // Section Settings - $oSettings = new Section(); - - $oSettings->setBorderColor('FF00AA'); - $this->assertEquals(array('FF00AA', 'FF00AA', 'FF00AA', 'FF00AA'), $oSettings->getBorderColor()); - $this->assertEquals('FF00AA', $oSettings->getBorderBottomColor()); - $this->assertEquals('FF00AA', $oSettings->getBorderLeftColor()); - $this->assertEquals('FF00AA', $oSettings->getBorderRightColor()); - $this->assertEquals('FF00AA', $oSettings->getBorderTopColor()); - - $oSettings->setBorderBottomColor('BBCCDD'); - $this->assertEquals('BBCCDD', $oSettings->getBorderBottomColor()); - - $oSettings->setBorderLeftColor('CCDDEE'); - $this->assertEquals('CCDDEE', $oSettings->getBorderLeftColor()); - - $oSettings->setBorderRightColor('11EE22'); - $this->assertEquals('11EE22', $oSettings->getBorderRightColor()); - - $oSettings->setBorderTopColor('22FF33'); - $this->assertEquals('22FF33', $oSettings->getBorderTopColor()); + $section = new Section(); + + $this->assertFalse($section->hasBorder()); + $borders = array('top', 'right', 'bottom', 'left'); + $borderSides = array( + array(Absolute::from('pt', rand(1, 20)), new Hex('f93de1'), new BorderStyle('double'), Absolute::from('pt', rand(1, 20)), true), + array(Absolute::from('twip', rand(1, 400)), new Hex('000000'), new BorderStyle('outset'), Absolute::from('twip', rand(1, 400)), false), + array(Absolute::from('eop', rand(1, 160)), new Rgb(255, 0, 100), new BorderStyle('dotted'), Absolute::from('eop', rand(1, 160)), true), + ); + $lastBorderSide = array(new Absolute(0), new Hex(null), new BorderStyle('single'), new Absolute(0), false); + foreach ($borderSides as $key => $borderSide) { + $newBorder = new BorderSide(...$borderSide); + + foreach ($borders as $side) { + $currentBorder = $section->getBorder($side); + $this->assertEquals($lastBorderSide[0], $currentBorder->getSize(), "Size for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[1], $currentBorder->getColor(), "Color for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[2], $currentBorder->getStyle(), "Style for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[3], $currentBorder->getSpace(), "Space for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[4], $currentBorder->getShadow(), "Shadow for border side #$key for side $side should match last border side still"); + + $section->setBorder($side, $newBorder); + $updatedBorder = $section->getBorder($side); + $this->assertEquals($borderSide[0], $updatedBorder->getSize(), "Size for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[1], $updatedBorder->getColor(), "Color for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[2], $updatedBorder->getStyle(), "Style for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[3], $updatedBorder->getSpace(), "Space for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[4], $updatedBorder->getShadow(), "Shadow for border side #$key for side $side should match new border"); + } + + $lastBorderSide = $borderSide; + } } /** @@ -246,14 +213,22 @@ public function testHeader() { $oSettings = new Section(); - $this->assertEquals(720, $oSettings->getHeaderHeight()); + $this->assertEquals(720, $oSettings->getHeaderHeight()->toInt('twip')); $iVal = rand(1, 1000); - $oSettings->setHeaderHeight($iVal); - $this->assertEquals($iVal, $oSettings->getHeaderHeight()); + $oSettings->setHeaderHeight(Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oSettings->getHeaderHeight()->toInt('twip')); + } + /** + * Set/get header height + * @expectedException \TypeError + * @expectedExceptionMessageRegExp /^(Argument 1 passed to PhpOffice\\PhpWord\\Style\\Section::setHeaderHeight\(\) must be an instance of PhpOffice\\PhpWord\\Style\\Lengths\\Absolute, none given|Too few arguments to function PhpOffice\\PhpWord\\Style\\Section::setHeaderHeight\(\), 0 passed)/ + */ + public function testHeaderMissing() + { + $oSettings = new Section(); $oSettings->setHeaderHeight(); - $this->assertEquals(720, $oSettings->getHeaderHeight()); } /** @@ -264,14 +239,22 @@ public function testFooter() // Section Settings $oSettings = new Section(); - $this->assertEquals(720, $oSettings->getFooterHeight()); + $this->assertEquals(720, $oSettings->getFooterHeight()->toInt('twip')); $iVal = rand(1, 1000); - $oSettings->setFooterHeight($iVal); - $this->assertEquals($iVal, $oSettings->getFooterHeight()); + $oSettings->setFooterHeight(Absolute::from('twip', $iVal)); + $this->assertEquals($iVal, $oSettings->getFooterHeight()->toInt('twip')); + } + /** + * Set/get footer height + * @expectedException \TypeError + * @expectedExceptionMessageRegExp /^(Argument 1 passed to PhpOffice\\PhpWord\\Style\\Section::setFooterHeight\(\) must be an instance of PhpOffice\\PhpWord\\Style\\Lengths\\Absolute, none given|Too few arguments to function PhpOffice\\PhpWord\\Style\\Section::setFooterHeight\(\), 0 passed)/ + */ + public function testFooterEmpty() + { + $oSettings = new Section(); $oSettings->setFooterHeight(); - $this->assertEquals(720, $oSettings->getFooterHeight()); } /** @@ -304,14 +287,23 @@ public function testColumnsSpace() $oSettings = new Section(); // Default - $this->assertEquals(720, $oSettings->getColsSpace()); + $this->assertEquals(720, $oSettings->getColsSpace()->toInt('twip')); $iVal = rand(1, 1000); - $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Section', $oSettings->setColsSpace($iVal)); - $this->assertEquals($iVal, $oSettings->getColsSpace()); + $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Section', $oSettings->setColsSpace(Absolute::from('twip', $iVal))); + $this->assertEquals($iVal, $oSettings->getColsSpace()->toInt('twip')); + } + /** + * Set/get column spacing + * PHP 7.1+ + * @expectedException \TypeError + * @expectedExceptionMessageRegExp /^(Argument 1 passed to PhpOffice\\PhpWord\\Style\\Section::setColsSpace\(\) must be an instance of PhpOffice\\PhpWord\\Style\\Lengths\\Absolute, none given|Too few arguments to function PhpOffice\\PhpWord\\Style\\Section::setColsSpace\(\), 0 passed)/ + */ + public function testColumnsSpaceEmpty() + { + $oSettings = new Section(); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Section', $oSettings->setColsSpace()); - $this->assertEquals(720, $oSettings->getColsSpace()); } /** diff --git a/tests/PhpWord/Style/ShadingTest.php b/tests/PhpWord/Style/ShadingTest.php index 7aba03a122..b37f771e88 100644 --- a/tests/PhpWord/Style/ShadingTest.php +++ b/tests/PhpWord/Style/ShadingTest.php @@ -1,4 +1,5 @@ array('clear', 'solid'), - 'color' => array(null, 'FF0000'), - 'fill' => array(null, 'FF0000'), + 'color' => array(null, new Hex('FF0000')), + 'fill' => array(null, new Hex('FF0000')), ); foreach ($properties as $property => $value) { list($default, $expected) = $value; - $get = "get{$property}"; - $set = "set{$property}"; - $this->assertEquals($default, $object->$get()); // Default value + $this->assertEquals($default, $this->get($object, $property)); // Default value + + $this->set($object, $property, $expected); + + if ($expected instanceof BasicColor) { + $expected = $expected->toHex(); + } + $this->assertEquals($expected, $this->get($object, $property)); // New value + } + } - $object->$set($expected); + private function get(Shading $object, string $property) + { + $get = "get{$property}"; - $this->assertEquals($expected, $object->$get()); // New value + $result = $object->$get(); + if ($result instanceof BasicColor) { + $result = $result->toHex(); } + + return $result; + } + + private function set(Shading $object, string $property, $expected) + { + $set = "set$property"; + + return $object->$set($expected); } } diff --git a/tests/PhpWord/Style/ShapeTest.php b/tests/PhpWord/Style/ShapeTest.php new file mode 100644 index 0000000000..72b1e56d6b --- /dev/null +++ b/tests/PhpWord/Style/ShapeTest.php @@ -0,0 +1,75 @@ +assertEquals(new Percent(0), $object->getRoundness()); + } + + /** + * @covers \PhpOffice\PhpWord\Style\Shape::setRoundness + * @depends testGetRoundness + */ + public function testSetRoundness() + { + $shape = new Shape(); + $this->assertEquals(new Percent(0), $shape->getRoundness()); + $shape->setRoundness(new Percent(50)); + $this->assertNotEquals(new Percent(0), $shape->getRoundness()); + $this->assertEquals(new Percent(50), $shape->getRoundness()); + } + + /** + * @covers \PhpOffice\PhpWord\Style\Shape::setRoundness + * @depends testSetRoundness + * @expectedException \PhpOffice\PhpWord\Exception\Exception + * @expectedExceptionMessage Provided roundness -1.000000% must be no less than 0% + */ + public function testSetRoundnessNegative() + { + $shape = new Shape(); + $shape->setRoundness(new Percent(-1)); + } + + /** + * @covers \PhpOffice\PhpWord\Style\Shape::setRoundness + * @depends testSetRoundness + * @expectedException \PhpOffice\PhpWord\Exception\Exception + * @expectedExceptionMessage Provided roundness 101.000000% must be no greater than 100% + */ + public function testSetRoundnessTooLarge() + { + $shape = new Shape(); + $shape->setRoundness(new Percent(101)); + } +} diff --git a/tests/PhpWord/Style/SpacingTest.php b/tests/PhpWord/Style/SpacingTest.php index f7402edd12..deca6ac5ff 100644 --- a/tests/PhpWord/Style/SpacingTest.php +++ b/tests/PhpWord/Style/SpacingTest.php @@ -1,4 +1,5 @@ array(null, 10), - 'after' => array(null, 10), - 'line' => array(null, 10), + 'before' => array(null, Absolute::from('twip', 10)), + 'after' => array(null, Absolute::from('twip', 10)), + 'line' => array(null, Absolute::from('pt', 10)), 'lineRule' => array('auto', 'exact'), ); foreach ($properties as $property => $value) { @@ -41,11 +44,20 @@ public function testGetSetProperties() $get = "get{$property}"; $set = "set{$property}"; - $this->assertEquals($default, $object->$get()); // Default value + $result = $object->$get(); + if ($expected instanceof Absolute) { + $result = $result->toInt('twip'); + } + $this->assertEquals($default, $result); // Default value $object->$set($expected); - $this->assertEquals($expected, $object->$get()); // New value + $result = $object->$get(); + if ($expected instanceof Absolute) { + $expected = $expected->toInt('twip'); + $result = $result->toInt('twip'); + } + $this->assertEquals($expected, $result); // New value } } } diff --git a/tests/PhpWord/Style/TOCTest.php b/tests/PhpWord/Style/TOCTest.php index 445c5e4315..a00d262ab4 100644 --- a/tests/PhpWord/Style/TOCTest.php +++ b/tests/PhpWord/Style/TOCTest.php @@ -1,4 +1,5 @@ array(TOC::TAB_LEADER_DOT, TOC::TAB_LEADER_UNDERSCORE), - 'tabPos' => array(9062, 10), - 'indent' => array(200, 10), + 'tabPos' => array(9062, Absolute::from('twip', 10)), + 'indent' => array(200, Absolute::from('twip', 10)), ); foreach ($properties as $property => $value) { list($default, $expected) = $value; $get = "get{$property}"; $set = "set{$property}"; - $this->assertEquals($default, $object->$get()); // Default value + $result = $object->$get(); + if ($result instanceof Absolute) { + $result = $result->toInt('twip'); + } + $this->assertEquals($default, $result); // Default value $object->$set($expected); - $this->assertEquals($expected, $object->$get()); // New value + $result = $object->$get(); + if ($expected instanceof Absolute) { + $expected = $expected->toInt('twip'); + $result = $result->toInt('twip'); + } + + $this->assertEquals($expected, $result); // New value } } } diff --git a/tests/PhpWord/Style/TabTest.php b/tests/PhpWord/Style/TabTest.php index 8d8d3f6cf0..36c003abc5 100644 --- a/tests/PhpWord/Style/TabTest.php +++ b/tests/PhpWord/Style/TabTest.php @@ -1,4 +1,5 @@ array(Tab::TAB_STOP_CLEAR, Tab::TAB_STOP_RIGHT), 'leader' => array(Tab::TAB_LEADER_NONE, Tab::TAB_LEADER_DOT), - 'position' => array(0, 10), + 'position' => array(0, Absolute::from('twip', 10)), ); foreach ($properties as $property => $value) { list($default, $expected) = $value; $get = "get{$property}"; $set = "set{$property}"; - $this->assertEquals($default, $object->$get()); // Default value + $result = $object->$get(); + if ($expected instanceof Absolute) { + $result = $result->toInt('twip'); + } + $this->assertEquals($default, $result); // Default value $object->$set($expected); - $this->assertEquals($expected, $object->$get()); // New value + $result = $object->$get(); + if ($expected instanceof Absolute) { + $expected = $expected->toInt('twip'); + $result = $result->toInt('twip'); + } + $this->assertEquals($expected, $result); // New value } } } diff --git a/tests/PhpWord/Style/TablePositionTest.php b/tests/PhpWord/Style/TablePositionTest.php index 1243c3d595..098b4e557f 100644 --- a/tests/PhpWord/Style/TablePositionTest.php +++ b/tests/PhpWord/Style/TablePositionTest.php @@ -1,4 +1,5 @@ TablePosition::VANCHOR_PAGE, 'bottomFromText' => 20); + $styleTable = array('vertAnchor' => TablePosition::VANCHOR_PAGE, 'bottomFromText' => Absolute::from('twip', 20)); $object = new TablePosition($styleTable); $this->assertEquals(TablePosition::VANCHOR_PAGE, $object->getVertAnchor()); - $this->assertEquals(20, $object->getBottomFromText()); + $this->assertEquals(20, $object->getBottomFromText()->toInt('twip')); } /** @@ -44,22 +47,53 @@ public function testSetGetNormal() $object = new TablePosition(); $attributes = array( - 'leftFromText' => 4, - 'rightFromText' => 4, - 'topFromText' => 4, - 'bottomFromText' => 4, + 'leftFromText' => Absolute::from('twip', 4), + 'rightFromText' => Absolute::from('twip', 4), + 'topFromText' => Absolute::from('twip', 4), + 'bottomFromText' => Absolute::from('twip', 4), 'vertAnchor' => TablePosition::VANCHOR_PAGE, 'horzAnchor' => TablePosition::HANCHOR_TEXT, 'tblpXSpec' => TablePosition::XALIGN_CENTER, - 'tblpX' => 5, + 'tblpX' => Absolute::from('twip', 5), 'tblpYSpec' => TablePosition::YALIGN_OUTSIDE, - 'tblpY' => 6, + 'tblpY' => Absolute::from('twip', 6), ); foreach ($attributes as $key => $value) { $set = "set{$key}"; $get = "get{$key}"; $object->$set($value); - $this->assertEquals($value, $object->$get()); + $result = $object->$get(); + if ($value instanceof Absolute) { + $value = $value->toInt('twip'); + $result = $result->toInt('twip'); + } + $this->assertEquals($value, $result, "Read value for attribute $key should be the same as the written value"); + } + } + + /** + * @covers \PhpOffice\PhpWord\Style\TablePosition + */ + public function testSetGetAbsolute() + { + $attributes = array( + 'TopFromText', + 'BottomFromText', + 'LeftFromText', + 'RightFromText', + 'TblpX', + 'TblpY', + ); + + $level = new TablePosition(); + foreach ($attributes as $attribute) { + $get = "get$attribute"; + $set = "set$attribute"; + + $this->assertEquals(new Absolute(null), $level->$get()); + $level->$set(Absolute::from('pt', 5)); + $this->assertNotEquals(new Absolute(null), $level->$get()); + $this->assertEquals(Absolute::from('pt', 5), $level->$get()); } } } diff --git a/tests/PhpWord/Style/TableTest.php b/tests/PhpWord/Style/TableTest.php index 91fc35509c..0a56d80006 100644 --- a/tests/PhpWord/Style/TableTest.php +++ b/tests/PhpWord/Style/TableTest.php @@ -1,4 +1,5 @@ 'FF0000'); - $styleFirstRow = array('borderBottomSize' => 3); + $styleTable = array('bgColor' => new Hex('FF0000')); + $styleFirstRow = array('bordersFromArray' => array( + 'top' => new BorderSide(Absolute::from('eop', 3)), + )); $object = new Table($styleTable, $styleFirstRow); - $this->assertEquals('FF0000', $object->getBgColor()); + $this->assertEquals('FF0000', $object->getBgColor()->toHex()); $firstRow = $object->getFirstRow(); $this->assertInstanceOf('PhpOffice\\PhpWord\\Style\\Table', $firstRow); - $this->assertEquals(3, $firstRow->getBorderBottomSize()); + $this->assertEquals(3, $firstRow->getBorder('top')->getSize()->toInt('eop')); + $this->assertEquals(0, $firstRow->getBorder('bottom')->getSize()->toInt('eop')); } /** @@ -55,10 +63,9 @@ public function testDefaultValues() { $object = new Table(); - $this->assertNull($object->getBgColor()); + $this->assertNull($object->getBgColor()->toHex()); $this->assertEquals(Table::LAYOUT_AUTO, $object->getLayout()); - $this->assertEquals(TblWidth::AUTO, $object->getUnit()); - $this->assertNull($object->getIndent()); + $this->assertInstanceOf(Auto::class, $object->getIndent()); } /** @@ -69,79 +76,119 @@ public function testSetGetNormal() $object = new Table(); $attributes = array( - 'bgColor' => 'FF0000', - 'borderTopSize' => 4, - 'borderTopColor' => 'FF0000', - 'borderLeftSize' => 4, - 'borderLeftColor' => 'FF0000', - 'borderRightSize' => 4, - 'borderRightColor' => 'FF0000', - 'borderBottomSize' => 4, - 'borderBottomColor' => 'FF0000', - 'borderInsideHSize' => 4, - 'borderInsideHColor' => 'FF0000', - 'borderInsideVSize' => 4, - 'borderInsideVColor' => 'FF0000', - 'cellMarginTop' => 240, - 'cellMarginLeft' => 240, - 'cellMarginRight' => 240, - 'cellMarginBottom' => 240, + 'bgColor' => new Hex('FF0000'), + 'cellMarginTop' => Absolute::from('eop', 240), + 'cellMarginLeft' => Absolute::from('eop', 240), + 'cellMarginRight' => Absolute::from('eop', 240), + 'cellMarginBottom' => Absolute::from('eop', 240), 'alignment' => JcTable::CENTER, - 'width' => 100, - 'unit' => 'pct', + 'width' => new Percent(100), 'layout' => Table::LAYOUT_FIXED, ); foreach ($attributes as $key => $value) { $set = "set{$key}"; $get = "get{$key}"; $object->$set($value); - $this->assertEquals($value, $object->$get()); + $result = $object->$get(); + if ($result instanceof Absolute) { + $result = $result->toInt('eop'); + $value = $value->toInt('eop'); + } elseif ($result instanceof Percent) { + $result = $result->toInt(); + $value = $value->toInt(); + } elseif ($result instanceof BasicColor) { + $result = $result->toHex(); + $value = $value->toHex(); + } + $this->assertEquals($value, $result); } } /** - * Test border color - * - * Set border color and test if each part has the same color - * While looping, push values array to be asserted with getBorderColor + * Test borders */ - public function testBorderColor() + public function testBorders() { - $object = new Table(); - $parts = array('Top', 'Left', 'Right', 'Bottom', 'InsideH', 'InsideV'); + $table = new Table(); - $value = 'FF0000'; - $object->setBorderColor($value); - $values = array(); - foreach ($parts as $part) { - $get = "getBorder{$part}Color"; - $values[] = $value; - $this->assertEquals($value, $object->$get()); + $this->assertFalse($table->hasBorder()); + $borders = array('top', 'bottom', 'start', 'end', 'insideH', 'insideV'); + $borderSides = array( + array(Absolute::from('pt', rand(1, 20)), new Hex('f93de1'), new BorderStyle('double'), Absolute::from('pt', rand(1, 20)), true), + array(Absolute::from('twip', rand(1, 400)), new Hex('000000'), new BorderStyle('outset'), Absolute::from('twip', rand(1, 400)), false), + array(Absolute::from('eop', rand(1, 160)), new Rgb(255, 0, 100), new BorderStyle('dotted'), Absolute::from('eop', rand(1, 160)), true), + ); + $lastBorderSide = array(new Absolute(0), new Hex(null), new BorderStyle('single'), new Absolute(0), false); + foreach ($borderSides as $key => $borderSide) { + $newBorder = new BorderSide(...$borderSide); + + foreach ($borders as $side) { + $currentBorder = $table->getBorder($side); + $this->assertEquals($lastBorderSide[0], $currentBorder->getSize(), "Size for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[1], $currentBorder->getColor(), "Color for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[2], $currentBorder->getStyle(), "Style for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[3], $currentBorder->getSpace(), "Space for border side #$key for side $side should match last border side still"); + $this->assertEquals($lastBorderSide[4], $currentBorder->getShadow(), "Shadow for border side #$key for side $side should match last border side still"); + + $table->setBorder($side, $newBorder); + $updatedBorder = $table->getBorder($side); + $this->assertEquals($borderSide[0], $updatedBorder->getSize(), "Size for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[1], $updatedBorder->getColor(), "Color for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[2], $updatedBorder->getStyle(), "Style for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[3], $updatedBorder->getSpace(), "Space for border side #$key for side $side should match new border"); + $this->assertEquals($borderSide[4], $updatedBorder->getShadow(), "Shadow for border side #$key for side $side should match new border"); + } + + $lastBorderSide = $borderSide; } - $this->assertEquals($values, $object->getBorderColor()); } /** - * Test border size - * - * Set border size and test if each part has the same size - * While looping, push values array to be asserted with getBorderSize - * Value is in eights of a point, i.e. 4 / 8 = .5pt + * Test invalid border + * @depends testBorders + * @expectedException \PhpOffice\PhpWord\Exception\Exception + * @expectedExceptionMessage Invalid side `badside` provided */ - public function testBorderSize() + public function testGetBorderInvalid() { - $object = new Table(); - $parts = array('Top', 'Left', 'Right', 'Bottom', 'InsideH', 'InsideV'); + $table = new Table(); + $table->getBorder('badside'); + } - $value = 4; - $object->setBorderSize($value); - $values = array(); - foreach ($parts as $part) { - $get = "getBorder{$part}Size"; - $values[] = $value; - $this->assertEquals($value, $object->$get()); - } - $this->assertEquals($values, $object->getBorderSize()); + /** + * Test invalid border + * @depends testBorders + * @expectedException \PhpOffice\PhpWord\Exception\Exception + * @expectedExceptionMessage Size must be specified + */ + public function testSetBorderNullSize() + { + $table = new Table(); + $table->getBorder('top')->setSize(new Absolute(null)); + } + + /** + * Test invalid border + * @depends testBorders + * @expectedException \PhpOffice\PhpWord\Exception\Exception + * @expectedExceptionMessage Space must be specified + */ + public function testSetBorderNullSpace() + { + $table = new Table(); + $table->getBorder('top')->setSpace(new Absolute(null)); + } + + /** + * Test invalid border + * @depends testBorders + * @expectedException \PhpOffice\PhpWord\Exception\Exception + * @expectedExceptionMessage Invalid side `badside` provided + */ + public function testSetBorderInvalid() + { + $table = new Table(); + $table->setBorder('badside', new BorderSide()); } /** @@ -157,14 +204,16 @@ public function testCellMargin() $parts = array('Top', 'Left', 'Right', 'Bottom'); $value = 240; - $object->setCellMargin($value); + $object->setCellMargin(Absolute::from('twip', $value)); $values = array(); foreach ($parts as $part) { $get = "getCellMargin{$part}"; $values[] = $value; - $this->assertEquals($value, $object->$get()); + $this->assertEquals($value, $object->$get()->toInt('twip')); } - $this->assertEquals($values, $object->getCellMargin()); + $this->assertEquals($values, array_map(function ($value) { + return $value->toInt('twip'); + }, $object->getCellMargin())); $this->assertTrue($object->hasMargin()); } @@ -174,16 +223,22 @@ public function testCellMargin() public function testSetStyleValue() { $object = new Table(); - $object->setStyleValue('borderSize', 120); - $object->setStyleValue('cellMargin', 240); - $object->setStyleValue('borderColor', '999999'); - - $this->assertEquals(array(120, 120, 120, 120, 120, 120), $object->getBorderSize()); - $this->assertEquals(array(240, 240, 240, 240), $object->getCellMargin()); - $this->assertEquals( - array('999999', '999999', '999999', '999999', '999999', '999999'), - $object->getBorderColor() - ); + $object->setStyleValue('bordersFromArray', array( + 'top' => new BorderSide(Absolute::from('twip', 120)), + 'bottom' => new BorderSide(Absolute::from('twip', 120)), + 'start' => new BorderSide(Absolute::from('twip', 120)), + 'end' => new BorderSide(Absolute::from('twip', 120)), + 'insideH' => new BorderSide(Absolute::from('twip', 120)), + 'insideV' => new BorderSide(Absolute::from('twip', 120)), + )); + $object->setStyleValue('cellMargin', Absolute::from('twip', 240)); + + $this->assertEquals(array('top' => 120, 'bottom' => 120, 'insideH' => 120, 'insideV' => 120, 'start' => 120, 'end' => 120), array_map(function ($value) { + return $value->getSize()->toInt('twip'); + }, $object->getBorders())); + $this->assertEquals(array(240, 240, 240, 240), array_map(function ($value) { + return $value->toInt('twip'); + }, $object->getCellMargin())); } /** @@ -192,10 +247,10 @@ public function testSetStyleValue() public function testTableCellSpacing() { $object = new Table(); - $this->assertNull($object->getCellSpacing()); + $this->assertNull($object->getCellSpacing()->toInt('twip')); - $object = new Table(array('cellSpacing' => 20)); - $this->assertEquals(20, $object->getCellSpacing()); + $object = new Table(array('cellSpacing' => Absolute::from('twip', 20))); + $this->assertEquals(20, $object->getCellSpacing()->toInt('twip')); } /** @@ -213,7 +268,7 @@ public function testTablePosition() public function testIndent() { - $indent = new TblWidthComplexType(100, TblWidth::TWIP); + $indent = Absolute::from('twip', 100); $table = new Table(array('indent' => $indent)); diff --git a/tests/PhpWord/Style/TextBoxTest.php b/tests/PhpWord/Style/TextBoxTest.php index 803189cd19..26bf205078 100644 --- a/tests/PhpWord/Style/TextBoxTest.php +++ b/tests/PhpWord/Style/TextBoxTest.php @@ -1,4 +1,5 @@ 200, - 'height' => 200, + // 'width' => Absolute::from("twip", 200), + // 'height' => Absolute::from("twip", 200), 'alignment' => Jc::START, - 'marginTop' => 240, - 'marginLeft' => 240, + 'marginTop' => Absolute::from('twip', 240), + 'marginLeft' => Absolute::from('twip', 240), 'wrappingStyle' => 'inline', 'positioning' => 'absolute', 'posHorizontal' => 'center', 'posVertical' => 'top', 'posHorizontalRel' => 'margin', 'posVerticalRel' => 'page', - 'innerMarginTop' => '5', - 'innerMarginRight' => '5', - 'innerMarginBottom' => '5', - 'innerMarginLeft' => '5', - 'borderSize' => '2', - 'borderColor' => 'red', + 'innerMarginTop' => Absolute::from('twip', 5), + 'innerMarginRight' => Absolute::from('twip', 5), + 'innerMarginBottom' => Absolute::from('twip', 5), + 'innerMarginLeft' => Absolute::from('twip', 5), + 'borderSize' => Absolute::from('twip', 2), + 'borderColor' => new HighlightColor('red'), ); foreach ($properties as $key => $value) { $set = "set{$key}"; $get = "get{$key}"; $object->$set($value); - $this->assertEquals($value, $object->$get()); + $result = $object->$get(); + if ($value instanceof Absolute) { + try { + $value = $value->toInt('twip'); + $result = $result->toInt('twip'); + } catch (Throwable $ex) { + throw new Exception("Failed to convert values for property `$key`", 1, $ex); + } + } + $this->assertEquals($value, $result); } } @@ -69,28 +83,33 @@ public function testSetStyleValue() $object = new TextBox(); $properties = array( - 'width' => 200, - 'height' => 200, + 'width' => Absolute::from('twip', 200), + 'height' => Absolute::from('twip', 200), 'alignment' => Jc::START, - 'marginTop' => 240, - 'marginLeft' => 240, + 'marginTop' => Absolute::from('twip', 240), + 'marginLeft' => Absolute::from('twip', 240), 'wrappingStyle' => 'inline', 'positioning' => 'absolute', 'posHorizontal' => 'center', 'posVertical' => 'top', 'posHorizontalRel' => 'margin', 'posVerticalRel' => 'page', - 'innerMarginTop' => '5', - 'innerMarginRight' => '5', - 'innerMarginBottom' => '5', - 'innerMarginLeft' => '5', - 'borderSize' => '2', - 'borderColor' => 'red', + 'innerMarginTop' => Absolute::from('twip', 5), + 'innerMarginRight' => Absolute::from('twip', 5), + 'innerMarginBottom' => Absolute::from('twip', 5), + 'innerMarginLeft' => Absolute::from('twip', 5), + 'borderSize' => Absolute::from('twip', 2), + 'borderColor' => new HighlightColor('red'), ); foreach ($properties as $key => $value) { $get = "get{$key}"; $object->setStyleValue("{$key}", $value); - $this->assertEquals($value, $object->$get()); + $result = $object->$get(); + if ($value instanceof Absolute) { + $result = $result->toInt('twip'); + $value = $value->toInt('twip'); + } + $this->assertEquals($value, $result); } } @@ -112,8 +131,8 @@ public function testSetGetWidth() { $expected = 200; $object = new TextBox(); - $object->setWidth($expected); - $this->assertEquals($expected, $object->getWidth()); + $object->setWidth(Absolute::from('twip', $expected)); + $this->assertEquals($expected, $object->getWidth()->toInt('twip')); } /** @@ -123,8 +142,8 @@ public function testSetGetHeight() { $expected = 200; $object = new TextBox(); - $object->setHeight($expected); - $this->assertEquals($expected, $object->getHeight()); + $object->setHeight(Absolute::from('twip', $expected)); + $this->assertEquals($expected, $object->getHeight()->toInt('twip')); } /** @@ -146,8 +165,8 @@ public function testSetGetMarginTop() { $expected = 5; $object = new TextBox(); - $object->setMarginTop($expected); - $this->assertEquals($expected, $object->getMarginTop()); + $object->setMarginTop(Absolute::from('twip', $expected)); + $this->assertEquals($expected, $object->getMarginTop()->toInt('twip')); } /** @@ -157,8 +176,8 @@ public function testSetGetMarginLeft() { $expected = 5; $object = new TextBox(); - $object->setMarginLeft($expected); - $this->assertEquals($expected, $object->getMarginLeft()); + $object->setMarginLeft(Absolute::from('twip', $expected)); + $this->assertEquals($expected, $object->getMarginLeft()->toInt('twip')); } /** @@ -168,8 +187,8 @@ public function testSetGetInnerMarginTop() { $expected = 5; $object = new TextBox(); - $object->setInnerMarginTop($expected); - $this->assertEquals($expected, $object->getInnerMarginTop()); + $object->setInnerMarginTop(Absolute::from('twip', $expected)); + $this->assertEquals($expected, $object->getInnerMarginTop()->toInt('twip')); } /** @@ -245,8 +264,8 @@ public function testSetGetInnerMarginRight() { $expected = 5; $object = new TextBox(); - $object->setInnerMarginRight($expected); - $this->assertEquals($expected, $object->getInnerMarginRight()); + $object->setInnerMarginRight(Absolute::from('twip', $expected)); + $this->assertEquals($expected, $object->getInnerMarginRight()->toInt('twip')); } /** @@ -256,8 +275,8 @@ public function testSetGetInnerMarginBottom() { $expected = 5; $object = new TextBox(); - $object->setInnerMarginBottom($expected); - $this->assertEquals($expected, $object->getInnerMarginBottom()); + $object->setInnerMarginBottom(Absolute::from('twip', $expected)); + $this->assertEquals($expected, $object->getInnerMarginBottom()->toInt('twip')); } /** @@ -267,8 +286,8 @@ public function testSetGetInnerMarginLeft() { $expected = 5; $object = new TextBox(); - $object->setInnerMarginLeft($expected); - $this->assertEquals($expected, $object->getInnerMarginLeft()); + $object->setInnerMarginLeft(Absolute::from('twip', $expected)); + $this->assertEquals($expected, $object->getInnerMarginLeft()->toInt('twip')); } /** @@ -278,8 +297,10 @@ public function testSetGetInnerMargin() { $expected = 5; $object = new TextBox(); - $object->setInnerMargin($expected); - $this->assertEquals(array($expected, $expected, $expected, $expected), $object->getInnerMargin()); + $object->setInnerMargin(Absolute::from('twip', $expected)); + $this->assertEquals(array($expected, $expected, $expected, $expected), array_map(function ($value) { + return $value->toInt('twip'); + }, $object->getInnerMargin())); } /** @@ -289,8 +310,8 @@ public function testSetGetBorderSize() { $expected = 2; $object = new TextBox(); - $object->setBorderSize($expected); - $this->assertEquals($expected, $object->getBorderSize()); + $object->setBorderSize(Absolute::from('twip', $expected)); + $this->assertEquals($expected, $object->getBorderSize()->toInt('twip')); } /** @@ -298,7 +319,7 @@ public function testSetGetBorderSize() */ public function testSetGetBorderColor() { - $expected = 'red'; + $expected = new HighlightColor('red'); $object = new TextBox(); $object->setBorderColor($expected); $this->assertEquals($expected, $object->getBorderColor()); diff --git a/tests/PhpWord/Style/Theme/BodyFontsTest.php b/tests/PhpWord/Style/Theme/BodyFontsTest.php new file mode 100644 index 0000000000..d3d2e4dcb3 --- /dev/null +++ b/tests/PhpWord/Style/Theme/BodyFontsTest.php @@ -0,0 +1,107 @@ +assertEquals('Calibri', $fontScheme->getLatin()); + $this->assertEquals('', $fontScheme->getEastAsian()); + $this->assertEquals('', $fontScheme->getComplexScript()); + $this->assertEquals('MS 明朝', $fontScheme->getFont('Jpan')); + $this->assertEquals('MS 明朝', $fontScheme->getFonts()['Jpan']); + } + + public function testCustomLatin() + { + $fontScheme = new BodyFonts(array('Latin' => 'Custom Font')); + + $this->assertEquals('Custom Font', $fontScheme->getLatin()); + $this->assertEquals('', $fontScheme->getEastAsian()); + $this->assertEquals('', $fontScheme->getComplexScript()); + $this->assertEquals('MS 明朝', $fontScheme->getFont('Jpan')); + $this->assertEquals('MS 明朝', $fontScheme->getFonts()['Jpan']); + } + + public function testCustomEastAsian() + { + $fontScheme = new BodyFonts(array('EastAsian' => 'Custom Font')); + + $this->assertEquals('Calibri', $fontScheme->getLatin()); + $this->assertEquals('Custom Font', $fontScheme->getEastAsian()); + $this->assertEquals('', $fontScheme->getComplexScript()); + $this->assertEquals('MS 明朝', $fontScheme->getFont('Jpan')); + $this->assertEquals('MS 明朝', $fontScheme->getFonts()['Jpan']); + } + + public function testCustomComplexScript() + { + $fontScheme = new BodyFonts(array('ComplexScript' => 'Custom Font')); + + $this->assertEquals('Calibri', $fontScheme->getLatin()); + $this->assertEquals('', $fontScheme->getEastAsian()); + $this->assertEquals('Custom Font', $fontScheme->getComplexScript()); + $this->assertEquals('MS 明朝', $fontScheme->getFont('Jpan')); + $this->assertEquals('MS 明朝', $fontScheme->getFonts()['Jpan']); + } + + public function testCustomFont() + { + $fontScheme = new BodyFonts(array('Jpan' => 'Custom Font')); + + $this->assertEquals('Calibri', $fontScheme->getLatin()); + $this->assertEquals('', $fontScheme->getEastAsian()); + $this->assertEquals('', $fontScheme->getComplexScript()); + $this->assertEquals('Custom Font', $fontScheme->getFont('Jpan')); + $this->assertEquals('Custom Font', $fontScheme->getFonts()['Jpan']); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Invalid script 'BadScript' provided + */ + public function testSettingBadFont() + { + new BodyFonts(array('BadScript' => 'Custom Font')); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage No font found for script 'BadScript' in color scheme 'PhpOffice\PhpWord\Style\Theme\BodyFonts' + */ + public function testGettingBadFont() + { + $bodyFonts = new BodyFonts(); + $bodyFonts->getFont('BadScript'); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage No font found for script 'BadScript' in color scheme 'PhpOffice\PhpWord\Style\Theme\Fonts' + */ + public function testGettingBadDefaultFont() + { + HeadingFonts::getDefaultFont('BadScript'); + } +} diff --git a/tests/PhpWord/Style/Theme/ColorSchemeTest.php b/tests/PhpWord/Style/Theme/ColorSchemeTest.php new file mode 100644 index 0000000000..2e614e0b4d --- /dev/null +++ b/tests/PhpWord/Style/Theme/ColorSchemeTest.php @@ -0,0 +1,154 @@ + new SystemColor('windowText', new Hex('000')), + 'dk2' => new Hex('1F497D'), + 'lt1' => new SystemColor('window', new Hex('fff')), + 'lt2' => new Hex('EEECE1'), + 'accent1' => new Hex('4F81BD'), + 'accent2' => new Hex('C0504D'), + 'accent3' => new Hex('9BBB59'), + 'accent4' => new Hex('8064A2'), + 'accent5' => new Hex('4BACC6'), + 'accent6' => new Hex('F79646'), + 'hlink' => new Hex('0000FF'), + 'folHlink' => new Hex('800080'), + ); + $colorScheme = new ColorScheme($colors); + + foreach ($colors as $name => $color) { + $this->assertInstanceOf(get_class($color), $colorScheme->getColor($name)); + $this->assertEquals($color->toHexOrName(), $colorScheme->getColor($name)->toHexOrName()); + $this->assertEquals($color->toHexOrName(), $colorScheme->getColors()[$name]->toHexOrName()); + } + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage 12 colors expected, but 1 colors provided + */ + public function testTooFewColors() + { + new ColorScheme(array( + 'dk1' => new SystemColor('windowText', new Hex('000')), + )); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage 12 colors expected, but 13 colors provided + */ + public function testTooManyColors() + { + new ColorScheme(array( + 'dk1' => new SystemColor('windowText', new Hex('000')), + 'dk2' => new Hex('1F497D'), + 'dk3' => new Hex('3D5CA3'), + 'lt1' => new SystemColor('window', new Hex('fff')), + 'lt2' => new Hex('EEECE1'), + 'accent1' => new Hex('4F81BD'), + 'accent2' => new Hex('C0504D'), + 'accent3' => new Hex('9BBB59'), + 'accent4' => new Hex('8064A2'), + 'accent5' => new Hex('4BACC6'), + 'accent6' => new Hex('F79646'), + 'hlink' => new Hex('0000FF'), + 'folHlink' => new Hex('800080'), + )); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Missing 'dk2' from provided color scheme + */ + public function testWrongName() + { + new ColorScheme(array( + 'dk1' => new SystemColor('windowText', new Hex('000')), + 'dk3' => new Hex('1F497D'), + 'lt1' => new SystemColor('window', new Hex('fff')), + 'lt2' => new Hex('EEECE1'), + 'accent1' => new Hex('4F81BD'), + 'accent2' => new Hex('C0504D'), + 'accent3' => new Hex('9BBB59'), + 'accent4' => new Hex('8064A2'), + 'accent5' => new Hex('4BACC6'), + 'accent6' => new Hex('F79646'), + 'hlink' => new Hex('0000FF'), + 'folHlink' => new Hex('800080'), + )); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Provided color for 'dk1' must be an instance of 'PhpOffice\PhpWord\Style\Colors\SpecialColor', 'string' provided + */ + public function testSettingBadColor() + { + new ColorScheme(array( + 'dk1' => '000000', + 'dk2' => new Hex('1F497D'), + 'lt1' => new SystemColor('window', new Hex('fff')), + 'lt2' => new Hex('EEECE1'), + 'accent1' => new Hex('4F81BD'), + 'accent2' => new Hex('C0504D'), + 'accent3' => new Hex('9BBB59'), + 'accent4' => new Hex('8064A2'), + 'accent5' => new Hex('4BACC6'), + 'accent6' => new Hex('F79646'), + 'hlink' => new Hex('0000FF'), + 'folHlink' => new Hex('800080'), + )); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage No color exists for 'dk3' + */ + public function testGettingBadColor() + { + $colorScheme = new ColorScheme(array( + 'dk1' => new SystemColor('windowText', new Hex('000')), + 'dk2' => new Hex('1F497D'), + 'lt1' => new SystemColor('window', new Hex('fff')), + 'lt2' => new Hex('EEECE1'), + 'accent1' => new Hex('4F81BD'), + 'accent2' => new Hex('C0504D'), + 'accent3' => new Hex('9BBB59'), + 'accent4' => new Hex('8064A2'), + 'accent5' => new Hex('4BACC6'), + 'accent6' => new Hex('F79646'), + 'hlink' => new Hex('0000FF'), + 'folHlink' => new Hex('800080'), + )); + $colorScheme->getColor('dk3'); + } +} diff --git a/tests/PhpWord/Style/Theme/FontSchemeTest.php b/tests/PhpWord/Style/Theme/FontSchemeTest.php new file mode 100644 index 0000000000..01754bfe1a --- /dev/null +++ b/tests/PhpWord/Style/Theme/FontSchemeTest.php @@ -0,0 +1,80 @@ +assertEquals('Office', $fontScheme->getName()); + + $this->assertInstanceOf(HeadingFonts::class, $fontScheme->getHeadingFonts()); + + $this->assertInstanceOf(BodyFonts::class, $fontScheme->getBodyFonts()); + } + + public function testCustomName() + { + $fontScheme = new FontScheme('New Name'); + + $this->assertEquals('New Name', $fontScheme->getName()); + } + + public function testCustomFonts() + { + $fontScheme = new FontScheme( + 'Custom Heading Fonts', + new HeadingFonts(array('Latin' => 'Custom Font')), + new BodyFonts(array('Latin' => 'Another Custom Font')) + ); + $this->assertEquals('Custom Font', $fontScheme->getHeadingFonts()->getLatin()); + $this->assertEquals('Another Custom Font', $fontScheme->getBodyFonts()->getLatin()); + } + + /** + * @expectedException \TypeError + * @expectedExceptionMessage Argument 2 passed to PhpOffice\PhpWord\Style\Theme\FontScheme::__construct() must be an instance of PhpOffice\PhpWord\Style\Theme\HeadingFonts + */ + public function testBodyAsHeading() + { + new FontScheme( + 'Body as Heading', + new BodyFonts(), + new BodyFonts() + ); + } + + /** + * @expectedException \TypeError + * @expectedExceptionMessage Argument 3 passed to PhpOffice\PhpWord\Style\Theme\FontScheme::__construct() must be an instance of PhpOffice\PhpWord\Style\Theme\BodyFonts + */ + public function testHeadingAsBody() + { + new FontScheme( + 'Heading as Body', + new HeadingFonts(), + new HeadingFonts() + ); + } +} diff --git a/tests/PhpWord/Style/Theme/HeadingFontsTest.php b/tests/PhpWord/Style/Theme/HeadingFontsTest.php new file mode 100644 index 0000000000..6911ab4f92 --- /dev/null +++ b/tests/PhpWord/Style/Theme/HeadingFontsTest.php @@ -0,0 +1,116 @@ +assertEquals('Cambria', $fontScheme->getLatin()); + $this->assertEquals('', $fontScheme->getEastAsian()); + $this->assertEquals('', $fontScheme->getComplexScript()); + $this->assertEquals('MS ゴシック', $fontScheme->getFont('Jpan')); + $this->assertEquals('MS ゴシック', $fontScheme->getFonts()['Jpan']); + } + + public function testCustomLatin() + { + $fontScheme = new HeadingFonts(array('Latin' => 'Custom Font')); + + $this->assertEquals('Custom Font', $fontScheme->getLatin()); + $this->assertEquals('', $fontScheme->getEastAsian()); + $this->assertEquals('', $fontScheme->getComplexScript()); + $this->assertEquals('MS ゴシック', $fontScheme->getFont('Jpan')); + $this->assertEquals('MS ゴシック', $fontScheme->getFonts()['Jpan']); + } + + public function testCustomEastAsian() + { + $fontScheme = new HeadingFonts(array('EastAsian' => 'Custom Font')); + + $this->assertEquals('Cambria', $fontScheme->getLatin()); + $this->assertEquals('Custom Font', $fontScheme->getEastAsian()); + $this->assertEquals('', $fontScheme->getComplexScript()); + $this->assertEquals('MS ゴシック', $fontScheme->getFont('Jpan')); + $this->assertEquals('MS ゴシック', $fontScheme->getFonts()['Jpan']); + } + + public function testCustomComplexScript() + { + $fontScheme = new HeadingFonts(array('ComplexScript' => 'Custom Font')); + + $this->assertEquals('Cambria', $fontScheme->getLatin()); + $this->assertEquals('', $fontScheme->getEastAsian()); + $this->assertEquals('Custom Font', $fontScheme->getComplexScript()); + $this->assertEquals('MS ゴシック', $fontScheme->getFont('Jpan')); + $this->assertEquals('MS ゴシック', $fontScheme->getFonts()['Jpan']); + } + + public function testCustomFont() + { + $fontScheme = new HeadingFonts(array('Jpan' => 'Custom Font')); + + $this->assertEquals('Cambria', $fontScheme->getLatin()); + $this->assertEquals('', $fontScheme->getEastAsian()); + $this->assertEquals('', $fontScheme->getComplexScript()); + $this->assertEquals('Custom Font', $fontScheme->getFont('Jpan')); + $this->assertEquals('Custom Font', $fontScheme->getFonts()['Jpan']); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Font name expected, 'object' provided + */ + public function testSettingFontWrong() + { + new HeadingFonts(array('Latin' => new HeadingFonts())); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage Invalid script 'BadScript' provided + */ + public function testSettingBadFont() + { + new HeadingFonts(array('BadScript' => 'Custom Font')); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage No font found for script 'BadScript' in color scheme 'PhpOffice\PhpWord\Style\Theme\HeadingFonts' + */ + public function testGettingBadFont() + { + $headingFonts = new HeadingFonts(); + $headingFonts->getFont('BadScript'); + } + + /** + * @expectedException \Exception + * @expectedExceptionMessage No font found for script 'BadScript' in color scheme 'PhpOffice\PhpWord\Style\Theme\Fonts' + */ + public function testGettingBadDefaultFont() + { + HeadingFonts::getDefaultFont('BadScript'); + } +} diff --git a/tests/PhpWord/Style/Theme/ThemeTest.php b/tests/PhpWord/Style/Theme/ThemeTest.php new file mode 100644 index 0000000000..88fc401ea2 --- /dev/null +++ b/tests/PhpWord/Style/Theme/ThemeTest.php @@ -0,0 +1,36 @@ +getColorScheme(); + $this->assertInstanceOf(ColorScheme::class, $colorScheme); + + $fontScheme = $theme->getFontScheme(); + $this->assertInstanceOf(FontScheme::class, $fontScheme); + } +} diff --git a/tests/PhpWord/StyleTest.php b/tests/PhpWord/StyleTest.php index cbc39c871e..816b5286f2 100644 --- a/tests/PhpWord/StyleTest.php +++ b/tests/PhpWord/StyleTest.php @@ -1,4 +1,5 @@ Jc::CENTER); $font = array('italic' => true, '_bold' => true); - $table = array('bgColor' => 'CCCCCC'); + $table = array('bgColor' => new Hex('CCCCCC')); $styles = array( 'Paragraph' => 'Paragraph', 'Font' => 'Font', diff --git a/tests/PhpWord/TemplateProcessorTest.php b/tests/PhpWord/TemplateProcessorTest.php index 4caca77aeb..a71e7b8d7c 100644 --- a/tests/PhpWord/TemplateProcessorTest.php +++ b/tests/PhpWord/TemplateProcessorTest.php @@ -1,4 +1,5 @@ 1, 'userName' => 'Batman', 'userLocation' => 'Gotham City'), - array('userId' => 2, 'userName' => 'Superman', 'userLocation' => 'Metropolis'), + array('userId' => '1', 'userName' => 'Batman', 'userLocation' => 'Gotham City'), + array('userId' => '2', 'userName' => 'Superman', 'userLocation' => 'Metropolis'), ); $templateProcessor->setValue('tableHeader', 'My clonable table'); $templateProcessor->cloneRowAndSetValues('userId', $values); @@ -310,6 +311,26 @@ public function testSetValue() ); } + /** + * @covers ::setValue + * @test + */ + public function testSetValueMultipleReplacements() + { + $mainPart = ' + + + Hello ${firstname} ${lastname} + + '; + + Settings::setOutputEscapingEnabled(true); + $templateProcessor = new TestableTemplateProcesor($mainPart); + $templateProcessor->setValue(array('firstname', 'lastname'), array('Jane', 'Doe')); + + $this->assertContains('Hello Jane Doe', $templateProcessor->getMainPart()); + } + public function testSetComplexValue() { $title = new TextRun(); @@ -382,6 +403,82 @@ public function testSetValues() $this->assertContains('Hello John Doe', $templateProcessor->getMainPart()); } + /** + * @covers \PhpOffice\PhpWord\TemplateProcessor::getSlice + */ + public function testGetSlice() + { + $mainPart = ' + + + Hello ${firstname} ${lastname} + + '; + + $stub = new TestableTemplateProcesor($mainPart); + $this->assertEquals(substr($mainPart, 0, 1), $stub->getSlice(0, 1)); + $this->assertEquals(substr($mainPart, 1, 1), $stub->getSlice(1, 2)); + $this->assertEquals(substr($mainPart, 1, 182), $stub->getSlice(1, 183)); + $this->assertEquals(substr($mainPart, 0, 183), $stub->getSlice(0, 183)); + } + + /** + * @depends testGetSlice + * @covers \PhpOffice\PhpWord\TemplateProcessor::getSlice + * @expectedException \PhpOffice\PhpWord\Exception\Exception + * @expectedExceptionMessage Start position must be at least 0. `-1` provided + */ + public function testGetSliceNegative() + { + $mainPart = ' + + + Hello ${firstname} ${lastname} + + '; + + $stub = new TestableTemplateProcesor($mainPart); + $stub->getSlice(-1); + } + + /** + * @depends testGetSlice + * @covers \PhpOffice\PhpWord\TemplateProcessor::getSlice + * @expectedException \PhpOffice\PhpWord\Exception\Exception + * @expectedExceptionMessage Provided end position `5` is smaller than the start position `9` + */ + public function testGetSliceFlipped() + { + $mainPart = ' + + + Hello ${firstname} ${lastname} + + '; + + $stub = new TestableTemplateProcesor($mainPart); + $stub->getSlice(9, 5); + } + + /** + * @depends testGetSlice + * @covers \PhpOffice\PhpWord\TemplateProcessor::getSlice + * @expectedException \PhpOffice\PhpWord\Exception\Exception + * @expectedExceptionMessage Provided end position `184` is longer than the length `183` + */ + public function testGetSliceLong() + { + $mainPart = ' + + + Hello ${firstname} ${lastname} + + '; + + $stub = new TestableTemplateProcesor($mainPart); + $stub->getSlice(0, 184); + } + /** * @covers ::setImageValue * @test @@ -394,7 +491,7 @@ public function testSetImageValue() $variablesReplace = array( 'headerValue' => $imagePath, 'documentContent' => array('path' => $imagePath, 'width' => 500, 'height' => 500), - 'footerValue' => array('path' => $imagePath, 'width' => 100, 'height' => 50, 'ratio' => false), + 'footerValue' => array('path' => $imagePath, 'width' => 100, 'height' => 50, 'ratio' => null), ); $templateProcessor->setImageValue(array_keys($variablesReplace), $variablesReplace); @@ -844,4 +941,18 @@ public function testShouldMakeFieldsUpdateOnOpen() $templateProcessor->setUpdateFields(false); $this->assertContains('', $templateProcessor->getSettingsPart()); } + + /** + * Helper function to call protected method + * + * @param string $method + */ + public static function callProtectedMethod($object, $method, array $args = array()) + { + $class = new \ReflectionClass(get_class($object)); + $method = $class->getMethod($method); + $method->setAccessible(true); + + return $method->invokeArgs($object, $args); + } } diff --git a/tests/PhpWord/Writer/HTML/ElementTest.php b/tests/PhpWord/Writer/HTML/ElementTest.php index 101e226f50..70404e3573 100644 --- a/tests/PhpWord/Writer/HTML/ElementTest.php +++ b/tests/PhpWord/Writer/HTML/ElementTest.php @@ -1,4 +1,5 @@ addSection(); $table = $section->addTable(); $row1 = $table->addRow(); - $cell11 = $row1->addCell(1000, array('gridSpan' => 2, 'bgColor' => '6086B8')); + $cell11 = $row1->addCell(Absolute::from('twip', 1000), array('gridSpan' => 2, 'bgColor' => new Hex('6086B8'))); $cell11->addText('cell spanning 2 bellow'); $row2 = $table->addRow(); - $cell21 = $row2->addCell(500, array('bgColor' => 'ffffff')); + $cell21 = $row2->addCell(Absolute::from('twip', 500), array('bgColor' => new Hex('ffffff'))); $cell21->addText('first cell'); - $cell22 = $row2->addCell(500); + $cell22 = $row2->addCell(Absolute::from('twip', 500)); $cell22->addText('second cell'); $dom = $this->getAsHTML($phpWord); @@ -102,8 +105,8 @@ public function testWriteColSpan() $this->assertEquals(2, $xpath->query('/html/body/table/tr[2]/td')->length); $this->assertEquals('#6086B8', $xpath->query('/html/body/table/tr[1]/td')->item(0)->attributes->getNamedItem('bgcolor')->textContent); - $this->assertEquals('#ffffff', $xpath->query('/html/body/table/tr[1]/td')->item(0)->attributes->getNamedItem('color')->textContent); - $this->assertEquals('#ffffff', $xpath->query('/html/body/table/tr[2]/td')->item(0)->attributes->getNamedItem('bgcolor')->textContent); + $this->assertEquals('#FFFFFF', $xpath->query('/html/body/table/tr[1]/td')->item(0)->attributes->getNamedItem('color')->textContent); + $this->assertEquals('#FFFFFF', $xpath->query('/html/body/table/tr[2]/td')->item(0)->attributes->getNamedItem('bgcolor')->textContent); $this->assertNull($xpath->query('/html/body/table/tr[2]/td')->item(0)->attributes->getNamedItem('color')); } @@ -117,16 +120,16 @@ public function testWriteRowSpan() $table = $section->addTable(); $row1 = $table->addRow(); - $row1->addCell(1000, array('vMerge' => 'restart'))->addText('row spanning 3 bellow'); - $row1->addCell(500)->addText('first cell being spanned'); + $row1->addCell(Absolute::from('twip', 1000), array('vMerge' => 'restart'))->addText('row spanning 3 bellow'); + $row1->addCell(Absolute::from('twip', 500))->addText('first cell being spanned'); $row2 = $table->addRow(); $row2->addCell(null, array('vMerge' => 'continue')); - $row2->addCell(500)->addText('second cell being spanned'); + $row2->addCell(Absolute::from('twip', 500))->addText('second cell being spanned'); $row3 = $table->addRow(); $row3->addCell(null, array('vMerge' => 'continue')); - $row3->addCell(500)->addText('third cell being spanned'); + $row3->addCell(Absolute::from('twip', 500))->addText('third cell being spanned'); $dom = $this->getAsHTML($phpWord); $xpath = new \DOMXPath($dom); diff --git a/tests/PhpWord/Writer/HTML/PartTest.php b/tests/PhpWord/Writer/HTML/PartTest.php index f83034143c..edb179f2da 100644 --- a/tests/PhpWord/Writer/HTML/PartTest.php +++ b/tests/PhpWord/Writer/HTML/PartTest.php @@ -1,4 +1,5 @@ addTitleStyle(1, array('bold' => true)); $phpWord->addFontStyle( 'Font', - array('name' => 'Verdana', 'size' => 11, 'color' => 'FF0000', 'fgColor' => 'FF0000') + array('name' => 'Verdana', 'size' => Absolute::from('pt', 11), 'color' => new Hex('FF0000')) ); - $phpWord->addParagraphStyle('Paragraph', array('alignment' => Jc::CENTER, 'spaceAfter' => 20, 'spaceBefore' => 20)); + $phpWord->addParagraphStyle('Paragraph', array('alignment' => Jc::CENTER, 'spaceAfter' => Absolute::from('eop', 20), 'spaceBefore' => Absolute::from('eop', 20))); $section = $phpWord->addSection(); $section->addBookmark('top'); $section->addText(htmlspecialchars('Test 1', ENT_COMPAT, 'UTF-8'), 'Font', 'Paragraph'); diff --git a/tests/PhpWord/Writer/ODText/ElementTest.php b/tests/PhpWord/Writer/ODText/ElementTest.php index 37f0d1ef52..12b7579e28 100644 --- a/tests/PhpWord/Writer/ODText/ElementTest.php +++ b/tests/PhpWord/Writer/ODText/ElementTest.php @@ -1,4 +1,5 @@ setCustomProperty('Company', 'PHPWord'); $phpWord->setDefaultFontName('Verdana'); - $phpWord->addFontStyle('Font', array('size' => 11)); + $phpWord->addFontStyle('Font', array('size' => Absolute::from('pt', 11))); $phpWord->addParagraphStyle('Paragraph', array('alignment' => Jc::CENTER)); - $phpWord->addTableStyle('tblStyle', array('width' => 100)); + $phpWord->addTableStyle('tblStyle', array('width' => Absolute::from('twip', 100))); $section = $phpWord->addSection(array('colsNum' => 2)); $section->addText($expected); @@ -65,14 +67,14 @@ public function testWriteContent() $section->addTextBreak(); $section->addPageBreak(); $section->addListItem('Test list item'); - $section->addImage($imageSrc, array('width' => 50)); + $section->addImage($imageSrc, array('width' => Absolute::from('twip', 50))); $section->addObject($objectSrc); $section->addTOC(); $textrun = $section->addTextRun(); $textrun->addText('Test text run'); - $table = $section->addTable(array('width' => 50)); + $table = $section->addTable(array('width' => Absolute::from('twip', 50))); $cell = $table->addRow()->addCell(); $cell = $table->addRow()->addCell(); $cell->addText('Test'); @@ -102,7 +104,7 @@ public function testWriteContent() public function testWriteNoStyle() { $phpWord = new PhpWord(); - $phpWord->addFontStyle('Font', array('size' => 11)); + $phpWord->addFontStyle('Font', array('size' => Absolute::from('pt', 11))); $doc = TestHelperDOCX::getDocument($phpWord, 'ODText'); diff --git a/tests/PhpWord/Writer/ODText/Style/TableTest.php b/tests/PhpWord/Writer/ODText/Style/TableTest.php new file mode 100644 index 0000000000..35933cca04 --- /dev/null +++ b/tests/PhpWord/Writer/ODText/Style/TableTest.php @@ -0,0 +1,99 @@ + array( + new Auto(), + Absolute::from('in', 1), + new Percent(15), + ), + ))); + $table->write(); + + $this->assertEquals("\n \n\n\n \n\n\n \n\n\n \n\n", $xmlWriter->getData()); + } + + /** + * Test wrong type of column width + * @expectedException \Exception + * @expectedExceptionMessage Column widths must be specified with `PhpOffice\PhpWord\Style\Lengths\Length` + */ + public function testWrongColumnWidthType() + { + new Table(new XMLWriter(), new TableStyle(array( + 'ColumnWidths' => array( + '5pt', + ), + ))); + } + + /** + * Test wrong type of column width + * @expectedException \Exception + * @expectedExceptionMessage Unsupported width `class@anonymous + */ + public function testWrongColumnWidthClass() + { + $table = new Table(new XMLWriter(), new TableStyle(array( + 'ColumnWidths' => array( + new class() extends Length { + public function isSpecified(): bool + { + return true; + } + }, + ), + ))); + $table->write(); + } + + /** + * Test wrong type of style + * @expectedException \Exception + * @expectedExceptionMessage Incorrect value provided for style. PhpOffice\PhpWord\Style\Table expected, PhpOffice\PhpWord\Style\Row provided + */ + public function testWrongStyle() + { + $xmlWriter = new XMLWriter(); + $table = new Table($xmlWriter, new RowStyle()); + $table->write(); + + $this->assertEquals('', $xmlWriter->getData()); + } +} diff --git a/tests/PhpWord/Writer/ODText/StyleTest.php b/tests/PhpWord/Writer/ODText/StyleTest.php index b1bf417d59..54931c276d 100644 --- a/tests/PhpWord/Writer/ODText/StyleTest.php +++ b/tests/PhpWord/Writer/ODText/StyleTest.php @@ -1,4 +1,5 @@ addFontStyle('Font', array('size' => 11)); + $phpWord->addFontStyle('Font', array('size' => Absolute::from('pt', 11))); $phpWord->addParagraphStyle('Paragraph', array('alignment' => Jc::CENTER)); $section = $phpWord->addSection(); $section->addText('Test 1', 'Font'); diff --git a/tests/PhpWord/Writer/PDF/DomPDFTest.php b/tests/PhpWord/Writer/PDF/DomPDFTest.php index bc229d5186..4768c978e8 100644 --- a/tests/PhpWord/Writer/PDF/DomPDFTest.php +++ b/tests/PhpWord/Writer/PDF/DomPDFTest.php @@ -1,4 +1,5 @@ setSizes(array(1, 2, 3, 4)); - $border->setColors(array('#FF0000', '#FF0000', '#FF0000', '#FF0000')); - $border->setSizes(array(20, 20, 20, 20)); + $border->setBorder('top', new BorderSide( + Absolute::from('pt', 1), + new Hex('FF0000'), + null, + Absolute::from('twip', 20) + )); + $border->setBorder('left', new BorderSide( + Absolute::from('pt', 1), + new Hex('FF0000'), + null, + Absolute::from('twip', 20) + )); + $border->setBorder('right', new BorderSide( + Absolute::from('pt', 1), + new Hex('FF0000'), + null, + Absolute::from('twip', 20) + )); + $border->setBorder('bottom', new BorderSide( + Absolute::from('pt', 1), + new Hex('FF0000'), + null, + Absolute::from('twip', 20) + )); $content = $border->write(); @@ -61,9 +86,9 @@ public function testBorderWithNonRegisteredColors() public function testIndentation() { $indentation = new \PhpOffice\PhpWord\Style\Indentation(); - $indentation->setLeft(1); - $indentation->setRight(2); - $indentation->setFirstLine(3); + $indentation->setLeft(Absolute::from('twip', 1)); + $indentation->setRight(Absolute::from('twip', 2)); + $indentation->setFirstLine(Absolute::from('twip', 3)); $indentWriter = new \PhpOffice\PhpWord\Writer\RTF\Style\Indentation($indentation); $indentWriter->setParentWriter(new RTF()); @@ -76,7 +101,7 @@ public function testRightTab() { $tabRight = new \PhpOffice\PhpWord\Style\Tab(); $tabRight->setType(\PhpOffice\PhpWord\Style\Tab::TAB_STOP_RIGHT); - $tabRight->setPosition(5); + $tabRight->setPosition(Absolute::from('twip', 5)); $tabWriter = new \PhpOffice\PhpWord\Writer\RTF\Style\Tab($tabRight); $tabWriter->setParentWriter(new RTF()); diff --git a/tests/PhpWord/Writer/RTFTest.php b/tests/PhpWord/Writer/RTFTest.php index 010720bd12..a18a4c2dee 100644 --- a/tests/PhpWord/Writer/RTFTest.php +++ b/tests/PhpWord/Writer/RTFTest.php @@ -1,4 +1,5 @@ addFontStyle( 'Font', - array('name' => 'Verdana', 'size' => 11, 'color' => 'FF0000', 'fgColor' => '00FF00') + array('name' => 'Verdana', 'size' => Absolute::from('pt', 11), 'color' => new Hex('FF0000')) ); $phpWord->addParagraphStyle('Paragraph', array('alignment' => Jc::CENTER)); $section = $phpWord->addSection(); diff --git a/tests/PhpWord/Writer/Word2007/ElementTest.php b/tests/PhpWord/Writer/Word2007/ElementTest.php index 6a295965f7..9a12af4ae9 100644 --- a/tests/PhpWord/Writer/Word2007/ElementTest.php +++ b/tests/PhpWord/Writer/Word2007/ElementTest.php @@ -1,4 +1,5 @@ addSection(); - $section->addLine(array('width' => 1000, 'height' => 1000, 'positioning' => 'absolute', 'flip' => true)); + $section->addLine(array('width' => Absolute::from('pt', 1000), 'height' => Absolute::from('pt', 1000), 'positioning' => 'absolute', 'flip' => true)); $doc = TestHelperDOCX::getDocument($phpWord); $element = '/w:document/w:body/w:p/w:r/w:pict/v:shapetype'; @@ -121,11 +125,11 @@ public function testTableElements() $section = $phpWord->addSection(); $table = $section->addTable(array('alignment' => \PhpOffice\PhpWord\SimpleType\JcTable::CENTER)); - $table->addRow(900); - $table->addCell(2000)->addText('Row 1'); - $table->addCell(2000)->addText('Row 2'); - $table->addCell(2000)->addText('Row 3'); - $table->addCell(2000)->addText('Row 4'); + $table->addRow(Absolute::from('twip', 900)); + $table->addCell(Absolute::from('twip', 2000))->addText('Row 1'); + $table->addCell(Absolute::from('twip', 2000))->addText('Row 2'); + $table->addCell(Absolute::from('twip', 2000))->addText('Row 3'); + $table->addCell(Absolute::from('twip', 2000))->addText('Row 4'); $doc = TestHelperDOCX::getDocument($phpWord); @@ -144,8 +148,8 @@ public function testTableWithStyleName() $section = $phpWord->addSection(); $table = $section->addTable('my_predefined_style'); - $table->setWidth(75); - $table->addRow(900); + $table->setWidth(Absolute::from('twip', 75)); + $table->addRow(Absolute::from('twip', 900)); $doc = TestHelperDOCX::getDocument($phpWord); @@ -167,8 +171,8 @@ public function testShapeElements() 'arc', array( 'points' => '-90 20', - 'frame' => array('width' => 120, 'height' => 120), - 'outline' => array('color' => '#333333', 'weight' => 2, 'startArrow' => 'oval', 'endArrow' => 'open'), + 'frame' => array('width' => Absolute::from('pt', 120), 'height' => Absolute::from('pt', 120)), + 'outline' => array('color' => new Hex('333333'), 'weight' => Absolute::from('pt', 2), 'startArrow' => 'oval', 'endArrow' => 'open'), ) ); @@ -176,10 +180,11 @@ public function testShapeElements() $section->addShape( 'curve', array( - 'points' => '1,100 200,1 1,50 200,50', 'connector' => 'elbow', - 'outline' => array( - 'color' => '#66cc00', - 'weight' => 2, + 'points' => '1,100 200,1 1,50 200,50', + 'connector' => 'elbow', + 'outline' => array( + 'color' => new Hex('66cc00'), + 'weight' => Absolute::from('emu', 2), 'dash' => 'dash', 'startArrow' => 'diamond', 'endArrow' => 'block', @@ -193,9 +198,9 @@ public function testShapeElements() array( 'points' => '1,1 150,30', 'outline' => array( - 'color' => '#cc00ff', + 'color' => new Hex('cc00ff'), 'line' => 'thickThin', - 'weight' => 3, + 'weight' => Absolute::from('emu', 3), 'startArrow' => 'oval', 'endArrow' => 'classic', 'endCap' => 'round', @@ -209,8 +214,8 @@ public function testShapeElements() array( 'points' => '1,30 20,10 55,20 75,10 100,40 115,50, 120,15 200,50', 'outline' => array( - 'color' => '#cc6666', - 'weight' => 2, + 'color' => new Hex('cc6666'), + 'weight' => Absolute::from('emu', 2), 'startArrow' => 'none', 'endArrow' => 'classic', ), @@ -221,11 +226,11 @@ public function testShapeElements() $section->addShape( 'rect', array( - 'roundness' => 0.2, - 'frame' => array('width' => 100, 'height' => 100, 'left' => 1, 'top' => 1), - 'fill' => array('color' => '#FFCC33'), - 'outline' => array('color' => '#990000', 'weight' => 1), - 'shadow' => array('color' => '#EEEEEE', 'offset' => '3pt,3pt'), + 'roundness' => new Percent(20), + 'frame' => array('width' => Absolute::from('pt', 100), 'height' => Absolute::from('pt', 100), 'left' => Absolute::from('pt', 1), 'top' => Absolute::from('pt', 1)), + 'fill' => array('color' => new Hex('FFCC33')), + 'outline' => array('color' => new Hex('990000'), 'weight' => Absolute::from('emu', 1)), + 'shadow' => array('color' => new Hex('EEEEEE'), 'offset' => '3pt,3pt'), ) ); @@ -233,10 +238,10 @@ public function testShapeElements() $section->addShape( 'oval', array( - 'frame' => array('width' => 100, 'height' => 70, 'left' => 1, 'top' => 1), - 'fill' => array('color' => '#33CC99'), - 'outline' => array('color' => '#333333', 'weight' => 2), - 'extrusion' => array('type' => 'perspective', 'color' => '#EEEEEE'), + 'frame' => array('width' => Absolute::from('pt', 100), 'height' => Absolute::from('pt', 70), 'left' => Absolute::from('pt', 1), 'top' => Absolute::from('pt', 1)), + 'fill' => array('color' => new Hex('33CC99')), + 'outline' => array('color' => new Hex('333333'), 'weight' => Absolute::from('emu', 2)), + 'extrusion' => array('type' => 'perspective', 'color' => new Hex('EEEEEE')), ) ); @@ -256,7 +261,7 @@ public function testChartElements() { $phpWord = new PhpWord(); $section = $phpWord->addSection(); - $style = array('width' => 1000000, 'height' => 1000000, 'showAxisLabels' => true, 'showGridX' => true, 'showGridY' => true); + $style = array('width' => Absolute::from('emu', 1000000), 'height' => Absolute::from('emu', 1000000), 'showAxisLabels' => true, 'showGridX' => true, 'showGridY' => true); $chartTypes = array('pie', 'doughnut', 'bar', 'line', 'area', 'scatter', 'radar'); $categories = array('A', 'B', 'C', 'D', 'E'); @@ -476,8 +481,8 @@ public function testTrackChange() public function testTitleAndHeading() { $phpWord = new PhpWord(); - $phpWord->addTitleStyle(0, array('size' => 14, 'italic' => true)); - $phpWord->addTitleStyle(1, array('size' => 20, 'color' => '333333', 'bold' => true)); + $phpWord->addTitleStyle(0, array('size' => Absolute::from('pt', 14), 'italic' => true)); + $phpWord->addTitleStyle(1, array('size' => Absolute::from('pt', 20), 'color' => new Hex('333333'), 'bold' => true)); $section = $phpWord->addSection(); $section->addTitle('This is a title', 0); @@ -517,7 +522,7 @@ public function testTextWithAmpersant() public function testListItemRunStyleWriting() { $phpWord = new PhpWord(); - $phpWord->addParagraphStyle('MyParagraphStyle', array('spaceBefore' => 400)); + $phpWord->addParagraphStyle('MyParagraphStyle', array('spaceBefore' => Absolute::from('twip', 400))); $section = $phpWord->addSection(); $listItemRun = $section->addListItemRun(0, null, 'MyParagraphStyle'); diff --git a/tests/PhpWord/Writer/Word2007/Part/AbstractPartTest.php b/tests/PhpWord/Writer/Word2007/Part/AbstractPartTest.php index fac94882d2..e580ba116b 100644 --- a/tests/PhpWord/Writer/Word2007/Part/AbstractPartTest.php +++ b/tests/PhpWord/Writer/Word2007/Part/AbstractPartTest.php @@ -1,4 +1,5 @@ getStyle(); $style->setLandscape(); $style->setPageNumberingStart(2); - $style->setBorderSize(240); + $style->setBorders(new BorderSide(Absolute::from('eop', 240))); $style->setBreakType('nextPage'); $doc = TestHelperDOCX::getDocument($phpWord); @@ -126,8 +131,8 @@ public function testElements() $objectSrc = __DIR__ . '/../../../_files/documents/sheet.xls'; $phpWord = new PhpWord(); - $phpWord->addTitleStyle(1, array('color' => '333333', 'bold' => true)); - $phpWord->addTitleStyle(2, array('color' => '666666')); + $phpWord->addTitleStyle(1, array('color' => new Colors\Hex('333333'), 'bold' => true)); + $phpWord->addTitleStyle(2, array('color' => new Colors\Hex('666666'))); $section = $phpWord->addSection(); $section->addTOC(); $section->addPageBreak(); @@ -147,9 +152,9 @@ public function testElements() 'positioning' => 'relative', 'posHorizontalRel' => 'margin', 'posVerticalRel' => 'margin', - 'innerMargin' => 10, - 'borderSize' => 1, - 'borderColor' => '#FF0', + 'innerMargin' => Absolute::from('eop', 10), + 'borderSize' => Absolute::from('eop', 1), + 'borderColor' => new Colors\Hex('FF0'), ) ); $section->addTextBox(array('wrappingStyle' => 'tight', 'positioning' => 'absolute', 'alignment' => Jc::CENTER)); @@ -172,13 +177,13 @@ public function testElements() $section->addField('PAGE', array('format' => 'ArabicDash')); $section->addLine( array( - 'width' => 10, - 'height' => 10, + 'width' => Absolute::from('eop', 10), + 'height' => Absolute::from('eop', 10), 'positioning' => 'absolute', 'beginArrow' => 'block', 'endArrow' => 'open', 'dash' => 'rounddot', - 'weight' => 10, + 'weight' => Absolute::from('eop', 10), ) ); @@ -214,30 +219,30 @@ public function testElementStyles() { $objectSrc = __DIR__ . '/../../../_files/documents/sheet.xls'; - $tabs = array(new \PhpOffice\PhpWord\Style\Tab('right', 9090)); + $tabs = array(new \PhpOffice\PhpWord\Style\Tab('right', Absolute::from('twip', 9090))); $phpWord = new PhpWord(); $phpWord->addParagraphStyle( 'pStyle', array( 'alignment' => Jc::CENTER, 'tabs' => $tabs, - 'shading' => array('fill' => 'FFFF99'), - 'borderSize' => 4, + 'shading' => array('fill' => new Colors\Hex('FFFF99')), + 'borderSize' => Absolute::from('eop', 4), ) ); // Style #1 $phpWord->addFontStyle( 'fStyle', array( - 'size' => '20', + 'size' => Absolute::from('pt', 20), 'bold' => true, 'allCaps' => true, - 'scale' => 200, - 'spacing' => 240, - 'kerning' => 10, + 'scale' => new Percent(200), + 'spacing' => Absolute::from('pt', 240), + 'kerning' => Absolute::from('pt', 10), ) ); // Style #2 - $phpWord->addTitleStyle(1, array('color' => '333333', 'doubleStrikethrough' => true)); // Style #3 - $phpWord->addTableStyle('tStyle', array('borderSize' => 1)); + $phpWord->addTitleStyle(1, array('color' => new Colors\Hex('333333'), 'doubleStrikethrough' => true)); // Style #3 + $phpWord->addTableStyle('tStyle', array('borderSize' => Absolute::from('eop', 1))); $fontStyle = new Font('text', array('alignment' => Jc::CENTER)); $section = $phpWord->addSection(); @@ -247,7 +252,7 @@ public function testElementStyles() $section->addTitle('Title 1', 1); $section->addTOC('fStyle'); $table = $section->addTable('tStyle'); - $table->setWidth(100); + $table->setWidth(Absolute::from('eop', 100)); $doc = TestHelperDOCX::getDocument($phpWord); // List item @@ -275,7 +280,7 @@ public function testWriteText() $phpWord = new PhpWord(); $phpWord->addFontStyle($rStyle, array('bold' => true)); - $phpWord->addParagraphStyle($pStyle, array('hanging' => 120, 'indent' => 120)); + $phpWord->addParagraphStyle($pStyle, array('hanging' => Absolute::from('eop', 120), 'indent' => Absolute::from('eop', 120))); $section = $phpWord->addSection(); $section->addText('Test', $rStyle, $pStyle); $doc = TestHelperDOCX::getDocument($phpWord); @@ -292,7 +297,7 @@ public function testWriteText() public function testWriteTextRun() { $pStyle = 'pStyle'; - $aStyle = array('alignment' => Jc::BOTH, 'spaceBefore' => 120, 'spaceAfter' => 120); + $aStyle = array('alignment' => Jc::BOTH, 'spaceBefore' => Absolute::from('eop', 120), 'spaceAfter' => Absolute::from('eop', 120)); $imageSrc = __DIR__ . '/../../../_files/images/earth.jpg'; $phpWord = new PhpWord(); @@ -363,8 +368,8 @@ public function testWritePreserveText() */ public function testWriteTextBreak() { - $fArray = array('size' => 12); - $pArray = array('spacing' => 240); + $fArray = array('size' => Absolute::from('pt', 12)); + $pArray = array('spacing' => Absolute::from('twip', 240)); $fName = 'fStyle'; $pName = 'pStyle'; @@ -389,7 +394,7 @@ public function testWriteTextBreak() public function testWriteImage() { $phpWord = new PhpWord(); - $styles = array('alignment' => Jc::START, 'width' => 40, 'height' => 40, 'marginTop' => -1, 'marginLeft' => -1); + $styles = array('alignment' => Jc::START, 'width' => Absolute::from('eop', 40), 'height' => Absolute::from('eop', 40), 'marginTop' => Absolute::from('eop', -1), 'marginLeft' => Absolute::from('eop', -1)); $wraps = array('inline', 'behind', 'infront', 'square', 'tight'); $section = $phpWord->addSection(); foreach ($wraps as $wrap) { @@ -437,7 +442,7 @@ public function testWriteWatermark() public function testWriteTitle() { $phpWord = new PhpWord(); - $phpWord->addTitleStyle(1, array('bold' => true), array('spaceAfter' => 240)); + $phpWord->addTitleStyle(1, array('bold' => true), array('spaceAfter' => Absolute::from('eop', 240))); $phpWord->addSection()->addTitle('Test', 1); $doc = TestHelperDOCX::getDocument($phpWord); @@ -454,8 +459,8 @@ public function testWriteCheckbox() $pStyle = 'pStyle'; $phpWord = new PhpWord(); - // $phpWord->addFontStyle($rStyle, array('bold' => true)); - // $phpWord->addParagraphStyle($pStyle, array('hanging' => 120, 'indent' => 120)); + $phpWord->addFontStyle($rStyle, array('bold' => true)); + $phpWord->addParagraphStyle($pStyle, array('hanging' => Absolute::from('eop', 120), 'indent' => Absolute::from('eop', 120))); $section = $phpWord->addSection(); $section->addCheckBox('Check1', 'Test', $rStyle, $pStyle); $doc = TestHelperDOCX::getDocument($phpWord); @@ -505,15 +510,15 @@ public function testWriteFontStyle() { $phpWord = new PhpWord(); $styles['name'] = 'Verdana'; - $styles['size'] = 14; + $styles['size'] = Absolute::from('pt', 14); $styles['bold'] = true; $styles['italic'] = true; $styles['underline'] = 'dash'; $styles['strikethrough'] = true; $styles['superScript'] = true; - $styles['color'] = 'FF0000'; - $styles['fgColor'] = 'yellow'; - $styles['bgColor'] = 'FFFF00'; + $styles['color'] = new Colors\Hex('FF0000'); + $styles['fgColor'] = new Colors\HighlightColor('yellow'); + $styles['bgColor'] = new Colors\Hex('FFFF00'); $styles['hint'] = 'eastAsia'; $styles['smallCaps'] = true; @@ -523,26 +528,28 @@ public function testWriteFontStyle() $parent = '/w:document/w:body/w:p/w:r/w:rPr'; $this->assertEquals($styles['name'], $doc->getElementAttribute("{$parent}/w:rFonts", 'w:ascii')); - $this->assertEquals($styles['size'] * 2, $doc->getElementAttribute("{$parent}/w:sz", 'w:val')); + $this->assertEquals($styles['size']->toInt('hpt'), $doc->getElementAttribute("{$parent}/w:sz", 'w:val')); $this->assertTrue($doc->elementExists("{$parent}/w:b")); $this->assertTrue($doc->elementExists("{$parent}/w:i")); $this->assertEquals($styles['underline'], $doc->getElementAttribute("{$parent}/w:u", 'w:val')); $this->assertTrue($doc->elementExists("{$parent}/w:strike")); $this->assertEquals('superscript', $doc->getElementAttribute("{$parent}/w:vertAlign", 'w:val')); - $this->assertEquals($styles['color'], $doc->getElementAttribute("{$parent}/w:color", 'w:val')); - $this->assertEquals($styles['fgColor'], $doc->getElementAttribute("{$parent}/w:highlight", 'w:val')); + $this->assertEquals($styles['color']->toHexOrName(), $doc->getElementAttribute("{$parent}/w:color", 'w:val')); + $this->assertEquals($styles['fgColor']->toHexOrName(), $doc->getElementAttribute("{$parent}/w:highlight", 'w:val')); $this->assertTrue($doc->elementExists("{$parent}/w:smallCaps")); } /** - * Tests that if no color is set on a cell a border gets writen with the default color + * Tests that if no color is set on a cell a border gets writen with auto color */ public function testWriteDefaultColor() { $phpWord = new PhpWord(); $section = $phpWord->addSection(); - $cStyles['borderTopSize'] = 120; + $cStyles['bordersFromArray'] = array( + 'top' => new BorderSide(Absolute::from('eop', 120)), + ); $table = $section->addTable(); $table->addRow(); @@ -550,7 +557,7 @@ public function testWriteDefaultColor() $cell->addText('Test'); $doc = TestHelperDOCX::getDocument($phpWord); - $this->assertEquals(Cell::DEFAULT_BORDER_COLOR, $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr/w:tc/w:tcPr/w:tcBorders/w:top', 'w:color')); + $this->assertEquals('auto', $doc->getElementAttribute('/w:document/w:body/w:tbl/w:tr/w:tc/w:tcPr/w:tcBorders/w:top', 'w:color')); } /** @@ -559,34 +566,34 @@ public function testWriteDefaultColor() public function testWriteTableStyle() { $phpWord = new PhpWord(); - $rHeight = 120; - $cWidth = 120; + $rHeight = Absolute::from('eop', 120); + $cWidth = Absolute::from('eop', 120); $imageSrc = __DIR__ . '/../../../_files/images/earth.jpg'; $objectSrc = __DIR__ . '/../../../_files/documents/sheet.xls'; - $tStyles['width'] = 50; - $tStyles['cellMarginTop'] = 120; - $tStyles['cellMarginRight'] = 120; - $tStyles['cellMarginBottom'] = 120; - $tStyles['cellMarginLeft'] = 120; + $tStyles['width'] = Absolute::from('eop', 50); + $tStyles['cellMarginTop'] = Absolute::from('eop', 120); + $tStyles['cellMarginRight'] = Absolute::from('eop', 120); + $tStyles['cellMarginBottom'] = Absolute::from('eop', 120); + $tStyles['cellMarginLeft'] = Absolute::from('eop', 120); $rStyles['tblHeader'] = true; $rStyles['cantSplit'] = true; $cStyles['valign'] = 'top'; $cStyles['textDirection'] = 'btLr'; - $cStyles['bgColor'] = 'FF0000'; - $cStyles['borderTopSize'] = 120; - $cStyles['borderBottomSize'] = 120; - $cStyles['borderLeftSize'] = 120; - $cStyles['borderRightSize'] = 120; - $cStyles['borderTopColor'] = 'FF0000'; - $cStyles['borderBottomColor'] = 'FF0000'; - $cStyles['borderLeftColor'] = 'FF0000'; - $cStyles['borderRightColor'] = 'FF0000'; + $cStyles['bgColor'] = new Colors\Hex('FF0000'); + $cStyles['borderTopSize'] = Absolute::from('eop', 120); + $cStyles['borderBottomSize'] = Absolute::from('eop', 120); + $cStyles['borderLeftSize'] = Absolute::from('eop', 120); + $cStyles['borderRightSize'] = Absolute::from('eop', 120); + $cStyles['borderTopColor'] = new Colors\Hex('FF0000'); + $cStyles['borderBottomColor'] = new Colors\Hex('FF0000'); + $cStyles['borderLeftColor'] = new Colors\Hex('FF0000'); + $cStyles['borderRightColor'] = new Colors\Hex('FF0000'); $cStyles['vMerge'] = 'restart'; $section = $phpWord->addSection(); $table = $section->addTable($tStyles); - $table->setWidth(100); + $table->setWidth(Absolute::from('eop', 100)); $table->addRow($rHeight, $rStyles); $cell = $table->addCell($cWidth, $cStyles); $cell->addText('Test'); @@ -603,12 +610,12 @@ public function testWriteTableStyle() $parent = '/w:document/w:body/w:tbl/w:tblPr/w:tblCellMar'; $parent = '/w:document/w:body/w:tbl/w:tr/w:trPr'; - $this->assertEquals($rHeight, $doc->getElementAttribute("{$parent}/w:trHeight", 'w:val')); + $this->assertEquals($rHeight->toInt('twip'), $doc->getElementAttribute("{$parent}/w:trHeight", 'w:val')); $this->assertEquals($rStyles['tblHeader'], $doc->getElementAttribute("{$parent}/w:tblHeader", 'w:val')); $this->assertEquals($rStyles['cantSplit'], $doc->getElementAttribute("{$parent}/w:cantSplit", 'w:val')); $parent = '/w:document/w:body/w:tbl/w:tr/w:tc/w:tcPr'; - $this->assertEquals($cWidth, $doc->getElementAttribute("{$parent}/w:tcW", 'w:w')); + $this->assertEquals($cWidth->toInt('twip'), $doc->getElementAttribute("{$parent}/w:tcW", 'w:w')); $this->assertEquals($cStyles['valign'], $doc->getElementAttribute("{$parent}/w:vAlign", 'w:val')); $this->assertEquals($cStyles['textDirection'], $doc->getElementAttribute("{$parent}/w:textDirection", 'w:val')); } @@ -624,18 +631,18 @@ public function testWriteCellStyleCellGridSpan() $table = $section->addTable(); $table->addRow(); - $cell = $table->addCell(200); + $cell = $table->addCell(Absolute::from('eop', 200)); $cell->getStyle()->setGridSpan(5); $table->addRow(); - $table->addCell(40); - $table->addCell(40); - $table->addCell(40); - $table->addCell(40); - $table->addCell(40); + $table->addCell(Absolute::from('eop', 40)); + $table->addCell(Absolute::from('eop', 40)); + $table->addCell(Absolute::from('eop', 40)); + $table->addCell(Absolute::from('eop', 40)); + $table->addCell(Absolute::from('eop', 40)); $table->addRow(); - $cell = $table->addCell(200, array('borderRightColor' => 'FF0000')); + $cell = $table->addCell(Absolute::from('eop', 200), array('borderRightColor' => new Colors\Hex('FF0000'))); $cell->getStyle()->setGridSpan(5); $doc = TestHelperDOCX::getDocument($phpWord); @@ -653,7 +660,7 @@ public function testWriteGutterAndLineNumbering() $lineNumberingPath = '/w:document/w:body/w:sectPr/w:lnNumType'; $phpWord = new PhpWord(); - $phpWord->addSection(array('gutter' => 240, 'lineNumbering' => array())); + $phpWord->addSection(array('gutter' => Absolute::from('twip', 240), 'lineNumbering' => array())); $doc = TestHelperDOCX::getDocument($phpWord); $this->assertEquals(240, $doc->getElement($pageMarginPath)->getAttribute('w:gutter')); diff --git a/tests/PhpWord/Writer/Word2007/Part/FooterTest.php b/tests/PhpWord/Writer/Word2007/Part/FooterTest.php index 1f9bba0dd0..45158202ac 100644 --- a/tests/PhpWord/Writer/Word2007/Part/FooterTest.php +++ b/tests/PhpWord/Writer/Word2007/Part/FooterTest.php @@ -1,4 +1,5 @@ addText(''); - $container->addPreserveText(''); - $container->addTextBreak(); - $container->addTextRun(); + // $container->addText(''); + // $container->addPreserveText(''); + // $container->addTextBreak(); + // $container->addTextRun(); $container->addTable()->addRow()->addCell()->addText(''); - $container->addImage($imageSrc); - + // $container->addImage($imageSrc); + // $writer = new Word2007(); - $writer->setUseDiskCaching(true); + // $writer->setUseDiskCaching(true); $object = new Footer(); $object->setParentWriter($writer); $object->setElement($container); diff --git a/tests/PhpWord/Writer/Word2007/Part/FootnotesTest.php b/tests/PhpWord/Writer/Word2007/Part/FootnotesTest.php index 4b0e94df30..3ce9169a39 100644 --- a/tests/PhpWord/Writer/Word2007/Part/FootnotesTest.php +++ b/tests/PhpWord/Writer/Word2007/Part/FootnotesTest.php @@ -1,4 +1,5 @@ 'space', 'text' => '%1.', 'alignment' => Jc::START, - 'left' => 360, - 'hanging' => 360, - 'tabPos' => 360, + 'left' => Absolute::from('twip', 360), + 'hanging' => Absolute::from('twip', 360), + 'tabPos' => Absolute::from('twip', 360), 'font' => 'Arial', 'hint' => 'default', ), diff --git a/tests/PhpWord/Writer/Word2007/Part/SettingsTest.php b/tests/PhpWord/Writer/Word2007/Part/SettingsTest.php index 8a21e827cd..f75102716b 100644 --- a/tests/PhpWord/Writer/Word2007/Part/SettingsTest.php +++ b/tests/PhpWord/Writer/Word2007/Part/SettingsTest.php @@ -1,4 +1,5 @@ getSettings()->setHyphenationZone(100); + $phpWord->getSettings()->setHyphenationZone(Absolute::from('twip', 100)); $doc = TestHelperDOCX::getDocument($phpWord); diff --git a/tests/PhpWord/Writer/Word2007/Part/StylesTest.php b/tests/PhpWord/Writer/Word2007/Part/StylesTest.php index 91f371841d..6afe207448 100644 --- a/tests/PhpWord/Writer/Word2007/Part/StylesTest.php +++ b/tests/PhpWord/Writer/Word2007/Part/StylesTest.php @@ -1,4 +1,5 @@ Jc::BOTH); $pBase = array('basedOn' => 'Normal'); $pNew = array('basedOn' => 'Base Style', 'next' => 'Normal'); - $rStyle = array('size' => 20); - $tStyle = array('bgColor' => 'FF0000', 'cellMargin' => 120, 'borderSize' => 120); - $firstRowStyle = array('bgColor' => '0000FF', 'borderSize' => 120, 'borderColor' => '00FF00'); + $rStyle = array('size' => Absolute::from('pt', 20)); + $tStyle = array('bgColor' => new Colors\Hex('FF0000'), 'cellMargin' => Absolute::from('eop', 120), 'borderSize' => Absolute::from('eop', 120)); + $firstRowStyle = array('bgColor' => new Colors\Hex('0000FF'), 'borderSize' => Absolute::from('eop', 120), 'borderColor' => new Colors\Hex('00FF00')); $phpWord->setDefaultParagraphStyle($pStyle); $phpWord->addParagraphStyle('Base Style', $pBase); $phpWord->addParagraphStyle('New Style', $pNew); @@ -88,11 +91,11 @@ public function testFontStyleBasedOn() $childFont = new Font(); $childFont->setParagraph($baseParagraphStyle); - $childFont->setSize(16); + $childFont->setSize(Absolute::from('pt', 16)); $childFont = $phpWord->addFontStyle('ChildFontStyle', $childFont); $otherFont = new Font(); - $otherFont->setSize(20); + $otherFont->setSize(Absolute::from('pt', 20)); $otherFont = $phpWord->addFontStyle('OtherFontStyle', $otherFont); $doc = TestHelperDOCX::getDocument($phpWord); @@ -124,7 +127,7 @@ public function testFontStyleBasedOnOtherFontStyle() $styleGeneration = new Font(); $styleGeneration->setParagraph($styleGenerationP); - $styleGeneration->setSize(9.5); + $styleGeneration->setSize(Absolute::from('pt', 9.5)); $phpWord->addFontStyle('Generation', $styleGeneration); $styleGenerationEteinteP = new Paragraph(); @@ -132,7 +135,7 @@ public function testFontStyleBasedOnOtherFontStyle() $styleGenerationEteinte = new Font(); $styleGenerationEteinte->setParagraph($styleGenerationEteinteP); - $styleGenerationEteinte->setSize(8.5); + $styleGenerationEteinte->setSize(Absolute::from('pt', 8.5)); $phpWord->addFontStyle('GeneratEteinte', $styleGenerationEteinte); $doc = TestHelperDOCX::getDocument($phpWord); diff --git a/tests/PhpWord/Writer/Word2007/PartTest.php b/tests/PhpWord/Writer/Word2007/PartTest.php index 277f61e1be..7158047aca 100644 --- a/tests/PhpWord/Writer/Word2007/PartTest.php +++ b/tests/PhpWord/Writer/Word2007/PartTest.php @@ -1,4 +1,5 @@ setWidth(Absolute::from('twip', 10.3)); + $width = $cellStyle->getWidth(); + $this->assertInstanceOf(Absolute::class, $width); + $this->assertEquals($width->toFloat('twip'), 10.3); + + $cellStyle->setWidth(Absolute::from('twip', 0)); + $width = $cellStyle->getWidth(); + $this->assertInstanceOf(Absolute::class, $width); + $this->assertEquals($width->toInt('twip'), 0); + + $cellStyle->setWidth(new Percent(10)); + $width = $cellStyle->getWidth(); + $this->assertInstanceOf(Percent::class, $width); + $this->assertEquals($width->toInt(), 10); + + $cellStyle->setWidth(new Percent(0)); + $width = $cellStyle->getWidth(); + $this->assertInstanceOf(Percent::class, $width); + $this->assertEquals($width->toInt(), 0); + + $cellStyle->setWidth(new Auto()); + $width = $cellStyle->getWidth(); + $this->assertInstanceOf(Auto::class, $width); + } + + /** + * @covers \PhpOffice\PhpWord\Writer\Word2007\Style\Cell + * @expectedException \Exception + * @expectedExceptionMessage Unsupported width `class@anonymous + */ + public function testSetBadWidth() + { + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $table = $section->addTable(new Table()); + $row = $table->addRow(); + $cellStyle = new CellStyle(); + $cellStyle->setWidth(new class() extends Length { + public function isSpecified(): bool + { + return true; + } + }); + $row->addCell(null, $cellStyle); + + TestHelperDOCX::getDocument($phpWord, 'Word2007'); + } + + public function testWidth() + { + $widths = array( + array(Absolute::from('twip', 10.3), 'dxa', 10.3), + array(Absolute::from('twip', 0), 'nil', null), + array(new Percent(50), 'pct', '50%'), + array(new Percent(0), 'nil', null), + array(new Auto(), 'auto', null), + ); + + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $table = $section->addTable(new Table()); + $row = $table->addRow(); + foreach ($widths as $info) { + list($width, $expectedType, $expectedWidth) = $info; + $cellStyle = new CellStyle(); + $cellStyle->setWidth($width); + $row->addCell(null, $cellStyle); + } + + $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); + + $cellId = 0; + foreach ($widths as $info) { + $cellId += 1; + list($width, $expectedType, $expectedWidth) = $info; + $message = get_class($width) . ' expecting [' . $expectedType . ', ' . $expectedWidth . '] should be valid'; + $path = '/w:document/w:body/w:tbl/w:tr/w:tc[' . $cellId . ']/w:tcPr/w:tcW'; + $this->assertTrue($doc->elementExists($path), $message); + $this->assertEquals($expectedType, $doc->getElementAttribute($path, 'w:type'), $message); + $this->assertEquals($expectedWidth, $doc->getElementAttribute($path, 'w:w'), $message); + } + } + + public function testSetBgColor() + { + $colors = array( + new Hex('0a1b2c'), + new HighlightColor('yellow'), + new Rgb(134, 230, 9), + new ThemeColor('dk1'), + ); + + $cellStyle = new CellStyle(); + + foreach ($colors as $color) { + $cellStyle->setBgColor($color); + $bgColor = $cellStyle->getBgColor(); + $this->assertEquals($color->toHexOrName(), $bgColor->toHexOrName()); + } + } + + /** + * @expectedException \TypeError + * @expectedExceptionMessage Argument 1 passed to PhpOffice\PhpWord\Style\Cell::setBgColor() must be an instance of PhpOffice\PhpWord\Style\Colors\BasicColor, instance of PhpOffice\PhpWord\Style\Colors\SystemColor given + */ + public function testSetBgColorSystemColor() + { + $cellStyle = new CellStyle(); + $cellStyle->setBgColor(new SystemColor('window', new Hex('123456'))); + } + + public function testBgColor() + { + $colors = array( + new Hex('0a1b2c'), + new HighlightColor('yellow'), + new Rgb(134, 230, 9), + new ThemeColor('dk1'), + ); + + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $table = $section->addTable(new Table()); + $row = $table->addRow(); + foreach ($colors as $color) { + $cellStyle = new CellStyle(); + $cellStyle->setBgColor($color); + $row->addCell(null, $cellStyle); + } + + $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); + + $cellId = 0; + foreach ($colors as $color) { + $cellId += 1; + $message = get_class($color) . '(' . $color->toHexOrName() . ') should be valid'; + $path = '/w:document/w:body/w:tbl/w:tr/w:tc[' . $cellId . ']/w:tcPr/w:shd'; + $this->assertTrue($doc->elementExists($path), $message); + $this->assertEquals('clear', $doc->getElementAttribute($path, 'w:val'), $message); + $this->assertEquals($color->toHexOrName(), $doc->getElementAttribute($path, 'w:fill'), $message); + } + } +} diff --git a/tests/PhpWord/Writer/Word2007/Style/FontTest.php b/tests/PhpWord/Writer/Word2007/Style/FontTest.php index ccfffbfb01..431fc52f40 100644 --- a/tests/PhpWord/Writer/Word2007/Style/FontTest.php +++ b/tests/PhpWord/Writer/Word2007/Style/FontTest.php @@ -1,4 +1,5 @@ addSection(); - $section->addText('This text is lowered', array('position' => -20)); + $section->addText('This text is lowered', array('position' => Absolute::from('hpt', -20))); $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); $path = '/w:document/w:body/w:p/w:r/w:rPr/w:position'; diff --git a/tests/PhpWord/Writer/Word2007/Style/ImageTest.php b/tests/PhpWord/Writer/Word2007/Style/ImageTest.php index efa0a10534..120d35958e 100644 --- a/tests/PhpWord/Writer/Word2007/Style/ImageTest.php +++ b/tests/PhpWord/Writer/Word2007/Style/ImageTest.php @@ -1,4 +1,5 @@ Image::WRAP_INLINE, - 'wrapDistanceLeft' => 10, - 'wrapDistanceRight' => 20, - 'wrapDistanceTop' => 30, - 'wrapDistanceBottom' => 40, + 'wrapDistanceLeft' => Absolute::from('pt', 10), + 'wrapDistanceRight' => Absolute::from('pt', 20), + 'wrapDistanceTop' => Absolute::from('pt', 30), + 'wrapDistanceBottom' => Absolute::from('pt', 40), ); $phpWord = new \PhpOffice\PhpWord\PhpWord(); @@ -61,9 +63,9 @@ public function testWrapping() $this->assertTrue($doc->elementExists($path)); $style = $doc->getElement($path)->getAttribute('style'); $this->assertNotNull($style); - $this->assertContains('mso-wrap-distance-left:10pt;', $style); - $this->assertContains('mso-wrap-distance-right:20pt;', $style); - $this->assertContains('mso-wrap-distance-top:30pt;', $style); - $this->assertContains('mso-wrap-distance-bottom:40pt;', $style); + $this->assertContains('mso-wrap-distance-left:13px;', $style); + $this->assertContains('mso-wrap-distance-right:27px;', $style); + $this->assertContains('mso-wrap-distance-top:40px;', $style); + $this->assertContains('mso-wrap-distance-bottom:53px;', $style); } } diff --git a/tests/PhpWord/Writer/Word2007/Style/ParagraphTest.php b/tests/PhpWord/Writer/Word2007/Style/ParagraphTest.php index 843f98807f..6a72fa54c5 100644 --- a/tests/PhpWord/Writer/Word2007/Style/ParagraphTest.php +++ b/tests/PhpWord/Writer/Word2007/Style/ParagraphTest.php @@ -1,4 +1,5 @@ addParagraphStyle('testStyle', array('indent' => '10')); + $phpWord->addParagraphStyle('testStyle', array('indent' => Absolute::from('twip', 10))); $section = $phpWord->addSection(); $section->addText('test', null, array('numStyle' => 'testStyle', 'numLevel' => '1')); $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); @@ -55,7 +60,7 @@ public function testLineSpacingExact() { $phpWord = new \PhpOffice\PhpWord\PhpWord(); $section = $phpWord->addSection(); - $section->addText('test', null, array('spacing' => 240, 'spacingLineRule' => 'exact')); + $section->addText('test', null, array('spacing' => Absolute::from('twip', 240), 'spacingLineRule' => 'exact')); $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); $path = '/w:document/w:body/w:p/w:pPr/w:spacing'; @@ -68,7 +73,7 @@ public function testLineSpacingAuto() { $phpWord = new \PhpOffice\PhpWord\PhpWord(); $section = $phpWord->addSection(); - $section->addText('test', null, array('spacing' => 240, 'spacingLineRule' => 'auto')); + $section->addText('test', null, array('spacing' => Absolute::from('twip', 240), 'spacingLineRule' => 'auto')); $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); $path = '/w:document/w:body/w:p/w:pPr/w:spacing'; @@ -90,4 +95,78 @@ public function testSuppressAutoHyphens() $path = '/w:document/w:body/w:p/w:pPr/w:suppressAutoHyphens'; $this->assertTrue($doc->elementExists($path)); } + + public function testDefaultNoBorders() + { + $paragraphStyle = new ParagraphStyle(); + + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $section->addText('test', null, $paragraphStyle); + $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); + + $path = '/w:document/w:body/w:p/w:pPr'; + $this->assertTrue($doc->elementExists($path)); + $this->assertFalse($doc->elementExists("$path/w:pBdr")); + } + + public function testDefaultBorders() + { + $paragraphStyle = new ParagraphStyle(); + $paragraphStyle->setBorders(new BorderSide(Absolute::from('pt', 1))); + + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $section->addText('test', null, $paragraphStyle); + $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); + + $path = '/w:document/w:body/w:p/w:pPr/w:pBdr'; + $this->assertTrue($doc->elementExists($path)); + $sides = array( + 'top', + 'bottom', + 'left', + 'right', + ); + foreach ($sides as $side) { + $sidePath = "$path/w:$side"; + $this->assertTrue($doc->elementExists($sidePath)); + $this->assertEquals('single', $doc->getElementAttribute($sidePath, 'w:val'), "$side was not set properly"); + $this->assertEquals('8', $doc->getElementAttribute($sidePath, 'w:sz'), "$side was not set properly"); + $this->assertEquals('auto', $doc->getElementAttribute($sidePath, 'w:color'), "$side was not set properly"); + $this->assertEquals('false', $doc->getElementAttribute($sidePath, 'w:shadow'), "$side was not set properly"); + } + } + + public function testBorders() + { + $paragraphStyle = new ParagraphStyle(); + $paragraphStyle->setBorders(new BorderSide( + Absolute::from('pt', 1), + new Rgb(255, 0, 0), + new BorderStyle('double') + )); + + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $section->addText('test', null, $paragraphStyle); + $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); + + $path = '/w:document/w:body/w:p/w:pPr/w:pBdr'; + $this->assertTrue($doc->elementExists($path)); + $sides = array( + 'top', + 'bottom', + 'left', + 'right', + ); + foreach ($sides as $side) { + $sidePath = "$path/w:$side"; + $this->assertTrue($doc->elementExists($sidePath)); + $this->assertEquals('double', $doc->getElementAttribute($sidePath, 'w:val'), "$side was not set properly"); + $this->assertEquals('8', $doc->getElementAttribute($sidePath, 'w:sz'), "$side was not set properly"); + $this->assertEquals('FF0000', $doc->getElementAttribute($sidePath, 'w:color'), "$side was not set properly"); + $this->assertEquals('false', $doc->getElementAttribute($sidePath, 'w:shadow'), "$side was not set properly"); + } + } } diff --git a/tests/PhpWord/Writer/Word2007/Style/TableTest.php b/tests/PhpWord/Writer/Word2007/Style/TableTest.php index 8e5cb63415..4792d9aa15 100644 --- a/tests/PhpWord/Writer/Word2007/Style/TableTest.php +++ b/tests/PhpWord/Writer/Word2007/Style/TableTest.php @@ -1,4 +1,5 @@ assertEquals(Table::LAYOUT_FIXED, $doc->getElementAttribute($path, 'w:type')); } + /** + * Test write styles + * @expectedException \Exception + * @expectedExceptionMessage Unsupported width `class@anonymous + */ + public function testWidths() + { + $widths = array( + array(new Auto(), 'auto', null), + array(Absolute::from('twip', 54), 'dxa', 54), + array(new Absolute(0), 'nil', ''), + array(new Percent(50), 'pct', 50), + + // Invalid class must be last + array(new class() extends Length { + public function isSpecified(): bool + { + return true; + } + }, null, null), + ); + foreach ($widths as $info) { + list($width, $expectedType, $expectedWidth) = $info; + + $tableStyle = new Table(); + $tableStyle->setWidth($width); + + $phpWord = new \PhpOffice\PhpWord\PhpWord(); + $section = $phpWord->addSection(); + $table = $section->addTable($tableStyle); + $table->addRow(); + + $doc = TestHelperDOCX::getDocument($phpWord, 'Word2007'); + + $path = '/w:document/w:body/w:tbl/w:tblPr/w:tblW'; + $this->assertTrue($doc->elementExists($path)); + $this->assertEquals($expectedType, $doc->getElementAttribute($path, 'w:type')); + $this->assertEquals($expectedWidth, $doc->getElementAttribute($path, 'w:w')); + } + } + /** * Test write styles */ public function testCellSpacing() { $tableStyle = new Table(); - $tableStyle->setCellSpacing(10.3); + $tableStyle->setCellSpacing(Absolute::from('twip', 10.3)); $phpWord = new \PhpOffice\PhpWord\PhpWord(); $section = $phpWord->addSection(); @@ -77,7 +121,6 @@ public function testCellSpacing() $path = '/w:document/w:body/w:tbl/w:tblPr/w:tblCellSpacing'; $this->assertTrue($doc->elementExists($path)); $this->assertEquals(10.3, $doc->getElementAttribute($path, 'w:w')); - $this->assertEquals(TblWidth::TWIP, $doc->getElementAttribute($path, 'w:type')); } /** @@ -86,16 +129,16 @@ public function testCellSpacing() public function testTablePosition() { $tablePosition = array( - 'leftFromText' => 10, - 'rightFromText' => 20, - 'topFromText' => 30, - 'bottomFromText' => 40, + 'leftFromText' => Absolute::from('twip', 10), + 'rightFromText' => Absolute::from('twip', 20), + 'topFromText' => Absolute::from('twip', 30), + 'bottomFromText' => Absolute::from('twip', 40), 'vertAnchor' => TablePosition::VANCHOR_PAGE, 'horzAnchor' => TablePosition::HANCHOR_MARGIN, 'tblpXSpec' => TablePosition::XALIGN_CENTER, - 'tblpX' => 50, + 'tblpX' => Absolute::from('twip', 50), 'tblpYSpec' => TablePosition::YALIGN_TOP, - 'tblpY' => 60, + 'tblpY' => Absolute::from('twip', 60), ); $tableStyle = new Table(); $tableStyle->setPosition($tablePosition); @@ -124,10 +167,9 @@ public function testTablePosition() public function testIndent() { $value = 100; - $type = TblWidth::TWIP; $tableStyle = new Table(); - $tableStyle->setIndent(new TblWidthComplexType($value, $type)); + $tableStyle->setIndent(Absolute::from('twip', $value)); $phpWord = new \PhpOffice\PhpWord\PhpWord(); $section = $phpWord->addSection(); @@ -139,10 +181,10 @@ public function testIndent() $path = '/w:document/w:body/w:tbl/w:tblPr/w:tblInd'; $this->assertTrue($doc->elementExists($path)); $this->assertSame($value, (int) $doc->getElementAttribute($path, 'w:w')); - $this->assertSame($type, $doc->getElementAttribute($path, 'w:type')); + $this->assertSame('dxa', $doc->getElementAttribute($path, 'w:type')); } - public function testRigthToLeft() + public function testRightToLeft() { $tableStyle = new Table(); $tableStyle->setBidiVisual(true); diff --git a/tests/PhpWord/Writer/Word2007/StyleTest.php b/tests/PhpWord/Writer/Word2007/StyleTest.php index 48cff8713a..650eaf2919 100644 --- a/tests/PhpWord/Writer/Word2007/StyleTest.php +++ b/tests/PhpWord/Writer/Word2007/StyleTest.php @@ -1,4 +1,5 @@ addFontStyle('Font', array('size' => 11)); + $phpWord->addFontStyle('Font', array('size' => Absolute::from('pt', 11))); $phpWord->addParagraphStyle('Paragraph', array('alignment' => Jc::CENTER)); $section = $phpWord->addSection(); $section->addText('Test 1', 'Font', 'Paragraph'); diff --git a/tests/PhpWord/_files/broken-phpword.ini b/tests/PhpWord/_files/broken-phpword.ini new file mode 100644 index 0000000000..c16f404428 --- /dev/null +++ b/tests/PhpWord/_files/broken-phpword.ini @@ -0,0 +1,17 @@ +; Default config file for PHPWord +; Copy this file into phpword.ini and use Settings::loadConfig to load + +[General] + +compatibility = true +zipClass = ZipArchive +pdfRendererName = DomPDF +pdfRendererPath = +; tempDir = "C:\PhpWordTemp" +outputEscapingEnabled = false + +[Font] + +defaultFontName = Arial +defaultFontSize = 10 +defaultFontClr = blue diff --git a/tests/PhpWord/_includes/AbstractTestReader.php b/tests/PhpWord/_includes/AbstractTestReader.php index d9097d717f..9888ad9d59 100644 --- a/tests/PhpWord/_includes/AbstractTestReader.php +++ b/tests/PhpWord/_includes/AbstractTestReader.php @@ -1,4 +1,5 @@