Skip to content

Model View Controller

In some cases can be useful to leverage the power of server-side rendered applications, to reduce the load on the client-side, or to improve the SEO of your application. Serinus provides a way to render views using the ViewEngine.

To create a view engine, you first need to create a class that extends the ViewEngine class and implement the render and renderString methods and also a template engine to render the views.

In this guide we will use the MustacheX package to render the views.

dart
import 'package:serinus/serinus.dart';
import 'package:mustachex/mustachex.dart';

class MustacheViewEngine extends ViewEngine{
  
  const MustacheViewEngine({
    super.viewFolder
  });

  @override
  Future<String> render(View view) async {
    final processor = MustachexProcessor(
      initialVariables: view.variables
    );
    final template = File('${Directory.current.path}/$viewFolder/${view.view}.mustache');
    final exists = await template.exists();
    if(exists){
      final content = await template.readAsString();
      final processed = await processor.process(content);
      return processed;
    }
    return await _notFoundView(view);
  }

  @override
  Future<String> renderString(ViewString view) async {
    final processor = MustachexProcessor(
      initialVariables: view.variables
    );
    return await processor.process(view.viewData);
  }

  Future<String> _notFoundView(String view) async {
    final processor = MustachexProcessor(
      initialVariables: {'view': view}
    );
    return await processor.process('View {{view}} not found');
  }
  
}

In the MustacheViewEngine class, you can pass the following parameters to the constructor:

  • viewFolder: The folder where the views are stored. By default it is the views folder.

We can now use the MustacheViewEngine in our application.

dart
import 'package:serinus/serinus.dart';

Future<void> main() async {
  final app = await serinus.createApplication(
      entrypoint: AppModule(), host: InternetAddress.anyIPv4, port: 3000);
  app.viewEngine = MustacheViewEngine();
  await app.serve();
}

Templates

Now let's create the view that will be rendered. Create a folder called views in the root of your project and create a file called home.mustache inside the views folder.

html
<!-- views/home.mustache -->
<!DOCTYPE html>
<html>
<head>
  <title>Home</title>
</head>
<body>
  <h1>Welcome to Serinus {{name}}</h1>
</body>
</html>

Now we can add the Route to render the view.

dart
import 'package:serinus/serinus.dart';

class HomeController extends Controller {
  HomeController() {
    on(Route.get('/'), (RequestContext context) async {
      return View('home', variables: {'name': 'Dear User'});
    });
  }
}

We can also render directly a string.

dart
import 'package:serinus/serinus.dart';

class HomeController extends Controller {
  HomeController() {
    on(Route.get('/'), (RequestContext context) async {
      return ViewString('Hello {{name}}', variables: {'name': 'Dear User'});
    });
  }
}

Now if you access the route / you will see the view rendered.

Built with 💙 and Dart 🎯 | One of the 🐤 of Avesbox