NAME

KelpX::Controller - Base custom controller for Kelp

SYNOPSIS

### application's configuration
modules_init => {
	Routes => {
		base => 'My::Controller',
		rebless => 1, # needed for historic reasons
	}
},

### your base controller
package My::Controller;

use Kelp::Base 'KelpX::Controller';

sub build
{
	my $self = shift;

	$self->add_route('/sub' => sub { ... });
	$self->add_route('/method_name', 'this_controller_method');
	$self->add_route('/hashref', {
		to => 'this_controller_method',
	});
}

sub this_controller_method
{
	my $self = shift;

	return 'Hello world from ' . ref $self;
}

### your main application class
package My::Kelp;

attr context_obj => 'KelpX::Controller::Context';

sub build
{
	my $self = shift;

	$self->context->build_controllers;
}

DESCRIPTION

Since Kelp 2.16 it's quite easy to introduce your own base controller class instead of subclassing the main application class.

This extension is a more modern approach to route handling, which lets you have a proper hierarchy of custom classes which serve as controllers. Enabling it is easy, and can be done as shown in "SYNOPSIS".

The controller will be built just like a regular object the first time it's used. It will not be cleared after the request, since the context object will have its persistent_controllers set to true by default. You may override "build" in the controller, but if you want to have it add any routes then you will have to instantiate it manually using $app->context->controller.

Building controllers

In order for routes defined in your controller to register, you need to build the controllers. You can done it manually by calling $self->context->controller for each of your controllers, or automatically by calling $self->context->build_controllers. The latter will load all modules in the namespace of your base controller using Module::Loader and build them. As long as all your controllers are in the same namespace under the base controller, the automatic method is recommended (though it does not allow for conditional disabling of the controllers).

ATTRIBUTES

context

Required. The app's context object.

app

The application object. Will be loaded from "context".

METHODS

These methods are provided for convenience, but they are not meant to fully replicate the API available when not using controllers or using reblessing. For this reason, this module is not a drop-in replacement for reblessing controllers.

build

A build method, which will be called right after the controller is instantiated. Similar to app build method. Takes no arguments and does nothing by default - it's up to you to override it.

Since controllers inherit from one another and each route should only be built once, a special check is implemented which will not run this method automatically if it was not implemented in a controller class (to avoid calling parent class method). Make sure not to call SUPER version of this method if you implement it. If you need code that must be run for every controller, it's recommended to override new.

add_route

Similar to "add_route" in Kelp::Module::Routes, but it modifies the destination so that the route dispatch will run in the context of current controller. Unlike core Kelp controllers, you can use simple function names or even plain subroutines and they will pass the instance of your controller as their first argument.

req

Proxy for reading the req attribute from "context".

res

Proxy for reading the res attribute from "context".

CAVEATS

  • When using this module, even subroutine based routes will be run using the application's main controller (instead of application instance). Thanks to this, main class will never be used as call context for route handlers, so any hooks like before_dispatch can be safely moved to the base controller.

SEE ALSO

Kelp

AUTHOR

Bartosz Jarzyna, <bbrtj.pro@gmail.com>

COPYRIGHT AND LICENSE

Copyright (C) 2024 by Bartosz Jarzyna

This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself.