UiBot is a small helper library to make using the Google Apps Script UI Service faster and more enjoyable. With the UiBot library, you can specify your interface using a simple declarative JSON syntax, rather then lots of chained function calls. UiBot also includes some helpers to reduce the amount of code needed to build forms.
While you can also create a UI with Google's recently released Html Service, by using the UI Service your dialogs will share a common design with Google Docs and will work smoothly across all of the browsers supported by Google Docs.
###UI Service without UiBot
function doGet() {
var app = UiApp.createApplication()
.setTitle('New app');
var grid = app.createGrid(4, 2);
var cityChoices = app.createListBox()
.setName('city').setId('city')
.addItem('New York').addItem('Paris')
.addItem('London').addItem('Tokyo');
cityChoices.setItemSelected(2, true);
grid.setWidget(0, 0, app.createLabel('Name:'));
grid.setWidget(0, 1, app.createTextBox()
.setName('userName').setId('userName'));
grid.setWidget(1, 0, app.createLabel('Age:'));
grid.setWidget(1, 1, app.createTextBox()
.setName('age').setId('age'));
grid.setWidget(3, 0, app.createLabel('City'));
grid.setWidget(3, 1, cityChoices);
var panel = app.createVerticalPanel();
app.add(panel);
panel.add(grid);
var button = app.createButton('submit');
var handler = app.createServerHandler('b');
handler.addCallbackElement(grid);
var handler2 = app.createClientHandler()
.forTargets(grid)
.setVisible(false);
button.addClickHandler(handler);
button.addClickHandler(handler2);
panel.add(button);
return app;
}
function b() {
Browser.msgBox('callback!');
}
###UI Service with UiBot
function doGet() {
var panel = {
wType: 'VerticalPanel',
items: [{
wType: 'HorizontalForm',
id: 'form',
items: [{
fieldLabel : 'Name:',
wType : 'TextBox',
name : 'userName'
},{
fieldLabel : 'Age:',
wType : 'TextBox',
name : 'age'
},{
blank : true
},{
fieldLabel : 'City:',
wType : 'ListBox',
name : 'city',
data : ['New York', 'Paris',
'London', 'Tokyo'],
selected : 'London'
}],
},{
wType : 'Button',
text : 'Submit',
onClick: {
callback : 'b',
cbElementId : 'form',
hide : 'form'
}
}] //panel items
}; //panel
bot = new UiBot.UiBot({
title: 'New App',
items: [ panel ]
});
return bot.getApp();
}
UiBot.b = function() {
Browser.msgBox('callback!');
}
This tutorial covers the use of UiBot and assumes some familiarity with Google Apps Script and the Google Apps Script UI Service. If you're new to either of these, you may want to take a look at the following tutorials before continuing.
If you'd like to follow along, be sure to add the UiBot library to your project. The project key is MdY_z6sqJlAAsbzYJGBr7WppQAdMKFO40 and instructions on adding a library to your development environment are here.
function doGet() {
var bot = new UiBot.UiBot();
return bot.getApp();
}
Google Apps Script requires you to return an App object to render the interface in a web apps. We could also show our interface in a spreadsheet by changing the last line of the code above.
function doGet() {
var bot = new UiBot.UiBot();
SpreadsheetApp
.getActiveSpreadsheet()
.show( bot.getApp() );
}
We can supply an optional configuration object to UiBot to set the title, width and height of our interface.
function doGet() {
var bot = new UiBot.UiBot({
title: 'Test App!',
width: 300,
height: 300
});
return bot.getApp();
}
When defining widgets with UiBot, you must specify a widget type using the wType property. Widgets may also have other properties (the part after the 'set' in the UI Service documentation). For example, the api function setText is equivalent to the UiBot property text). A full list of the widget types and their properties is here.
Let's create a label on our interface
function doGet() {
var bot = new UiBot.UiBot();
bot.addWidget({
wType : 'Label',
text : 'This is a label!'
});
return bot.getApp();
}
Widgets that can contain other widgets (such as panels) have an items property that can be filled with an array of other widgets. Let's create a panel with our label inside.
function doGet() {
var bot = new UiBot.UiBot();
bot.addWidget({
wType : 'VerticalPanel',
items : [{
wType : 'Label',
text : 'This is a label!'
}]
});
return bot.getApp();
}
Of course, there's no reason to create a panel for a single widget...
function doGet() {
var bot = new UiBot.UiBot();
bot.addWidget({
wType : 'VerticalPanel',
items : [{
wType : 'Label',
text : 'This is a label!'
},{
wType : 'Label',
text : 'This is a second label!'
}]
});
return bot.getApp();
}
We can nest widgets inside of the UiBot object too...
function doGet() {
var bot = new UiBot.UiBot({
title: 'Test App!',
items: [{
wType : 'VerticalPanel',
items : [{
wType : 'Label',
text : 'This is a label!'
},{
wType : 'Label',
text : 'This is a second label!'
}] //panel items
}] //app items
});
return bot.getApp();
}
Nesting is a choice though - we can also break out each widget into separate objects (somewhere between these extremes is usually the best choice to write clear code).
function doGet() {
var label1 = {
wType : 'Label',
text : 'This is a label!'
}
var label2 = {
wType : 'Label',
text : 'This is a second label!'
}
var panel = {
wType : 'VerticalPanel',
items : [ label1, label2 ]
}
var bot = new UiBot.UiBot({
title: 'Test App!',
items: [ panel ]
});
return bot.getApp();
}
We often need create forms with labels on the left and widgets on the right, so UiBot has a special helper for this. The HorizontalForm widget type automatically creates a grid with two columns and the correct number of rows. A new fieldLabel property in each widget makes it easy to set a label for the widget. You can also add a blank row to separate sections of your form using the blank property.
function doGet() {
var form = {
wType: 'HorizontalForm',
items: [{
fieldLabel : 'Name:',
wType : 'TextBox',
name : 'userName'
},{
fieldLabel : 'Age:',
wType : 'TextBox',
name : 'age'
},{
blank : true
},{
fieldLabel : 'City:',
wType : 'TextBox',
name : 'city'
}]
};
var bot = new UiBot.UiBot({
title: 'Test App!',
items: [ form ]
});
return bot.getApp();
}
It's also pretty common to create listboxes with data in them, so UiBot adds optional data and selected properties to the ListBox widget.
function doGet() {
var form = {
wType : 'HorizontalForm',
items : [{
fieldLabel : 'Period:',
wType : 'ListBox',
name : 'periodList',
data : ["Day", "Month","Quarter", "Year"],
selected : "Month"
}]
};
var bot = new UiBot.UiBot({
title: 'Test App!',
items: [ form ]
});
return bot.getApp();
}
To change the look of each widget, you can use the optional styles property which accepts a set of css styles.
function doGet() {
var bot = new UiBot.UiBot({
title: 'Test App!',
items: [{
wType : 'Label',
text : 'This is a label!',
styles : {
'font-size' : 20,
'color' : 'blue'
}
}]
});
return bot.getApp();
}
onClick handlers can also work on the client side to offer a faster response to the user. Let's create handlers that toggle two labels. The show and hide properties accept an array of widget ids. There are also properties to enable and disable widgets, which also take an array of widget ids.
function doGet() {
var bot = new UiBot.UiBot({
title: 'ui test',
items: [{
wType : 'Label',
text : 'This is label 1',
id : 'label1',
styles : { 'font-size' : 30 },
onClick : { 'show' : 'label2' }
},{
wType : 'Label',
text : 'This is label 2',
id : 'label2',
visible : false,
styles : { 'font-size' : 30 },
onClick : { 'hide' : ['label1', 'label2'] }
}]
});
return bot.getApp();
}
UiBot also makes it simple to add server side click handlers to buttons and other widgets. An onClick property can accept an object with the name of a callback function and an optional element id to be passed to this function (the cbElementId property). Note that Google Apps Script scopes the callback to the libary where it was created, so for now your server callback must be placed in the UiBot namespace (see the example below).
function doGet() {
var buttons = {
wType : 'HorizontalPanel',
items : [{
wType : 'Button',
text : 'Submit',
onClick : {
callback: 'submitHandler',
cbElementId: 'mainForm'
}
}]
};
var form = {
wType : 'HorizontalForm',
id : 'mainForm',
items : [{
fieldLabel : 'Period:',
wType : 'ListBox',
name : 'period',
data : ["Day", "Month","Quarter", "Year"],
selected : "Month"
}]
};
var bot = new UiBot.UiBot({
title: 'Test App!',
items: [ form, buttons ]
});
return bot.getApp();
}
UiBot.submitHandler = function(e) {
Browser.msgBox(e.parameter.period);
}
The code for UiBot is released under the permissive MIT license and was written in CoffeeScript and compiled to Javascript. It is available on Github at https://github.com/gotdan/UiBot. Inspiration for the design came from ExtJs
Future additions to UiBot may include support for other handlers and for validators. For now, you can build most of your interface in UiBot and then write non-click related handlers and validators using the traditional UI Service api.
Please feel free contact me with any questions or suggestions - dan at dan-gottlieb dot com.