Yii
Yii Framework Related Posts
Free PHP, HTML, CSS, JavaScript editor (IDE) – Codelobster PHP Edition
2
Introduction
I do not normally post anything related to IDE’s but I thought this article was worth as it supports Yii
as one of the frameworks of the IDE. Thought this could be a good IDE for those Window Yii’ers out there.
Codelobster
For valuable work on creation of sites you need a good comfortable editor necessarily. There are many requiring paid products for this purpose, but we would like to select free of charge very functional and at the same time of simple in the use editor - Codelobster PHP Edition.
Let us consider some important possibilities and advantages of this program:
- All code highlights depending on a type, the also mixed code is thus supported, so the area of HTML will be highlighted as HTML, PHP as PHP, and Javascript as Javascript in a the same file. Thre is possibility of choice from color schemes, including popular IDEs.
- Powerful autocompletion for HTML, PHP, CSS and Javascript, including HTML5 and CSS3. For PHP the structure of project is fully recognized, and the complete list of methods falls out in the proper places.
- HTML/CSS inspector on the type of Firebug, which allows easily to correlate the selected elements of page with a code and proper style.
- Context help on all supported languages. By pressing F1 key the page with detailed description for current tag, attribute or function will be opened.
- PHP debugger. PHP debugger allows to execute PHP scripts incrementally, watching the values of all variables in every line.
- SQL manager allows to produce all necessary actions with a database – to add, delete, edit a structure and records in tables, to export data, execute SQL queries. Highlighting and autocompletion works for SQL files also.
- Support of FTP allows to work straight with a remote server and to do all necessary changes with files;
- The portable option allows to use editor without the preliminary installation.
- Other useful utilities: pair highlighting, possibility of blocks selection, collapsing, tooltips, navigation on descriptions of functions and included files at withholding of the key of CTRL, viewing of structure of files and project, preview in a browser, book-marks, and all other standard possibilities for work with a code.
Also there are special plugins for work with
- CMS: Drupal, Joomla
- PHP frameworks: CakePHP, CodeIgniter, Symfony, Yii
- JavaScript libraly: JQuery
- WordPress blogging engine
- Smarty template engine
How to work with Yii and CodeLobster
Tweet this!
EScriptBoost Extension for Yii
3Introduction
Probably a lot of you would wonder why, having so many good extensions related to minifying/compressing/packing your javascript code and your css files at the Yii Extensions Repository, here comes this guy offering us another solution.
I did check out all the extension in our repository, just to name some of them:
All of them are great, but none were filling the requirements we had. I did not have any issue compressing all our files as our team, will use the YUI compressor jar file to create our compressed javascript versions and then use the wonderful mapping features of CClientScript. The issue was with the assets of external, or own developed, extensions and the javascript code that, even Yii own widgets, were writing to the POS_BEGIN, POS_END, POS_HEAD, POS_LOAD, POS_READY positions. Thats exactly what this library is doing: allowing Yii coders to minify those scripts.
Library
The library comes with three flavors:
- EScriptBoost Component
- EClientScriptBoost Extension
- AssetManagerBoost Extension
EScriptBoost Component
This is a very easy to use component to compress your Javascript or CSS code at your will. The minifiers used are:
- For CSS- CssMin and CssMinify (with CssCompressor and CssUriRewriter classes that you can also use independently)
- For JS- JsMin, JsMinPlus and JavaScriptPacker
Usage
// this is a very simple example// we use cache as we do not want to // compress/minify all the time our // script $js = Yii::app()->cache->get('scriptID'); if(!$js) { $cacheDuration = 30; $js = <<<EOD // my long and uncompressed code here EOD; // $js = EScriptBoost::packJs($js); // $js = EScriptBoost::minifyJs($js, EScriptBoost::JS_MIN_PLUS); $js = EScriptBoost::minifyJs($js, EScriptBoost::JS_MIN); // see Cache guide for more options | dependencies Yii::app()->cache->set('scriptID', $cacheDuration); } Yii::app()->clientScript->registerScript('scriptID', $js);
That was troublesome right? No worries, if you don’t really care about using JS_MIN, or JS_MIN_PLUS, you can use its helper function registerScript, it will handle all of the above automatically:
$js = <<<EOD
// my long and uncompressed code here
EOD;
EScriptBoost::registerScript('scriptID', $js);
EClientScriptBoost Extension
EScriptBoost was good for the javascript code written by me but what about the ones written by Yii widgets?EClientScriptBoost was developed to solve that:
Usage
On your main.php config file:
'import' => array(
// ... other configuration settings on main.php
// ... importing the folder where scriptboost is
'application.extensions.scriptboost.*',
// ... more configuration settings
),
// ... other configuration settings on main.php
'components' => array(
'clientScript' => array(
// ... assuming you have previously imported the folder
// where EClientScriptBoost is
'class'=>'EClientScriptBoost',
'cacheDuration'=>30,
// ... more configuration settings
Done! now, every time you or other component on your application will be minified and cached as you specify on your cache settings. Easy right?
EAssetManagerBoost Extension
But there was one more challenge to solve. Some extensions, widgets, etc, do publish a whole bunch of files in our assets that are not minified. This is where EAssetManagerBoost comes handy.
This extension does only minify javascript/css files, and also makes sure that the, about to be compressed, file do not match any of its $minifiedExtensionFlags so minified/compressed files are not processed at all.
Usage
Make sure you have deleted your previous assets folder contents.
'import' => array(
// ... other configuration settings on main.php
// ... importing the folder where scriptboost is
'application.extensions.scriptboost.*',
// ... more configuration settings
),
// ... other configuration settings on main.php
'components' => array(
'assetManager' => array(
// ... assuming you have previously imported the folder
'class' => 'EAssetManagerBoost',
'minifiedExtensionFlags'=>array('min.js','minified.js','packed.js')
),
// ... more configuration settings
Important Note There is a small drawback to use EAssetManagerBoost and is that, the first time your application is requested, it will take a bit of time as it will go throughout all your asset files to be published and minify them.
Resources
Tweet this!
Using events with CAction classes
2Introduction
There are some good guides out there explaining how to work with events and the ways to attach them to your components, but none (that I know) explain the following way to configure your events with CAction classes on your controllers.
As you know, events are used by:
- Declaring an event in your component adding its method (ie. function onClick($event))
- Attach it to event handlers (ie. $object->onClick=array($handlerObject,’staticmethod’);)
- Raising it from your component to call all subscribed handlers (ie. $this->raiseEvent(‘onClick’,$event)). Remember, that for a handler, you can write an object with static methods, an object with a method, create a function (create_function) and even attach a function directly (since PHP 5.3)
Tip
If we look at the magic method __set of CComponent, we will see that event handlers are actually set like properties.
Having that into account, the following is my quick tip to set your event handlers when working with CAction classes, which I think is far much better to organize your code in your controllers.
The CAction class
Lets write a simple CAction class for the example and save it as EMyAction.php:
class EMyAction extends CAction{
public function onTest($event){
$this->raiseEvent('onTest', $event);
}
public function run() {
$event = new CEvent($this);
$this->onTest($event);
}
}
The Controller
Now in our controller, for the sake of the example, lets write a method handler and configure the action (assumed to be on actions folder under, which is in controllers folder).
// our event handler method, that, for simplicity,
// we set it in our controller
public function eventHandlerMethod($event)
{
echo 'TESTING Handler';
}
// declaring actions and its event handlers
public function actions()
{
return array(
// test is the action name <controller/action>
'test'=>array(
'class'=>'actions.EMyAction',
'onTest'=>array($this,'eventHandlerMethod')
)
);
}
And that’s it, call the controller’s action as you would with any other in your preferred browser to test the results.
Tweet this!
Yii 1.1 Application Development Cookbook -by Alexander Makarov
2
As most of you know, I am a heavy user of the Yii’s forum. Lots of times, people keep asking what is the best way to learn Yii, what should be the steps? Where are good resources to look for their answers… I once wrote an article: Yii Learning Guide, that should be now updated with a new great resource, the new book written by Alexander Makarov (samdark), Yii 1.1 Application Development Cookbook.
I had the pleasure to be one of the reviewers of the book, and I must admit that I felt more like a student than a reviewer
. The book is an excellent guide to solve, through practical examples, some of the challenges a Yii programmer would face during its Yii application development:
- How events work
- Using Exceptions
- Components, Widgets, Collections
- Deep on Router, Controllers and Views
- AJAX and JQuery with Yii
- Forms, ActiveRecord, Models and Database
- Zii Components
- Extensions
- Error Handling, debugging and Logging
- Security and performance tuning
- Using external libraries
- Deployment
In addition, the book is not only a good practical reference to the points above, but also directs to external references on the web for further reading. It is a perfect companion for our online Yii site, class reference, wiki, and forum… I have mine already, when are you going to get yours?
Tweet this!
Yiianswers, a new site for yii lovers
0
I would like to introduce a new site that a good friend of mine and work colleague Maurizio Domba (mdomba) and I have setup in order to solve an issue that actually occurs at the Yii forum and help the Yii community with what we think could be a great tool for their learning or development processes: Yiianswers.com
Don’t get us wrong, the forum is amazing, highly addictive (you can check our profiles -mdomba an tonydspaniard to find out how we love that forum) and very useful thanks to the support of a lot of good programmers that freely give aways their time to help others get the right answers, in order to push Yii to the level where it supposed to be in terms of PHP community acceptance.
Nevertheless, one of the things that we face as the forum grows is that the same questions are repeated again and again, and we believe that this is not because users do not know how to search for a solution but for the structure and functionality of a forum itself. We always thought that could be great to have a site to keep questions and its solutions in a categorized tree where people could easily browse, search, and/or find what they are looking for, and this is why Yiianswers.com is here.
Yiianswers.com is an instant Questions and Answers site, so popular nowadays. We really hope that will help you out and will easy the task of finding the right answers to your Yii questions.
Tweet this!
Avoiding duplicate script download when using CActiveForm on Ajax calls
0Introduction
Sometimes the active form we wish to use to edit/add a new element on our database is too small and we believe that is much better to use an AJAX’ed dialog/slide form rather than reloading the page to just display one or two fields.
The only thing required is simple, we just need to create a view that will be partially rendered by a call to a controller (using renderPartial) and make sure that we process output -setting to true the parameter on the function. Everything will work as expected but…
The issue
If we open firebug (firefox), or developer tools (chrome), or whatever the tool you use in order to see the XmlHttpRequest object calls and resources downloaded, you will see that every time we do call the controller to display the active form, different Yii “core JS” files keeps being downloaded to the client. The JS files downloaded depends on your code but there are at least jquery.js, jquery-ui.js and jquery.yiiactiveform.js.
The solution
The solution is a bit tricky but simple. We need to pre-render the jquery.yiiactiveform.js on the view where we are going to place the AJAX functionality (the button that opens the modal dialog or slides/shows a layer with AJAX’ed form contents). For example, on index.php view file:
cs()->registerCoreScript('yiiactiveform');
Now, I assume that you have created your function to display the AJAX’ed active form and its contents are returned by a call to a controller’s action that will partially render a view. This is what we have to do in our action:
// Just before rendering the view that // has our activeform Yii::app()->clientScript->corePackages = array();
It is very important that we set corePackages to array() instead of null, as setting it to null will make CClientScript to reload the packages.php file (located in framework/web/js/) and we won’t stop the duplication of the script.
And that’s it, everything is working as it should.
Tweet this!
Custom Autocomplete Display and Value Submission
4Introduction
How many of us has wondered how to create an autocomplete that will display the names of a related models but do require the id of that selected name to be submitted for model creation/update?
I was looking around wiki and found that was no approach as the one I did so I guessed this is worth to write.
Requirements
For our example, I want to be able to:
- Have an autocomplete field in our form
- Once user selects an item in the dropdown list and fill a hidden box with the id of the selected item for submission
Making the right choice
To setup the autocomplete was a very straight forward operation, but I couldn’t figure out how to get values from a custom JSON response and then fill the correspondent hidden fields.
CAutoComplete does has a way to do it, but I wanted to use CJuiAutoComplete to get all the cool features of its JQuery Ui and by looking at his code there was no method chain, something that is required to work with custom JSON responses as we need to override some methods.
My Solution
After doing some research I decided to:
- extend from CJuiAutoComplete
- include the required property for method chain and modify its ‘run’ function
- then initialize the newly created property with the javascript functions that handle my custom JSON
Extending from CJuiAutoComplete and make required modifications
Very simple, we are going to add a methodChain property and modify the run function to include it (zii is not a major concern to Yii, but main developers should think about this minor change).
class myAutoComplete extends CJuiAutoComplete
{
/**
* @var string the chain of method calls that would be appended at the end of the autocomplete constructor.
* For example, ".result(function(...){})" would cause the specified js function to execute
* when the user selects an option.
*/
public $methodChain;
/**
* Run this widget.
* This method registers necessary javascript and renders the needed HTML code.
*/
public function run()
{
list($name,$id)=$this->resolveNameID();
if(isset($this->htmlOptions['id']))
$id=$this->htmlOptions['id'];
else
$this->htmlOptions['id']=$id;
if(isset($this->htmlOptions['name']))
$name=$this->htmlOptions['name'];
if($this->hasModel())
echo CHtml::activeTextField($this->model,$this->attribute,$this->htmlOptions);
else
echo CHtml::textField($name,$this->value,$this->htmlOptions);
if($this->sourceUrl!==null)
$this->options['source']=CHtml::normalizeUrl($this->sourceUrl);
else
$this->options['source']=$this->source;
$options=CJavaScript::encode($this->options);
$js = "jQuery('#{$id}').autocomplete($options){$this->methodChain};";
$cs = Yii::app()->getClientScript();
$cs->registerScript(__CLASS__.'#'.$id, $js);
}
}
Using our widget
Now that we have our beautiful widget that handles method chain in our Autocomplete, let’s assume a couple of things:
- We saved our class onto a folder in our application -ie protected/extensions
- We have a hidden INPUT HTML element with model’s attribute_id
- We have created an action on our testController named autocomplete that returns a JSON object on the following format:
// This function will echo a JSON object
// on this format:
// [{id:id, name: 'name'}]
public function actionAutocomplete(){
$res = array();
$term = Yii::app()->getRequest()->getParam('term', false);
if ($term)
{
// test table is for the sake of this example
$sql = 'SELECT id, name FROM {{test}} where LCASE(name) LIKE :name';
$cmd = Yii::app()->db->createCommand($sql);
$cmd->bindValue(":name","%".strtolower($term)."%", PDO::PARAM_STR);
$res = $cmd->queryAll();
}
echo CJSON::encode($res);
Yii::app()->end();
}
We have everything, let’s use our widget in our view:
// REMEMBER, we have a hidden
// input HTML element with model's attribute_id
<?php echo $form->hiddenField($model, 'attribute_id'); ?>
<?php
// ext is a shortcut for application.extensions
$this->widget('ext.myAutoComplete', array(
'name' => 'test_autocomplete',
'source' => $this->createUrl('test/autocomplete'),
// attribute_value is a custom property that returns the
// name of our related object -ie return $model->related_model->name
'value' => $model->isNewRecord ? '': $model->attribute_value,
'options' => array(
'minChars'=>3,
'autoFill'=>false,
'focus'=> 'js:function( event, ui ) {
$( "#test_autocomplete" ).val( ui.item.name );
return false;
}',
'select'=>'js:function( event, ui ) {
$("#'.CHtml::activeId($model,'attribute_id').'")
.val(ui.item.id);
return false;
}'
),
'htmlOptions'=>array('class'=>'input-1', 'autocomplete'=>'off'),
'methodChain'=>'.data( "autocomplete" )._renderItem = function( ul, item ) {
return $( "<li></li>" )
.data( "item.autocomplete", item )
.append( "<a>" + item.name + "</a>" )
.appendTo( ul );
};'
));
?>
Done! Just make sure that when you do submit your form, you get the value from the hidden field instead of the autocomplete element
Final Notes
I do not know if there are other ways of doing the same thing (apart from pure Javascript) to have the same results. If you know, with CJuiAutoComplete widget, let us know here.
Hope you find this useful.
Cheers
Tweet this!
Implementing a User Level Access System
2I would like to explain this time how easy is to implement a level access system with Yii framework.
Please note that this article is a simple example and good security should be taken into account when we play with authentication systems.
Step 1: Setting Up
a. Include a field on your user’s table named, yep you guessed, level
b. Create an object ‘LevelLookUp’ that will tell us who is who on this system
class LevelLookUp{
const MEMBER = 0;
const ADMIN = 2;
// For CGridView, CListView Purposes
public static function getLabel( $level ){
if($level == self::MEMBER)
return 'Member';
if($level == self::ADMIN)
return 'Administrator';
return false;
}
// for dropdown lists purposes
public static function getLevelList(){
return array(
self::MEMBER=>'Member',
self::ADMIN=>'Administrator');
}
}
c.Modifying UserIdentity so we can store the user Model id on authentication, as CWebUser’s default id is set to the user Model’s username:
class UserIdentity extends CUserIdentity
{
private $_id;
/**
* Authenticates a user.
* @return boolean whether authentication succeeds.
*/
public function authenticate()
{
$username = strtolower($this->username);
// from database... change to suite your authentication criteria
$user = User::model()->find('LOWER(username)=?', array($username));
if($user===null)
$this->errorCode=self::ERROR_USERNAME_INVALID;
else if(!$user->validatePassword($this->password))
$this->errorCode = self::ERROR_PASSWORD_INVALID;
else{
$this->_id = $user->id;
$this->username = $user->username;
$this->errorCode = self::ERROR_NONE;
}
return $this->errorCode == self::ERROR_NONE;
}
public function getId()
{
return $this->_id;
}
}
d. Modify CWebUser’s application in order to hold the ‘level’ property. We are going to call it EWebUser and will extend from CWebUser, and save it on protected/components to be loaded automatically by our default’s configuration file.
class EWebUser extends CWebUser{
protected $_model;
function isAdmin(){
$user = $this->loadUser();
if ($user)
return $user->level==LevelLookUp::ADMIN;
return false;
}
// Load user model.
protected function loadUser()
{
if ( $this->_model === null ) {
$this->_model = User::model()->findByPk( $this->id );
}
return $this->_model;
}
}
e. Modify our main.php config file (this file is in protected/config folder)
// go to the 'user' section
// application components
'components'=>array(
'user'=>array(
// There you go, use our 'extended' version
'class'=>'application.components.EWebUser',
// enable cookie-based authentication
'allowAutoLogin'=>true,
),
Step 2: Putting everything together
Now that we know, who logged, we could easily find out if it is just a member or an administrator and render the elements by checking the level as simple as this:
// for normal content
if(Yii::app()->user->isAdmin())
echo 'Is administrator';
// for CMenus
$this->widget('zii.widgets.CMenu',array(
array('label'=>'Categories',
'url'=>array('/category/index'),
'visible'=>(Yii::app()->user->isAdmin()),
//... More stuff
//...
// for data chuncks
<?php if(Yii::app()->user->isAdmin():?>
<b>My HTML</b>
<?php endif;?>
// for access rules
return array(
array('allow',
'actions'=>array('create','delete','update'),
'expression'=>'$user->isAdmin()'
),
// ...
Tweet this!
Actions code reuse with CAction
3Introduction
We all know how good ‘gii’ automates the code for us and we normally tend to be happy with what that tool offers at the beginning of our Yii learning curve. But as soon as you start working in larger and larger projects, you realize that its code is too repetitive to maintain and having a small pitfall in general actions means to go over and over through them to fix the issues.
CAction to the Rescue
I have already explained how to use widgets as action providers to encapsulate the actions. What I am going to explain here is how can we easily create an action to work throughout different controllers.
Gii provides us normally with the following code on the ‘actionCreate’:
public function actionCreate()
{
$model=new ModelName;
// Uncomment the following line if AJAX validation is needed
// $this->performAjaxValidation($model);
if(isset($_POST['ModelName']))
{
$model->attributes=$_POST['ModelName'];
if($model->save())
$this->redirect(array('view','id'=>$model->id));
}
$this->render('create',array(
'model'=>$model,
));
}
For a normal project and with the default CMS layout of Yii, does suit our regular needs and we tend to leave it as it is. But, as I said before, imagine that we need to include a new parameter in our redirection for example? In order to avoid that we can tweak a bit the code and develop a general action.
Step 1 – Creating the Action
For the sake of the example, create the following action and save it on your protected/components/actions folder
class Create extends CAction {
public function run() {
$controller = $this->getController();
// get the Model Name
$model_class = ucfirst($controller->getId());
// create the Model
$model = new $model_class();
// Uncomment the following line if AJAX validation is needed
// $this->performAjaxValidation($model);
if (isset($_POST[$model_class])) {
$model->attributes = $_POST[$model_class];
if ($model->save())
$controller->redirect(array('view', 'id' => $model->id));
}
$controller->render('create', array(
'model' => $model,
));
}
}
Step 2 – Declare the action on the Controller
Once we have the action class created, the only thing we need to do is declare it in our controller’s actions function in order to use it.
public function actions(){
return array(
'create'=>'application.components.actions.create',
);
}
After declaring the action we can call it: http://myhost/index.php?r=controller/create, just like any other.
Final Notes
In the example above I have used ‘getController()’ and ‘getId()’ in order to access the model, but we can actually use properties as CAction is a class. This could be the declaration of an action passing the model name to load:
// Assuming the action class has the
// following public properties:
// public model_name
// -----------------
// ModelClass is a test model class name
// -----------------
// On the controller:
public function actions(){
return array(
'create'=>array(
'class'=>'application.components.actions.create',
'model_name'=>'ModelClass',
);
}
Tweet this!
EGMaps 2.0 News: Layers, Polygons and Rectangles
3The new version of EGMaps 2.0 is about to see the light. In the meantime new features have been included and are available through the SVN source at google’s code. Even though we are planning many more features, the following are the list of the newly inserted features of the Yii extension till date:
- Polygons (committer Matthias Kay)
- Rectangles
- Circles
- Bicycling Layer
- Traffic Layer
- Panoramio Layer
Polygon Example
Yii::import('ext.egmap.*');
$gMap = new EGMap();
$gMap->setWidth(588);
$gMap->setHeight(345);
$gMap->zoom = 3;
$gMap->mapTypeControlOptions = array(
'position'=> EGMapControlPosition::RIGHT_TOP,
'style'=>EGMap::MAPTYPECONTROL_STYLE_DROPDOWN_MENU
);
$gMap->setCenter(34.04924594193164, -118.24104309082031);
$coords = array();
$coords[] = new EGMapCoord(25.774252, -80.190262);
$coords[] = new EGMapCoord(18.466465, -66.118292);
$coords[] = new EGMapCoord(32.321384, -64.75737);
$coords[] = new EGMapCoord(25.774252, -80.190262);
$polygon = new EGMapPolygon($coords);
$gMap->addPolygon($polygon);
$gMap->centerOnPolygons();
$gMap->zoomOnPolygons(0.1);
$gMap->renderMap(array(),'en','ES');
Circle and Rectangle Example
Yii::import('ext.egmap.*');
$gMap = new EGMap();
$gMap->setWidth(588);
$gMap->setHeight(345);
$gMap->zoom = 3;
$gMap->mapTypeControlOptions = array(
'position'=> EGMapControlPosition::RIGHT_TOP,
'style'=>EGMap::MAPTYPECONTROL_STYLE_DROPDOWN_MENU
);
$gMap->setCenter(34.04924594193164, -118.24104309082031);
$circle = new EGMapCircle(new EGMapCoord(34.04924594193164, -118.24104309082031));
$circle->radius = 300000;
// we can even attach info windows to the overlay!
$circle->addHtmlInfoWindow(new EGMapInfoWindow('Hey! I am a circlel!'));
$gMap->addCircle($circle);
$bounds = new EGMapBounds(new EGMapCoord(25.774252, -80.190262),new EGMapCoord(32.321384, -64.75737) );
$rec = new EGMapRectangle($bounds);
$rec->addHtmlInfoWindow(new EGMapInfoWindow('Hey! I am a rectangle!'));
$gMap->addRectangle($rec);
$gMap->renderMap(array(),'en','ES');
Panoramio Layer Example
$gMap = new EGMap(); $gMap->setWidth(588); $gMap->setHeight(345); $gMap->zoom = 3; $gMap->mapTypeControlOptions = array( 'position'=> EGMapControlPosition::RIGHT_TOP, 'style'=>EGMap::MAPTYPECONTROL_STYLE_DROPDOWN_MENU ); $gMap->setCenter(34.04924594193164, -118.24104309082031); // we can also use the same way TRAFFIC and BICYCLING layers $gMap->setLayer(new EGMapLayer(EGMapLayer::PANORAMIO)); $gMap->renderMap(array(),'en','ES');
The extension has also been updated its reverse geocoding (committer Say_Ten).
Please, remember that in order to work with the above examples you require to download the svn source from the google code, not the zipped package on Yii’s repository. The extension will be updated when we reach the following goals:
- Elevation Paths
- Polylines
- Ground Overlays
- Animations
- Map Styling
Tweet this! 
Try to register and click again. Sometimes the plugin i have to count downloads do fail.There was an issue with the cache, solved by […] 10 hours ago