Thursday, November 5, 2009

Strings.jsm library

function: Strings aPropertiesFile
Construct an instance of the Strings class.
aPropertiesFile An URI for a properties file. String.

function: get aName &optional aArgs
Get a localized string.
aName A name of key. String.
aArgs An arguments for a formatted string like the printf. Array of string.

Wednesday, November 4, 2009

API Reference: Database.jsm library

Database.jsm library
OR Mapper for the SQLite RDBMS builted in the Firefox.
Developped by tombloo.

Example.
// Make a Model for "Bookmark".
var Bookmark = Entity({
name : 'bookmarks',
fields : {
id : 'INTEGER PRIMARY KEY',
url : 'TEXT UNIQUE NOT NULL',
title : 'TEXT',
date : 'TIMESTAMP NOT NULL',
last_visited : 'TIMESTAMP',
comment : 'TEXT',
}
})
// Get the file of database(at "ProfD/foobar/foobar.sqlite").
function dbFile() {
var pd = DirectoryService.get("ProfD", Ci.nsIFile);
pd.append("foobar");
if (!pd.exists() || !pd.isDirectory()) {
pd.create(Ci.nsIFile.DIRECTORY_TYPE, 0755);
}
pd.append("foobar.sqlite");
return pd;
}

// Construct an instance of the Database class.
var db = new Database(dbFile());
// Set the instance to the model of Bookmark.
Bookmark.db = db;
// initialize the model.
Bookmark.initialize();
...
//Bookmark.insert, find, update, deleteById, and so on.
...
// Close the database.
db.close();

Remarks#1: TIMESTAMP type, LIST type.

