Interactable
public protocol Interactable<Request> : AbstractInteractable
This protocol represents Interactors that handle requests using a synchronous context, typically providing results for requests via a InteractorResponseClosure
.
Interactors provide an interface to perform some business logic, often acting as a datasource by interfacing with an API, handling system APIs, or some other self-contained work.
The concept of Interactors comes from Clean Swift, used in its architecture as a way to move logic and datasource management out of view controllers. In Destinations, the Interactable
protocol represents Interactor objects which provide an interface to perform some logic or data request, typically by interfacing with an backend API, handling system APIs, or some other self-contained work. Datasourceable
inherits from this protocol and should be used for objects which specifically represent a datasource of some kind. There are also Actor-based async versions of these protocols available.
Though requests to Interactors can be made using a Destination’s performRequest
method, in general one should use the performInterfaceAction
method. This abstracts the specific implementation details of an interactor away from the interface and lets it focus on making requests through a standardized request.
The recommended way is to assign a user interaction type to your request and using an InteractorAssisting
-conforming assistant to configure the request, leaving the Destination’s interface free of associated business logic.
Interactors are typically created by Destination providers and stored in the Destination. Like with presentations of new Destinations, Interactor requests are associated with a particular InterfaceAction
object and are represented by an InteractorConfiguration
model object. Action types for each Interactor are defined as enums, keeping Interactor-specific knowledge out of the interface.
For example, here we’re creating an Interactor action for making a pagination request to the ColorsDatasource, and then assign it to a moreButton
user interaction type:
let paginateAction = InteractorConfiguration<ColorsListProvider.InteractorType, ColorsDatasource>(interactorType: .colors, actionType: .paginate, assistantType: .basic)
let colorsProvider = ColorsListProvider(interactorsData: [.moreButton: paginateAction])
And here is that user interaction type being called by a Destination’s interface. Note that performInterfaceAction
can throw and the Destination method handleThrowable
automatically handles that for you, logging any errors to the console with the built-in Logger class.
destination().handleThrowable {
try destination().performInterfaceAction(interactionType: .moreButton)
}
-
A dictionary of Interactor request responses, whose keys are the Interactor action type they are associated with.
Declaration
Swift
var requestResponses: [Request.ActionType : InteractorResponseClosure<Request>] { get set }
-
Requests the Interactor to perform an operation.
Declaration
Swift
func perform(request: Request)
Parameters
request
A configuration model which defines the request.
-
assignResponseForAction(response:
Default implementationfor: ) Assigns a response closure for an Interactor action type, which is called after the request has completed.
Default Implementation
Declaration
Swift
func assignResponseForAction(response: @escaping InteractorResponseClosure<Request>, for action: Request.ActionType)
Parameters
response
The response closure to be called.
action
The Interactor action type to be associated with this response.
-
responseForAction(action:
Default implementation) Returns a response closure for the specified Interactor action.
Default Implementation
Declaration
Swift
func responseForAction(action: Request.ActionType) -> InteractorResponseClosure<Request>?
Parameters
action
The interactor action to find a response for.
Return Value
The response closure, if one was found.