| 
<?phpnamespace eMacros\Runtime\Property;
 
 use eMacros\Applicable;
 use eMacros\Scope;
 use eMacros\GenericList;
 use eMacros\Symbol;
 
 class PropertyAssign implements Applicable {
 /**
 * Key/Property name
 * @var mixed
 */
 public $property;
 
 public function __construct($property = null) {
 $this->property = $property;
 }
 
 /**
 * Sets a key/property value
 * Usage: (#= 'name' _obj "emma") (#surname= _obj "doe")
 * Returns: the assigned value
 * (non-PHPdoc)
 * @see \eMacros\Applicable::apply()
 */
 public function apply(Scope $scope, GenericList $arguments) {
 $nargs = count($arguments);
 
 if (is_null($this->property)) {
 if ($nargs == 0) {
 throw new \BadFunctionCallException("PropertyAssign: No key defined.");
 }
 elseif ($nargs == 1) {
 throw new \BadFunctionCallException("PropertyAssign: No target specified.");
 }
 elseif ($nargs == 2) {
 throw new \BadFunctionCallException("PropertyAssign: No value specified.");
 }
 
 $target = $arguments[1];
 
 if (!($target instanceof Symbol)) {
 throw new \InvalidArgumentException(sprintf("PropertyAssign: Expected symbol as second argument but %s was found instead.", substr(strtolower(strstr(get_class($arguments[1]), '\\')), 1)));
 }
 
 $property = $arguments[0]->evaluate($scope);
 $value = $arguments[2]->evaluate($scope);
 $ref = $target->symbol;
 }
 else {
 if ($nargs == 0) {
 throw new \BadFunctionCallException("PropertyAssign: No target found.");
 }
 elseif ($nargs == 1) {
 throw new \BadFunctionCallException("PropertyAssign: No value specified.");
 }
 
 $target = $arguments[0];
 
 if (!($target instanceof Symbol)) {
 throw new \InvalidArgumentException(sprintf("PropertyAssign: Expected symbol as last argument but %s was found instead.", substr(strtolower(strstr(get_class($arguments[0]), '\\')), 1)));
 }
 
 $property = $this->property;
 $value = $arguments[1]->evaluate($scope);
 $ref = $target->symbol;
 }
 
 if (is_array($scope->symbols[$ref]) || $scope->symbols[$ref] instanceof \ArrayAccess || $scope->symbols[$ref] instanceof \ArrayObject) {
 $scope->symbols[$ref][$property] = $value;
 return $value;
 }
 elseif (is_object($scope->symbols[$ref])) {
 if (!property_exists($scope->symbols[$ref], $property)) {
 if ($scope->symbols[$ref] instanceof \stdClass) {
 $scope->symbols[$ref]->$property = $value;
 return $value;
 }
 
 if (method_exists($scope->symbols[$ref], '__set')) {
 $scope->symbols[$ref]->__set($property, $value);
 return $value;
 }
 
 throw new \UnexpectedValueException(sprintf("PropertyAssign: Property '$property' not found on class %s.", get_class($scope->symbols[$ref])));
 }
 
 $rp = new \ReflectionProperty($scope->symbols[$ref], $property);
 
 if (!$rp->isPublic()) {
 $rp->setAccessible(true);
 }
 
 $rp->setValue($scope->symbols[$ref], $value);
 return $value;
 }
 
 throw new \InvalidArgumentException(sprintf("PropertyAssign: Expected array/object as last argument but %s was found instead.", gettype($scope->symbols[$ref])));
 }
 }
 ?>
 |