|   |   | 
| (6 intermediate revisions by 4 users not shown) | 
| Line 1: | Line 1: | 
|  | {{Stability-stable}}
 |  | The content on this page has moved to https://documentation.open-xchange.com/7.10.2/ui/ | 
|  | 
 |  | 
 | 
|  | <div class="title">UI Development Style Guide</div>
 |  | Note: Open-Xchange is in the process of migrating all its technical documentation to a new and improved documentation system (documentation.open-xchange.com). Please note as the migration takes place more information will be available on the new system and less on this system. Thank you for your understanding during this period of transition. | 
|  |   |  | 
|  | __TOC__
 |  | 
|  |   |  | 
|  | There lots of stuff we don't need to talk about because JSHint will bug you! (white space, formatting, undefined variables, globals, etc.)
 |  | 
|  |   |  | 
|  | ''Eclipse seems to have a bug on MacOS that makes theeditor really slow when showing white-spaces. However, as long as you configure Eclipse properly to use spaces instead oftabs and remove alltrailing white-space on save (yep, there's a flag for that), you can leave that option turned off.''
 |  | 
|  |   |  | 
|  | == Javascript coding guidelines ==
 |  | 
|  |   |  | 
|  | ===Basic rules===
 |  | 
|  | * '''Indentation is 4 spaces'''<br>No tabs! And there's no need for two consecutive empty lines. If possible turn on "show white-spaces" in your editor.
 |  | 
|  | * '''Use underscore's high-level functions (e.g. each, filter, map)'''<br>Because it's nice toread and it uses native code if available. For example, iterating an array via .each() is faster than aclassic for-loop (in Chrome at least). Don't use jQuery's each (except for DOM nodes). If you need "break" you have to use a classic for-loop.
 |  | 
|  | * '''Don't make functions within a loop'''<br>For most cases, JSHint will bug you. But when using .each(), for example, it won't. However, you might still create functions over andover again - so avoid that. And if there no good reason, try to avoid creating nested sub functions (bit slower; might leak memory).
 |  | 
|  | * '''Require modules only when they're required!'''<br>Review your code if your module really needs all required modules upfront. Check if some dependencies can be resolved at runtime, e.g. event handlers or functions that are working asynchronously.
 |  | 
|  | ** ''Hint: We patched require.js, so require() returns a deferred object.''
 |  | 
|  | * '''Use jQuery's .on() and .off() instead of .bind() .unbind() .delegate()'''<br>Because the new event systemof jQuery 1.7 was completely redesigned and bind/unbind are now marked as deprecated.
 |  | 
|  | * '''Use delegated event handlers if possible'''<br>Instead of adding tons of click handlers for each element, use one (!) delegate on the parent element (VGrid uses that technique for example).
 |  | 
|  | * '''Don't create global code'''<br>underscore.js is an exception. There some basic jQuery plugin that extend jQuery.fn (that's global as well).Even for rarely used jQuery plugins create AMDs (Asynchronous Module Definition) and load them via require().
 |  | 
|  | * '''Naming'''<br>Use camelCase for variables (e.g. variableName). Use upper-case/underscores for constants (e.g.MAX_WIDTH).Use camel-case with upper-case first char for class names (e.g. ClassDefinition). Don't use special notations for jQuery-Objects: var node = $(…) is better than var $node = $(…);
 |  | 
|  | * '''Try to define all variables at the beginning of a function'''<br>And please just use one (!) "var" statement.
 |  | 
|  | ** An exception could be if you plan to break from thecontrol flow before doing an expensive call in the variable definition.
 |  | 
|  | * '''Use $.Deferred() whenever possible'''<br>Instead of using simple callbacks. Remember that your functions might need an error callback as well.
 |  | 
|  | ** ''Hint: You can use "return $.when();" instead of "return $.Deferred().resolve();"''
 |  | 
|  | * '''Use options instead of long argument lists'''<br>Avoid creating functions like foo("1234", true, null, null, cb_success). Instead use: foo({ folder: "1234", cache: true }).done(….);
 |  | 
|  | *'''Don't use $('<div class="foo" style="float: left">Hello</div>')'''<br> This is hard to read, hard to extend, doesn't allow i18n. Might become evil once 'Hello' is replaced by a variable (evil HTML injection). Just use $('<div>') plus proper helpers like addClass(), css(), text() etc.
 |  | 
|  | * '''Use $('<div>') instead of $('<div/>')'''<br>There's no need for XHTML.
 |  | 
|  | * Prefer $('<input>', { type: 'radio' }); over $('<input>').attr('type', 'radio');<br>Actually there's a semantic difference (not just syntax) - IE willteach you this! Never please write $('<label ...></label>'), $('<div></div>') or even $('<input...></input>')
 |  | 
|  | * Try to write readable code - even if JSHint is already happy. Example: { a: 1000, b: 2000 } has better readability than {a:1000,b:2000}.
 |  | 
|  | * '''Don't COPY/PASTE code you don't understand!'''<br>'''Never ever do COPY/PASTE inheritance'''<br>Always try to code stuff by yourself. Don't just imitate what others developed. If it looks like what you need, understand it first. If you can use it, try to really reuse it (define & require are your friends). If you need a slight modification, try to add that via options. Talk to the author if possible.
 |  | 
|  | * '''Of course… Don't repeat yourself!'''<br>Try to besmart. Look for patterns. Create local helper functions once you have to do stuff twice.
 |  | 
|  |   |  | 
|  | ===Quotes===
 |  | 
|  | Use single quotes, unless you are writing JSON.
 |  | 
|  | <pre class="language-javascript"> 
 |  | 
|  | // right
 |  | 
|  | var foo = 'bar';
 |  | 
|  | // wrong
 |  | 
|  | var foo = "bar";
 |  | 
|  | </pre>
 |  | 
|  |   |  | 
|  | ===Braces===
 |  | 
|  | Your opening braces go on thesame line as the statement.
 |  | 
|  |   |  | 
|  | <pre class="language-javascript"> 
 |  | 
|  | // Right
 |  | 
|  | if (true) {
 |  | 
|  |   console.log('winning');
 |  | 
|  | }
 |  | 
|  | // Wrong
 |  | 
|  | if (true)
 |  | 
|  | {
 |  | 
|  |   console.log('losing');
 |  | 
|  | }
 |  | 
|  | </pre>
 |  | 
|  | Also, notice the use of whitespace before andafter the condition statement.
 |  | 
|  |   |  | 
|  | ===Conditions===
 |  | 
|  | Any non-trivial conditions should be assigned to a descriptive variable:
 |  | 
|  |   |  | 
|  | <pre class="language-javascript"> 
 |  | 
|  | // Right
 |  | 
|  | var isAuthorized = (user.isAdmin() || user.isModerator());
 |  | 
|  | if (isAuthorized) {
 |  | 
|  |   console.log('winning');
 |  | 
|  | }
 |  | 
|  |   |  | 
|  | // Wrong
 |  | 
|  | if (user.isAdmin() || user.isModerator()) {
 |  | 
|  |   console.log('losing');
 |  | 
|  | }
 |  | 
|  | </pre>
 |  | 
|  |   |  | 
|  | ===Function length===
 |  | 
|  | Keep your functions short. A good function fits on a slide that the people in the last row of a big room can comfortably read. So don't count onthem having perfect vision and limit yourself to ~10 lines of code per function.
 |  | 
|  |   |  | 
|  | ===Return statements===
 |  | 
|  | To avoid deep nesting of if-statements, always return a functions value as early as possible.
 |  | 
|  |   |  | 
|  | <pre class="language-javascript"> 
 |  | 
|  | // Right
 |  | 
|  | function isPercentage(val) {
 |  | 
|  |   if (val < 0) {
 |  | 
|  |     return false;
 |  | 
|  |   }
 |  | 
|  |   if (val > 100) {
 |  | 
|  |     return false;
 |  | 
|  |   }
 |  | 
|  |   return true;
 |  | 
|  | }
 |  | 
|  |   |  | 
|  | // Wrong
 |  | 
|  | function isPercentage(val) {
 |  | 
|  |   if (val >= 0) {
 |  | 
|  |     if (val < 100) {
 |  | 
|  |       return true;
 |  | 
|  |     } else {
 |  | 
|  |       return false;
 |  | 
|  |     }
 |  | 
|  |   } else {
 |  | 
|  |     return false;
 |  | 
|  |   }
 |  | 
|  | }
 |  | 
|  |   |  | 
|  | // Or for thisparticular example it may also be fine to shorten things even further
 |  | 
|  | function isPercentage(val) {
 |  | 
|  |   var isInRange = (val >= 0 && val <= 100);
 |  | 
|  |   return isInRange;
 |  | 
|  | }
 |  | 
|  | </pre>
 |  | 
|  |   |  | 
|  | ===Creating instances===
 |  | 
|  | <pre class="language-javascript"> 
 |  | 
|  | // Simple one-time instances
 |  | 
|  | var obj = { foo: 'bar' };
 |  | 
|  |   |  | 
|  | // Classes.Must start with upper camel case (JSHint will bug you):
 |  | 
|  | var Klass = function () {
 |  | 
|  |   var local = 'foo';
 |  | 
|  |   this.doSomething = function () {
 |  | 
|  |     // has access to private variable local
 |  | 
|  |   };
 |  | 
|  | };
 |  | 
|  |   |  | 
|  | // Do NOT use (except forreally good performance reasons):
 |  | 
|  | var Klass = function () {
 |  | 
|  |   this.local = 'foo';
 |  | 
|  | };
 |  | 
|  | Klass.prototype.doSomething = function () {
 |  | 
|  | };
 |  | 
|  | </pre>
 |  | 
|  |   |  | 
|  | ==Commit log & Bugzilla==
 |  | 
|  | Please use the commit log to generally explain what has changed. We need to use a consistent notation for bugs:
 |  | 
|  |   |  | 
|  | <pre class="language-text"> 
 |  | 
|  | Fixed: <just copy&paste bugzilla title>
 |  | 
|  | Fixed: Bug 12345 - Nothing works, all broken
 |  | 
|  | </pre>
 |  | 
|  |   |  | 
|  | Uppercase "Fixed", a colon, a space. For L3 bugs, please add a short explanation (1-2 sentences) to your bugzilla comment. Separate that using a complete newline (because several git tools send commit messages as e-mails, where the first line is the subject and the following empty line denotes the begin ofthe message body). This helps QA a lot.
 |  | 
|  |   |  | 
|  |   |  | 
|  | [[Category:AppSuite]]
 |  | 
|  | [[Category:UI]]
 |  | 
|  | [[Category:Development process]]
 |  |