Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Confusing class and method call in OpenCart

I have a framework (OpenCart) Controller class (like: catalog/controller/product/product.php) the code looks like:

class ControllerProductProduct extends Controller {
    public function index() {
      //some code
      $this->response->setOutput($this->render());
      //some more code
    }
}

there is an expression like $this->response->setOutput($this->render());. I know what this expression is used for, but I am pretty confused in how it works.

$this refers to current class i.e. ControllerProductProduct, it means $this->response object must exist in either ControllerProductProduct or its parent class Controller. But this is not the case. This object actually exists in a protected property of parent class Controller as Controller::registry->data['response']->setOutput(). So shouldn't it say like:

$this->registry->data['response']->setOutput();

instead of $this->response->setOutput();

I am also giving a snippet of Controller class so you can have idea.

abstract class Controller {
    protected $registry;    
    //Other Properties
    public function __construct($registry) {
        $this->registry = $registry;
    }
    public function __get($key) {
        //get() returns registry->data[$key];
        return $this->registry->get($key);
    }
    public function __set($key, $value) {
        $this->registry->set($key, $value);
    }
    //Other methods
}

I've no clue how this expression is working? Any idea how this is possible?

Thanks.

like image 522
8thperson Avatar asked Nov 12 '22 07:11

8thperson


1 Answers

This is working very easily using magic methods __get() and __set().

If You are trying to get an inaccessible class variable (e.g. that is not declared) a magic __get('property_name') method is invoked.

Thus when You are trying to retrieve $response, a magic method __get() is called and $this->registry->get('response') is returned instead (as there is no $response property declared).

Yes, You could write $this->registry->get('response')->setOutput($this->render()); instead, but this would be of not much use and more writing. It is OK to let the PHP retrieve the variable using it's __get() method, though it is not so clean.

Anyway, there is nothing wrong with the solution.

EDIT: a little bit cleaner solution would be this:

class Controller {
    //...
    function getResponse() {
        return $this->registry->get('response');
    }
    //...
}

Then You could call a concrete method in Your code and it would be clear enough:

class ControllerProductProduct extends Controller {
    public function index()
        //...
        $this->getResponse()->setOutput($this->render());
    }
}

But this would mean that there would be a need of getXYZ method for each possible property while __get() allows You to extend the $registry with no further work needed (in the case I described if You would add another property to $register You would have to add another getProperty() method - but still this would be more clear/clean solution).

like image 144
shadyyx Avatar answered Nov 14 '22 23:11

shadyyx