If you like Blapy, I'll be pleased that you star it :-)
Blapy is a jQuery plugin that helps you to create and manage ajax and single page web applications (SPA) with almost no javascript coding to do it.
Your web application is built the usual way of generating web pages like with php or any standard CMS and Blapy will transform it into a dynamic web application with ajaxified contents.
So, it may help you to transform your "normal" web site in a web application without the hassle of changing the way you develop websites.
Blapy will speed up your page load too as it won't load all the internet files every page change but only the requested new contents.
Blapy is json compliant too and eases the integration of ajax/rest API with templating and routing services very simple to use... Blapy will be the "V-iew" in an MVC application approach...
The best of all is that your web application will be fully SEO compliant as Blapy uses normal URLs to do the routing and your html rendering output from your server to update your application blocks!
Go and see the demo: https://www.intersel.fr/assets_intersel_a/gitdemos/Blapy/demos/helloworld/
You can have a look on a more complete example based on the SB Admin 2 - Free Bootstrap Admin Theme (you can see the original one here)
or test the "To do list" web Apps demo made with Blapy and inspired from TodoMVC
This one allows to dynamically load and display blapy block contents that were hidden and which need to be displayed as they became visible.
All the demos found in the demos directory can be tested there : https://www.intersel.fr/assets_intersel_a/gitdemos/Blapy/demos/
We invite you to have a deep look in the code source of the demos as they use quite every possible features and configurations available in Blapy as they are used to test the library.
As it is a simple jQuery plugin... copy all the provided Blapy directories somewhere in your project, then include the needed javascript libraries in your code then call the Blapy function... and you're done...
As an "Hello world" example:
<!DOCTYPE html>
<!-- To run the current sample code in your own environment, copy this to an html page. -->
<html>
<head>
<!-- load of the external libraries needed by Blapy (provided in the package) -->
<script type="text/javascript" src="<myrootdir>/extlib/jquery-3.3.1.min.js"></script>
<script type="text/javascript" src="<myrootdir>/extlib/sammy/lib/sammy.js"></script>
<script type="text/javascript" src="<myrootdir>/extlib/iFSM/extlib/jquery.dotimeout.js" ></script>
<script type="text/javascript" src="<myrootdir>/extlib/iFSM/extlib/jquery.attrchange.js"></script>
<script type="text/javascript" src="<myrootdir>/extlib/json2html/json2html.js"></script>
<script type="text/javascript" src="<myrootdir>/extlib/iFSM/iFSM.js"></script>
<!-- load of the Blapy script -->
<script type="text/javascript" src="<myrootdir>/Blapy.js"></script>
</head>
<!-- start Blapy -->
<body id="myBlapy">
<div id="mainContainer"
data-blapy-container="true"
data-blapy-container-name="mainContainer"
data-blapy-container-content="helloWorld Test1">
Hello World...
</div>
<script>
$( document ).ready(function() {
$('#myBlapy').Blapy();
});
</script>
</body>
</html>
- Blapy
- Have a look on the "Hello World" demo and other demos
- Who may need it?
- Why would I use that?!
- How does it work?
- How to configure my pages to become pages of a web application?
- General algorithm of Blapy
- .Blapy(options)
- Blapy Blocks
- Blapy Links
- Triggered events
- Sending events to Blapy
- Blapy parameters sent when calling a URL
- Blapy animation plugin functions
- LIBRARY DEPENDENCIES
- FAQ
- Problem resolutions
- Contact
Everyone using a CMS that generates web pages from a server and would like to transform his website to a client application-like website, ie that does not reload each page during the user navigation but only the needed blocks within the page.
Everyone who...
- would like to keep the way he builds websites but would like to have it behaves like an ajax web application.
- gave up with AngularJS and other javascript frameworks to build web app... like me ;-)
- wants to integrate json REST API with a simple approach without coding...
- is starting with Javascript...
- does not want to learn a Nth new framework to do websites...
- wants to keep the application SEO compliant with google...
And, as I just discovered it recently, for those who are using pjax, Blapy may be a good alternative too... You'll tell me!
The concept of a web application getting data through REST Api with a client application that is doing the job of connecting the whole to build an application is quite a difficult job with a steep learning curve...
Whereas PHP -or whatever web server languages- websites built on a standard CMS are easier to handle... Any standard CMS does the page rendering job quite naturally for years... Except that it reloads pages when clicking a link... or it needs to develop ajax calls to dynamically update parts of the page...
So, the idea is to provide a simple environment that don't change your habits when creating your website without having the hassle of creating ajax calls.
The benefits of using Blapy?
- no difficult framework to understand how to build a web application
- no REST or Ajax url end points to develop. Of course, you can do that if you like to have your application that way ;-)
- no change in your habit to develop website: building the pages with Blapy don't change from the "static" usual way of doing a website, meaning you can continue to use your standard LAMP and CMS environement
- configuration is simple and quite natural: it uses html5 "data" attributes to configure the Blapy configuration and there is quite nothing to do from an existing website :-)
- the ajax things are done behind the scene with no js lines of code to implement them
- the history of browsing is kept as any smart framework
- completly compliant with any existing html/js code
- SEO compliant as the server keeps the control on the application behaviour on the loaded pages and blocks, and so it is able to deliver correct contents to search engines as google.
The main simple idea behind Blapy is to automatically and dynamically bind and update html blocks in ajax during the web navigation from page to page.
Simple ;-) but powerful...
Rules defined on the html blocks with data attributes will specify how the blocks should be updated with their new dynamic contents.
let's have a first html file test1.html with some blocks with special attributes we will see later on...
<body id="myBlapy">
<h1>I'm test1.html file</h1>
<ul>
<li><a href="test1.html" data-blapy-link="true">Hello World!</a></li>
<li><a href="test2.html" data-blapy-link="true">How is it going?</a></li>
</ul>
<div id="mainContainer" data-blapy-container="true" data-blapy-container-name="mainContainer" data-blapy-container-content="helloWorld">
Hello World
</div>
</body>
Let's imagine now that you would like that the website loads and updates only the "mainContainer" part without updating the whole page when we click on the test2.html link...
You surely know that you would need to call a URL in ajax, get the new content from the server and update the container with jQuery html function... Perhaps meet some problem with the browser history when going back... etc... etc...
With Blapy, just create a second test2.html file as usual: it will be quite the same than test1.html (let's imagine the files are php generated...) with a new content in the "mainContainer" part :
<body id="myBlapy">
<h1>I am test2.html file</h1>
<ul>
<li><a href="test1.html" data-blapy-link="true">Hello World!</a></li>
<li><a href="test2.html" data-blapy-link="true">How is it going?</a></li>
</ul>
<div data-blapy-container="true" data-blapy-container-name="mainContainer" data-blapy-container-content="Howisitgoing?">
How is it going?
</div>
</body>
These two html files will load and behave normally if you load them and click the links.
Well, just add at the end of your files this little script :
<!-- load of the external libraries needed by Blapy (provided in the package) -->
<script type="text/javascript" src="extlib/jquery-3.3.1.min.js" ></script>
<script type="text/javascript" src="extlib/sammy/lib/sammy.js" ></script>
<script type="text/javascript" src="extlib/iFSM/extlib/jquery.dotimeout.js" ></script>
<script type="text/javascript" src="extlib/iFSM/extlib/jquery.attrchange.js"></script>
<script type="text/javascript" src="extlib/iFSM/iFSM.js" ></script>
<!-- load of the Blapy script -->
<script type="text/javascript" src="Blapy.js"></script>
<!-- start Blapy -->
<script>
$( document ).ready(function() {
$('#myBlapy').Blapy();
});
</script>
You will then see that when clicking on the page links, only the 'data-blapy-container' block is changed without reloading the whole page! You can see that as the title has not changed...
Tada! you've got a client web application :-)
- Identify the common blocks (div, p, ... html tags) between your pages. When you use a CMS, these blocks are the same ones than those you identified when building your website.
<div id="myContainer">
How is it going?
</div>
- Tell to Blapy that these blocks are the ones that may be updated from page to page:
- add a "data-blapy-container" attribute set to true in order to configure this container as a Blapy container
- give a name identifier to the container with the "data-blapy-container-name" attribute in order to identify this content block as unique.
- give a content name to identify each unique page content to be used with the "data-blapy-container-content" attribute.
<div id="myContainer" data-blapy-container="true" data-blapy-container-name="mainContainer" data-blapy-container-content="Howisitgoing">
How is it going?
</div>
You can create as many Blapy containers as you need parts of your page to be updated.
- Identify the links pointing to pages that have contents you would like to use to update the content blocks of your current page.
<a href="test1.html">Hello World!</a>
- Tell the links that they need to be handled by Blapy by using the "data-blapy-link" attribute.
<a href="test1.html" data-blapy-link="true">Hello World!</a>
- Call Blapy jquery script on the body of your page (or any main div in your page)... The html block should have an "id" attribute.
<html>
...
<body id="myBlapy">
...
<div id="myContainer" data-blapy-container="true" data-blapy-container-name="mainContainer" data-blapy-container-content="Howisitgoing">
How is it going?
</div>
...
<script type="text/javascript" src="../Blapy.js"></script>
<script>
$( document ).ready(function() {
$('#myBlapy').Blapy();
});
</script>
</body>
</html>
That's it: your blocks will be automatically updated by ajax calls to the other pages according to your configuration.
The general algorithm of Blapy is the following:
- when a link is called (clicked) and if it is a "blapy-link",
- get the new content from the clicked page through an ajax call
- get all the "blapy-container" of the current page
- for each "blapy-container" of the current page
- Get the "blapy-container" with the same name ("blapy-container-name") from the new page
- Compare "blapy-container-content" : is the current "blapy-container-content" different from the new one?
- if "yes", then change the "blapy-container" with the new one
Blapy is the jQuery function that starts your web application.
$("#<an id>").Blapy(options);
Options is a javascript object. It can take as input the following possible option parameters :
- debug: (default:false) if true, then log things in the console
- LogLevel: (default:1) log level: 1: error ; 2: warning; 3: notice
- AlertError: (default:false) show an alert box on error messages if true
- activeSammy: (default:false) if set to true, will use Sammy for URL routing
- pageLoadedFunction: (default:null) function to call when the page is loaded
- pageReadyFunction: (default:null) function to call when the page is ready
- beforePageLoad: (default:null) function to call before the page load the new content
- beforeContentChange: (default:null) function to call when a Blapy bloc will have its content changed
- afterContentChange: (default:null) function to call after a Blapy bloc has its content changed
- afterPageChange: (default:null) function to call when the page had all its content changed
- onErrorOnPageChange: (default:null) function to call when the page got an error during change
- fsmExtension: (default:null) Finite State Machine (iFSM) definition in order to extend the default blapy's iFSM engine
$( document ).ready(function() {
$('#bodyId').Blapy({activeSammy:true,debug:true,LogLevel:2})
});
Blapy absolutely needs to be called on an object with an "id" set.
That means that if you'd like to bind Blapy to the "body", you have to set an id on the body element:
<body id="bodyId">
...
</body>
You can have as many separate Blapy blocks as you like.
$( document ).ready(function() {
$('#myBlapyApp1,#myBlapyApp2,#myBlapyApp3').Blapy();
});
This way, you will be able to tell which application a Blapy block should update when loaded.
See "data-blapy-applyon" option on Blapy blocks.
Blapy blocks are the parts where you would like the content to be updated from external contents downloaded through ajax calls by Blapy.
A Blapy block may be any html element where you have set Blapy attributes that define the behaviour as Blapy blocks.
These attributes are analysed from the external Blapy block in order to know how the updating process should be applied on the current block.
To define a Blapy Block, you need to use the following attributes:
- data-blapy-container: set to "true", tells that the current html tag is Blapy block.
- data-blapy-container-name: gives the name of the Blapy block. It will identify the block.
- data-blapy-container-content: gives the subject of the content. It will identify the content of the block.
- data-blapy-update (option, default:'update'): tells how Blapy should update the Blapy block when an external page or content is loaded.
- update: if the container-name is found from the external content and its container-content is different from the current page, the Blapy block of the current page is to be replaced by the new one
- force-update: if the container-name is found from the external content, the Blapy block of the current page is to be replaced by the new one
- append: if the container-name is found from the external content, the external content should be added to the end of the current Blapy block content.
- prepend: if the container-name is found from the external content, the external content should be added before the current Blapy block content.
- replace: if the container-name is found from the external content, the inner content of the external content should replace the current Blapy block content.
- json: the content of the current container is considered to be a (Mustache or json2html) template to apply on json data.
When the blapy block needs to be updated, then it is considered that the new content is a json object or an array of json objects that will be parsed with the given template.
These json data will be applied on the template. These parameters allows the json configuration:
- data-blapy-template-file (option): defines a URL called to get the template to apply on json data if the container (that is used to define the template) is empty
- data-blapy-template-wrap (option): once the json data are rendered, it is possible to wrap the result by giving the wrap html tag (ex: "
<table>
") - data-blapy-template-header (option): once the json data are rendered, it is possible to set a header (ex: "
<tr><th>header</th></tr>
") - data-blapy-template-footer (option): once the json data are rendered, it is possible to set a footer (ex: "
<tr><th>footer</th></tr>
") - data-blapy-template-default-id (option): if multiple templates are set, set the default one to use on initialization. Default to the first found one.
- data-blapy-template-mustache-delimiterStart (option): available for mustache template, blapy will change the default start and end delimiters and use the new mustache delimiters when parsing the block. Actually, the new delimiters are set at the beginning of the template just before rendering, allowing blocks parsed within blocks with different rendering tags. Example: "<%"
- data-blapy-template-mustache-delimiterEnd (option): available for mustache template and should be defined if data-blapy-template-mustache-delimiterStart is defined. Example: "%>"
- data-blapy-template-init (option): a (REST) URL to get json data to use to initialize the block
- data-blapy-template-init-params (option): json string of the parameters to send to the URL
- data-blapy-template-init-method (option): 'GET' (default) || 'POST' || 'PUT' || 'DELETE'
- data-blapy-template-init-fromproperty (option): path to the property in the returned json that should be used as data input. eg: "data.results" will use the object found in jsonReturnData.data.results as the json data to use
- data-blapy-template-init-search (option):
"<property>==<value>[,<property>==<value>,...]"
will get the objects that match the query. Examples:- id==2, will get all objects that have their 'id' property equals to '2'
- id== , will get all objects that have an 'id' property
- ==2, will get all objects that have properties equals to '2'
- data-blapy-template-init-processdata (option): a function name that take a json object and should return a json object. It will be called once the json data is received from the url and before to be processed in its blapy block.
- data-blapy-template-init-purejson (option): '0' ("blapy oriented" json) || '1' (default) (not "blapy oriented" json),
- remove: if the container-name is found from the external content, then the Blapy block is to be removed.
- custom: if the container-name is found from the external content, then we call the custom change 'doCustomChange' if defined and send the Blapy_doCustomChange event.
- [an animation plugin function name]: if the container-name is found from the external content, function to call and apply to do the content change. The available plugin functions may be found in the Blapy_AnimationPlugins.js file.
- data-blapy-update-rule: ['local'|'external' (default)], if 'local', will use the data-blapy-update rule defined in the current block, else will use the one defined in the external block. Exception to the default value, a "json" block is always "local".
- data-blapy-applyon (option, default:'all'): By default, the Blapy blocks loaded by a Blapy link will be tried on all Blapy blocks. If defined, the external container will only be applied on the matched Blapy blocks contained in the given application id element.
- data-blapy-href (option): a URL to call on data-blapy-updateblock-time (if set) or on data-blapy-updateblock-ondisplay
- data-blapy-updateblock-time (option): a time in milliseconds when the URL set in 'data-blapy-href' should be called to update the block.
- data-blapy-updateblock-ondisplay (option): if set to true, the block will be initialized from data-blapy-href or from data-blapy-template-init (if data-blapy-update is set to "json") when the element becomes visible (after a scroll).
<div data-blapy-container="true"
data-blapy-container-name="mainContainerApp3"
data-blapy-container-content="aContent2"
data-blapy-applyon="myBlapyApp1,myBlapyApp3"
>
<h3>a Content</h3>
This is content...
</div>
<div data-blapy-container="true"
data-blapy-container-name="submainContainerApp1"
data-blapy-container-content="aSubContent"
data-blapy-update="remove"
>
</div>
A Blapy Link is a url link that should be handled by Blapy.
A Blapy link may be attached to the html "<a>
" or "<form>
" tags by specifying a "data-blapy-link" attribute on it.
It may be attached to other kind of tag, then you will have to specify the "data-blapy-href" attribute to explicit the hyperlink.
Remarks on blapy links on tags that are not "a" or "form"
- In this case, Blapy automatically binds a click event on this element in order to simulate an anchor.
- The routing mecanism done with Sammy does not apply on them.
To define a Blapy Link, here are its attributes:
- data-blapy-link: tells that the current link has to be handled by Blapy.
- set to "true" (or void), Blapy will "get" the content of the link and will process the blapy blocks matching with the current application content
Remarks: if data-blapy-link is set to a form, the "method" configuration (eg. "GET", "POST", "PUT", "DELETE") is used to get the content. You can add this proprety "method" to a A dom element to set the method to use when the link is clicked.
-
data-blapy-href: if not bound to a "A" or "FORM" tag, it tells the hyperlink to use.
-
data-blapy-params: a json string to set paramaters to send along to the link to call
-
data-blapy-embedding-blockid: tells to embed the return source of the link in a blapy block of the given name. Usefull for return sources that are not 'blapy' formatted and that address a specific block...
-
data-blapy-active-blapyId: useful in the case the link is embedded in several different blapy objects, it gives the correct blapy object for the link. If not set, the link will be processed by all the blapy objects that contain the link.
-
data-blapy-noblapydata: if set to "1", no blapy information data are sent to the URL.
<ul>
<li><a href="content1.php" data-blapy-link="true">Content 1</a></li>
<li data-blapy-href="content2.php" data-blapy-link="true">Content 2</a></li>
<li>
<a href="jsoncontent3.php"
data-blapy-link="true"
data-blapy-embedding-blockid="mainContainerApp3"
data-blapy-params='{"id":"12","label":"test"}'
method="POST"
>
Content 3
</a>
</li>
</ul>
Blapy generates the specific events during the Blapy object change processing.
Some events are sent to the DOM element on which you create the jquery blapy object, some apply on the blapy block where occurs the processing.
Triggered when a page is fully and normally loaded and sent to the blapy object once when it is initialized.
Triggered when the page is ready after a change in the page, and sent to the blapy object.
It is triggered after a "loadURL" or "postData" event sent to Blapy (see hereafter).
Triggered before a page loads its new content and sent to the blapy object.
- Parameters: the Blapy json data sent to the URL
- [data].aUrl: url to call
- [data].params: json of the parameters to send to aUrl
Triggered after a page loaded its new content and sent to the blapy object.
- Parameters: the Blapy json data sent to the URL
- [data].aUrl: url to call
- [data].params: json of the parameters to send to aUrl
Triggered before a Blapy block content change and sent to the (jquery) blapy Block that will change.
- Parameters:
- the Blapy block
Triggered after a Blapy block content has changed and sent to the (jquery) blapy Block that has changed..
- Parameters:
- the Blapy block
Triggered after a Blapy block content has its template loaded and ready, sent to the (jquery) blapy Block..
- Parameters:
- the Blapy block
Triggered if data-blapy-update='custom', sent to the object that should change its content
- Parameters:
- newObject : the new Blapy Block
Triggered when an error occured during a page content change.
- Parameters:
- anError: the error message
To listen to Blapy events, you may use the jQuery 'on' function as in this example:
$("#myBlapy").on( "Blapy_ErrorOnPageChange", function(event,anError) {
alert( anError );
});
$("#myBlapy").on( "Blapy_beforePageLoad", function(event,anError) {
alert( 'call before loading new blocks...' );
});
//assure that the event will be received by the new DOM object #mainContainer if it has been replaced by Blapy
$(document).on( "Blapy_afterContentChange","#mainContainer", function(event,aBlock) {
alert( 'Blapy_afterContentChange of block '+$(aBlock).attr('id'));//aBlock refer to the #mainContainer dom object
});
You can activate some features of Blapy by sending events to it with the 'trigger' function of jQuery:
$('#<id of the blapy application tag>').trigger(<anEvent>,{
aUrl:<aURL to call>,
params:{<someParameters>}
...
});
This event allows you to call a URL.
$('#<id of the blapy application tag>').trigger('loadUrl',{
aUrl:<aURL to call>,
params:{
embeddingBlockId:<a Blapy Block Id>,
templateId:<a xmp id of a json template>
},
noBlapyData:"<0|1>"
});
- aUrl: the Url to call
- params:
- embeddingBlockId (optional): a block container name (data-blapy-container-name)
- templateId (optional): an id of an xmp object that describes a json template of a json block. Remarks: "embeddingBlockId" needs to be defined.
- noBlapyData: if set to "1", no blapy information are sent (default: blapy data are sent)
$('#myBlapy').trigger('loadUrl',{
aUrl:"helloworld_2.php",
});
$('#<id of the blapy application tag>').trigger('postData',{
aUrl:"<aURL to call>",
params:{
embeddingBlockId:"<aContainerName>",
templateId:<a xmp id of a json template>
},
method:"<http method>",
noBlapyData:"<0|1>"
});
- aUrl: the Url to call
- params:
- embeddingBlockId (optional): a block container name (data-blapy-container-name)
- templateId (optional): an id of an xmp object that describes a json template of a json block. Remarks: "embeddingBlockId" needs to be defined.
- any property/value to send to the server
- method (option)
- 'post' (default)
- 'put'
- 'delete'
- 'get' (same behaviour than "loadURL" event in this case)
- noBlapyData: if set to "1", no blapy information are sent (default: blapy data are sent)
$("#myBlapy").trigger('postData',{
aUrl:"testForm.php",
params:{fname:'Emmanuel',lname:'Podvin'}
});
This event allows you to call Blapy to directly update a Blapy block.
$('#<id of the blapy application tag>').trigger('updateBlock',{
html:<a blapy content>,
params:{
embeddingBlockId:<a Blapy Block Container Name>,
templateId:<a xmp id of a json template>
}
});
- html
- any blapy content (blapy blocks, json string or objects, ...)
- params:
- embeddingBlockId (optional): a block container name (data-blapy-container-name)
- templateId (optional): an id of an xmp object that describes a json template of a json block. Remarks: "embeddingBlockId" needs to be defined.
$('#myBlapy').trigger('updateBlock',{
html:"[{name:"John Doe"}]",
params:{embeddingBlockId:'myBlapyBlock'}
});
This event allows you to reload the Blapy blocks using their init configuration (init url).
$('#<id of the blapy application tag>').trigger('reloadBlock',{
params:{
embeddingBlockId:<a Blapy Block Container Name>,
templateId:<a xmp id of a json template>
}
});
- params:
- embeddingBlockId (optional): a block container name (data-blapy-container-name). If none given, all the json blocks are reloaded.
- templateId (optional): an id of an xmp object that describes a json template of a json block. Remarks: "embeddingBlockId" needs to be defined.
$('#myBlapy').trigger('reloadBlock',{
params:{embeddingBlockId:'myBlapyBlock'}
});
When Blapy calls a 'Blapy Link', the following parameters are sent along the other GET/POST/PUT/DELETE parameters:
- blapycall=1 - tells that the call is coming from Blapy
- blapyaction=[anAction] -tells the kind of action that will be operated
- update: blocks will be updated
Knowing these parameters allows you to optimize the generated html returned by the server to the client, so sending back only the useful html blocks instead of the full html page.
You can deactivate the sending of these data by configuring the data-blapy-noblapydata parameter.
The json template are the content definition of a blapy json block type (cf. data-blapy-update) that will be parsed on json data sent to the block.
A json template may use "Mustache" or "json2html" tags.
When the blapy block has its property "data-blapy-update" set to "json", the content of the block is considered to be a template.
<section id="jsonTPLExample"
data-blapy-container="true"
data-blapy-container-name="jsonTPLExample"
data-blapy-container-content="aTPL"
data-blapy-update="json"
data-blapy-template-init="myJsonData.json"
>
My name is {{firstname}}!<br>
</section>
if the response of "myJsonData.json" is an array of json data, something like that:
[
{"firstname":"Emmanuel"},
{"firstname":"Maryse"},
{"firstname":"Augustin"},
]
The resulting parse of the blapy block in the browser will be:
<section id="jsonTPLExample"
data-blapy-container="true"
data-blapy-container-name="jsonTPLExample"
data-blapy-container-content="aTPL"
data-blapy-update="json"
data-blapy-template-init="myJsonData.json"
>
My name is Emmanuel!<br>
My name is Maryse!<br>
My name is Augustin!<br>
</section>
It is possible to have blapy blocks inside your template. They will be parsed once the template is applied on the json data. This way, you can have blapy blocks that are configured with your json data...
If the received json data is an array (like in the above example), Blapy will automatically add a property "blapyIndex" setting the index of the item in the array.
For the above example, the json data to be parsed on the template will actually be
[
{"firstname":"Emmanuel","blapyIndex":"1"},
{"firstname":"Maryse","blapyIndex":"2"},
{"firstname":"Augustin","blapyIndex":"3"},
]
This "blapyIndex" may be used in your template as any other of properties (${blapyIndex} or {{blapyIndex}})...
It starts from 1... if json data is not an array, then blapyIndex is set to 0.
According to the context, you may need to change the template of your block to display differently your data.
It is so possible to define several templates for the same block. These templates can be selected through the Blapy API with the messages "updateBlock", "postData", "reloadBlock"...
The way to describe them needs the use of the XMP tag and the "data-blapy-container-tpl" (to be set to true) and "data-blapy-container-tpl-id" properties, like in the following example:
The "data-blapy-template-default-id" property may be used to set the default template by default.
<section id="jsonTPLExample"
data-blapy-container="true"
data-blapy-container-name="jsonTPLExample"
data-blapy-container-content="aTPL"
data-blapy-update="json"
data-blapy-template-init="myJsonData.json"
data-blapy-template-default-id="secondTPL"
>
<xmp style="display:none" data-blapy-container-tpl="true" data-blapy-container-tpl-id="firstTPL">
My name is {{firstname}}!<br>
</xmp>
<xmp style="display:none" data-blapy-container-tpl="true" data-blapy-container-tpl-id="secondTPL">
Is {{firstname}} your firstname?<br>
</xmp>
</section>
You may call the "reloadBlock", "postData" or "updateBlock" event messages to change the template to use. It will reload the json data from the server and display them with the updated template.
eg: this example will use "secondTPL" template to display the data in "results" block.
$('#myBlapy').trigger('postData',
{
"aUrl": "testForm.php",
"params":{
"embeddingBlockId":"results",
"templateId": "secondTPL",
"firstname":$("select[name=firstname]",'#myForm').val(),
"lastname":$('input[name=lastname]:checked', '#myForm').val()
}
}
);
It is possible to create its own animation plugin functions on Blapy blocks when they are loaded.
It is also a way to hook features on the content that will be placed in a Blapy block...
The prototype of an animation plugin function is :
theBlapy.prototype.myAnimationFunction = function (oldContainer,newContainer) {}
Have a look in the Blapy_AnimationPlugins.js and add your new functions in it inspired by the existing functions.
To work properly, you need to include the following javascript libraries:
- jQuery (>= 3.x)
<script type="text/javascript" src="extlib/jquery-3.3.1.min.js"></script>
- iFSM by intersel
- this library manages finite state machines and needs these libraries:
- doTimeout by "Cowboy" Ben Alman
- this library brings some very usefull feature on the usual javascript setTimeout function like Debouncing, Delays & Polling Loops, Hover Intent...
<script type="text/javascript" src="extlib/jquery.dotimeout.js"></script>
- attrchange by Selvakumar Arumugam
- a simple jQuery function to bind a listener function to any HTML element on attribute change
<script type="text/javascript" src="extlib/jquery.attrchange.js"></script>
- this library manages finite state machines and needs these libraries:
- json2html (optional if blapy block does not use json feature or use "Mustache" template engine)
- json2html is a javascript HTML templating library used to transform JSON objects into HTML using a template.
- used for json parsing and templating
<script type="text/javascript" src="../../extlib/json2html/json2html.js"></script>
- Mustache (optional if blapy block does not use json feature or use "json2html" template engine)
- Mustache is a javascript HTML templating library used to transform JSON objects into HTML using a template.
- used for json parsing and templating
<script type="text/javascript" src="../../extlib/mustache/mustache.js"></script>
- Sammy (optional if you don't need routing management)
- Sammy is a small framework to make web application providing simple but efficient 'route' services
<script type="text/javascript" src="extlib/sammy/lib/sammy.js"></script>
- json5 (optional if your json are "straight" json)
- expands the syntax of JSON in order to be able to process less strict json input (made by humans for example)
<script type="text/javascript" src="extlib/json5/index.min.js"></script>
- jquery.appear (optional if you don't need to init blocks when they become visible after a scroll)
<script type="text/javascript" src="extlib/jquery.appear/jquery.appear.js"></script>
If you have questions or unsolved problems, you can have a look on the our FAQs or leave a message on the Issue board.
No, you can optimize your code by only sending the useful Blapy blocks.
Yes, but in order to have the Blapy see them, set an id on the html tag and call Blapy on it:
<html id="myBlapy">
<head>
<title data-blapy-container="true"
data-blapy-container-name="Title"
data-blapy-container-content="myTitle">This is a title page</title>
<script>
$( document ).ready(function() {
$('#myBlapy').Blapy();
});
</script>
</head>
<body>
<!-- body part -->
</body>
</html>
The syntax follows the one defined by json2html library: ${myVariableName} OR the one defined by Mustache: {{myVariableName}} according to the library you have set.
With json2html:
First name: ${fname}<br>
Last name: ${lname}<br>
With Mustache:
First name: {{fname}}<br>
Last name: {{lname}}<br>
Let's say we would like to create a dynamic table as the following example:
<section id="tableExample">
<table>
<tr><td>John</td><td>Doe</td></tr>
<tr><td>Bob</td><td>Dylan</td></tr>
....
</table>
</section>
First step is to define the blapy block:
<section id="tableExample"
data-blapy-container="true"
data-blapy-container-name="results"
data-blapy-container-content="void"
data-blapy-update="json"
data-blapy-template-wrap="<table>"
data-blapy-template-init="arrayvalues.data.php"
>
<tr><td>${firstname}</td><td>${lastname}</td></tr>
</section>
Some explanations on the defined attributes:
- data-blapy-update="json": the blapy block is filled with json data. So the content in the block is the template to apply on the json data. The template may be given by an external file with the property "data-blapy-template-file".
- data-blapy-template-wrap="
<table>
": the template is wrapped with a 'table' tag - data-blapy-template-init="arrayvalues.data.php": the values to init our block are to be found in this file
The returned content of the "data-blapy-template-init" file should contain the addressed blapy block definition ("data-blapy-container-name") with a json struture with the data to display, and so should be something like the following example:
<section id="tableExample"
data-blapy-container="true"
data-blapy-container-name="results"
data-blapy-container-content="initContent"
>
[
{firstname: "John",lastname: "Doe"},
{firstname: "Bob",lastname: "Dylan"},
...
]
</section>
Of course, this content may be dynamically generated by the server side.
Once you start blapy like with "('#tableExample').Blapy();", blapy will automatically read the json content of the "data-blapy-template-init" file and will apply its content to the html template.
The resulting content will be like the example to produced...
Instead of having a HTML template, you can set a javascript that will be interprated to generate the DOM.
The javascript should be inserted with the specific tag "<blapyScriptJs>
".
This example shows how "<li>
" statement will be inserted according to the statut of "dontdisplay" variable.
Let's define a UL statement that we want be filled with LI statements.
It will be initialized from the "data-blapy-template-init" variable with the content returned by myInitFile.php file. It could have been initialized with a "loadUrl" or "postData" calls.
<ul id="MenuExampleWithInitializedWithJSScript"
data-blapy-container="true"
data-blapy-container-name="MenuExampleWithInitializedWithJSScript"
data-blapy-container-content="MenuExampleVoid"
data-blapy-update="json"
data-blapy-template-init="myInitFile.php"
>
<blapyScriptJS>
if (!"${dontdisplay}")
jQuery('#MenuExampleWithInitializedWithJSScript').append('<li class="${class}"><a href="${url}">${action}</a></li>');
</blapyScriptJS>
</ul>
Please note that what myIntFile.php returns as data content should have only data initialization for this block.
<ul id="MenuExampleWithInitializedWithJSScript"
data-blapy-container="true"
data-blapy-container-name="MenuExampleWithInitializedWithJSScript"
data-blapy-update="json"
>
[
{class: "",url: "#",action:"Action"},
{class: "",url: "#",action:"Action Not Shown",dontdisplay:'1'},
{class: "",url: "#",action:"Another action",dontdisplay:'1'},
{class: "myClass",url: "#",action:"Something else here"},
]
</ul>
or the same description but in full json description:
[
{ "blapy-container-name":"MenuExampleWithInitializedWithJSScript",
"blapy-data":
[
{class: "",url: "#",action:"Action"},
{class: "",url: "#",action:"Action Not Shown",dontdisplay:'1'},
{class: "",url: "#",action:"Another action",dontdisplay:'1'},
{class: "myClass",url: "#",action:"Something else here"},
]
}
]
The result will be processed as:
<blapyScriptJS>
if (!"")
jQuery('#MenuExampleWithInitializedWithJSScript').append('<li class=""><a href="#">Action</a></li>');
</blapyScriptJS>
<blapyScriptJS>
if (!"1")
jQuery('#MenuExampleWithInitializedWithJSScript').append('<li class=""><a href="#">Action Not Shown</a></li>');
</blapyScriptJS>
<blapyScriptJS>
if (!"1")
jQuery('#MenuExampleWithInitializedWithJSScript').append('<li class=""><a href="#">Another action</a></li>');
</blapyScriptJS>
<blapyScriptJS>
if (!"")
jQuery('#MenuExampleWithInitializedWithJSScript').append('<li class="myClass"><a href="#">Something else here</a></li>');
</blapyScriptJS>
and so, giving the following processed DOM:
<ul id="MenuExampleWithInitializedWithJSScript"
data-blapy-container="true"
data-blapy-container-name="MenuExampleWithInitializedWithJSScript"
data-blapy-update="json"
>
<li class=""><a href="#">Action</a></li>
<li class="myClass"><a href="#">Something else here</a></li>
</ul>
You just defined an array the way you would do in javascript with your json objects
<div
data-blapy-container="true"
data-blapy-container-name="resultFormJson"
data-blapy-container-content="resultFormJson"
data-blapy-update="json"
>
[
{fname: "Emmanuel",lname: "Durand"},
{fname: "Maryse",lname: "Dupond"}
]
</div>
By default, when there is a blapy call, you define a blapy block and put your json data inside it and send the result back.
It is possible to return "pure" json to be handled by Blapy.
There are two possible configurations:
- the json content returned has no information on the Blapy blocks that should be updated
- the json content is blapy oriented
It applies to content that comes from outside your web application, that are not aware of Blapy ;-)
In order to tell to Blapy on what blapy block this kind of content should update, you need to configure the Blapy link that calls the json content.
Use the "data-blapy-embedding-blockid" parameter and configure it to the name of your Blapy block.
That's it! When the blapy link will call your json content, all your blapy block with the given name will be updated with the json content.
HTML File
<a href="myJsonDataFile.json"
data-blapy-link="true"
data-blapy-embedding-blockid="aBlapyBlockToUpdate"
>Click here to update my "aBlapyBlockToUpdate" to update
</a>
<div
data-blapy-container="true"
data-blapy-container-name="aBlapyBlockToUpdate"
data-blapy-update="json"
>
This is ${jsonContent}<br>
The other content is: ${anotherContent}
</div>
myJsonDataFile.json
{"jsonContent":"a content for myJsonDataFile...","anotherContent":"another content for the example"}
Remark: You can have a look on the "demos/verifyEmails" demo that use this feature.
It is possible to configure your json content in order to give all the blapy blocks information and the data to transmit to the blapy blocks.
To do so, send an array of objects, each one will describe a blapy block. All the blapy attributes may be given to configure your blapy block. The names are those described in this document without the "data-" at the beginning of the name. For example, "data-blapy-container" will have "blapy-container" as attribute object.
The necessary attributes are:
- blapy-container-name: name of the blapy container
- blapy-container-content: identifier name of the content
- blapy-data: the json data to send to the blapy block
It is expected that that the answer returns an array of blapy objects.
The following blapy block definitions are the same:
- usual blapy block
<div
data-blapy-container-name="fnameOptions"
data-blapy-container-content="fnameOptionsNew"
>
[
{"firstname": "John","lastname": "Doe","selected":"false"},
{"firstname": "Bob","lastname": "Dylan","selected":"false"},
{"firstname": "Peter","lastname": "Rabbit","selected":"false"},
]
</div>
- pure json blapy block corresponding to the above example
[
{ "blapy-container-name":"fnameOptions",
"blapy-container-content":"fnameOptionsNew",
"blapy-data":
[
{"firstname": "John","lastname": "Doe","selected":"false"},
{"firstname": "Bob","lastname": "Dylan","selected":"false"},
{"firstname": "Peter","lastname": "Rabbit","selected":"false"},
]
}
]
The idea is to directly call the url that will initialize your page by calling one of the Blapy API functions "loadURL" or "postData" just after the Blapy initialization.
Example:
$( document ).ready(function() {
$('#myBlapy').Blapy();
$('#myBlapy').trigger('loadUrl',{aUrl:"init_my_page.php"})
});
You have two attributes for Blapy blocks named "data-blapy-href " and "data-blapy-updateblock-time" that let you configure the URL to call and the period to call it.
This example will update the block every second (1000ms) from index.php:
<div data-blapy-container="true"
data-blapy-container-name="dateContainer"
data-blapy-container-content="aContent_<?php echo uniqid();?>"
data-blapy-href = "index.php"
data-blapy-updateblock-time = "1000"
>
<b>Time is:</b> <?php echo date('d-M-Y H:i:s');?>
</div>
Use the events "Blapy_beforePageLoad" and "Blapy_afterPageChange" in order to active/deactivate your loader, as in the following example:
<script>
$("#myBlapy").on( "Blapy_beforePageLoad", function(event) {
$('#loader').show();
});
$("#myBlapy").on( "Blapy_afterPageChange", function(event) {
$('#loader').hide();
});
</script>
Generally, the new block will replace the old one, and so, the id will follow... and that's mainly ok...
Sometime, if there are several blocks with the same data-blapy-container-name in order to update several blocks with the same info, it could be a problem that several new blocks get the same id after processing...
You can give no id on the new sent blocks, this way the system will set the id of the old block to change to the new one...
The idea is to add behaviour and features to our blapy object when it's ready, in order to do things like:
$("<myBlypyObject>").trigger('alertUs',"Hey there!");
To do that in the context of the FSM used in the Blapy object, we will extend it using fsmExtension option.
The safe state to extend is "PageReady", set when the blapy block object is ready to accept messages like "loadUrl" or "postData"...
In your blapy initialization, add your FSM extension as in this example:
$('#myBlapy').Blapy({
fsmExtension:{
'PageReady':{
'alertUs': {
init_function: function(parameters, event, data){
alert('alertUs called in '+this.myUIObject.attr('id')+' says: '+data);
}
}
}
}
});
remarks:
- 'this' refers to the FSM
- The Blapy object may be accessed with "this.opts.theBlapy".
- if you add new iFSM states, think to come back to the "PageReady" state at the end of your processing.
Thanks to the "data-blapy-template-init-processdata" parameter, you can give the name of function to do the preprocessing of your data and change and add any new properties to be processed by Mustache.
<div ....
data-blapy-template-init-processdata="initMyJsonData"
>
color is {{#greenColor}}green{{/greenColor}}{{^greenColor}}... well... I don't know...{{/greenColor}}
</div>
//....
<script>
// declare a function visible from Blapy
// aJson is an array of element(s) to process
window.initMyJsonData = function(aJson)
{
//for each elements, do an analysis and add/remove or do whatever you need to be done...
aJson = aJson.map(aData => {
if (aData["myPropertyColor"] == "green")
{
aData["greenColor"]=true;//add a new property
delete aData.myPropertyColor; //remove a property
}
return aData;//returns the modified array item
});
//return our modified json data array
return aJson;
}
</script>
The problem addressed here is to be able to have json blocks initialized according to an upper block configuration.
For example, if you have a request that gives you a list of people with their id like [{"id":0,"name":"Paul"}, ...]
, you perhaps would like get their details from another request that would need the id of the person to get the information.
To do that, you would need to nest your blapy blocks: the first level will get the list of your people, the second level will get for each one their details...
Using Mustache, You will need to accomplish this:
- the data-blapy-template-mustache-delimiterStart and data-blapy-template-mustache-delimiterEnd modifiers
- they will allow to distinguish between the first template (using the default Mustache delimiters "{{" and "}}") and the second one using an other set of tags like "{%" "%}"
- to define data-blapy-container-name according to upper identifier tags. you can use the "blapyIndex" tag to get different names for your second level block.
<div data-blapy-container="true"...blapy block 1st level definition...>
The id is {{id}}.<br>
The details are:<br>
<div data-blapy-container="true"
data-blapy-container-name="details_{{blapyIndex}}"
data-blapy-template-mustache-delimiterStart="{%"
data-blapy-template-mustache-delimiterEnd="%}"
data-blapy-template-init="/people/getDetails/?id={{id}}"
...Blapy block 2nd level definition...>
age of {{name}} is {%age%}<br>...
</div>
</div>
In a first pass, Blapy will get you the first level block parsed for every people using the "{{ delimiters }}", configuring your "data-blapy-template-init" URL with the id sent in the url. In a second pass, all the sub blocks will be parsed by Blapy, getting all the details for each sub blocks and parsed the results with the new "{% delimiters %}"...
As recommanded, you can use <XMP>
html tag to escape your json templates when needed... But there is a drawback as you can not nest this tag...
<XMP>
Hello,
<XMP>
I'm nested
</XMP>
xmp...
</XMP>
Won't work as expected... and the text after the first closing XMP tag will close the whole xmp blocks...
The solution proposed by Blapy is to escape your sub XMP blocks. This is done by adding as many '|' as there is sub XMP level as in this example:
//I'm a json template
<XMP>
Hello,
....
//I'm a sub json template
<|XMP>
I'm nested
....
//I'm a sub sub json template
<||XMP>
and subnested...
<||/XMP>
<|/XMP>
xmp...
</XMP>
Blapy will remove the "|" according to the level of parsing... and this way to nest as many json templates you like...
Have a look to the full demonstration in demos/demo_json_nested_blocks/index.html
- Did you verify that your external block has a different data-blapy-container-content content than the current one? If not, the content is not updated as it is considered to be the same... The content of data-blapy-container-content may be any name. You can use the current time or a unique id to set the name as in this example:
<div data-blapy-container="true"
data-blapy-container-name="dateContainer"
data-blapy-container-content="aContent_<?php echo uniqid();?>"
>
<b>Time is:</b> <?php echo date('d-M-Y H:i:s');?>
</div>
- Maybe, the code of your external block is not a valid HTML code. For instance, if you use tbody as a blapy block, don't set it alone, but embed it within a table tag.
Clicking to a blapy link generates several ajax calls though it should generate only a unique call...
- Verify your html return of the first ajax call. For instance,
<img src="" alt="">
will generate a second ajax call to index.html...
- It's generally a problem linked to base URL. YOu can fix it by setting a
<base>
html tag in your html head part:
<base href="/demos/todomvc/" target="_blank">
Blapy expects that you define the A and Form tags as blapy links if you want them to be handled by blapy.
Hereafter, you can add this little script to automate that every A / Form tags become "blapy-link=true":
// every new page load, will assure that every new links will have the "blapy-link" attribute
$(document).on( "Blapy_PageReady","body", function(event,anError) {
$('#[[+BlapyApplicationId]]').find('a,form').attr('data-blapy-link','true');
var myBlapy = $('#<You Blapy DOM Object>').getFSM();//get the FSM working behind the scene for blapy
myBlapy[0].opts.theBlapy.setBlapyUrl(); // call the function that will make blapy handle the url links
});
Once in place, every url links will be considered as Blapy Links...
When my template contains "img" tag with the name file defined by a placeholder, I've got a 404 error
As the file is parsed as HTML, img tag will try to load the image that does not exist as the image name is not the placeholder name.
To fix this, simply wrap your html template with the tag "xmp" which will neutralize html analysis.
My Blapy block does not appear when I change its style from "display:off" to "display:block" when "data-blapy-updateblock-ondisplay" is set
The jquery.appear object is not aware of a change in the display...
In order to alert it, you can simulate a scroll on the window just after changing the display status of your block with :
$(window).scroll();
If you have any ideas, feedback, requests or bug reports, you can reach me at github@intersel.org, or via my website: http://www.intersel.fr