sokil / frontend-bundle
Frontend components to quick start frontend app
Installs: 73
Dependents: 2
Suggesters: 2
Security: 0
Stars: 1
Watchers: 1
Forks: 0
Open Issues: 3
Language:JavaScript
pkg:composer/sokil/frontend-bundle
Requires
- php: ^5.6 || ^7.0
- friendsofsymfony/rest-bundle: ~1.7 || ~2.0
- symfony/framework-bundle: 2.*
Requires (Dev)
- phpunit/phpunit: 5.4.*
- satooshi/php-coveralls: 0.7.*
Suggests
- sokil/deploy-bundle: Deploy symfony2 apps
README
Single Page Application based on Backbone, Marionette and Twitter Bootstrap
Installation
Add composer dependency:
composer require sokil/frontend-bundle
Add bundle to AppKernel:
<?php class AppKernel extends Kernel { public function registerBundles() { $bundles = array( new Sokil\FrontendBundle\FrontendBundle(), ); } }
Build resources:
bower install
npm install
grunt
Bundle uses assetic so you need to register it in assetic config:
assetic: bundles: - FrontendBundle
You can use DeployBundle to handle process of deployment static resources.
Controller
Single Page Application
We may configure pre-defined controller, responsible for rendering single page application, as service:
acme.spa.controller: class: Sokil\FrontendBundle\Controller\IndexController arguments: - 'AcmeBundle:Spa:index.html.twig' calls: - [setContainer, ["@service_container"]]
Or use your own controller:
<?php namespace Acme\SiteBundle\Controller; use Symfony\Bundle\FrameworkBundle\Controller\Controller; use Symfony\Component\HttpFoundation\Request; class SpaController extends Controller { public function indexAction(Request $request) { if (!$this->isGranted('IS_AUTHENTICATED_REMEMBERED')) { throw $this->createAccessDeniedException(); } // render response return $this->render('SiteBundle:Spa:index.html.twig', [ 'applicationData' => $this->get('acme.spa.app_data')->getData(), // optional appdata ]); } }
See how to convigure spa in view SiteBundle:Spa:index.html.twig below in View section .
If some additional data required to be passed from backend to frontend, this may be done through Application data.
Application data
Application data may be used to pass some data from backend to frontend. Create service, responsible for obtaining application data from different sources:
acme.spa.app_data: class: Sokil\FrontendBundle\Spa\ApplicationData
Now pass name of this service to definition of your spa action:
acme.spa.controller: class: Sokil\FrontendBundle\Controller\IndexController arguments: - 'AcmeBundle:Spa:index.html.twig' - '@acme.spa.app_data' calls: - [setContainer, ["@service_container"]]
Application data generated from number of providers. provider is a servise identified by tag frontend.spa.app_data_provider.
Also app data service must be defined by key app_data, pointing to the instance of application data service.
acme.spa.some_app_data_provider; class: Acme\Spa\SomeApplicationDataprovider tags: - {name: frontend.spa.app_data_provider, app_data: acme.spa.app_data}
Provider class must implement Sokil\FrontendBundle\Spa\ApplicationDataProviderInterface. It must implement one method getData()
which return map of application parameters.
Model
Handling json requests
Backbone models send json requests.
To enable support, you may add FOSRestBundle, and configure serializer.
app/AppKernel.php:
new FOS\RestBundle\FOSRestBundle(),
app/config/config.yml:
framework: serializer: enabled: true enable_annotations: true
It you don't need whole bundle, just register listener, which converts json request to array:
services:
    json_request_listener:
        class: Sokil\FrontendBundle\EventListener\JsonRequestListener
        tags:
            - {name: kernel.event_listener, event: kernel.request, method: onKernelRequest, priority: -255 }
View
Template
View renders Marionette 2 application and starts it. Bootstrap 3 used as UI framework. For adding some CSS and JS resorses
on page, use macro from src/Resources/views/macro.html.twig:
{% import "@FrontendBundle/Resources/views/macro.html.twig" as frontend %}
<!DOCTYPE html>
<html>
    <head>
        {{ frontend.commonCssResources() }}
    </head>
    <body>
        <div id="content"></div>
        <div id="popup"></div>
    </body>
    {{ frontend.commonJsResources() }}
    <script type="text/javascript">
        (function() {
            // app options may be accessed through applicationData variable
            var options = {{ applicationData|json_encode|raw }};
            // router may be passed as option
            options.router = [
                AcmeRouter
            ];
            // container with fromtend services may be passed as option
            options.serviceDefinition = [
                AcmeServiceDefinition
            ];
            // root element of SPA app
            // optional, `body` used by default
            options.root = 'body';
            // requirejs
            options.requirejs = [
                AcmeRequireJsConfig
            ]
            // regions of root app, optional
            options.regions = {
                content: '#content', // region for content of app
                popup: '#popup'      // region for popup rendering
            }
            // start app
            window.app = new Application(options);
            window.app.start();
        })();
    </script>
</html>
Regions
Application refers to the root app.rootView view , which handles root element of the SPA application.
Root element may be configured in app options or used default value body. Root element holds some
regions where different functinnality rendered. Main contebnt renders to app.rootView.content region,
and popup renders to  app.rootView.popup region. You can pass your own regions and refers to them
through app.rootView:
Router
Router is instance of Backbone.Router. Also you can use Marionette.AppRouter.
If you have few routes, you can pass them all in routers option:
window.app = new Application({ routers: [ Bundle1Router, Bundle2Router ] });
To set default route, use option defaultRoute:
var Bundle1Router = new Backbone.Router({ routes: { '/some/route': 'someRoute' }, someRoute: function() { // ... } }); window.app = new Application({ routers: [ Bundle1Router, Bundle2Router ], defaultRoute: [Bundle1Router, 'someRoute']; });
Service container
Container is a registry to build and get already built services.
Service definition is just an object with methods to build service
instances, where this  refers to Container instance:
AcmeServiceDefinition = { someService: function() { return new SomeService(this.get('otherService')); }, otherService: function() { return new OtherService(); } }
Definitions passed to container by serviceDefinitions configuration parameter:
options.serviceDefinitions = [ Bundle1ServiceDefinition, Bundle2ServiceDefinition ];
Services then may be get from container:
var someService = app.container.get('someService');
RequireJs
Dependencies may be merged and passed to container:
options.requireJs = [ Bundle1RequireJsConfig, Bundle2RequireJsConfig ];
Every config may math one or both path and shim parameters:
var Bundle1RequireJsConfig = { paths: { 'bundle1_tinymce': 'staticpage/js/tinymce/tinymce.min' }, shim: { 'bundle1_tinymce': { exports: 'tinymce' } } };
By default, Application already has pre-configured FrontendRequireJsConfig. It has some dependencies, e.g. Twitter Typeahead.
Frontend Components
Popup
Popups must extend PopupView:
var MyPopupView = PopupView.extend({ events: { 'click .save': 'saveButtonClickListener' }, title: 'My Popup', buttons: [ {class: 'btn-primary save', title: 'Save'} ] }
List
This component allows to generate collection of models.
var SomeListView = ListView.extend({ showColumnHeader: true, // show table header /** * Defines list of columns * Array, or function that return array * @return {array} */ columns: function() { return [ { caption: app.t('SomeListView.some_column_name'), // column caption name: 'some_column_name' // column name }, ]; }, /** * Defines list of buttons, rendered in all rows * Array, or function that return array * @return {array} */ buttons: function() { return [ { name: 'edit', // button name class: 'btn btn-default btn-xs', // button class icon: 'glyphicon glyphicon-pencil', // button icon caption: app.t('SomeListView.edit'), // button cation click: function(e, itemId, view) { // click handler alert('edit'); } } ]; } });
MultiTypeahead
Wraps ListView, and adds typeahead functionality to add new elements to list.