User Tools

Site Tools

Translations of this page?:

en:tutorial:tutorial01

This is an old revision of the document!


A PCRE internal error occured. This might be caused by a faulty plugin

<WRAP center round tip 60%> **Troubleshooting** - During the course of this tutorial, it could happen that your template modifications are not updated. First, check your Json code with a tool like [[http://jsonlint.com/|Json Validator]]. Maybe a special character is missing ? This tool will help you find it. It could also be that something is not working good in the software. Remember that it's still in a beta version. Create a [[https://github.com/Gulix/geckos/issues/new|new Issue]] and describe your problem, with the name & verison of your browser, the content of your Javascript console (if you know how to get it), and the code of your template. </WRAP> <WRAP center round important 60%> **Out of date** - Since the release of the beta version of Geckos, some elements of this tutorial are now obsolete. A new version is on its way. </WRAP> ====== Tutorial - A simple template from scratch ====== Welcome to this first tutorial for Geckos, which will explain, step by step, how to create a simple template. The card that we'll create through this process is a card for the Blood Bowl Starplayers. You can then have your own Star Players cards, or have the classic ones with your minis photos on it. {{:tutorial:01_nobbla_original.jpg?500|Nobbla, Classic Chainsaw Player}} ===== A Blank Template ===== Let's start with a "blank" template. We need a simple text file, that will be filled with a json object. Just call it ''myTemplate.json'', and open it (with atom.io, notepad++ or another text editor of your choice). We'll start with the required fields. We need to set a ''styles'' array, that will contain our ''style'' object, with required properties. Our first field will be the one we'll use to store/set the name of our Star Players : <code javascript template-skeleton.json> { "styles": [ { "fields": [ { "name": "name", "label": "Name", "default": "Nobbla " } ], "canvasFields": [ ], "canvasBackground": "#FF0000", "canvasWidth": 536, "canvasHeight": 750 } ] } </code> So, this is our starting point. Let's put that template in the Geckos. Open up the [[http://gulix.github.io/geckos|demo page]], display the **Template** pane, and paste the above code in the template area. Click the button with the cogs (//Apply template to cards//)to validate the changes. {{:en:tutorial:01_ui01.png?200|}} Go back to the **Card** pane, the existing card should be selected. It's been updated but its old name (//Captain Wolf//) has been kept. It's because the default template and the one you created share the same field : ''name''. The existing card has been updated to your new template, but the data has stayed the same. //Create// a new card by clicking on the plus button. Nobbla is now in the list ! It's the default value we set in the field ''name'' for our template. If you set another value to this ''default'' property, any new card you'll add will have that new name. {{:en:tutorial:01_ui02.png?200|}} ===== A Background and a Name ===== Right now, our card is a big red rectangle. That's because our ''canvasFields'' section is empty. Using FabricJS (and its [[http://fabricjs.com/kitchensink|kitchensink]]), we could build the items of the card : stars, boxes, ... But that will be for another card and another tutorial. Instead, I've got a card background set in a png file. We'll set it as background for our card. Here's [[http://tof.canardpc.com/view/2346e727-216f-4841-bff4-8041db9d94e2.jpg|the image file]] we'll use. As the file is hosted on a web server, we could use its url to use it, and get this image field to add to the canvas (the canvas is where the card is drawn) : <code javascript> "canvasFields": [ { "type": "image", "width": 536, "height":750, "src": "http://tof.canardpc.com/view/2346e727-216f-4841-bff4-8041db9d94e2.jpg" } ], </code> That'll work, but what if the webserver is down ? Or if the image on the server is deleted ? Then, our template becomes useless. And that's not good. But there is a way to //embed// images into template. It's called [[https://en.wikipedia.org/wiki/Data_URI_scheme|DataUrl]]. The goal here is to //translate// a file into a string, that the browser will be able to //untranslate//, getting back the original file. To achieve this, you can use the [[http://dataurl.net/#dataurlmaker|DataUrl Maker]] and drop your file on it to get your DataUrl. The string you'll get will replace the url of the file. It will increase the size of your template greatly, but your template will be self-contained : <code javascript> "canvasFields": [ { "type": "image", "width": 536, "height":750, "src": "" } ], </code> Paste this new code section in place of the old ''canvasFields'' section. Then press the **Set** button, and watch your card being updated. We now have the background of our card, but we want to be able to edit it. We already have an [[..:template:styles#editable_fields|editable field]] with the ''name'' variable. We'll work with it. As this is an [[..:template:input-text|Input Text Field]], it generates a variable with the name ''$name''. Let's add a **textbox** element to the card, between the top stars. Why not a **text** element ? Because the textbox element will allow us more flexibility, with the alignment and the wrapping. Add this code to the ''canvasFields'' section, separated from the background image definition with a comma. <code javascript> { "type": "textbox", "text": "$name", "left": 91, "top": 11, "width": 351, "textAlign": "center" } </code> As you can see, it's pretty simple : we add a ''textbox'', at the position ''{91, 11}'', with a width of 351 pixels, and a text-alignment centered. This render our card name correctly thanks to the variable **$name**, who gets the value set in the corresponding field. But the text is poorly rendered. The [[http://fabricjs.com/docs/fabric.Textbox.html|FabricJS documentation]] lists many properties for the textbox item. <code javascript> { "type": "textbox", "text": "$name", "left": 91, "top": 11, "width": 351, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700 } </code> We've just added three properties : * **fill** sets the color of the text. It accepts any color in the form of a RGB hexadecimal value. * **fontFamily** indicates which font will be used. It looks on the user font, so make sure it's a common font (or [[..:template:custom-fonts|use a Custom Font]]). * **fontWeight** can accept numeric values (400, 600, 800, ...) or text values ("bold", "normal", ...). It defines the boldness of the text. We've not added a property to set the **fontSize**. It exists, and is set by default to 40 (see [[http://fabricjs.com/docs/fabric.Textbox.html#fontSize|FabricJS doc]]), and this size is well suited for our needs. But let's set it to 38 to see it in effect. Our whole ''canvasFields'' section now looks like this (I've reduced the DataUrl for the code to be easier to read, consider to restore it if you're testing it) : <code javascript> "canvasFields": [ { "type": "image", "width": 536, "height":750, "src": ""...." }, { "type": "textbox", "text": "$name", "left": 91, "top": 11, "width": 351, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 38 } ], </code> Our card now displays the name of our Star Player ! If you change it in the left editable field, it changes as well on the card image (as soon as you leave the field). Great ! Let's add some more fields ! {{:tutorial:01_ui_preview.jpg|What it should look like}} ===== Some selection to make ===== Before considering adding some more **editable fields** to our template, let's complete the card. The different boxes which will contain the editable values have no text to describe them. We will add them by adding some **textbox** elements to the ''canvasFields'' section. Those elements will not be linked to a variable, and we could have set them on the background image. But what if we want to make our template accessible to other languages ? We'll just have to change the value of those fields instead of having to edit the image, and get the new //dataurl// for the background image. So, **textbox** elements it is ! <code javascript> { "type": "textbox", "text": "$name", "left": 91, "top": 11, "width": 351, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 38 }, { "type": "textbox", "text": "MOVEMENT", "left": 386, "top": 126, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "STRENGTH", "left": 386, "top": 218, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "AGILITY", "left": 386, "top": 310, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "ARMOUR", "left": 386, "top": 402, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "TEAMS", "left": 386, "top": 494, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "SKILLS", "left": 18, "top": 494, "width": 347, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 } ], </code> I've repeated the **$name** field to show you what part of the template I've modified. Don't repeat it. To know where to put those new fields, open up your background image in an image editor, like [[http://www.getpaint.net/index.html|Paint.NET]], and select the area where you want to put your field. The coordinates of your selection are usually shown on the bottom info bar. No height is set for the TextBoxes, as they auto-adjust their height, depending on how much text they got. Now, let's think of our 4 fields for the player's attributes : Movement, Strength, Agility & Armour. We could use a **text input** field, but the user could then put an Agility value of "XX", which is meaningless. We'll opt instead for a [[..:template:choice-input|dropdown menu]], with only the existing and accepted values proposed. First, we'll go back to the ''fields'' section of the template. We'll add four **options fields**. For each, we define the options that the user will be able to select. Each option has a value and a text. The text is (for now) only used as a way to display the options. <code javascript> "fields": [ { "name": "name", "label": "Name", "default": "Nobbla" }, { "type": "options", "name": "movement", "label": "Movement", "default": "6", "options": [ { "option" : "1", "text": "1" }, { "option" : "2", "text": "2" }, { "option" : "3", "text": "3" }, { "option" : "4", "text": "4" }, { "option" : "5", "text": "5" }, { "option" : "6", "text": "6" }, { "option" : "7", "text": "7" }, { "option" : "8", "text": "8" }, { "option" : "9", "text": "9" }, { "option" : "10", "text": "10" } ] }, { "type": "options", "name": "strength", "label": "Strength", "default": "3", "options": [ { "option" : "1", "text": "1" }, { "option" : "2", "text": "2" }, { "option" : "3", "text": "3" }, { "option" : "4", "text": "4" }, { "option" : "5", "text": "5" }, { "option" : "6", "text": "6" }, { "option" : "7", "text": "7" } ] }, { "type": "options", "name": "agility", "label": "Agility", "default": "3", "options": [ { "option" : "1", "text": "1" }, { "option" : "2", "text": "2" }, { "option" : "3", "text": "3" }, { "option" : "4", "text": "4" }, { "option" : "5", "text": "5" } ] }, { "type": "options", "name": "armour", "label": "Armour", "default": "8", "options": [ { "option" : "5", "text": "5" }, { "option" : "6", "text": "6" }, { "option" : "7", "text": "7" }, { "option" : "8", "text": "8" }, { "option" : "9", "text": "9" }, { "option" : "10", "text": "10" } ] } ], </code> Edit your template code, and click the **Set** button. Under the previous **Name** field, 4 new fields appears : that's our **options** fields ! {{:tutorial:01_new_fields.jpg|Our new fields}} Now, like we did previously, we'll add four new **Textboxes** in our ''canvasFields'' section. Those elements will have a **text** property set with the variables that are generated by our options. Right now, the only variable generated by an option field is the text of the ''value'' property. So, we add four new fields for our canvas, linked to our four new variables. We use again **textboxes**, with an adjustment of the color and the size. Here is the added code, the first line being one already existing for you to know where to paste this code : <code javascript> { "type": "textbox", "text": "SKILLS", "left": 18, "top": 494, "width": 347, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "$movement", "left": 389, "top": 160, "width": 124, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 36 }, { "type": "textbox", "text": "$strength", "left": 389, "top": 252, "width": 124, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 36 }, { "type": "textbox", "text": "$agility", "left": 389, "top": 344, "width": 124, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 36 }, { "type": "textbox", "text": "$armour", "left": 389, "top": 436, "width": 124, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 36 } </code> We've now got an editable card, but some fields are still missing. Let's illustrate our card ! ===== Mug Shot ===== Our Star Player need a photo to be correctly identified. For this purpose, we'll use an [[..:template:image-input|Image field]]. This type of field allows the user to load an image for each card. Right now, the Image input is pretty basic. But it's [[https://github.com/Gulix/geckos/issues/37|planned to evolve]]. For the moment, the user has to ensure his image file corresponds, in ratio, to the image frame on the card. To add the **Image input** field, just add this line to the ''fields'' part of the template : <code javascript> { "type": "image", "name": "portray", "label": "Portray" } </code> This field will generate a **$portray** variable, which will contain the image field in the form of a dataurl (remember ?). Just what we need to add a field image to our canvas : <code javascript> { "type": "image", "left": 15, "top": 124, "width": 354, "height":352, "src": "$portray" } </code> **Set** the template to update your modifications, and a new field appears. {{:tutorial:01_select_your_image.jpg|Select your image file}} Load your image and see it on the card : {{:tutorial:01_nobblamugshot.jpg|Nobbla in person !}} ===== Skills to complete the card ===== Three fields are still missing. Let's start with the easier one. Each Star Player requires a certain amount of Gold Pieces to be recruited. Let's add an input field that will appear in the bottom of the card. The editable field : <code javascript> { "name": "cost", "label": "Cost", "default": "80.000 GP" } </code> And the canvas field : <code javascript> { "type": "textbox", "text": "$cost", "left": 18, "top": 684, "width": 504, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 40, "fontStyle": "italic" } </code> The new thing here is the use of the **fontStyle** property in the canvas field. It sets the text to italic. The next input field will be a new one : the [[..:template:multiline-text-input|Multiline Text Input]] field. It works like the classic **Text input** field, but allows the user to add line-breaks. We'll add two fields : <code javascript> { "name": "skills", "label": "Skills", "default": "", "type": "multiline" }, { "name": "teams", "label": "Teams", "default": "All", "type": "multiline" } </code> We now add two more **textboxes** that will be linked to these two fields : <code javascript> { "type": "textbox", "text": "$skills", "left": 20, "top": 524, "width": 342, "textAlign": "left", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 500, "fontSize": 28 }, { "type": "textbox", "text": "$teams", "left": 388, "top": 524, "width": 128, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 600, "fontSize": 22 } </code> Those fields use an automatic wrapping method : lines that are too long are broken to be displayed on several lines. ===== That's it ? ===== For this tutorial, yes, that's it. You know have a functional template that allows you to get Star Player cards. Of course, with the help of this tutorial, I encourage you to create your own template for your own needs. I will soon open a place to share the templates, and the demo page will host various templates (when [[https://github.com/Gulix/geckos/issues/30|this issue]] is completed). If you have any trouble with this tutorial or the software, feel free to ask me for help, via the creation of an [[https://github.com/Gulix/geckos/issues/new|Issue]], or via a message on a forum in which I'll participate. ===== Full code of the template ===== <code javascript template01-code.json> { "fields": [ { "name": "name", "label": "Name", "default": "Nobbla" }, { "type": "options", "name": "movement", "label": "Movement", "default": "6", "options": [ { "option" : "1", "text": "1" }, { "option" : "2", "text": "2" }, { "option" : "3", "text": "3" }, { "option" : "4", "text": "4" }, { "option" : "5", "text": "5" }, { "option" : "6", "text": "6" }, { "option" : "7", "text": "7" }, { "option" : "8", "text": "8" }, { "option" : "9", "text": "9" }, { "option" : "10", "text": "10" } ] }, { "type": "options", "name": "strength", "label": "Strength", "default": "3", "options": [ { "option" : "1", "text": "1" }, { "option" : "2", "text": "2" }, { "option" : "3", "text": "3" }, { "option" : "4", "text": "4" }, { "option" : "5", "text": "5" }, { "option" : "6", "text": "6" }, { "option" : "7", "text": "7" } ] }, { "type": "options", "name": "agility", "label": "Agility", "default": "3", "options": [ { "option" : "1", "text": "1" }, { "option" : "2", "text": "2" }, { "option" : "3", "text": "3" }, { "option" : "4", "text": "4" }, { "option" : "5", "text": "5" } ] }, { "type": "options", "name": "armour", "label": "Armour", "default": "8", "options": [ { "option" : "5", "text": "5" }, { "option" : "6", "text": "6" }, { "option" : "7", "text": "7" }, { "option" : "8", "text": "8" }, { "option" : "9", "text": "9" }, { "option" : "10", "text": "10" } ] }, { "type": "image", "name": "portray", "label": "Portray" }, { "name": "cost", "label": "Cost", "default": "80.000 GP" }, { "name": "skills", "label": "Skills", "default": "", "type": "multiline" }, { "name": "teams", "label": "Teams", "default": "All", "type": "multiline" } ], "canvasFields": [ { "type": "image", "width": 536, "height":750, "src": "" }, { "type": "textbox", "text": "$name", "left": 91, "top": 11, "width": 351, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 38 }, { "type": "textbox", "text": "MOVEMENT", "left": 386, "top": 126, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "STRENGTH", "left": 386, "top": 218, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "AGILITY", "left": 386, "top": 310, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "ARMOUR", "left": 386, "top": 402, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "TEAMS", "left": 386, "top": 494, "width": 132, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "SKILLS", "left": 18, "top": 494, "width": 347, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 22 }, { "type": "textbox", "text": "$movement", "left": 389, "top": 160, "width": 124, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 36 }, { "type": "textbox", "text": "$strength", "left": 389, "top": 252, "width": 124, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 36 }, { "type": "textbox", "text": "$agility", "left": 389, "top": 344, "width": 124, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 36 }, { "type": "textbox", "text": "$armour", "left": 389, "top": 436, "width": 124, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 36 }, { "type": "image", "left": 15, "top": 124, "width": 354, "height":352, "src": "$portray" }, { "type": "textbox", "text": "$cost", "left": 18, "top": 684, "width": 504, "textAlign": "center", "fill": "#FFD800", "fontFamily": "Arial", "fontWeight": 700, "fontSize": 40, "fontStyle": "italic" }, { "type": "textbox", "text": "$skills", "left": 20, "top": 524, "width": 342, "textAlign": "left", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 500, "fontSize": 28 }, { "type": "textbox", "text": "$teams", "left": 388, "top": 524, "width": 128, "textAlign": "center", "fill": "#000000", "fontFamily": "Arial", "fontWeight": 600, "fontSize": 22 } ], "canvasBackground": "#FF0000", "canvasWidth": 536, "canvasHeight": 750 } </code>

en/tutorial/tutorial01.1471007275.txt.gz · Last modified: 2016/08/12 15:07 by Nicolas Ronvel