objectEncoders = [ $ns . 'Items\\Regexp' => [$this, 'encodeRegexp'], $ns . 'JavaScript\\Code' => [$this, 'encodeCode'], $ns . 'JavaScript\\ConfigValue' => [$this, 'encodeConfigValue'], $ns . 'JavaScript\\Dictionary' => [$this, 'encodeDictionary'] ]; $this->typeEncoders = [ 'array' => [$this, 'encodeArray'], 'boolean' => [$this, 'encodeBoolean'], 'double' => [$this, 'encodeScalar'], 'integer' => [$this, 'encodeScalar'], 'object' => [$this, 'encodeObject'], 'string' => [$this, 'encodeScalar'] ]; } /** * Encode a value into JavaScript * * @param mixed $value * @return string */ public function encode($value) { $type = gettype($value); if (!isset($this->typeEncoders[$type])) { throw new RuntimeException('Cannot encode ' . $type . ' value'); } return $this->typeEncoders[$type]($value); } /** * Encode an array to JavaScript * * @param array $array * @return string */ protected function encodeArray(array $array) { return ($this->isIndexedArray($array)) ? $this->encodeIndexedArray($array) : $this->encodeAssociativeArray($array); } /** * Encode an associative array to JavaScript * * @param array $array * @param bool $preserveNames * @return string */ protected function encodeAssociativeArray(array $array, $preserveNames = false) { ksort($array); $src = '{'; $sep = ''; foreach ($array as $k => $v) { $src .= $sep . $this->encodePropertyName("$k", $preserveNames) . ':' . $this->encode($v); $sep = ','; } $src .= '}'; return $src; } /** * Encode a boolean value into JavaScript * * @param bool $value * @return string */ protected function encodeBoolean($value) { return ($value) ? '!0' : '!1'; } /** * Encode a Code instance into JavaScript * * @param Code $code * @return string */ protected function encodeCode(Code $code) { return (string) $code; } /** * Encode a ConfigValue instance into JavaScript * * @param ConfigValue $configValue * @return string */ protected function encodeConfigValue(ConfigValue $configValue) { return ($configValue->isDeduplicated()) ? $configValue->getVarName() : $this->encode($configValue->getValue()); } /** * Encode a Dictionary object into a JavaScript object * * @param Dictionary $dict * @return string */ protected function encodeDictionary(Dictionary $dict) { return $this->encodeAssociativeArray($dict->getArrayCopy(), true); } /** * Encode an indexed array to JavaScript * * @param array $array * @return string */ protected function encodeIndexedArray(array $array) { return '[' . implode(',', array_map([$this, 'encode'], $array)) . ']'; } /** * Encode an object into JavaScript * * @param object $object * @return string */ protected function encodeObject($object) { foreach ($this->objectEncoders as $className => $callback) { if ($object instanceof $className) { return $callback($object); } } throw new RuntimeException('Cannot encode instance of ' . get_class($object)); } /** * Encode an object property name into JavaScript * * @param string $name * @param bool $preserveNames * @return string */ protected function encodePropertyName($name, $preserveNames) { return ($preserveNames || !$this->isLegalProp($name)) ? json_encode($name) : $name; } /** * Encode a Regexp object into JavaScript * * @param Regexp $regexp * @return string */ protected function encodeRegexp(Regexp $regexp) { return $regexp->getJS(); } /** * Encode a scalar value into JavaScript * * @param mixed $value * @return string */ protected function encodeScalar($value) { return json_encode($value); } /** * Test whether given array is a numerically indexed array * * @param array $array * @return bool */ protected function isIndexedArray(array $array) { if (empty($array)) { return true; } if (isset($array[0]) && array_keys($array) === range(0, count($array) - 1)) { return true; } return false; } /** * Test whether a string can be used as a property name, unquoted * * @link http://es5.github.io/#A.1 * * @param string $name Property's name * @return bool */ protected function isLegalProp($name) { /** * @link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Reserved_Words * @link http://www.crockford.com/javascript/survey.html */ $reserved = ['abstract', 'boolean', 'break', 'byte', 'case', 'catch', 'char', 'class', 'const', 'continue', 'debugger', 'default', 'delete', 'do', 'double', 'else', 'enum', 'export', 'extends', 'false', 'final', 'finally', 'float', 'for', 'function', 'goto', 'if', 'implements', 'import', 'in', 'instanceof', 'int', 'interface', 'let', 'long', 'native', 'new', 'null', 'package', 'private', 'protected', 'public', 'return', 'short', 'static', 'super', 'switch', 'synchronized', 'this', 'throw', 'throws', 'transient', 'true', 'try', 'typeof', 'var', 'void', 'volatile', 'while', 'with']; if (in_array($name, $reserved, true)) { return false; } return (bool) preg_match('#^(?![0-9])[$_\\pL][$_\\pL\\pNl]+$#Du', $name); } }