array ('fo:line-height', 'paragraph', true), 'line-height-at-least' => array ('style:line-height-at-least', 'paragraph', true), 'line-spacing' => array ('style:line-spacing', 'paragraph', true), 'font-independent-line-spacing' => array ('style:font-independent-line-spacing', 'paragraph', true), 'text-align' => array ('fo:text-align', 'paragraph', true), 'text-align-last' => array ('fo:text-align-last', 'paragraph', true), 'justify-single-word' => array ('style:justify-single-word', 'paragraph', true), 'keep-together' => array ('fo:keep-together', 'paragraph', true), 'widows' => array ('fo:widows', 'paragraph', true), 'orphans' => array ('fo:orphans', 'paragraph', true), 'tab-stop-distance' => array ('style:tab-stop-distance', 'paragraph', true), 'hyphenation-keep' => array ('fo:hyphenation-keep', 'paragraph', true), 'hyphenation-ladder-count' => array ('fo:hyphenation-ladder-count', 'paragraph', true), 'register-true' => array ('style:register-true', 'paragraph', true), 'text-indent' => array ('fo:text-indent', 'paragraph', true), 'auto-text-indent' => array ('style:auto-text-indent', 'paragraph', true), 'margin' => array ('fo:margin', 'paragraph', true), 'margin-top' => array ('fo:margin-top', 'paragraph', true), 'margin-right' => array ('fo:margin-right', 'paragraph', true), 'margin-bottom' => array ('fo:margin-bottom', 'paragraph', true), 'margin-left' => array ('fo:margin-left', 'paragraph', true), 'break-before' => array ('fo:break-before', 'paragraph', true), 'break-after' => array ('fo:break-after', 'paragraph', true), 'background-color' => array ('fo:background-color', 'paragraph', true), 'border' => array ('fo:border', 'paragraph', true), 'border-top' => array ('fo:border-top', 'paragraph', true), 'border-right' => array ('fo:border-right', 'paragraph', true), 'border-bottom' => array ('fo:border-bottom', 'paragraph', true), 'border-left' => array ('fo:border-left', 'paragraph', true), 'border-line-width' => array ('style:border-line-width', 'paragraph', true), 'border-line-width-top' => array ('style:border-line-width-top', 'paragraph', true), 'border-line-width-bottom' => array ('style:border-line-width-bottom', 'paragraph', true), 'border-line-width-left' => array ('style:border-line-width-left', 'paragraph', true), 'border-line-width-right' => array ('style:border-line-width-right', 'paragraph', true), 'join-border' => array ('style:join-border', 'paragraph', true), 'padding' => array ('fo:padding', 'paragraph', true), 'padding-top' => array ('fo:padding-top', 'paragraph', true), 'padding-bottom' => array ('fo:padding-bottom', 'paragraph', true), 'padding-left' => array ('fo:padding-left', 'paragraph', true), 'padding-right' => array ('fo:padding-right', 'paragraph', true), 'shadow' => array ('style:shadow', 'paragraph', true), 'keep-with-next' => array ('fo:keep-with-next', 'paragraph', true), 'number-lines' => array ('text:number-lines', 'paragraph', true), 'line-number' => array ('text:line-number', 'paragraph', true), 'text-autospace' => array ('style:text-autospace', 'paragraph', true), 'punctuation-wrap' => array ('style:punctuation-wrap', 'paragraph', true), 'line-break' => array ('style:line-break', 'paragraph', true), 'vertical-align' => array ('style:vertical-align', 'paragraph', true), 'writing-mode' => array ('style:writing-mode', 'paragraph', true), 'writing-mode-automatic' => array ('style:writing-mode-automatic', 'paragraph', true), 'snap-to-layout-grid' => array ('style:snap-to-layout-grid', 'paragraph', true), 'page-number' => array ('style:page-number', 'paragraph', true), 'background-transparency' => array ('style:background-transparency', 'paragraph', true), ); // Additional fields for child element tab-stop. static $tab_stop_fields = array( 'style-position' => array ('style:position', 'tab-stop', true), 'style-type' => array ('style:type', 'tab-stop', true), 'style-leader-type' => array ('style:leader-type', 'tab-stop', true), 'style-leader-style' => array ('style:leader-style', 'tab-stop', true), 'style-leader-width' => array ('style:leader-width', 'tab-stop', true), 'style-leader-color' => array ('style:leader-color', 'tab-stop', true), 'style-leader-text' => array ('style:leader-text', 'tab-stop', true), ); protected $style_properties = array(); protected $text_properties = array(); protected $tab_stops = array(); /** * Constructor. */ public function __construct() { parent::__construct(); } /** * Set style properties by importing values from a properties array. * Properties might be disabled by setting them in $disabled. * The style must have been previously created. * * @param $properties Properties to be imported * @param $disabled Properties to be ignored */ public function importProperties($properties, $disabled=array()) { foreach ($properties as $property => $value) { if ($disabled [$property] == 0) { $this->setProperty($property, $value); } } } /** * Check if a style is a common style. * * @return bool Is common style */ public function mustBeCommonStyle() { return false; } /** * Get the style family of a style. * * @return string Style family */ static public function getFamily() { return 'paragraph'; } /** * Set a property. * * @param $property The name of the property to set * @param $value New value to set */ public function setProperty($property, $value) { $style_fields = ODTStyleStyle::getStyleProperties (); if (array_key_exists ($property, $style_fields)) { $this->setPropertyInternal ($property, $style_fields [$property][0], $value, $style_fields [$property][1], $this->style_properties); return; } // FIXME: currently with setProperty there always will only be one tab-stop. // Maybe in the future supply a function "add tab stop" or something. if (array_key_exists ($property, self::$tab_stop_fields)) { if (!isset($this->tab_stops [0])) { $this->tab_stops [0] = array(); } $this->setPropertyInternal ($property, self::$tab_stop_fields [$property][0], $value, self::$tab_stop_fields [$property][1], $this->tab_stops[0]); return; } // Compare with paragraph fields before text fields first! // So, paragraph properties get precedence. if (array_key_exists ($property, self::$paragraph_fields)) { $this->setPropertyInternal ($property, self::$paragraph_fields [$property][0], $value, self::$paragraph_fields [$property][1]); return; } $text_fields = ODTTextStyle::getTextProperties (); if (array_key_exists ($property, $text_fields)) { $this->setPropertyInternal ($property, $text_fields [$property][0], $value, $text_fields [$property][1], $this->text_properties); return; } } /** * Get the value of a property. * * @param $property The property name * @return string The current value of the property */ public function getProperty($property) { $style_fields = ODTStyleStyle::getStyleProperties (); if (array_key_exists ($property, $style_fields)) { return $this->style_properties [$property]['value']; } $paragraph_fields = self::$paragraph_fields; if (array_key_exists ($property, $paragraph_fields)) { return parent::getProperty($property); } $text_fields = ODTTextStyle::getTextProperties (); if (array_key_exists ($property, $text_fields)) { return $this->text_properties [$property]['value']; } return parent::getProperty($property); } /** * Create new style by importing ODT style definition. * * @param $xmlCode Style definition in ODT XML format * @return ODTStyle New specific style */ static public function importODTStyle($xmlCode) { $style = new ODTParagraphStyle(); $attrs = 0; $open = XMLUtil::getElementOpenTag('style:style', $xmlCode); if (!empty($open)) { $attrs += $style->importODTStyleInternal(ODTStyleStyle::getStyleProperties (), $open, $style->style_properties); } else { $open = XMLUtil::getElementOpenTag('style:default-style', $xmlCode); if (!empty($open)) { $style->setDefault(true); $attrs += $style->importODTStyleInternal(ODTStyleStyle::getStyleProperties (), $open, $style->style_properties); } } $open = XMLUtil::getElementOpenTag('style:paragraph-properties', $xmlCode); if (!empty($open)) { $attrs += $style->importODTStyleInternal(self::$paragraph_fields, $xmlCode); } $open = XMLUtil::getElementOpenTag('style:text-properties', $xmlCode); if (!empty($open)) { $attrs += $style->importODTStyleInternal(ODTTextStyle::getTextProperties (), $open, $style->text_properties); } // Get all tab-stops. $tabs = XMLUtil::getElementContent('style:tab-stops', $xmlCode); if (isset($tabs)) { $max = strlen($tabs); $pos = 0; $index = 0; $tab = XMLUtil::getElement('style:tab-stop', $tabs, $end); $pos = $end; while (isset($tab)) { $style->tab_stops [$index] = array(); $attrs += $style->importODTStyleInternal(self::$tab_stop_fields, $tab, $style->tab_stops [$index]); $index++; $tab = XMLUtil::getElement('style:tab-stop', substr ($tabs, $pos), $end); $pos += $end; } } // If style has no meaningfull content then throw it away if ( $attrs == 0 ) { return NULL; } return $style; } static public function getParagraphProperties () { return self::$paragraph_fields; } /** * Encode current style values in a string and return it. * * @return string ODT XML encoded style */ public function toString() { $style = ''; $para_props = ''; $text_props = ''; $tab_stops_props = ''; // Get style contents foreach ($this->style_properties as $property => $items) { if ($items ['odt_property'] != 'style:family') { $style .= $items ['odt_property'].'="'.$items ['value'].'" '; } } $style .= 'style:family="'.$this->getFamily().'" '; // Get paragraph properties ODT properties foreach ($this->properties as $property => $items) { $para_props .= $items ['odt_property'].'="'.$items ['value'].'" '; } // Get text properties foreach ($this->text_properties as $property => $items) { $text_props .= $items ['odt_property'].'="'.$items ['value'].'" '; } // Get tab-stops properties for ($index = 0 ; $index < count($this->tab_stops) ; $index++) { $tab_stops_props .= 'tab_stops[$index] as $property => $items) { $tab_stops_props .= $items ['odt_property'].'="'.$items ['value'].'" '; } $tab_stops_props .= '/>'; } // Build style. if (!$this->isDefault()) { $style = '\n"; } else { $style = '\n"; } if (!empty($para_props) || !empty($tab_stops_props)) { if (empty($tab_stops_props)) { $style .= '\n"; } else { $style .= '\n"; $style .= ''."\n"; $style .= $tab_stops_props."\n"; $style .= ''."\n"; $style .= ''."\n"; } } if (!empty($text_props)) { $style .= '\n"; } if (!$this->isDefault()) { $style .= ''."\n"; } else { $style .= ''."\n"; } return $style; } /** * This function creates a paragraph style using the style as set in the assoziative array $properties. * The parameters in the array should be named as the CSS property names e.g. 'color' or 'background-color'. * Properties which shall not be used in the style can be disabled by setting the value in disabled_props * to 1 e.g. $disabled_props ['color'] = 1 would block the usage of the color property. * * The currently supported properties are: * background-color, color, font-style, font-weight, font-size, border, font-family, font-variant, letter-spacing, * vertical-align, line-height, background-image * * The function returns the name of the new style or NULL if all relevant properties are empty. * * @author LarsDW223 * @param $properties * @param null $disabled_props * @return ODTParagraphStyle or NULL */ public static function createParagraphStyle(array $properties, array $disabled_props = NULL, ODTDocument $doc=NULL){ // Convert 'text-decoration'. if ( $properties ['text-decoration'] == 'line-through' ) { $properties ['text-line-through-style'] = 'solid'; } if ( $properties ['text-decoration'] == 'underline' ) { $properties ['text-underline-style'] = 'solid'; } if ( $properties ['text-decoration'] == 'overline' ) { $properties ['text-overline-style'] = 'solid'; } // If the property 'vertical-align' has the value 'sub' or 'super' // then for ODT it needs to be converted to the corresponding 'text-position' property. // Replace sub and super with text-position. $valign = $properties ['vertical-align']; if (!empty($valign)) { if ( $valign == 'sub' ) { $properties ['text-position'] = '-33% 100%'; unset($properties ['vertical-align']); } elseif ( $valign == 'super' ) { $properties ['text-position'] = '33% 100%'; unset($properties ['vertical-align']); } } // Separate country from language $lang = $properties ['lang']; $country = $properties ['country']; if ( !empty($lang) ) { $parts = preg_split ('/-/', $lang); $lang = $parts [0]; $country = $parts [1]; $properties ['country'] = trim($country); $properties ['lang'] = trim($lang); } if (!empty($properties ['country'])) { if (empty($properties ['country-asian'])) { $properties ['country-asian'] = $properties ['country']; } if (empty($properties ['country-complex'])) { $properties ['country-complex'] = $properties ['country']; } } // Always set 'auto-text-indent = false' if 'text-indent' is set. if (!empty($properties ['text-indent'])) { $properties ['auto-text-indent'] = 'false'; $length = strlen ($properties ['text-indent']); if ( $length > 0 && $properties ['text-indent'] [$length-1] == '%' && isset($doc) ) { // Percentage value needs to be converted to absolute value. // ODT standard says that percentage value should work if used in a common style. // This did not work with LibreOffice 4.4.3.2. $value = trim ($properties ['text-indent'], '%'); $properties ['text-indent'] = $doc->getAbsWidthMindMargins ($value).'cm'; } } // Eventually create parent for font-size $save = $disabled_props ['font-size']; $odt_fo_size = ''; if ( empty ($disabled_props ['font-size']) ) { $odt_fo_size = $properties ['font-size']; } $parent = ''; $length = strlen ($odt_fo_size); if ( $length > 0 && $odt_fo_size [$length-1] == '%' && isset($doc)) { // A font-size in percent is only supported in common style definitions, not in automatic // styles. Create a common style and set it as parent for this automatic style. $name = 'Size'.trim ($odt_fo_size, '%').'pc'; $style_obj = ODTTextStyle::createSizeOnlyTextStyle ($name, $odt_fo_size); $doc->addStyle($style_obj); $parent = $style_obj->getProperty('style-name'); if (!empty($parent)) { $properties ['style-parent'] = $parent; } } // Create style name (if not given). $style_name = $properties ['style-name']; if ( empty($style_name) ) { $style_name = self::getNewStylename ('Paragraph'); $properties ['style-name'] = $style_name; } // FIXME: fix missing tab stop handling... //case 'tab-stop': // $tab .= $params [$property]['name'].'="'.$value.'" '; // $tab .= self::writeExtensionNames ($params [$property]['name'], $value); // break; // Create empty paragraph style. $object = new ODTParagraphStyle(); if (!isset($object)) { return NULL; } // Import our properties $object->importProperties($properties, $disabled_props); // Restore $disabled_props $disabled_props ['font-size'] = $save; return $object; } /** * Simple helper function for creating a paragrapg style for a pagebreak. * * @author LarsDW223 * * @param string $parent Name of the parent style to set * @param string $before Pagebreak before or after? * @return ODTParagraphStyle */ public static function createPagebreakStyle($style_name, $parent=NULL,$before=true) { $properties = array(); $properties ['style-name'] = $style_name; if ( !empty($parent) ) { $properties ['style-parent'] = $parent; } if ($before == true ) { $properties ['break-before'] = 'page'; } else { $properties ['break-after'] = 'page'; } return self::createParagraphStyle($properties); } /** * Set a property. * * @param $property The name of the property to set * @param $value New value to set */ public static function copyLayoutProperties(ODTParagraphStyle $source, ODTParagraphStyle $dest, array $disabled=NULL) { // DO NOT COPY STYLE FIELDS/PROPERTIES // Copy $tab_stop_fields foreach (self::$tab_stop_fields as $property => $fields) { $value = $source->getProperty($property); if (isset($value) && $disabled [$property] == 0) { $dest -> setProperty($property, $value); } } // Copy $paragraph_fields foreach (self::$paragraph_fields as $property => $fields) { $value = $source->getProperty($property); if (isset($value) && $disabled [$property] == 0) { $dest -> setProperty($property, $value); } } // Copy $text_fields $text_fields = ODTTextStyle::getTextProperties (); foreach ($text_fields as $property => $fields) { $value = $source->getProperty($property); if (isset($value) && $disabled [$property] == 0) { $dest -> setProperty($property, $value); } } } }