mirror of
				https://github.com/ezyang/htmlpurifier.git
				synced 2025-10-24 18:16:19 +02:00 
			
		
		
		
	Merged r1746: Length and UnitConverter implementation.
git-svn-id: http://htmlpurifier.org/svnroot/htmlpurifier/branches/php4@1783 48356398-32a2-884e-a903-53898d9a118a
This commit is contained in:
		
							
								
								
									
										168
									
								
								library/HTMLPurifier/UnitConverter.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								library/HTMLPurifier/UnitConverter.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,168 @@ | ||||
| <?php | ||||
|  | ||||
| require_once 'HTMLPurifier/Length.php'; | ||||
|  | ||||
| /** | ||||
|  * Class for converting between different unit-lengths as specified by | ||||
|  * CSS. | ||||
|  */ | ||||
| class HTMLPurifier_UnitConverter | ||||
| { | ||||
|      | ||||
|     // For documentation purposes: these are magical numbers | ||||
|     // const ENGLISH = 1; | ||||
|     // const METRIC = 2; | ||||
|      | ||||
|     /** | ||||
|      * Minimum bcmath precision for output. | ||||
|      */ | ||||
|     var $outputPrecision; | ||||
|      | ||||
|     /** | ||||
|      * Bcmath precision for internal calculations. | ||||
|      */ | ||||
|     var $internalPrecision; | ||||
|      | ||||
|     function HTMLPurifier_UnitConverter($output_precision = 4, $internal_precision = 10) { | ||||
|         $this->outputPrecision = $output_precision; | ||||
|         $this->internalPrecision = $internal_precision; | ||||
|     } | ||||
|      | ||||
|     /** | ||||
|      * Converts a length object of one unit into another unit. | ||||
|      * @note | ||||
|      *      About precision: This conversion function pays very special | ||||
|      *      attention to the incoming precision of values and attempts | ||||
|      *      to maintain a number of significant figure. Results are | ||||
|      *      fairly accurate up to nine digits. Some caveats: | ||||
|      *          - If a number is zero-padded as a result of this significant | ||||
|      *            figure tracking, the zeroes will be eliminated. | ||||
|      *          - If a number contains less than four sigfigs ($outputPrecision) | ||||
|      *            and this causes some decimals to be excluded, those | ||||
|      *            decimals will be added on. | ||||
|      *          - Significant digits will be ignored for quantities greater | ||||
|      *            than one. This is a limitation of BCMath and I don't | ||||
|      *            feel like coding around it. | ||||
|      */ | ||||
|     function convert($length, $to_unit) { | ||||
|          | ||||
|         /** | ||||
|          * Units information array. Units are grouped into measuring systems | ||||
|          * (English, Metric), and are assigned an integer representing | ||||
|          * the conversion factor between that unit and the smallest unit in | ||||
|          * the system. Numeric indexes are actually magical constants that | ||||
|          * encode conversion data from one system to the next, with a O(n^2) | ||||
|          * constraint on memory (this is generally not a problem, since | ||||
|          * the number of measuring systems is small.) | ||||
|          */ | ||||
|         static $units = array( | ||||
|             1 => array( | ||||
|                 'pt' => 1, | ||||
|                 'pc' => 12, | ||||
|                 'in' => 72, | ||||
|                 2 => array('pt', '0.352777778', 'mm'), | ||||
|             ), | ||||
|             2 => array( | ||||
|                 'mm' => 1, | ||||
|                 'cm' => 10, | ||||
|                 1 => array('mm', '2.83464567', 'pt'), | ||||
|             ), | ||||
|         ); | ||||
|          | ||||
|         if ($length->n === '0' || $length->unit === false) { | ||||
|             return new HTMLPurifier_Length('0', $unit); | ||||
|         } | ||||
|          | ||||
|         $state = $dest = false; | ||||
|         foreach ($units as $k => $x) { | ||||
|             if (isset($x[$length->unit])) $state = $k; | ||||
|             if (isset($x[$to_unit])) $dest_state = $k; | ||||
|         } | ||||
|         if (!$state || !$dest_state) return false; | ||||
|          | ||||
|         $n    = $length->n; | ||||
|         $unit = $length->unit; | ||||
|          | ||||
|         // Some calculations about the initial precision of the number; | ||||
|         // this will be useful when we need to do final rounding. | ||||
|         $log = (int) floor(log($n, 10)); | ||||
|         if (strpos($n, '.') === false) { | ||||
|             $sigfigs = strlen(trim($n, '0+-')); | ||||
|         } else { | ||||
|             $sigfigs = strlen(ltrim($n, '0+-')) - 1; // eliminate extra decimal character | ||||
|         } | ||||
|         if ($sigfigs < $this->outputPrecision) $sigfigs = $this->outputPrecision; | ||||
|          | ||||
|         // BCMath's internal precision deals only with decimals. Use | ||||
|         // our default if the initial number has no decimals, or increase | ||||
|         // it by how ever many decimals, thus, the number of guard digits | ||||
|         // will always be greater than or equal to internalPrecision. | ||||
|         $cp = ($log < 0) ? $this->internalPrecision - $log : $this->internalPrecision; // internal precision | ||||
|          | ||||
|         for ($i = 0; $i < 2; $i++) { | ||||
|              | ||||
|             // Determine what unit IN THIS SYSTEM we need to convert to | ||||
|             if ($dest_state === $state) { | ||||
|                 // Simple conversion | ||||
|                 $dest_unit = $to_unit; | ||||
|             } else { | ||||
|                 // Convert to the smallest unit, pending a system shift | ||||
|                 $dest_unit = $units[$state][$dest_state][0]; | ||||
|             } | ||||
|              | ||||
|             // Do the conversion if necessary | ||||
|             if ($dest_unit !== $unit) { | ||||
|                 $factor = bcdiv($units[$state][$unit], $units[$state][$dest_unit], $cp); | ||||
|                 $n = bcmul($n, $factor, $cp); | ||||
|                 $unit = $dest_unit; | ||||
|             } | ||||
|              | ||||
|             // Output was zero, so bail out early | ||||
|             if ($n === '') { | ||||
|                 $n = '0'; | ||||
|                 $unit = $to_unit; | ||||
|                 break; | ||||
|             } | ||||
|              | ||||
|             // It was a simple conversion, so bail out | ||||
|             if ($dest_state === $state) { | ||||
|                 break; | ||||
|             } | ||||
|              | ||||
|             if ($i !== 0) { | ||||
|                 // Conversion failed! Apparently, the system we forwarded | ||||
|                 // to didn't have this unit. This should never happen! | ||||
|                 return false; | ||||
|             } | ||||
|              | ||||
|             // Pre-condition: $i == 0 | ||||
|              | ||||
|             // Perform conversion to next system of units | ||||
|             $n = bcmul($n, $units[$state][$dest_state][1], $cp); | ||||
|             $unit = $units[$state][$dest_state][2]; | ||||
|             $state = $dest_state; | ||||
|              | ||||
|             // One more loop around to convert the unit in the new system. | ||||
|              | ||||
|         } | ||||
|          | ||||
|         // Post-condition: $unit == $to_unit | ||||
|         if ($unit !== $to_unit) return false; | ||||
|          | ||||
|         // Calculate how many decimals we need ($rp) | ||||
|         // Calculations will always be carried to the decimal; this is | ||||
|         // a limitation with BC (we can't set the scale to be negative) | ||||
|         $new_log = (int) floor(log($n, 10)); | ||||
|          | ||||
|         $rp = $sigfigs - $new_log - $log - 1; | ||||
|         if ($rp < 0) $rp = 0; | ||||
|          | ||||
|         $n = bcadd($n, '0.' .  str_repeat('0', $rp) . '5', $rp + 1); | ||||
|         $n = bcdiv($n, '1', $rp); | ||||
|         if (strpos($n, '.') !== false) $n = rtrim($n, '0'); | ||||
|         $n = rtrim($n, '.'); | ||||
|          | ||||
|         return new HTMLPurifier_Length($n, $unit); | ||||
|     } | ||||
|      | ||||
| } | ||||
		Reference in New Issue
	
	Block a user