By specifing a TIMESTAMP type, you can read and write the Javascript's Date object transparently.
(Note: Actually the SQLite doesn't have the Date type, so the library store the value of "Date.getTime()" as INTEGER type.)

Similarly, by specifing a LIST type, you can do the Array object transparently.
(Note: The library store the value of "A CSV-format string made from the array" as TEXT type.)

Remarks#2: findByFoo, findByFooAndBar, countByFoo、countByFooAndBar

You can find and count the model with its value of "Foo" or "Bar". This is realized by hooking the __noSuchMethod__.

Reference Manual
Database Class

function: Database aFile
Construct an instance of the Database class.
aFile a file path of the database file of the SQLite. An instance of nsIFlie.

getter: version
Get the version of the database, corresponding to the PRAGMA's user_version(not to the schema_version).

setter: version
Set the version of the database.

function: getPragma aName
Get the value of the PRAGMA.
aName A name of the PRAGMA. String.

function: setPragma aName aVal
Set the value of the PRAGMA.
aName A name of the PRAGMA. String.
aVal A value to set. String.

function: createStatement aSQL
Create a statement. It returns a mozIStorageStatementWrapper.
aSQL A SQL statement. String.

function: bindParams aWrapper aParams
Bind a parameter to the statement. It returns a mozIStorageStatementWrapper of the binded statement.
aWrapper A statement. An instance of the mozIStorageStatementWrapper.
aParams A parameter. Dispatch for its type in the followings manner.

object: bind as a named parameter.
array: bind the elements in order in the array.
value: bind to the first parameter.
null: do nothing.

function: getParamNames aWrapper
Get an array of the named parameters of the statement.
aWrapper A statement. An instance of the mozIStorageStatementWrapper.

function: getColumnNames aStatement
Get an array of the column names of the statement.
aStatement A statement. An instance of the mozIStorageStatement or the mozIStorageStatementWrapper.

function: getRow aRow aColumnNames
Get a object which converted the table rows.
aRow A table row. An instance of the mozIStorageStatementRow.
aColumnNames A list of the column names. Array.

function: execute aSQL aParams
Execute the SQL query. You can use both DDL and DML.
aSQL A SQL statement. String or mozIStorageStatementWrapper.
aParams A parameter for the SQL statement. Object or Array or Statement, same with the bindParams.

function: transaction aProc
Execute operations in a transaction.
Use for a batch processing which requires high performance.
When it raise an error, we'll rollback the transaction.
Otherwise, we'll commit it automatically.
If another transaction was already started, we don't start any transaction.
aProc A oparation. A function with no arguments.

function: beginTransaction
Begin a transaction.
We don't raise error still the transaction is already started.

function: commitTransaction
Commit a transaction.
We don't raise error still the transaction is not started.

function: rollbackTransaction
Rollback a transaction.
We don't raise error still the transaction is not started.

function: throwException aError
Interpret an error of the database and throw it again.
aError An error of the database.

function: close
Close the database.
You have to close it to delete the file, as it's locked.

function: tableExists aName
Check wheather a database exists or not.
aName A name of the tabel. String.

function: vacuum
Vacuum a waste of the data area in the database.

Model Class

function: save
Save a model to the database. Firstly do with "insert" with a generated id, nextly with "update".

function: remove
Remove a model from the database. We use deleteById internally.

property: definitions
An object indicates the entity definition of the model.

function: initialize
CREATE a table.
We execute the following SQL internally.

CREATE TABLE IF NOT EXISTS {def.name} (
{def.fields.join(', ')}
)

function: deinitialize
DROP a table.
We execute the following SQL internally.

DROP TABLE {def.name}

function: insert
INSERT a model.
We execute the following SQL internally.

INSERT INTO {def.name} (
{fields.join(', ')}
) VALUES (
{params.join(', ')}
)
function: update
UPDATE a model.
We execute the following SQL internally.

UPDATE {def.name}
SET {fields}
WHERE id = :id

function: deleteById aId
DELETE a model by id.
aId An id of the model. Integer.
We execute the following SQL internally.

DELETE FROM {def.name}
WHERE id = :id

function: deleteAll
DELETE all of models.
We execute the following SQL internally.

DELETE FROM {def.name}

function: countAll
Count all of models.
We execute the following SQL internally.

SELECT count(*) AS count
FROM {def.name}

function: findAll
Find all of models.
We execute the following SQL internally.

SELECT *
FROM {def.name}

function: findFirst aParams
Find a model with a parameter "LIMIT 1".
aParams A parameter, same with Database.execute.
We execute the following SQL internally.
SELECT *
FROM {def.name}
...Contents of aParams...
LIMIT 1 OFFSET 0

function: find aSQL aParams
Find models with a parameter.
aSQL An object or an instance of the mozIStorageStatementWrapper.
aParams A parameter, same with Database.execute.
We execute the following SQL internally when aParams is null, aSQL is object, and aSQL.where is string.

SELECT *
FROM {def.name}
WHERE {sql.where}

We execute the following SQL internally when aParams is null, aSQL is object.

SELECT *
FROM {def.name}

Otherwise, We do the Database.execute internally.

Model.db.execute(sql, params).map(Model.rowToObject);

function: findByFoo aFoo
function: findByFooAndBarAnd... aFoo aBar ...
Find models by a value of the Foo or the Bar. This is realized by hooking the __noSuchMethod__.
We execute the following SQL internally.

SELECT *
FROM {def.name}
WHERE foo = aFoo and bar = aBar and ...

function: countByFoo aFoo
function: countByFooAndBar... aFoo aBar ...
Count models by a value of the Foo or the Bar. This is realized by hooking the __noSuchMethod__.

We execute the following SQL internally.

SELECT count(id) AS count
FROM {def.name}
WHERE foo = aFoo and bar = aBar and ...

function: rowToObject aObject
Construct an instance of the model from an object, except when you call the "save" method, we will call the "update".
aObject An object with proper elements for the entity definition.

Entity Class


function: Entity aDefinition
Construct an instance of the model with the entity definition.
aDefinition An object indicates the entity definition.
(name property is a name of the table, fields property is a mapping between a name of entity and its type.

Example.
var Bookmark = Entity({
name : 'bookmarks',
fields : {
id : 'INTEGER PRIMARY KEY',
url : 'TEXT UNIQUE NOT NULL',
title : 'TEXT',
date : 'TIMESTAMP NOT NULL',
last_visited : 'TIMESTAMP',
comment : 'TEXT',
}
})

function: createWhereClause aFields
Create a string for a WHERE clause.
aFields An array of parameters' name for WHERE.

function: createInitializeSQL aDefinition
Create a string for a "CREATE TABLE IF NOT EXISTS..." statement.
aDefinition An object indicates the entity definition, same with Entity.

function: createInsertSQL aDefinition
Create a string for a "INSERT INTO..." statement.
aDefinition An object indicates the entity definition, same with Entity.

function: createUpdateSQL aDefinition
Create a string for a "UPDATE ..." statement.
aDefinition An object indicates the entity definition, same with Entity.

function: compactSQL aSQL
Remove useless whitespaces from a SQL statement and create a compact string of it.
This aims to hold the variation of the statement's representation to a minimum and increase a cache-hit rate of interpreted statements.

aSQL A string of the SQL statements.

Bibliography

* Storage - MDC
* mozIStorageStatement - MDC
* mozIStorageStatementWrapper - MDC

Tuesday, November 3, 2009

API Reference: Prefs.jsm library

API Reference

Prefs.jsm library

What is the User Preferences?
Remarks #1

Every preferences have a type. There are three simple types and some complex types.
  • boolean
  • integer
  • string
  • and some complex types.
Remarks #2

Don't overlap your preference's name to others. Keep to use your original branch (ex. "extensions." or something like that) is a good manner.

Example #1: How to handle the "javascript.options.strict" preference.

  • Construct a instance of Prefs Class.
//Specify a branch. Note: you must terminate it with a period.
var prefs = new Prefs("javascript.options.");
  • Set the value of preference to true.
prefs.set("strict", true);
  • Get the value of preference.
var v = prefs.get("strict");
That's all. The library can infer a proper type for the preference automatically (and you can specify it manually if you want).

Reference Manual

function: Prefs aBranchName
Construct a new instance.
aBranchName A branch's name. String. You must terminate it with a period.

function: get aPrefName &optional aDefaultValue aType
Get the value of preference.
aPrefName A preference's name. String.
aDefaultValue A return value when the function failed to get.
aType A type of preference. String. Please see the table.1 below.

function: set aPrefName aValue &optional aType aRelFileRelativeToKey
Set the value of preference.
aPrefName A preference's name. String.
aValue A value to set.
aType A type of preference. String. Please see the table.2 below.
aRelFileRelativeToKey A keyword for the relative file path's root directory(ex. "ProfD", etc). String.

function: clear aPrefName
Clear the preference.
aPrefName A preference's name. String.

function: getChildList &optional aStartingAt
Get the array of preferences' name of below the branch.
aStartingAt A branch's name. String.

Table.1: aType for get
type
aType
calls internally
boolean"boolean"getBoolPref(aPrefName)
integer"integer"getIntPref(aPrefName)
string"string" getComplexValue(aPrefName, Ci.nsISupportsString).data
localized string"localized" getComplexValue(aPrefName, Ci.nsIPrefLocalizedString).data
absolute file path"file"getComplexValue(aPrefName, Ci.nsILocalFile)
relative file path"relFile" getComplexValue(aPrefName, Ci.nsIRelativeFilePref)

Table.2: aType for set

type
aType
calls internally
boolean"boolean" setBoolPref(aPrefName, !!aValue)
integer"integer"setIntPref(aPrefName, +aValue)
string"string" setComplexValue(aPrefName, Ci.nsISupportsString, nsISupportsString's instance)
localized string"localized" setComplexValue(aPrefName, Ci.nsIPrefLocalizedString, nsIPrefLocalizedString's instance
absolute file path"file" setComplexValue(aPrefName, Ci.nsILocalFile, aValue)
relative file path"relFile" setComplexValue(aPrefName, Ci.nsIRelativeFilePref, nsIRelativeFilePref's instance)

Please refer the File I/O - MDC about the relative file path.

Friday, February 29, 2008

Generate Rogue-like dungeon in Javascirpt

........................................
........................................
..###........############...............
..###.###################...............
..###.#......############...............
..#####......############...............
..###.#......############.#####...####..
..###.#......############.#.###...####..
..#####......#....#.......#.###...####..
..###.#......#....#.......#.#####.####..
..###.#......##...#.......#.###.#.####..
..###.#.......#...#.......#.###.#.####..
..###.#.....###############.###.#.####..
..###.#.....##########......###.#.####..
..###.#.....##########......###.#.####..
..###.################......###.######..
..###.......##########......###...####..
............##########..................
........................................
........................................

Example
I'll explain how to generate a rogue-like dungeon (maze) randomly.

First of all, get a 2-dimensional array, dungeon[40][80].
Then initialize it and show the zero-cleared dungeon.


<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
<title>Dungeon</title>
<meta http-equiv="content-type" content="text/html; charset=utf-8" />
<script src="./lib/prototype.js" type="text/javascript"></script>
<script type="text/javascript" language="javascript">
// <![CDATA[

var dungeon_width = 80;
var dungeon_height = 40;

dungeon = new Array(dungeon_height);
for (j=0; j < dungeon_height; j++) {
dungeon[j] = new Array(dungeon_width)
for (i=0; i < dungeon_width; i++) {
dungeon[j][i] = '.';
}
}

function main() {
dungeon.each(function(x) {
var elt = document.createElement('code');
elt.appendChild(document.createTextNode(x.join('')));
$('main').appendChild(elt);
$('main').appendChild(document.createElement('br'));
});
}

window.onload = main;
// ]]>
</script>
</head>
<body>
<div id="main"></div>
</body>
</html>


Secondly, divides rectangles recursively to split the dungeon.
Example reload page to make new splitting.


function split(rect) {
if ((rect.x1 - rect.x0 < margin * 2) ||
(rect.y1 - rect.y0 < margin * 2) ||
(random_range(0,5) === 0)) {
  //above is a ending condition.
return [rect];
} else {
if (random_range(0,2) === 0) {
//split vertical
var a = random_range(rect.y0 + margin, rect.y1 - margin);
return [new rectangle(rect.x0, rect.y0, rect.x1, a),
new rectangle(rect.x0, a, rect.x1, rect.y1)].map(split).flatten();
} else {
//split horizonal
var a = random_range(rect.x0 + margin, rect.x1 - margin);
return [new rectangle(rect.x0, rect.y0, a, rect.y1),
new rectangle(a, rect.y0, rect.x1, rect.y1)].map(split).flatten();
}
}
}


Recursively calls "split" divides rectangles more and more. And the last "flatten" makes them to a list of rects.
Make a room in each rects. These rects help to avoid overlapping rooms each other.
I'm going to connects them with corriders like this.


Now we get "Every rooms is always connected each other at least once".
From this, we can avoid a bad patten of below, it doesn't connected.


Finally, makes corrider between the rooms.
To connect rooms divided horizonal, Use a "N-letter" corrider, and for vertical one use "Z-letter" corrider.
Decide whether horizonal or vertical by their coordinate of rectangles.
Example (reloading page will generate a new dungeon.)

function make_corrider(partitions, rooms) {
var connect_list = new Array();

for (var i = 0; i < partitions.length - 1; i++)
connect_list.push([i, i + 1]);

var connect = function(p0,p1,r0,r1) {
if (p0.y1 == p1.y0) {
var a = random_range(r0.x0, r0.x1);
var b = random_range(r1.x0, r1.x1);
fill(new rectangle(a, half(p0.y0 + p0.y1), a, p0.y1));
fill(new rectangle(b, half(p1.y0 + p1.y1), b, p1.y0));
fill(new rectangle(a, p0.y1, b, p1.y0));
} else if (p0.x1 == p1.x0) {
var a = random_range(r0.y0, r0.y1);
var b = random_range(r1.y0, r1.y1);
fill(new rectangle(half(r0.x0 + r0.x1), a, p0.x1, a));
fill(new rectangle(half(r1.x0 + r1.x1), b, p1.x0, b));
fill(new rectangle(p0.x1, a, p1.x0, b));
}
};

connect_list.each(function(x){
connect(partitions[x[0]],partitions[x[1]],rooms[x[0]],rooms[x[1]])});
}




Additionaly, add corriders to give more randomness.


for (var i = 0; i < partitions.length; i++)
for (var j = i + 1; j < partitions.length; j++)
if (random_range(0,5) == 0)
if (partitions[i].x0 == partitions[j].x1 ||
partitions[i].x1 == partitions[j].x0 ||
partitions[i].y0 == partitions[j].y1 ||
partitions[i].y1 == partitions[j].y0)
connect_list.push([i,j]);


Complete.
Example

Minesweeper in Javascript

http://www.geocities.jp/teruakigemma/nikoniko/minesweeper.html


var field;
var closed_field;

function field_onclick(p) {
switch(field[p]) {
case '*':
field.each(function(x,i) {
if (x == '*') {
$('img' + i).src = 'mine.png';
new Effect.Shake('img' + i);
}
});
alert('Game Over');
main();
break;
case 0:
zero_region(p).map(neighbor).flatten().uniq().each(function(x) {
closed_field[x] = null;
$('img' + x).src = 'mine' + field[x] + '.png';
$('' + x).disabled = 'disabled';
});
break;
default:
closed_field[p] = null;
$('img' + p).src = 'mine' + field[p] + '.png';
$('' + p).disabled = 'disabled';
break;
}
if (closed_field.compact().length == 10) {
field.each(function(x,i) {
if (x == '*') {
$('img' + i).src = 'mine.png';
new Effect.Puff('img' + i);
}
});
alert('Congratulation!');
}
}

function zero_region(p) {
var result = [];
var f = function(n) {
var l = [].without.apply(neighbor(n).select(function(x) {return (field[x] == 0)}),result);
if (l.length != 0) {
result = result.concat(l);
l.each(f);
}
}
f(p);
return result;
}

function neighbor(p) {
var x = Math.floor(p / 9);
var y = p % 9;
var f = function(i,j) {
if (i < 0 || i > 8 || j < 0 || j > 8)
return null;
else
return i * 9 + j;
};
return [f(x-1,y-1),f(x,y-1),f(x+1,y-1),
f(x-1,y),f(x,y),f(x+1,y),
f(x-1,y+1),f(x,y+1),f(x+1,y+1)].compact();
}

function make_mine_field() {
field = $R(0,81,true).map(function(x) {return 0});
closed_field = $R(0,81,true).map(function(x) {return true});
var i = 0;
while(i < 10) {
var p = Math.floor(Math.random() * 81);
if (field[p] != '*') {
field[p] = '*';
i++;
}
}
field.each(function(x,i) {
if (x != '*')
field[i] = neighbor(i).select(function(y) {
return (field[y] == '*')}).length});
}

function main() {
make_mine_field();
Builder.dump();
var e = $('main');
while (e.firstChild) e.removeChild(e.firstChild);
field.inGroupsOf(9).each(function(x,i){
x.each(function(y,j) {
var num = i * 9 + j;
e.appendChild(BUTTON({id: num, onclick: 'field_onclick(' + num + ')'},
IMG({id: 'img' + num, src: 'field.png'})))});
e.appendChild(BR())});
}

window.onload = main;

Depth-first search with append-map.

As Haskell demonstrating depth-first search in the List monad(concatMap), we can easily do it in javascript with append-map.

Search the pair (a,b) such that a + b % 2 == 0.
(prototype.js is required.)


Array.prototype.append = function() {
return [].concat.apply([],this);
}

function solve() {
var a = [0,1,2];
var b = [1,2,4];
return a.map(function(x) {
return b.map(function(y) {
if ((x + y) % 2)
return [[x,y]];
else
return [];
}).append()}).append();
}


And now, let's solve the problem "SEND+MORE=MONEY" in javascript. I used "flatten" in this case.


function solve() {
// var digs = [0,1,2,3,4,5,6,7,8,9];
var digs = $A($R(0,9));
var m = 1;
var o = 0;
var s = 9;
return digs.without(m,o,s).map(function(e) {
return digs.without(m,o,s,e).map(function(n) {
return digs.without(m,o,s,e,n).map(function(d) {
return digs.without(m,o,s,e,n,d).map(function(r) {
return digs.without(m,o,s,e,n,d,r).map(function(y) {
if (1000 * s + 100 * e + 10 * n + d +
1000 * m + 100 * o + 10 * r + e ==
10000 * m + 1000 * o + 100 * n + 10 * e + y)
return ''+s+e+n+d+'+'+m+o+r+e+'='+m+o+n+e+y;
else
return [];
})})})})}).flatten();
}

15 puzzle in javascript

http://www.geocities.jp/teruakigemma/nikoniko/puzzle.html
used prototype.js.

var puzzle;
var zeropos;

function swap_puzzle(x,flag) {
var tmp = puzzle[x];
puzzle[x] = puzzle[zeropos];
puzzle[zeropos] = tmp;
if (flag) {
$('cell' + zeropos).src = puzzle[zeropos] + '.png';
$('cell' + x).src = puzzle[x] + '.png';
if ($A($R(0,15)).all(function(x) {return x == puzzle[x]}))
$('main').appendChild(new Element('span').update('nicely done!'));
};
zeropos = x;
}

function shuffle() {
for (var i = 0; i < 5; i++) {
var a = [];
if (Math.floor(zeropos / 4) != 0) a.push(zeropos - 4);
if (Math.floor(zeropos / 4) != 3) a.push(zeropos + 4);
if (zeropos % 4 != 0) a.push(zeropos - 1);
if (zeropos % 4 != 3) a.push(zeropos + 1);
swap_puzzle(a[Math.floor(Math.random() * a.length)],false);
};
}

function click_cell(x) {
if(x == zeropos - 4) swap_puzzle(zeropos - 4, true);
if(x == zeropos + 4) swap_puzzle(zeropos + 4, true);
if(x == zeropos - 1) swap_puzzle(zeropos - 1, true);
if(x == zeropos + 1) swap_puzzle(zeropos + 1, true);
}

function main() {
puzzle = $A($R(0,15));
zeropos = 0;
shuffle();
var e = $('main');
while (e.firstChild) e.removeChild(e.firstChild);
$A($R(0,15)).inGroupsOf(4).each(function(x) {
x.each(function(y) {
var img = new Element('img',{id:'cell' + y,src:puzzle[y] + '.png',width:64,height:64});
img.observe('click',function () {click_cell(y)});
e.appendChild(img)});
e.appendChild(new Element('br'))});
}

window.onload = main;