Browse Source

update ICP

leonlin14 9 years ago
commit
9a5cd11522
100 changed files with 30787 additions and 0 deletions
  1. 38 0
      .gitignore
  2. 22 0
      LICENSE
  3. 9 0
      README.md
  4. 161 0
      app.js
  5. 52 0
      configs/general.json.default
  6. 12 0
      data/modules.json
  7. 14 0
      lib/config.js
  8. 16 0
      lib/database.js
  9. 61 0
      lib/mailer.js
  10. 370 0
      lib/member.js
  11. 11 0
      lib/middleware.js
  12. 137 0
      lib/passport.js
  13. 20 0
      lib/utils.js
  14. 24 0
      models/member.js
  15. 63 0
      package.json
  16. 11 0
      routes/codebender.js
  17. 3 0
      src/blockly/.gitignore
  18. 177 0
      src/blockly/COPYING
  19. 8 0
      src/blockly/README.md
  20. 43 0
      src/blockly/appengine/README.txt
  21. 82 0
      src/blockly/appengine/app.yaml
  22. BIN
      src/blockly/appengine/apple-touch-icon.png
  23. BIN
      src/blockly/appengine/favicon.ico
  24. 11 0
      src/blockly/appengine/index.yaml
  25. 2 0
      src/blockly/appengine/index_redirect.py
  26. 68 0
      src/blockly/appengine/redirect.html
  27. 2 0
      src/blockly/appengine/robots.txt
  28. 194 0
      src/blockly/appengine/storage.js
  29. 85 0
      src/blockly/appengine/storage.py
  30. 70 0
      src/blockly/arduino_compressed.js
  31. 1307 0
      src/blockly/blockly_compressed.js
  32. 1605 0
      src/blockly/blockly_uncompressed.js
  33. 232 0
      src/blockly/blocks/base.js
  34. 117 0
      src/blockly/blocks/colour.js
  35. 505 0
      src/blockly/blocks/grove.js
  36. 681 0
      src/blockly/blocks/lists.js
  37. 464 0
      src/blockly/blocks/logic.js
  38. 318 0
      src/blockly/blocks/loops.js
  39. 481 0
      src/blockly/blocks/math.js
  40. 767 0
      src/blockly/blocks/procedures.js
  41. 664 0
      src/blockly/blocks/text.js
  42. 140 0
      src/blockly/blocks/variables.js
  43. 204 0
      src/blockly/blocks_compressed.js
  44. 440 0
      src/blockly/build.py
  45. 1258 0
      src/blockly/core/block.js
  46. 2247 0
      src/blockly/core/block_svg.js
  47. 652 0
      src/blockly/core/blockly.js
  48. 48 0
      src/blockly/core/blocks.js
  49. 578 0
      src/blockly/core/bubble.js
  50. 241 0
      src/blockly/core/comment.js
  51. 925 0
      src/blockly/core/connection.js
  52. 141 0
      src/blockly/core/contextmenu.js
  53. 732 0
      src/blockly/core/css.js
  54. 414 0
      src/blockly/core/field.js
  55. 279 0
      src/blockly/core/field_angle.js
  56. 117 0
      src/blockly/core/field_checkbox.js
  57. 242 0
      src/blockly/core/field_colour.js
  58. 350 0
      src/blockly/core/field_date.js
  59. 321 0
      src/blockly/core/field_dropdown.js
  60. 168 0
      src/blockly/core/field_image.js
  61. 107 0
      src/blockly/core/field_label.js
  62. 328 0
      src/blockly/core/field_textinput.js
  63. 197 0
      src/blockly/core/field_variable.js
  64. 731 0
      src/blockly/core/flyout.js
  65. 328 0
      src/blockly/core/generator.js
  66. 222 0
      src/blockly/core/icon.js
  67. 542 0
      src/blockly/core/inject.js
  68. 239 0
      src/blockly/core/input.js
  69. 62 0
      src/blockly/core/msg.js
  70. 303 0
      src/blockly/core/mutator.js
  71. 143 0
      src/blockly/core/names.js
  72. 284 0
      src/blockly/core/procedures.js
  73. 500 0
      src/blockly/core/realtime-client-utils.js
  74. 869 0
      src/blockly/core/realtime.js
  75. 523 0
      src/blockly/core/scrollbar.js
  76. 475 0
      src/blockly/core/toolbox.js
  77. 438 0
      src/blockly/core/tooltip.js
  78. 297 0
      src/blockly/core/trashcan.js
  79. 553 0
      src/blockly/core/utils.js
  80. 189 0
      src/blockly/core/variables.js
  81. 165 0
      src/blockly/core/warning.js
  82. 152 0
      src/blockly/core/widgetdiv.js
  83. 187 0
      src/blockly/core/workspace.js
  84. 1031 0
      src/blockly/core/workspace_svg.js
  85. 552 0
      src/blockly/core/xml.js
  86. 219 0
      src/blockly/core/zoom_controls.js
  87. 82 0
      src/blockly/dart_compressed.js
  88. 754 0
      src/blockly/demos/blockfactory/blocks.js
  89. 793 0
      src/blockly/demos/blockfactory/factory.js
  90. BIN
      src/blockly/demos/blockfactory/icon.png
  91. 220 0
      src/blockly/demos/blockfactory/index.html
  92. BIN
      src/blockly/demos/blockfactory/link.png
  93. 533 0
      src/blockly/demos/code/code.js
  94. BIN
      src/blockly/demos/code/icon.png
  95. BIN
      src/blockly/demos/code/icons.png
  96. 290 0
      src/blockly/demos/code/index.html
  97. 25 0
      src/blockly/demos/code/msg/ar.js
  98. 25 0
      src/blockly/demos/code/msg/be-tarask.js
  99. 25 0
      src/blockly/demos/code/msg/br.js
  100. 0 0
      src/blockly/demos/code/msg/ca.js

+ 38 - 0
.gitignore

@@ -0,0 +1,38 @@
1
+# Logs
2
+logs
3
+*.log
4
+
5
+# Runtime data
6
+pids
7
+*.pid
8
+*.seed
9
+
10
+# Directory for instrumented libs generated by jscoverage/JSCover
11
+lib-cov
12
+
13
+# Coverage directory used by tools like istanbul
14
+coverage
15
+
16
+# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
17
+.grunt
18
+
19
+# node-waf configuration
20
+.lock-wscript
21
+
22
+# Compiled binary addons (http://nodejs.org/api/addons.html)
23
+build/Release
24
+
25
+# Dependency directory
26
+# https://www.npmjs.org/doc/misc/npm-faq.html#should-i-check-my-node_modules-folder-into-git
27
+node_modules
28
+
29
+# webpack
30
+public/assets
31
+locales
32
+
33
+# local configuration file
34
+configs/general.json
35
+
36
+public
37
+server
38
+build

+ 22 - 0
LICENSE

@@ -0,0 +1,22 @@
1
+The MIT License (MIT)
2
+
3
+Copyright (c) 2015 Fred Chien
4
+
5
+Permission is hereby granted, free of charge, to any person obtaining a copy
6
+of this software and associated documentation files (the "Software"), to deal
7
+in the Software without restriction, including without limitation the rights
8
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+copies of the Software, and to permit persons to whom the Software is
10
+furnished to do so, subject to the following conditions:
11
+
12
+The above copyright notice and this permission notice shall be included in all
13
+copies or substantial portions of the Software.
14
+
15
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+SOFTWARE.
22
+

+ 9 - 0
README.md

@@ -0,0 +1,9 @@
1
+# Lantern
2
+
3
+An isomorphic web application with modern technologies which can be used to create your project.
4
+
5
+ - [Materialize](http://materializecss.com/)
6
+ - [Material icon](http://google.github.io/material-design-icons/#icon-images-for-the-web)
7
+ - [Awesome react](https://github.com/enaqx/awesome-react)
8
+ - [React components](http://react-components.com)
9
+ - [React DnD](http://gaearon.github.io/react-dnd/)

+ 161 - 0
app.js

@@ -0,0 +1,161 @@
1
+var path = require('path');
2
+var koa = require('koa');
3
+var Router = require('koa-router');
4
+var bodyParser = require('koa-bodyparser');
5
+var views = require('koa-views');
6
+var serve = require('koa-static');
7
+var session = require('koa-session');
8
+var passport = require('koa-passport');
9
+var co = require('co');
10
+
11
+// React
12
+var ReactApp = require('./build/server.js');
13
+
14
+// Loading settings
15
+var settings = require('./lib/config.js');
16
+if (!settings) {
17
+	console.error('Failed to load settings');
18
+	process.exit(1);
19
+}
20
+
21
+// Libraries
22
+var Utils = require('./lib/utils');
23
+//var Mailer = require('./lib/mailer');
24
+//var Database = require('./lib/database');
25
+var Passport = require('./lib/passport');
26
+
27
+var app = koa();
28
+
29
+// Static file path
30
+app.use(serve(path.join(__dirname, 'public')));
31
+
32
+// Enabling BODY
33
+app.use(bodyParser());
34
+
35
+
36
+// Initializing authenication
37
+Passport.init(passport);
38
+Passport.local(passport);
39
+
40
+// Setup 3rd-party authorization
41
+if (settings.general.authorization.github.enabled)
42
+	Passport.github(passport);
43
+
44
+if (settings.general.authorization.facebook.enabled)
45
+	Passport.facebook(passport);
46
+
47
+if (settings.general.authorization.google.enabled)
48
+	Passport.google(passport);
49
+
50
+if (settings.general.authorization.linkedin.enabled)
51
+	Passport.linkedin(passport);
52
+
53
+app.use(passport.initialize());
54
+app.use(passport.session());
55
+
56
+// Create render
57
+app.use(views(__dirname + '/views', {
58
+	ext: 'jade',
59
+	map: {
60
+		html: 'jade'
61
+	}
62
+}));
63
+
64
+// Initializing session mechanism
65
+app.keys = settings.general.session.keys || [];
66
+app.use(session(app));
67
+
68
+// Initializing locals to make template be able to get
69
+app.use(function *(next) {
70
+	this.state.user = this.req.user || undefined;
71
+	yield next;
72
+});
73
+
74
+// Routes
75
+app.use(require('./routes/codebender').middleware());
76
+/*
77
+app.use(require('./routes/auth').middleware());
78
+app.use(require('./routes/user').middleware());
79
+app.use(require('./routes/admin/dashboard').middleware());
80
+app.use(require('./routes/admin/users').middleware());
81
+*/
82
+co(function *() {
83
+
84
+	// Initializing APIs
85
+//	yield Mailer.init();
86
+//	yield Database.init();
87
+
88
+	// Initializing routes for front-end rendering
89
+	var router = new Router();
90
+	for (var index in ReactApp.routes) {
91
+		var route = ReactApp.routes[index];
92
+
93
+		// NotFound Page
94
+		if (!route.path) {
95
+			app.use(function *pageNotFound(next) {
96
+
97
+				// Be the last handler
98
+				yield next;
99
+
100
+				if (this.status != 404)
101
+					return;
102
+
103
+				if (this.json || this.body || !this.idempotent)
104
+					return;
105
+
106
+				// Rendering
107
+				var page = yield ReactApp.render(this.request.path);
108
+				yield this.render('index', {
109
+					title: settings.general.service.name,
110
+					content: page.content,
111
+					state: page.state
112
+				});
113
+
114
+				// Do not trigger koa's 404 handling
115
+				this.message = null;
116
+			});
117
+			continue;
118
+		}
119
+
120
+		// Redirect
121
+		if (route.redirect) {
122
+			(function(route) {
123
+				router.get(route.path, function *() {
124
+					this.redirect(route.redirect);
125
+				});
126
+			})(route);
127
+			continue;
128
+		}
129
+
130
+		// Register path for pages
131
+		router.get(route.path, function *() {
132
+
133
+			// It must create a new instance for rending react page asynchronously
134
+			delete require.cache[require.resolve('./build/server.js')];
135
+			var ReactApp = require('./build/server.js');
136
+			ReactApp.init({
137
+				externalUrl: Utils.getExternalUrl(),
138
+				cookie: this.req.headers.cookie
139
+			});
140
+
141
+			// Reset initial state with session for new page
142
+			var curState = {
143
+				User: this.state.user || {}
144
+			};
145
+			curState.User.logined = this.isAuthenticated();
146
+
147
+			// Rendering page and pass state to client-side
148
+			var page = yield ReactApp.render(this.request.path, curState);
149
+			yield this.render('index', {
150
+				title: settings.general.service.name,
151
+				content: page.content,
152
+				state: page.state
153
+			});
154
+		});
155
+	}
156
+	app.use(router.middleware());
157
+
158
+	app.listen(settings.general.server.port, function() {
159
+		console.log('server is ready');
160
+	});
161
+});

+ 52 - 0
configs/general.json.default

@@ -0,0 +1,52 @@
1
+{
2
+	"service": {
3
+		"name": "Genuino",
4
+		"external_url": "http://localhost:3001"
5
+	},
6
+	"server": {
7
+		"port": 3001
8
+	},
9
+	"mailer": {
10
+		"service": "Gmail",
11
+		"auth": {
12
+			"user": "youruser@gmail.com",
13
+			"clientId": "1062945721841-gv9knqff750pt1jfdnlnppqp18i8df68.apps.googleusercontent.com",
14
+			"clientSecret": "BBcspOhl_prNwcxOQtEKVtbe",
15
+			"refreshToken": "1/rvU5OkrToA7AujYucB9dagEHT4KoBbiXzP31aaJWweVIgOrJDtdun6zK6XiATCKT"
16
+		},
17
+		"sender": {
18
+			"name": "Genuino Service",
19
+			"address": "service@yourdomain.com"
20
+		}
21
+	},
22
+	"session": {
23
+		"keys": [
24
+			"YOUneverKnowWhoYOUaRe1234567890"
25
+		]
26
+	},
27
+	"database": {
28
+		"uri": "mongodb://localhost:27017/genuino"
29
+	},
30
+	"authorization": {
31
+		"github": {
32
+			"enabled": true,
33
+			"clientID": "f338ce33075757bfa59e",
34
+			"clientSecret": "571c81750b388c6233ffe060e491f822e617b402"
35
+		},
36
+		"facebook": {
37
+			"enabled": true,
38
+			"clientID": "511679122323080",
39
+			"clientSecret": "ffec2c1f7ea4b0e32ff31af0d0e275d6"
40
+		},
41
+		"google": {
42
+			"enabled": true,
43
+			"clientID": "1062945721841-5hm41ltvoahmepdi822v40iue8hufgeq.apps.googleusercontent.com",
44
+			"clientSecret": "y_QDeNpKoQS-P2u8SegRGiXE"
45
+		},
46
+		"linkedin": {
47
+			"enabled": true,
48
+			"clientID": "757uag6nqoq4r8",
49
+			"clientSecret": "0nx4svG7uesLlPai"
50
+		}
51
+	}
52
+}

+ 12 - 0
data/modules.json

@@ -0,0 +1,12 @@
1
+[
2
+  {
3
+    "sku": "101020037",
4
+	"cname": "馬達",
5
+	"ename": "Motor",
6
+	"color": 1,
7
+    "types": 1,
8
+	"slots": [ "D2", "D3" ],
9
+    "desc": "A Motor",
10
+	"func": "DigitalWrite($slot)"
11
+  }
12
+]

+ 14 - 0
lib/config.js

@@ -0,0 +1,14 @@
1
+var path = require('path');
2
+var fs = require('fs');
3
+var settings = {};
4
+
5
+try {
6
+	var data = fs.readFileSync(path.join(__dirname, '..', 'configs', 'general.json'));
7
+	settings.general = JSON.parse(data);
8
+} catch(e) {
9
+	console.warn('[warning]', 'No configuration file exists, using default settings.');
10
+	var data = fs.readFileSync(path.join(__dirname, '..', 'configs', 'general.json.default'));
11
+	settings.general = JSON.parse(data);
12
+}
13
+
14
+module.exports = settings;

+ 16 - 0
lib/database.js

@@ -0,0 +1,16 @@
1
+var settings = require('./config');
2
+var mongoose = require('mongoose');
3
+
4
+module.exports = {
5
+	init: function() {
6
+
7
+		return function(done) {
8
+
9
+			mongoose.connect(settings.general.database.uri);
10
+			var db = mongoose.connection;
11
+			db.once('open', function() {
12
+				done();
13
+			});
14
+		};
15
+	}
16
+};

+ 61 - 0
lib/mailer.js

@@ -0,0 +1,61 @@
1
+var settings = require('./config');
2
+var nodemailer = require('nodemailer');
3
+var smtpPool = require('nodemailer-smtp-pool');
4
+
5
+var transporter;
6
+
7
+var Mailer = module.exports = {
8
+	init: function() {
9
+
10
+		return function(done) {
11
+
12
+			if (settings.general.mailer.service == 'Gmail') {
13
+
14
+				// Initializing XOAuth2
15
+				var generator = require('xoauth2')
16
+					.createXOAuth2Generator(settings.general.mailer.auth);
17
+
18
+				// Create a pool of connections
19
+				var pool = smtpPool({
20
+					service: settings.general.mailer.service,
21
+					auth: {
22
+						xoauth2: generator
23
+					}
24
+				});
25
+
26
+				// Create a transport
27
+				transporter = nodemailer.createTransport(pool);
28
+
29
+				done();
30
+				return;
31
+			}
32
+
33
+			// Create a pool of connections for specific mail service
34
+			var pool = smtpPool({
35
+				service: settings.general.mailer.service,
36
+				auth: settings.general.mailer.auth
37
+			});
38
+
39
+			// Create a transport
40
+			transporter = nodemailer.createTransport(pool);
41
+
42
+			done();
43
+		};
44
+	},
45
+	sendMail: function(msg) {
46
+
47
+		return function(done) {
48
+
49
+			transporter.sendMail(msg, done);
50
+		};
51
+	},
52
+	sendMailFromService: function(to, subject, html) {
53
+
54
+		return Mailer.sendMail({
55
+			from: settings.general.mailer.sender.name + ' <' + settings.general.mailer.sender.address + '>',
56
+			to: to,
57
+			subject: subject,
58
+			html: html
59
+		});
60
+	}
61
+};

+ 370 - 0
lib/member.js

@@ -0,0 +1,370 @@
1
+var crypto = require('crypto');
2
+var mongoose = require('mongoose');
3
+var Member = require('../models/member');
4
+var settings = require('./config');
5
+var utils = require('./utils');
6
+
7
+module.exports = {
8
+	create: function(member) {
9
+		return function(done) {
10
+			var _member = new Member({
11
+				name: member.name,
12
+				email: member.email,
13
+				phone: member.phone,
14
+				gender: member.gender,
15
+				idno: member.idno,
16
+				salt: utils.generateSalt(),
17
+				birthday: member.birthday,
18
+				cardno: member.cardno,
19
+				tokens: member.tokens
20
+			});
21
+
22
+			if (member.password) {
23
+				// Encrypt plain password
24
+				_member.password = utils.encryptPassword(_member.salt, member.password);
25
+			} else {
26
+				_member.password = '';
27
+			}
28
+
29
+			_member.save(function(err) {
30
+				done(err, _member);
31
+			});
32
+		};
33
+	},
34
+	getMember: function(id) {
35
+		return function(done) {
36
+			Member.findOne({ _id: id }, function(err, member) {
37
+				if (err)
38
+					return done(err);
39
+
40
+				if (!member)
41
+					return done();
42
+
43
+				return done(null, member);
44
+			});
45
+		};
46
+	},
47
+	getMemberByEmail: function(email) {
48
+		return function(done) {
49
+			Member.findOne({ email: email}, function(err, member) {
50
+				if (err)
51
+					return done(err);
52
+
53
+				if (!member)
54
+					return done();
55
+
56
+				return done(null, member);
57
+			});
58
+		};
59
+	},
60
+	deleteMembers: function(ids) {
61
+		return function(done) {
62
+			Member.remove({
63
+				_id: {
64
+					$in: ids
65
+				}
66
+			}, function(err) {
67
+				done(err);
68
+			});
69
+		};
70
+	},
71
+	insert: function(members) {
72
+		return function(done) {
73
+
74
+			Member.collection.insert(members, done);
75
+		};
76
+	},
77
+	updateByEmail: function(email, member, opts) {
78
+		
79
+		return function(done) {
80
+
81
+			// Update time
82
+			member.updated = Date.now();
83
+
84
+			Member.update({ email: email }, member, opts, done);
85
+		};
86
+	},
87
+	changePassword: function(id, password) {
88
+		return function(done) {
89
+
90
+			// Generate a new salt for encryption
91
+			var salt = utils.generateSalt();
92
+			var newPassword = utils.encryptPassword(salt, password);
93
+
94
+			// Update password
95
+			Member.findOneAndUpdate({ _id: id }, {
96
+				salt: salt,
97
+				password: newPassword,
98
+				updated: Date.now()
99
+			}, { new: true }, function(err, member) {
100
+
101
+				if (err)
102
+					return done(err);
103
+
104
+				done(null, member ? true : false);
105
+			});
106
+		};
107
+	},
108
+	changePasswordWithToken: function(id, token, password) {
109
+		return function(done) {
110
+
111
+			// Generate a new salt for encryption
112
+			var salt = utils.generateSalt();
113
+			var newPassword = utils.encryptPassword(salt, password);
114
+
115
+			// TODO: Should check expired time of token
116
+			// Update password
117
+			Member.findOneAndUpdate({
118
+				_id: id,
119
+				'rule_tokens.name': 'reset_password'
120
+			}, {
121
+				$pull: {
122
+					rule_tokens: {
123
+						name: 'reset_password'
124
+					}
125
+				},
126
+				salt: salt,
127
+				password: newPassword,
128
+				updated: Date.now()
129
+			}, { new: true }, function(err, member) {
130
+
131
+				if (err)
132
+					return done(err);
133
+
134
+				done(null, member ? true : false);
135
+			});
136
+		};
137
+	},
138
+	checkCard: function(token) {
139
+		return function(done) {
140
+
141
+			Member.findOne({ tokens: token }, function(err, member) {
142
+				if (err)
143
+					return done(err);
144
+
145
+				if (!member)
146
+					return done(new Error('Not Found'));
147
+
148
+				return done(null, member);
149
+			});
150
+		};
151
+	},
152
+	authorizeMember: function(username, password) {
153
+		return function(done) {
154
+
155
+			Member.findOne({ email: username }, function(err, member) {
156
+				if (err)
157
+					return done(err);
158
+
159
+				// Found nothing
160
+				if (!member)
161
+					return done();
162
+
163
+				// First time to login
164
+				if (!member.password) {
165
+
166
+					// Using phone to be password
167
+					if (member.phone == password)
168
+						return done(null, member);
169
+					else
170
+						return done();
171
+				}
172
+
173
+				// Check password
174
+				if (utils.encryptPassword(member.salt, password) == member.password)
175
+					return done(null, member);
176
+				else
177
+					return done();
178
+			});
179
+		};
180
+	},
181
+	save: function(id, member) {
182
+
183
+		return function(done) {
184
+			var updated = Date.now();
185
+
186
+			var m = {
187
+				name: member.name || undefined,
188
+				email: member.email || undefined,
189
+				phone: member.phone || undefined,
190
+				gender: member.gender || undefined,
191
+				idno: member.idno || undefined,
192
+				birthday: member.birthday || undefined,
193
+				tokens: member.tokens || undefined,
194
+				updated: updated
195
+			};
196
+
197
+			// Remove fields which is unset
198
+			for (var key in m) {
199
+				if (m[key] == undefined)
200
+					delete m[key];
201
+			}
202
+
203
+			Member.findOneAndUpdate({ _id: id }, m, { new: true }, function(err, _member) {
204
+
205
+				if (err)
206
+					return done(err);
207
+
208
+				done(null, _member);
209
+			});
210
+		};
211
+	},
212
+	count: function() {
213
+		return function(done) {
214
+			Member.count({}, done);
215
+		};
216
+	},
217
+	list: function() {
218
+
219
+		var conditions = {};
220
+		var columns;
221
+		var opts = {};
222
+		if (arguments.length == 3) {
223
+			conditions = arguments[0];
224
+			columns = arguments[1];
225
+			opts = arguments[2];
226
+		} else if (arguments.length == 2) {
227
+			if (arguments[0] instanceof Array) {
228
+				columns = arguments[0];
229
+				opts = arguments[1];
230
+			} else if (arguments[1] instanceof Array) {
231
+				conditions = arguments[0];
232
+				columns = arguments[1];
233
+			} else {
234
+				conditions = arguments[0];
235
+				opts = arguments[1];
236
+			}
237
+		} else if (arguments.length == 1) {
238
+			columns = null;
239
+			opts = arguments[0];
240
+		}
241
+
242
+		return function(done) {
243
+
244
+			var cols = null;
245
+			if (columns)
246
+				cols = columns.join(' ');
247
+
248
+			Member.count(conditions, function(err, count) {
249
+				if (err) {
250
+					done(err);
251
+					return;
252
+				}
253
+
254
+				if (!count) {
255
+					done(err, { count: 0 });
256
+					return;
257
+				}
258
+
259
+				Member.find(conditions, cols, opts, function(err, members) {
260
+
261
+					done(err, {
262
+						count: count,
263
+						members: members
264
+					});
265
+				});
266
+			});
267
+		};
268
+	},
269
+	setupRuleToken: function(id, name, expired) {
270
+
271
+		return function(done) {
272
+
273
+			var token = utils.generateToken();
274
+
275
+			// Remove old token
276
+			Member.findOneAndUpdate({
277
+				_id: id,
278
+				'rule_tokens.name': name
279
+			}, {
280
+				$pull: {
281
+					'rule_tokens.$.name': name
282
+				}
283
+			}, function(err) {
284
+				if (err) {
285
+					return done(err);
286
+				}
287
+
288
+				// Update rule token. add a new one if no key exists
289
+				Member.findOneAndUpdate({
290
+					_id: id
291
+				}, {
292
+					$addToSet: {
293
+						rule_tokens: {
294
+							name: name,
295
+							token: token,
296
+							expired: expired
297
+						}
298
+					}
299
+				}, function(err, member) {
300
+
301
+					done(err, member ? {
302
+						token: token,
303
+						id: member._id
304
+					} : null);
305
+				});
306
+			});
307
+			
308
+		};
309
+	},
310
+	setupRuleTokenByEmail: function(email, name, expired) {
311
+
312
+		return function(done) {
313
+
314
+			var token = utils.generateToken();
315
+
316
+			// Remove old token
317
+			Member.findOneAndUpdate({
318
+				email: email,
319
+				'rule_tokens.name': name
320
+			}, {
321
+				$pull: {
322
+					'rule_tokens.$.name': name
323
+				}
324
+			}, function(err) {
325
+				if (err) {
326
+					return done(err);
327
+				}
328
+
329
+				// Update rule token. add a new one if no key exists
330
+				Member.findOneAndUpdate({
331
+					email: email
332
+				}, {
333
+					$addToSet: {
334
+						rule_tokens: {
335
+							name: name,
336
+							token: token,
337
+							expired: expired
338
+						}
339
+					}
340
+				}, function(err, member) {
341
+
342
+					done(err, member ? {
343
+						token: token,
344
+						id: member._id
345
+					} : null);
346
+				});
347
+			});
348
+			
349
+		};
350
+	},
351
+	updateCardno: function(id, cardno) {
352
+
353
+		return function(done) {
354
+			Member.update({ _id: id }, {
355
+				cardno: cardno,
356
+				updated: Date.now()
357
+			}, done);
358
+		};
359
+	},
360
+	updateCardnoByEmail: function(email, token, cardno) {
361
+
362
+		return function(done) {
363
+			Member.update({ email: email }, {
364
+				tokens: [ token ],
365
+				cardno: cardno,
366
+				updated: Date.now()
367
+			}, done);
368
+		};
369
+	}
370
+};

+ 11 - 0
lib/middleware.js

@@ -0,0 +1,11 @@
1
+
2
+module.exports = {
3
+	requireAuthorized: function *(next) {
4
+		if (!this.isAuthenticated()) {
5
+			this.status = 404;
6
+			return;
7
+		}
8
+
9
+		yield next;
10
+	}
11
+};

+ 137 - 0
lib/passport.js

@@ -0,0 +1,137 @@
1
+var co = require('co');
2
+var crypto = require('crypto');
3
+var LocalStrategy = require('passport-local').Strategy;
4
+var GithubStrategy = require('passport-github2').Strategy;
5
+var FacebookStrategy = require('passport-facebook').Strategy;
6
+var GoogleStrategy = require('passport-google-oauth').OAuth2Strategy;
7
+var LinkedinStrategy = require('passport-linkedin-oauth2').Strategy;
8
+var Utils = require('./utils');
9
+var Member = require('./member');
10
+
11
+// Loading settings
12
+var settings = require('./config.js');
13
+if (!settings) {
14
+	console.error('Failed to load settings');
15
+	return;
16
+}
17
+
18
+var protocol = settings.general.server.secure ? 'https://' : 'http://';
19
+var port = (settings.general.server.port == 80) ? '' : ':' + settings.general.server.port;
20
+
21
+module.exports = {
22
+	init: function(passport) {
23
+
24
+		passport.serializeUser(function(user, done) {
25
+			done(null, user);
26
+		})
27
+
28
+		passport.deserializeUser(function(user, done) {
29
+			done(null, user);
30
+		})
31
+	},
32
+	local: function(passport) {
33
+
34
+		passport.use(new LocalStrategy(function(username, password, done) {
35
+
36
+			co(function *() {
37
+				// Using own library to authorize
38
+				return yield Member.authorizeMember(username, password);
39
+			}).then(function(member) {
40
+				done(null, member);
41
+			}, function(err) {
42
+				done(err);
43
+			});
44
+		}));
45
+	},
46
+	github: function(passport) {
47
+
48
+		passport.use(new GithubStrategy({
49
+				clientID: settings.general.authorization.github.clientID,
50
+				clientSecret: settings.general.authorization.github.clientSecret,
51
+				callbackURL: Utils.getExternalUrl() + '/auth/github/callback'
52
+			},
53
+			function(accessToken, refreshToken, profile, done) {
54
+				var user = {
55
+					name: profile.displayName,
56
+					email: profile.emails[0].value
57
+				};
58
+				done(null, user);
59
+			}
60
+		));
61
+	},
62
+	facebook: function(passport) {
63
+
64
+		passport.use(new FacebookStrategy({
65
+				clientID: settings.general.authorization.facebook.clientID,
66
+				clientSecret: settings.general.authorization.facebook.clientSecret,
67
+				callbackURL: Utils.getExternalUrl() + '/auth/facebook/callback',
68
+				profileFields: [ 'displayName', 'emails' ],
69
+				enableProof: false
70
+			},
71
+			function(accessToken, refreshToken, profile, done) {
72
+
73
+				var user = {
74
+					name: profile.displayName,
75
+					email: profile.emails[0].value
76
+				};
77
+				done(null, user);
78
+			}
79
+		));
80
+	},
81
+	google: function(passport) {
82
+
83
+		passport.use(new GoogleStrategy({
84
+				clientID: settings.general.authorization.google.clientID,
85
+				clientSecret: settings.general.authorization.google.clientSecret,
86
+				callbackURL: Utils.getExternalUrl() + '/auth/google/callback'
87
+			},
88
+			function(accessToken, refreshToken, profile, done) {
89
+
90
+				var user = {
91
+					name: profile.displayName,
92
+					email: profile.emails[0].value
93
+				};
94
+				done(null, user);
95
+			}
96
+		));
97
+	},
98
+	linkedin: function(passport) {
99
+
100
+		passport.use(new LinkedinStrategy({
101
+				clientID: settings.general.authorization.linkedin.clientID,
102
+				clientSecret: settings.general.authorization.linkedin.clientSecret,
103
+				callbackURL: Utils.getExternalUrl() + '/auth/linkedin/callback',
104
+				scope: [ 'r_emailaddress', 'r_basicprofile' ],
105
+				state: true
106
+			},
107
+			function(accessToken, refreshToken, profile, done) {
108
+
109
+				var user = {
110
+					name: profile.displayName,
111
+					email: profile.emails[0].value
112
+				};
113
+				done(null, user);
114
+			}
115
+		));
116
+	},
117
+	login: function(ctx, user) {
118
+
119
+		return function(done) {
120
+
121
+			co(function *() {
122
+				var member = {
123
+					id: user.id,
124
+					name: user.name,
125
+					email: user.email,
126
+					login_time: Date.now(),
127
+					avatar_hash: crypto.createHash('md5').update(user.email).digest('hex')
128
+				}
129
+
130
+				// User information will be stored in session
131
+				yield ctx.login(member);
132
+
133
+				done(null, member);
134
+			});
135
+		};
136
+	}
137
+};

+ 20 - 0
lib/utils.js

@@ -0,0 +1,20 @@
1
+var settings = require('./config');
2
+var crypto = require('crypto');
3
+
4
+module.exports = {
5
+	getServiceName: function() {
6
+		return settings.general.service.name;
7
+	},
8
+	getExternalUrl: function() {
9
+		return settings.general.service.external_url;
10
+	},
11
+	generateToken: function() {
12
+		return crypto.randomBytes(16).toString('base64');
13
+	},
14
+	generateSalt: function() {
15
+		return crypto.randomBytes(12).toString('base64');
16
+	},
17
+	encryptPassword: function(salt, password) {
18
+		return crypto.createHmac('sha256', password + salt || '').digest('hex');
19
+	}
20
+};

+ 24 - 0
models/member.js

@@ -0,0 +1,24 @@
1
+var mongoose = require('mongoose');
2
+
3
+var Member = new mongoose.Schema({
4
+	name: String,
5
+	email: { type: String, unique: true },
6
+	password: String,
7
+	salt: String,
8
+	gender: Number,
9
+	birthday: Date,
10
+	hashname: { type: String },
11
+	phone: String,
12
+	idno: String,
13
+	tokens: [ String ],
14
+	rule_tokens: [{ name: String, token: String, expired: Date }],
15
+	disabled: { type: Boolean, default: false },
16
+	created: { type: Date, default: Date.now },
17
+	updated: { type: Date, default: Date.now }
18
+});
19
+
20
+Member.methods.validPassword = function(password) {
21
+    return (this.password == password) ? true : false;
22
+};
23
+
24
+module.exports = mongoose.model('Member', Member);

+ 63 - 0
package.json

@@ -0,0 +1,63 @@
1
+{
2
+  "name": "genuino",
3
+  "version": "1.0.0",
4
+  "description": "Genuino Online service",
5
+  "main": "app.js",
6
+  "scripts": {
7
+    "test": "echo \"Error: no test specified\" && exit 1",
8
+    "start": "npm run serve | npm run dev",
9
+    "serve": "./node_modules/.bin/http-server -p 8080",
10
+    "dev": "webpack-dev-server --progress --colors --port 8090"
11
+  },
12
+  "repository": {
13
+    "type": "git",
14
+    "url": "https://github.com/cfsghost/lantern.git"
15
+  },
16
+  "keywords": [
17
+    "web"
18
+  ],
19
+  "author": "Fred Chien <cfsghost@gmail.com>",
20
+  "license": "MIT",
21
+  "bugs": {
22
+    "url": "https://github.com/cfsghost/lantern/issues"
23
+  },
24
+  "homepage": "https://github.com/cfsghost/lantern",
25
+  "devDependencies": {
26
+    "css-loader": ">=0.15.6",
27
+    "file-loader": "^0.8.4",
28
+    "jsx-loader": ">=0.13.2",
29
+    "less": ">=2.5.1",
30
+    "less-loader": ">=2.2.0",
31
+    "node-libs-browser": ">=0.5.2",
32
+    "style-loader": ">=0.12.3",
33
+    "url-loader": "^0.5.6",
34
+    "webpack": ">=1.10.1"
35
+  },
36
+  "dependencies": {
37
+    "babel-core": "^5.8.23",
38
+    "babel-loader": "^5.3.2",
39
+    "babel-runtime": "^5.8.20",
40
+    "co": "^4.6.0",
41
+    "fluky": "^0.1.12",
42
+    "jade": ">=1.11.0",
43
+    "koa": ">=0.21.0",
44
+    "koa-bodyparser": ">=2.0.0",
45
+    "koa-passport": "^1.1.6",
46
+    "koa-router": ">=5.1.2",
47
+    "koa-session": ">=3.3.1",
48
+    "koa-static": ">=1.4.9",
49
+    "koa-views": ">=3.1.0",
50
+    "mongoose": "^4.1.5",
51
+    "passport-facebook": "^2.0.0",
52
+    "passport-github2": "^0.1.9",
53
+    "passport-google-oauth": "^0.2.0",
54
+    "passport-linkedin-oauth2": "^1.2.1",
55
+    "passport-local": "^1.0.0",
56
+    "react": "^0.13.3",
57
+    "react-dnd": "^1.1.4",
58
+    "react-router": "^0.13.3",
59
+    "react-tap-event-plugin": "^0.1.7",
60
+    "react-tools": ">=0.13.3",
61
+    "scriptjs": "^2.5.7"
62
+  }
63
+}

+ 11 - 0
routes/codebender.js

@@ -0,0 +1,11 @@
1
+var Router = require('koa-router');
2
+
3
+var router = module.exports = new Router();
4
+
5
+router.get('/js/codebender/chrome-client.js', function *() {
6
+	this.response.redirect('http://codebender.cc/js/codebender/chrome-client.js');
7
+});
8
+
9
+router.get('/js/codebender/firefox-client.js', function *() {
10
+	this.response.redirect('http://codebender.cc/js/codebender/firefox-client.js');
11
+});

+ 3 - 0
src/blockly/.gitignore

@@ -0,0 +1,3 @@
1
+.DS_Store
2
+*.pyc
3
+*.komodoproject

+ 177 - 0
src/blockly/COPYING

@@ -0,0 +1,177 @@
1
+
2
+                                 Apache License
3
+                           Version 2.0, January 2011
4
+                        http://www.apache.org/licenses/
5
+
6
+   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
7
+
8
+   1. Definitions.
9
+
10
+      "License" shall mean the terms and conditions for use, reproduction,
11
+      and distribution as defined by Sections 1 through 9 of this document.
12
+
13
+      "Licensor" shall mean the copyright owner or entity authorized by
14
+      the copyright owner that is granting the License.
15
+
16
+      "Legal Entity" shall mean the union of the acting entity and all
17
+      other entities that control, are controlled by, or are under common
18
+      control with that entity. For the purposes of this definition,
19
+      "control" means (i) the power, direct or indirect, to cause the
20
+      direction or management of such entity, whether by contract or
21
+      otherwise, or (ii) ownership of fifty percent (50%) or more of the
22
+      outstanding shares, or (iii) beneficial ownership of such entity.
23
+
24
+      "You" (or "Your") shall mean an individual or Legal Entity
25
+      exercising permissions granted by this License.
26
+
27
+      "Source" form shall mean the preferred form for making modifications,
28
+      including but not limited to software source code, documentation
29
+      source, and configuration files.
30
+
31
+      "Object" form shall mean any form resulting from mechanical
32
+      transformation or translation of a Source form, including but
33
+      not limited to compiled object code, generated documentation,
34
+      and conversions to other media types.
35
+
36
+      "Work" shall mean the work of authorship, whether in Source or
37
+      Object form, made available under the License, as indicated by a
38
+      copyright notice that is included in or attached to the work
39
+      (an example is provided in the Appendix below).
40
+
41
+      "Derivative Works" shall mean any work, whether in Source or Object
42
+      form, that is based on (or derived from) the Work and for which the
43
+      editorial revisions, annotations, elaborations, or other modifications
44
+      represent, as a whole, an original work of authorship. For the purposes
45
+      of this License, Derivative Works shall not include works that remain
46
+      separable from, or merely link (or bind by name) to the interfaces of,
47
+      the Work and Derivative Works thereof.
48
+
49
+      "Contribution" shall mean any work of authorship, including
50
+      the original version of the Work and any modifications or additions
51
+      to that Work or Derivative Works thereof, that is intentionally
52
+      submitted to Licensor for inclusion in the Work by the copyright owner
53
+      or by an individual or Legal Entity authorized to submit on behalf of
54
+      the copyright owner. For the purposes of this definition, "submitted"
55
+      means any form of electronic, verbal, or written communication sent
56
+      to the Licensor or its representatives, including but not limited to
57
+      communication on electronic mailing lists, source code control systems,
58
+      and issue tracking systems that are managed by, or on behalf of, the
59
+      Licensor for the purpose of discussing and improving the Work, but
60
+      excluding communication that is conspicuously marked or otherwise
61
+      designated in writing by the copyright owner as "Not a Contribution."
62
+
63
+      "Contributor" shall mean Licensor and any individual or Legal Entity
64
+      on behalf of whom a Contribution has been received by Licensor and
65
+      subsequently incorporated within the Work.
66
+
67
+   2. Grant of Copyright License. Subject to the terms and conditions of
68
+      this License, each Contributor hereby grants to You a perpetual,
69
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
70
+      copyright license to reproduce, prepare Derivative Works of,
71
+      publicly display, publicly perform, sublicense, and distribute the
72
+      Work and such Derivative Works in Source or Object form.
73
+
74
+   3. Grant of Patent License. Subject to the terms and conditions of
75
+      this License, each Contributor hereby grants to You a perpetual,
76
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
77
+      (except as stated in this section) patent license to make, have made,
78
+      use, offer to sell, sell, import, and otherwise transfer the Work,
79
+      where such license applies only to those patent claims licensable
80
+      by such Contributor that are necessarily infringed by their
81
+      Contribution(s) alone or by combination of their Contribution(s)
82
+      with the Work to which such Contribution(s) was submitted. If You
83
+      institute patent litigation against any entity (including a
84
+      cross-claim or counterclaim in a lawsuit) alleging that the Work
85
+      or a Contribution incorporated within the Work constitutes direct
86
+      or contributory patent infringement, then any patent licenses
87
+      granted to You under this License for that Work shall terminate
88
+      as of the date such litigation is filed.
89
+
90
+   4. Redistribution. You may reproduce and distribute copies of the
91
+      Work or Derivative Works thereof in any medium, with or without
92
+      modifications, and in Source or Object form, provided that You
93
+      meet the following conditions:
94
+
95
+      (a) You must give any other recipients of the Work or
96
+          Derivative Works a copy of this License; and
97
+
98
+      (b) You must cause any modified files to carry prominent notices
99
+          stating that You changed the files; and
100
+
101
+      (c) You must retain, in the Source form of any Derivative Works
102
+          that You distribute, all copyright, patent, trademark, and
103
+          attribution notices from the Source form of the Work,
104
+          excluding those notices that do not pertain to any part of
105
+          the Derivative Works; and
106
+
107
+      (d) If the Work includes a "NOTICE" text file as part of its
108
+          distribution, then any Derivative Works that You distribute must
109
+          include a readable copy of the attribution notices contained
110
+          within such NOTICE file, excluding those notices that do not
111
+          pertain to any part of the Derivative Works, in at least one
112
+          of the following places: within a NOTICE text file distributed
113
+          as part of the Derivative Works; within the Source form or
114
+          documentation, if provided along with the Derivative Works; or,
115
+          within a display generated by the Derivative Works, if and
116
+          wherever such third-party notices normally appear. The contents
117
+          of the NOTICE file are for informational purposes only and
118
+          do not modify the License. You may add Your own attribution
119
+          notices within Derivative Works that You distribute, alongside
120
+          or as an addendum to the NOTICE text from the Work, provided
121
+          that such additional attribution notices cannot be construed
122
+          as modifying the License.
123
+
124
+      You may add Your own copyright statement to Your modifications and
125
+      may provide additional or different license terms and conditions
126
+      for use, reproduction, or distribution of Your modifications, or
127
+      for any such Derivative Works as a whole, provided Your use,
128
+      reproduction, and distribution of the Work otherwise complies with
129
+      the conditions stated in this License.
130
+
131
+   5. Submission of Contributions. Unless You explicitly state otherwise,
132
+      any Contribution intentionally submitted for inclusion in the Work
133
+      by You to the Licensor shall be under the terms and conditions of
134
+      this License, without any additional terms or conditions.
135
+      Notwithstanding the above, nothing herein shall supersede or modify
136
+      the terms of any separate license agreement you may have executed
137
+      with Licensor regarding such Contributions.
138
+
139
+   6. Trademarks. This License does not grant permission to use the trade
140
+      names, trademarks, service marks, or product names of the Licensor,
141
+      except as required for reasonable and customary use in describing the
142
+      origin of the Work and reproducing the content of the NOTICE file.
143
+
144
+   7. Disclaimer of Warranty. Unless required by applicable law or
145
+      agreed to in writing, Licensor provides the Work (and each
146
+      Contributor provides its Contributions) on an "AS IS" BASIS,
147
+      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
148
+      implied, including, without limitation, any warranties or conditions
149
+      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
150
+      PARTICULAR PURPOSE. You are solely responsible for determining the
151
+      appropriateness of using or redistributing the Work and assume any
152
+      risks associated with Your exercise of permissions under this License.
153
+
154
+   8. Limitation of Liability. In no event and under no legal theory,
155
+      whether in tort (including negligence), contract, or otherwise,
156
+      unless required by applicable law (such as deliberate and grossly
157
+      negligent acts) or agreed to in writing, shall any Contributor be
158
+      liable to You for damages, including any direct, indirect, special,
159
+      incidental, or consequential damages of any character arising as a
160
+      result of this License or out of the use or inability to use the
161
+      Work (including but not limited to damages for loss of goodwill,
162
+      work stoppage, computer failure or malfunction, or any and all
163
+      other commercial damages or losses), even if such Contributor
164
+      has been advised of the possibility of such damages.
165
+
166
+   9. Accepting Warranty or Additional Liability. While redistributing
167
+      the Work or Derivative Works thereof, You may choose to offer,
168
+      and charge a fee for, acceptance of support, warranty, indemnity,
169
+      or other liability obligations and/or rights consistent with this
170
+      License. However, in accepting such obligations, You may act only
171
+      on Your own behalf and on Your sole responsibility, not on behalf
172
+      of any other Contributor, and only if You agree to indemnify,
173
+      defend, and hold each Contributor harmless for any liability
174
+      incurred by, or claims asserted against, such Contributor by reason
175
+      of your accepting any such warranty or additional liability.
176
+
177
+   END OF TERMS AND CONDITIONS

+ 8 - 0
src/blockly/README.md

@@ -0,0 +1,8 @@
1
+# Blockly
2
+
3
+Google's Blockly is a web-based, visual programming editor.  Users can drag
4
+blocks together to build programs.  All code is free and open source.
5
+
6
+**The project page is https://developers.google.com/blockly/**
7
+
8
+![](https://developers.google.com/blockly/sample.png)

+ 43 - 0
src/blockly/appengine/README.txt

@@ -0,0 +1,43 @@
1
+
2
+  Running an App Engine server
3
+
4
+This directory contains the files needed to setup the optional Blockly server.
5
+Although Blockly itself is 100% client-side, the server enables cloud storage
6
+and sharing.  Store your programs in Datastore and get a unique URL that allows
7
+you to load the program on any computer.
8
+
9
+To run your own App Engine instance you'll need to create this directory
10
+structure:
11
+
12
+blockly/
13
+ |- app.yaml
14
+ |- index.yaml
15
+ |- index_redirect.py
16
+ |- README.txt
17
+ |- storage.js
18
+ |- storage.py
19
+ |- closure-library/  (Optional)
20
+ `- static/
21
+     |- blocks/
22
+     |- core/
23
+     |- demos/
24
+     |- generators/
25
+     |- media/
26
+     |- msg/
27
+     |- tests/
28
+     |- blockly_compressed.js
29
+     |- blockly_uncompressed.js  (Optional)
30
+     |- blocks_compressed.js
31
+     |- dart_compressed.js
32
+     |- javascript_compressed.js
33
+     |- php_compressed.js
34
+     `- python_compressed.js
35
+
36
+Instructions for fetching the optional Closure library may be found here:
37
+  https://developers.google.com/blockly/hacking/closure
38
+
39
+Go to https://appengine.google.com/ and create your App Engine application.
40
+Modify the 'application' name of app.yaml to your App Engine application name.
41
+
42
+Finally, upload this directory structure to your App Engine account,
43
+wait a minute, then go to http://YOURAPPNAME.appspot.com/

+ 82 - 0
src/blockly/appengine/app.yaml

@@ -0,0 +1,82 @@
1
+application: blockly-demo
2
+version: 1
3
+runtime: python27
4
+api_version: 1
5
+threadsafe: no
6
+
7
+handlers:
8
+# Redirect obsolete URLs.
9
+# Blockly files moved from /blockly to /static on 5 Dec 2012.
10
+- url: /blockly/.*
11
+  static_files: redirect.html
12
+  upload: redirect.html
13
+# Code, Maze and Turtle moved from demos on 29 Dec 2012.
14
+- url: /static/demos/(maze|turtle)/.*
15
+  static_files: redirect.html
16
+  upload: redirect.html
17
+# Apps was disbanded on 20 Nov 2014.
18
+- url: /static/apps/.*
19
+  static_files: redirect.html
20
+  upload: redirect.html
21
+
22
+
23
+# Storage API.
24
+- url: /storage
25
+  script: storage.py
26
+  secure: always
27
+- url: /storage\.js
28
+  static_files: storage.js
29
+  upload: storage\.js
30
+  secure: always
31
+
32
+# Blockly files.
33
+- url: /static
34
+  static_dir: static
35
+  secure: always
36
+
37
+# Closure library for uncompiled Blockly.
38
+- url: /closure-library
39
+  static_dir: closure-library
40
+  secure: always
41
+
42
+# Redirect for root directory.
43
+- url: /
44
+  script: index_redirect.py
45
+  secure: always
46
+
47
+# Favicon.
48
+- url: /favicon\.ico
49
+  static_files: favicon.ico
50
+  upload: favicon\.ico
51
+  secure: always
52
+  expiration: "30d"
53
+
54
+# Apple icon.
55
+- url: /apple-touch-icon\.png
56
+  static_files: apple-touch-icon.png
57
+  upload: apple-touch-icon\.png
58
+  secure: always
59
+  expiration: "30d"
60
+
61
+# robot.txt
62
+- url: /robots\.txt
63
+  static_files: robots.txt
64
+  upload: robots\.txt
65
+  secure: always
66
+
67
+
68
+skip_files:
69
+# App Engine default patterns.
70
+- ^(.*/)?#.*#$
71
+- ^(.*/)?.*~$
72
+- ^(.*/)?.*\.py[co]$
73
+- ^(.*/)?.*/RCS/.*$
74
+- ^(.*/)?\..*$
75
+# Custom skip patterns.
76
+- ^static/appengine/.*$
77
+- ^static/demos/plane/soy/.+\.jar$
78
+- ^static/demos/plane/template.soy$
79
+- ^static/demos/plane/xlf/.*$
80
+- ^static/i18n/.*$
81
+- ^static/msg/json/.*$
82
+- ^.+\.soy$

BIN
src/blockly/appengine/apple-touch-icon.png


BIN
src/blockly/appengine/favicon.ico


+ 11 - 0
src/blockly/appengine/index.yaml

@@ -0,0 +1,11 @@
1
+indexes:
2
+
3
+# AUTOGENERATED
4
+
5
+# This index.yaml is automatically updated whenever the dev_appserver
6
+# detects that a new type of query is run.  If you want to manage the
7
+# index.yaml file manually, remove the above marker line (the line
8
+# saying "# AUTOGENERATED").  If you want to manage some indexes
9
+# manually, move them above the marker line.  The index.yaml file is
10
+# automatically uploaded to the admin console when you next deploy
11
+# your application using appcfg.py.

+ 2 - 0
src/blockly/appengine/index_redirect.py

@@ -0,0 +1,2 @@
1
+print("Status: 302")
2
+print("Location: /static/demos/index.html")

+ 68 - 0
src/blockly/appengine/redirect.html

@@ -0,0 +1,68 @@
1
+<!DOCTYPE html>
2
+<html>
3
+  <head>
4
+    <script>
5
+var loc = location.href;
6
+
7
+// Blockly files moved from /blockly to /static on 5 Dec 2012.
8
+if (loc.match('/blockly/')) {
9
+  loc = loc.replace('/blockly/', '/static/');
10
+}
11
+
12
+// Maze and Turtle moved from demos to apps on 29 Dec 2012.
13
+if (loc.match(/\/demos\/(maze|turtle)\//)) {
14
+  loc = loc.replace('/demos/', '/apps/');
15
+}
16
+
17
+// Vietnamese apps moved from vn to vi on 9 Jun 2012.
18
+if (loc.match('/vn.html')) {
19
+  loc = loc.replace('/vn.html', '/vi.html');
20
+}
21
+
22
+if (loc.match('/code/code.html')) {
23
+  // Code moved to index.html on 7 Aug 2013.
24
+  loc = loc.replace('/code.html', '/index.html');
25
+} else if (loc.match('/apps/code/zh_tw.html')) {
26
+  // zh-tw was changed to zh-hans on 25 Nov 2013.
27
+  loc = loc.replace('/zh_tw.html', '/index.html?lang=zh-hans');
28
+} else if (loc.match('/apps/code/index.html')) {
29
+  // NOP.
30
+} else if (loc.match(/\/apps\/code\/[-a-z]+\.html/)) {
31
+  // Code became language-agnostic on 20 Jul 2013.
32
+  loc = loc.replace(/\/([-a-z]+)\.html/, '/index.html?lang=$1');
33
+}
34
+
35
+if (loc.match('/apps/plane/plane.html')) {
36
+  // Plane moved to index.html on 7 Aug 2013.
37
+  loc = loc.replace('/plane.html', '/index.html');
38
+} else if (loc.match('/apps/code/plane.html')) {
39
+  // NOP.
40
+} else if (loc.match(/\/apps\/plane\/[\d_]*[-a-z]+\.html/)) {
41
+  // Plane became language-agnostic on 20 Jul 2013.
42
+  loc = loc.replace('vn.html', 'vi.html');
43
+  if (location.search) {
44
+    loc = loc.replace(/\/[\d_]*([-a-z]+)\.html\?/, '/index.html?lang=$1&');
45
+  } else {
46
+    loc = loc.replace(/\/[\d_]*([-a-z]+)\.html/, '/index.html?lang=$1');
47
+  }
48
+}
49
+
50
+if (loc.match('/apps/puzzle/')) {
51
+  // Puzzle moved to Blockly Games on 15 Oct 2014.
52
+  loc = 'https://blockly-games.appspot.com/puzzle';
53
+} else if (loc.match('/apps/maze/')) {
54
+  // Maze moved to Blockly Games on 10 Nov 2014.
55
+  loc = 'https://blockly-games.appspot.com/maze';
56
+} else if (loc.match('/apps/turtle/')) {
57
+  // Turtle moved to Blockly Games on 10 Nov 2014.
58
+  loc = 'https://blockly-games.appspot.com/turtle';
59
+} else if (loc.match('/apps/')) {
60
+  // Remaining apps moved to demos on 20 Nov 2014.
61
+  loc = loc.replace('/apps/', '/demos/');
62
+}
63
+
64
+location = loc;
65
+
66
+    </script>
67
+  </head>
68
+</html>

+ 2 - 0
src/blockly/appengine/robots.txt

@@ -0,0 +1,2 @@
1
+User-agent: *
2
+Disallow: /storage

+ 194 - 0
src/blockly/appengine/storage.js

@@ -0,0 +1,194 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Loading and saving blocks with localStorage and cloud storage.
23
+ * @author q.neutron@gmail.com (Quynh Neutron)
24
+ */
25
+'use strict';
26
+
27
+// Create a namespace.
28
+var BlocklyStorage = {};
29
+
30
+/**
31
+ * Backup code blocks to localStorage.
32
+ * @param {!Blockly.WorkspaceSvg} workspace Workspace.
33
+ * @private
34
+ */
35
+BlocklyStorage.backupBlocks_ = function(workspace) {
36
+  if ('localStorage' in window) {
37
+    var xml = Blockly.Xml.workspaceToDom(workspace);
38
+    // Gets the current URL, not including the hash.
39
+    var url = window.location.href.split('#')[0];
40
+    window.localStorage.setItem(url, Blockly.Xml.domToText(xml));
41
+  }
42
+};
43
+
44
+/**
45
+ * Bind the localStorage backup function to the unload event.
46
+ * @param {Blockly.WorkspaceSvg=} opt_workspace Workspace.
47
+ */
48
+BlocklyStorage.backupOnUnload = function(opt_workspace) {
49
+  var workspace = opt_workspace || Blockly.getMainWorkspace();
50
+  window.addEventListener('unload',
51
+      function() {BlocklyStorage.backupBlocks_(workspace);}, false);
52
+};
53
+
54
+/**
55
+ * Restore code blocks from localStorage.
56
+ * @param {Blockly.WorkspaceSvg=} opt_workspace Workspace.
57
+ */
58
+BlocklyStorage.restoreBlocks = function(opt_workspace) {
59
+  var url = window.location.href.split('#')[0];
60
+  if ('localStorage' in window && window.localStorage[url]) {
61
+    var workspace = opt_workspace || Blockly.getMainWorkspace();
62
+    var xml = Blockly.Xml.textToDom(window.localStorage[url]);
63
+    Blockly.Xml.domToWorkspace(workspace, xml);
64
+  }
65
+};
66
+
67
+/**
68
+ * Save blocks to database and return a link containing key to XML.
69
+ * @param {Blockly.WorkspaceSvg=} opt_workspace Workspace.
70
+ */
71
+BlocklyStorage.link = function(opt_workspace) {
72
+  var workspace = opt_workspace || Blockly.getMainWorkspace();
73
+  var xml = Blockly.Xml.workspaceToDom(workspace);
74
+  var data = Blockly.Xml.domToText(xml);
75
+  BlocklyStorage.makeRequest_('/storage', 'xml', data, workspace);
76
+};
77
+
78
+/**
79
+ * Retrieve XML text from database using given key.
80
+ * @param {string} key Key to XML, obtained from href.
81
+ * @param {Blockly.WorkspaceSvg=} opt_workspace Workspace.
82
+ */
83
+BlocklyStorage.retrieveXml = function(key, opt_workspace) {
84
+  var workspace = opt_workspace || Blockly.getMainWorkspace();
85
+  BlocklyStorage.makeRequest_('/storage', 'key', key, workspace);
86
+};
87
+
88
+/**
89
+ * Global reference to current AJAX request.
90
+ * @type {XMLHttpRequest}
91
+ * @private
92
+ */
93
+BlocklyStorage.httpRequest_ = null;
94
+
95
+/**
96
+ * Fire a new AJAX request.
97
+ * @param {string} url URL to fetch.
98
+ * @param {string} name Name of parameter.
99
+ * @param {string} content Content of parameter.
100
+ * @param {!Blockly.WorkspaceSvg} workspace Workspace.
101
+ * @private
102
+ */
103
+BlocklyStorage.makeRequest_ = function(url, name, content, workspace) {
104
+  if (BlocklyStorage.httpRequest_) {
105
+    // AJAX call is in-flight.
106
+    BlocklyStorage.httpRequest_.abort();
107
+  }
108
+  BlocklyStorage.httpRequest_ = new XMLHttpRequest();
109
+  BlocklyStorage.httpRequest_.name = name;
110
+  BlocklyStorage.httpRequest_.onreadystatechange =
111
+      BlocklyStorage.handleRequest_;
112
+  BlocklyStorage.httpRequest_.open('POST', url);
113
+  BlocklyStorage.httpRequest_.setRequestHeader('Content-Type',
114
+      'application/x-www-form-urlencoded');
115
+  BlocklyStorage.httpRequest_.send(name + '=' + encodeURIComponent(content));
116
+  BlocklyStorage.httpRequest_.workspace = workspace;
117
+};
118
+
119
+/**
120
+ * Callback function for AJAX call.
121
+ * @private
122
+ */
123
+BlocklyStorage.handleRequest_ = function() {
124
+  if (BlocklyStorage.httpRequest_.readyState == 4) {
125
+    if (BlocklyStorage.httpRequest_.status != 200) {
126
+      BlocklyStorage.alert(BlocklyStorage.HTTPREQUEST_ERROR + '\n' +
127
+          'httpRequest_.status: ' + BlocklyStorage.httpRequest_.status);
128
+    } else {
129
+      var data = BlocklyStorage.httpRequest_.responseText.trim();
130
+      if (BlocklyStorage.httpRequest_.name == 'xml') {
131
+        window.location.hash = data;
132
+        BlocklyStorage.alert(BlocklyStorage.LINK_ALERT.replace('%1',
133
+            window.location.href));
134
+      } else if (BlocklyStorage.httpRequest_.name == 'key') {
135
+        if (!data.length) {
136
+          BlocklyStorage.alert(BlocklyStorage.HASH_ERROR.replace('%1',
137
+              window.location.hash));
138
+        } else {
139
+          BlocklyStorage.loadXml_(data, BlocklyStorage.httpRequest_.workspace);
140
+        }
141
+      }
142
+      BlocklyStorage.monitorChanges_(BlocklyStorage.httpRequest_.workspace);
143
+    }
144
+    BlocklyStorage.httpRequest_ = null;
145
+  }
146
+};
147
+
148
+/**
149
+ * Start monitoring the workspace.  If a change is made that changes the XML,
150
+ * clear the key from the URL.  Stop monitoring the workspace once such a
151
+ * change is detected.
152
+ * @param {!Blockly.WorkspaceSvg} workspace Workspace.
153
+ * @private
154
+ */
155
+BlocklyStorage.monitorChanges_ = function(workspace) {
156
+  var startXmlDom = Blockly.Xml.workspaceToDom(workspace);
157
+  var startXmlText = Blockly.Xml.domToText(startXmlDom);
158
+  function change() {
159
+    var xmlDom = Blockly.Xml.workspaceToDom(workspace);
160
+    var xmlText = Blockly.Xml.domToText(xmlDom);
161
+    if (startXmlText != xmlText) {
162
+      window.location.hash = '';
163
+      workspace.removeChangeListener(bindData);
164
+    }
165
+  }
166
+  var bindData = workspace.addChangeListener(change);
167
+};
168
+
169
+/**
170
+ * Load blocks from XML.
171
+ * @param {string} xml Text representation of XML.
172
+ * @param {!Blockly.WorkspaceSvg} workspace Workspace.
173
+ * @private
174
+ */
175
+BlocklyStorage.loadXml_ = function(xml, workspace) {
176
+  try {
177
+    xml = Blockly.Xml.textToDom(xml);
178
+  } catch (e) {
179
+    BlocklyStorage.alert(BlocklyStorage.XML_ERROR + '\nXML: ' + xml);
180
+    return;
181
+  }
182
+  // Clear the workspace to avoid merge.
183
+  workspace.clear();
184
+  Blockly.Xml.domToWorkspace(workspace, xml);
185
+};
186
+
187
+/**
188
+ * Present a text message to the user.
189
+ * Designed to be overridden if an app has custom dialogs, or a butter bar.
190
+ * @param {string} message Text to alert.
191
+ */
192
+BlocklyStorage.alert = function(message) {
193
+  window.alert(message);
194
+};

+ 85 - 0
src/blockly/appengine/storage.py

@@ -0,0 +1,85 @@
1
+"""Blockly Demo: Storage
2
+
3
+Copyright 2012 Google Inc.
4
+https://developers.google.com/blockly/
5
+
6
+Licensed under the Apache License, Version 2.0 (the "License");
7
+you may not use this file except in compliance with the License.
8
+You may obtain a copy of the License at
9
+
10
+  http://www.apache.org/licenses/LICENSE-2.0
11
+
12
+Unless required by applicable law or agreed to in writing, software
13
+distributed under the License is distributed on an "AS IS" BASIS,
14
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
+See the License for the specific language governing permissions and
16
+limitations under the License.
17
+"""
18
+
19
+"""Store and retrieve XML with App Engine.
20
+"""
21
+
22
+__author__ = "q.neutron@gmail.com (Quynh Neutron)"
23
+
24
+import cgi
25
+from random import randint
26
+from google.appengine.ext import db
27
+from google.appengine.api import memcache
28
+import logging
29
+
30
+print "Content-Type: text/plain\n"
31
+
32
+def keyGen():
33
+  # Generate a random string of length KEY_LEN.
34
+  KEY_LEN = 6
35
+  CHARS = "abcdefghijkmnopqrstuvwxyz23456789" # Exclude l, 0, 1.
36
+  max_index = len(CHARS) - 1
37
+  return "".join([CHARS[randint(0, max_index)] for x in range(KEY_LEN)])
38
+
39
+class Xml(db.Model):
40
+  # A row in the database.
41
+  xml_hash = db.IntegerProperty()
42
+  xml_content = db.TextProperty()
43
+
44
+forms = cgi.FieldStorage()
45
+if "xml" in forms:
46
+  # Store XML and return a generated key.
47
+  xml_content = forms["xml"].value
48
+  xml_hash = hash(xml_content)
49
+  lookup_query = db.Query(Xml)
50
+  lookup_query.filter("xml_hash =", xml_hash)
51
+  lookup_result = lookup_query.get()
52
+  if lookup_result:
53
+    xml_key = lookup_result.key().name()
54
+  else:
55
+    trials = 0
56
+    result = True
57
+    while result:
58
+      trials += 1
59
+      if trials == 100:
60
+        raise Exception("Sorry, the generator failed to get a key for you.")
61
+      xml_key = keyGen()
62
+      result = db.get(db.Key.from_path("Xml", xml_key))
63
+    xml = db.Text(xml_content, encoding="utf_8")
64
+    row = Xml(key_name = xml_key, xml_hash = xml_hash, xml_content = xml)
65
+    row.put()
66
+  print xml_key
67
+
68
+if "key" in forms:
69
+  # Retrieve stored XML based on the provided key.
70
+  key_provided = forms["key"].value
71
+  # Normalize the string.
72
+  key_provided = key_provided.lower().strip()
73
+  # Check memcache for a quick match.
74
+  xml = memcache.get("XML_" + key_provided)
75
+  if xml is None:
76
+    # Check datastore for a definitive match.
77
+    result = db.get(db.Key.from_path("Xml", key_provided))
78
+    if not result:
79
+      xml = ""
80
+    else:
81
+      xml = result.xml_content
82
+    # Save to memcache for next hit.
83
+    if not memcache.add("XML_" + key_provided, xml, 3600):
84
+      logging.error("Memcache set failed.")
85
+  print xml.encode("utf-8")

File diff suppressed because it is too large
+ 70 - 0
src/blockly/arduino_compressed.js


File diff suppressed because it is too large
+ 1307 - 0
src/blockly/blockly_compressed.js


File diff suppressed because it is too large
+ 1605 - 0
src/blockly/blockly_uncompressed.js


+ 232 - 0
src/blockly/blocks/base.js

@@ -0,0 +1,232 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Fred Lin.
6
+ * https://github.com/gasolin/BlocklyDuino
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Helper functions for generating Arduino blocks.
23
+ * @author gasolin@gmail.com (Fred Lin)
24
+ */
25
+'use strict';
26
+
27
+//To support syntax defined in http://arduino.cc/en/Reference/HomePage
28
+
29
+goog.provide('Blockly.Blocks.base');
30
+
31
+goog.require('Blockly.Blocks');
32
+
33
+
34
+Blockly.Blocks['base_delay'] = {
35
+  helpUrl: 'http://arduino.cc/en/Reference/delay',
36
+  init: function() {
37
+    this.setColour(120);
38
+    this.appendDummyInput()
39
+        .appendField(Blockly.Msg.BASE_DELAY);
40
+    this.appendValueInput("DELAY_TIME", 'Number')
41
+        .setCheck('Number');
42
+    this.appendDummyInput()
43
+        .appendField(Blockly.Msg.BASE_DELAY_UNIT);
44
+    this.setInputsInline(true);
45
+    this.setPreviousStatement(true, null);
46
+    this.setNextStatement(true, null);
47
+    this.setTooltip('Delay specific time');
48
+  }
49
+};
50
+
51
+Blockly.Blocks['base_map'] = {
52
+  helpUrl: 'http://arduino.cc/en/Reference/map',
53
+  init: function() {
54
+    this.setColour(230);
55
+    this.appendValueInput("NUM", 'Number')
56
+        .appendField("Map ")
57
+        .setCheck('Number');
58
+    this.appendValueInput("DMAX", 'Number')
59
+        .appendField("value to [0-")
60
+        .setCheck('Number');
61
+    this.appendDummyInput()
62
+	      .appendField("]");
63
+    this.setInputsInline(true);
64
+    this.setOutput(true);
65
+    this.setTooltip('Re-maps a number from [0-1024] to another.');
66
+  }
67
+};
68
+
69
+Blockly.Blocks['inout_buildin_led'] = {
70
+   helpUrl: 'http://arduino.cc/en/Reference/DigitalWrite',
71
+   init: function() {
72
+     this.setColour(190);
73
+     this.appendDummyInput()
74
+	       .appendField("Build-in LED Stat")
75
+	       .appendField(new Blockly.FieldDropdown([["HIGH", "HIGH"], ["LOW", "LOW"]]), "STAT");
76
+     this.setPreviousStatement(true, null);
77
+     this.setNextStatement(true, null);
78
+     this.setTooltip('light or off the build-in LED');
79
+   }
80
+};
81
+
82
+Blockly.Blocks['inout_digital_write'] = {
83
+  helpUrl: 'http://arduino.cc/en/Reference/DigitalWrite',
84
+  init: function() {
85
+    this.setColour(230);
86
+    this.appendDummyInput()
87
+	      .appendField("DigitalWrite PIN#")
88
+	      .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN")
89
+      	.appendField("Stat")
90
+      	.appendField(new Blockly.FieldDropdown([["HIGH", "HIGH"], ["LOW", "LOW"]]), "STAT");
91
+    this.setPreviousStatement(true, null);
92
+    this.setNextStatement(true, null);
93
+    this.setTooltip('Write digital value to a specific Port');
94
+  }
95
+};
96
+
97
+Blockly.Blocks['inout_digital_read'] = {
98
+  helpUrl: 'http://arduino.cc/en/Reference/DigitalRead',
99
+  init: function() {
100
+    this.setColour(230);
101
+    this.appendDummyInput()
102
+	      .appendField("DigitalRead PIN#")
103
+	      .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
104
+    this.setOutput(true, 'Boolean');
105
+    this.setTooltip('');
106
+  }
107
+};
108
+
109
+Blockly.Blocks['inout_analog_write'] = {
110
+  helpUrl: 'http://arduino.cc/en/Reference/AnalogWrite',
111
+  init: function() {
112
+    this.setColour(230);
113
+    this.appendDummyInput()
114
+        .appendField("AnalogWrite PIN#")
115
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
116
+    this.appendValueInput("NUM", 'Number')
117
+        .appendField("value")
118
+        .setCheck('Number');
119
+    this.setInputsInline(true);
120
+    this.setPreviousStatement(true, null);
121
+    this.setNextStatement(true, null);
122
+    this.setTooltip('Write analog value between 0 and 255 to a specific Port');
123
+  }
124
+};
125
+
126
+Blockly.Blocks['inout_analog_read'] = {
127
+  helpUrl: 'http://arduino.cc/en/Reference/AnalogRead',
128
+  init: function() {
129
+    this.setColour(230);
130
+    this.appendDummyInput()
131
+        .appendField("AnalogRead PIN#")
132
+        .appendField(new Blockly.FieldDropdown(profile.default.analog), "PIN");
133
+    this.setOutput(true, 'Number');
134
+    this.setTooltip('Return value between 0 and 1024');
135
+  }
136
+};
137
+
138
+Blockly.Blocks['inout_tone'] = {
139
+  helpUrl: 'http://www.arduino.cc/en/Reference/Tone',
140
+  init: function() {
141
+    this.setColour(230);
142
+    this.appendDummyInput()
143
+        .appendField("Tone PIN#")
144
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
145
+    this.appendValueInput("NUM", "Number")
146
+        .appendField("frequency")
147
+        .setCheck("Number");
148
+    this.setInputsInline(true);
149
+    this.setPreviousStatement(true, null);
150
+    this.setNextStatement(true, null);
151
+    this.setTooltip("Generate audio tones on a pin");
152
+  }
153
+};
154
+
155
+Blockly.Blocks['inout_notone'] = {
156
+  helpUrl: 'http://www.arduino.cc/en/Reference/NoTone',
157
+  init: function() {
158
+    this.setColour(230);
159
+    this.appendDummyInput()
160
+        .appendField("No tone PIN#")
161
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
162
+    this.setInputsInline(true);
163
+    this.setPreviousStatement(true, null);
164
+    this.setNextStatement(true, null);
165
+    this.setTooltip("Stop generating a tone on a pin");
166
+  }
167
+};
168
+
169
+Blockly.Blocks['inout_highlow'] = {
170
+  helpUrl: 'http://arduino.cc/en/Reference/Constants',
171
+  init: function() {
172
+    this.setColour(230);
173
+    this.appendDummyInput()
174
+        .appendField(new Blockly.FieldDropdown([["HIGH", "HIGH"], ["LOW", "LOW"]]), 'BOOL')
175
+    this.setOutput(true, 'Boolean');
176
+    this.setTooltip('');
177
+  }
178
+};
179
+
180
+//servo block
181
+//http://www.seeedstudio.com/depot/emax-9g-es08a-high-sensitive-mini-servo-p-760.html?cPath=170_171
182
+Blockly.Blocks['servo_move'] = {
183
+  helpUrl: 'http://www.arduino.cc/playground/ComponentLib/servo',
184
+  init: function() {
185
+    this.setColour(190);
186
+    this.appendDummyInput()
187
+        .appendField("Servo")
188
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/depot/images/product/a991.jpg", 64, 64))
189
+        .appendField("PIN#")
190
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN")
191
+    this.appendValueInput("DEGREE", 'Number')
192
+        .setCheck('Number')
193
+        .setAlign(Blockly.ALIGN_RIGHT)
194
+        .appendField("Degree (0~180)");
195
+    this.appendValueInput("DELAY_TIME", 'Number')
196
+        .setCheck('Number')
197
+        .setAlign(Blockly.ALIGN_RIGHT)
198
+        .appendField("Delay");
199
+    this.setPreviousStatement(true, null);
200
+    this.setNextStatement(true, null);
201
+    this.setTooltip('move between 0~180 degree');
202
+  }
203
+};
204
+
205
+Blockly.Blocks['servo_read_degrees'] = {
206
+  helpUrl: 'http://www.arduino.cc/playground/ComponentLib/servo',
207
+  init: function() {
208
+    this.setColour(190);
209
+    this.appendDummyInput()
210
+        .appendField("Servo")
211
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/depot/images/product/a991.jpg", 64, 64))
212
+        .appendField("PIN#")
213
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
214
+    this.appendDummyInput()
215
+        .setAlign(Blockly.ALIGN_RIGHT)
216
+        .appendField("Read Degrees")
217
+    this.setOutput(true, 'Number');
218
+    this.setTooltip('return that degree with the last servo move.');
219
+  }
220
+};
221
+
222
+Blockly.Blocks['serial_print'] = {
223
+  helpUrl: 'http://www.arduino.cc/en/Serial/Print',
224
+  init: function() {
225
+    this.setColour(230);
226
+    this.appendValueInput("CONTENT", 'String')
227
+        .appendField("Serial Print");
228
+    this.setPreviousStatement(true, null);
229
+    this.setNextStatement(true, null);
230
+    this.setTooltip('Prints data to the console/serial port as human-readable ASCII text.');
231
+  }
232
+};

+ 117 - 0
src/blockly/blocks/colour.js

@@ -0,0 +1,117 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Colour blocks for Blockly.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Blocks.colour');
28
+
29
+goog.require('Blockly.Blocks');
30
+
31
+
32
+/**
33
+ * Common HSV hue for all blocks in this category.
34
+ */
35
+Blockly.Blocks.colour.HUE = 20;
36
+
37
+Blockly.Blocks['colour_picker'] = {
38
+  /**
39
+   * Block for colour picker.
40
+   * @this Blockly.Block
41
+   */
42
+  init: function() {
43
+    this.setHelpUrl(Blockly.Msg.COLOUR_PICKER_HELPURL);
44
+    this.setColour(Blockly.Blocks.colour.HUE);
45
+    this.appendDummyInput()
46
+        .appendField(new Blockly.FieldColour('#ff0000'), 'COLOUR');
47
+    this.setOutput(true, 'Colour');
48
+    this.setTooltip(Blockly.Msg.COLOUR_PICKER_TOOLTIP);
49
+  }
50
+};
51
+
52
+Blockly.Blocks['colour_random'] = {
53
+  /**
54
+   * Block for random colour.
55
+   * @this Blockly.Block
56
+   */
57
+  init: function() {
58
+    this.setHelpUrl(Blockly.Msg.COLOUR_RANDOM_HELPURL);
59
+    this.setColour(Blockly.Blocks.colour.HUE);
60
+    this.appendDummyInput()
61
+        .appendField(Blockly.Msg.COLOUR_RANDOM_TITLE);
62
+    this.setOutput(true, 'Colour');
63
+    this.setTooltip(Blockly.Msg.COLOUR_RANDOM_TOOLTIP);
64
+  }
65
+};
66
+
67
+Blockly.Blocks['colour_rgb'] = {
68
+  /**
69
+   * Block for composing a colour from RGB components.
70
+   * @this Blockly.Block
71
+   */
72
+  init: function() {
73
+    this.setHelpUrl(Blockly.Msg.COLOUR_RGB_HELPURL);
74
+    this.setColour(Blockly.Blocks.colour.HUE);
75
+    this.appendValueInput('RED')
76
+        .setCheck('Number')
77
+        .setAlign(Blockly.ALIGN_RIGHT)
78
+        .appendField(Blockly.Msg.COLOUR_RGB_TITLE)
79
+        .appendField(Blockly.Msg.COLOUR_RGB_RED);
80
+    this.appendValueInput('GREEN')
81
+        .setCheck('Number')
82
+        .setAlign(Blockly.ALIGN_RIGHT)
83
+        .appendField(Blockly.Msg.COLOUR_RGB_GREEN);
84
+    this.appendValueInput('BLUE')
85
+        .setCheck('Number')
86
+        .setAlign(Blockly.ALIGN_RIGHT)
87
+        .appendField(Blockly.Msg.COLOUR_RGB_BLUE);
88
+    this.setOutput(true, 'Colour');
89
+    this.setTooltip(Blockly.Msg.COLOUR_RGB_TOOLTIP);
90
+  }
91
+};
92
+
93
+Blockly.Blocks['colour_blend'] = {
94
+  /**
95
+   * Block for blending two colours together.
96
+   * @this Blockly.Block
97
+   */
98
+  init: function() {
99
+    this.setHelpUrl(Blockly.Msg.COLOUR_BLEND_HELPURL);
100
+    this.setColour(Blockly.Blocks.colour.HUE);
101
+    this.appendValueInput('COLOUR1')
102
+        .setCheck('Colour')
103
+        .setAlign(Blockly.ALIGN_RIGHT)
104
+        .appendField(Blockly.Msg.COLOUR_BLEND_TITLE)
105
+        .appendField(Blockly.Msg.COLOUR_BLEND_COLOUR1);
106
+    this.appendValueInput('COLOUR2')
107
+        .setCheck('Colour')
108
+        .setAlign(Blockly.ALIGN_RIGHT)
109
+        .appendField(Blockly.Msg.COLOUR_BLEND_COLOUR2);
110
+    this.appendValueInput('RATIO')
111
+        .setCheck('Number')
112
+        .setAlign(Blockly.ALIGN_RIGHT)
113
+        .appendField(Blockly.Msg.COLOUR_BLEND_RATIO);
114
+    this.setOutput(true, 'Colour');
115
+    this.setTooltip(Blockly.Msg.COLOUR_BLEND_TOOLTIP);
116
+  }
117
+};

+ 505 - 0
src/blockly/blocks/grove.js

@@ -0,0 +1,505 @@
1
+//http://www.seeedstudio.com/wiki/GROVE_System
2
+//http://www.seeedstudio.com/depot/index.php?main_page=advanced_search_result&search_in_description=1&keyword=grovefamily
3
+//support starter bundle example http://www.seeedstudio.com/wiki/GROVE_-_Starter_Kit_V1.1b
4
+
5
+/**
6
+ * @license
7
+ * Visual Blocks Editor
8
+ *
9
+ * Copyright 2012 Fred Lin.
10
+ * https://github.com/gasolin/BlocklyDuino
11
+ *
12
+ * Licensed under the Apache License, Version 2.0 (the "License");
13
+ * you may not use this file except in compliance with the License.
14
+ * You may obtain a copy of the License at
15
+ *
16
+ *   http://www.apache.org/licenses/LICENSE-2.0
17
+ *
18
+ * Unless required by applicable law or agreed to in writing, software
19
+ * distributed under the License is distributed on an "AS IS" BASIS,
20
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
21
+ * See the License for the specific language governing permissions and
22
+ * limitations under the License.
23
+ */
24
+
25
+/**
26
+ * @fileoverview Helper functions for generating seeeduino grove blocks.
27
+ * @author gasolin@gmail.com (Fred Lin)
28
+ */
29
+
30
+goog.provide('Blockly.Blocks.grove');
31
+
32
+goog.require('Blockly.Blocks');
33
+
34
+
35
+Blockly.Blocks['grove_led'] = {
36
+  helpUrl: 'http://www.seeedstudio.com/wiki/index.php?title=GROVE_-_Starter_Bundle_V1.0b#LED',
37
+  init: function() {
38
+    this.setColour(190);
39
+    this.appendDummyInput()
40
+        .appendField("LED")
41
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/e/e0/LED1.jpg/400px-LED1.jpg", 64, 64))
42
+        .appendField("PIN#")
43
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN")
44
+        .appendField("stat")
45
+        .appendField(new Blockly.FieldDropdown([["HIGH", "HIGH"], ["LOW", "LOW"]]), "STAT");
46
+    this.setPreviousStatement(true, null);
47
+    this.setNextStatement(true, null);
48
+    this.setTooltip('green LED');
49
+  }
50
+};
51
+
52
+Blockly.Blocks['grove_button'] = {
53
+  helpUrl: 'http://www.seeedstudio.com/wiki/index.php?title=GROVE_-_Starter_Bundle_V1.0b#Button',
54
+  init: function() {
55
+    this.setColour(190);
56
+    this.appendDummyInput()
57
+        .appendField("Button")
58
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/9/93/Button1.jpg/400px-Button1.jpg", 64, 64))
59
+        .appendField("PIN#")
60
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
61
+    this.setOutput(true, 'Boolean');
62
+    this.setTooltip('Basic digital input');
63
+  }
64
+};
65
+
66
+Blockly.Blocks['grove_rotary_angle'] = {
67
+  helpUrl: 'http://www.seeedstudio.com/wiki/index.php?title=GROVE_-_Starter_Bundle_V1.0b#Potentiometer',
68
+  init: function() {
69
+    this.setColour(10);
70
+    this.appendDummyInput()
71
+        .appendField("Rotary Angle")
72
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/5/59/Potentiometer1.jpg/400px-Potentiometer1.jpg", 64, 64))
73
+        .appendField("PIN#")
74
+        .appendField(new Blockly.FieldDropdown(profile.default.analog), "PIN");
75
+    this.setOutput(true, 'Number');
76
+    this.setTooltip('Analog output between 0 and Vcc');
77
+  }
78
+};
79
+
80
+Blockly.Blocks['grove_tilt_switch'] = {
81
+  helpUrl: 'http://www.seeedstudio.com/wiki/index.php?title=GROVE_-_Starter_Bundle_V1.0b#Tilt_switch',
82
+  init: function() {
83
+    this.setColour(190);
84
+    this.appendDummyInput()
85
+        .appendField("Tilt Switch")
86
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/9/95/Tilt1.jpg/400px-Tilt1.jpg", 64, 64))
87
+        .appendField("PIN#")
88
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
89
+    this.setOutput(true, 'Boolean');
90
+    this.setTooltip('When the switch is level it is open, and when tilted, the switch closes.');
91
+  }
92
+};
93
+
94
+Blockly.Blocks['grove_piezo_buzzer'] = {
95
+  helpUrl: 'http://www.seeedstudio.com/wiki/GROVE_-_Starter_Kit_V1.1b#Grove_.E2.80.93_Buzzer',
96
+  init: function() {
97
+    this.setColour(190);
98
+    this.appendDummyInput()
99
+        .appendField("Piezo Buzzer")
100
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/e/ed/Buzzer1.jpg/400px-Buzzer1.jpg", 64, 64))
101
+        .appendField("PIN#")
102
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN")
103
+        .appendField("stat")
104
+        .appendField(new Blockly.FieldDropdown([["HIGH", "HIGH"], ["LOW", "LOW"]]), "STAT");
105
+    this.setPreviousStatement(true, null);
106
+    this.setNextStatement(true, null);
107
+    this.setTooltip('Emit a tone when the output is high');
108
+  }
109
+};
110
+
111
+Blockly.Blocks['grove_relay'] = {
112
+  helpUrl: 'http://www.seeedstudio.com/wiki/Grove_-_Relay',
113
+  init: function() {
114
+    this.setColour(190);
115
+    this.appendDummyInput()
116
+        .appendField("Relay")
117
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/0/04/Twig-Relay1.jpg/400px-Twig-Relay1.jpg", 64, 64))
118
+        .appendField("PIN#")
119
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN")
120
+        .appendField("stat")
121
+        .appendField(new Blockly.FieldDropdown([["HIGH", "HIGH"], ["LOW", "LOW"]]), "STAT");
122
+    this.setPreviousStatement(true, null);
123
+    this.setNextStatement(true, null);
124
+    this.setTooltip('capable of switching a much higher voltages and currents. The maximum voltage and current that can be controlled by this module upto 250V at 10 amps.');
125
+  }
126
+};
127
+
128
+Blockly.Blocks['grove_temporature_sensor'] = {
129
+  helpUrl: 'http://www.seeedstudio.com/wiki/Project_Seven_-_Temperature',
130
+  init: function() {
131
+    this.setColour(10);
132
+    this.appendDummyInput()
133
+        .appendField("Temporature Sensor")
134
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/b/b0/Temperature1.jpg/400px-Temperature1.jpg", 64, 64))
135
+        .appendField("PIN#")
136
+        .appendField(new Blockly.FieldDropdown(profile.default.analog), "PIN")
137
+    this.setOutput(true, 'Number');
138
+    this.setTooltip('return number of ambient temperature in ℃');
139
+  }
140
+};
141
+
142
+Blockly.Blocks['grove_serial_lcd_print'] = {
143
+  helpUrl: 'http://www.seeedstudio.com/wiki/index.php?title=GROVE_-_Starter_Bundle_V1.0b#Serial_LCD',
144
+  init: function() {
145
+    this.setColour(190);
146
+    this.appendDummyInput()
147
+        .appendField("Serial LCD")
148
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/6/6a/LCD1.jpg/400px-LCD1.jpg", 64, 64))
149
+        .appendField("PIN#")
150
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
151
+    this.appendValueInput("TEXT", 'String')
152
+        .setCheck('String')
153
+        .setAlign(Blockly.ALIGN_RIGHT)
154
+        .appendField("print line1");
155
+    this.appendValueInput("TEXT2", 'String')
156
+        .setCheck('String')
157
+        .setAlign(Blockly.ALIGN_RIGHT)
158
+        .appendField("print line2")
159
+    this.appendValueInput("DELAY_TIME", 'Number')
160
+        .setCheck('Number')
161
+        .setAlign(Blockly.ALIGN_RIGHT)
162
+        .appendField("Delay");
163
+    this.setPreviousStatement(true, null);
164
+    this.setNextStatement(true, null);
165
+    this.setTooltip('print text on an 16 character by 2 line LCD.');
166
+  }
167
+};
168
+
169
+//grove lcd power on/off
170
+Blockly.Blocks['grove_serial_lcd_power'] = {
171
+  helpUrl: 'http://www.seeedstudio.com/wiki/index.php?title=GROVE_-_Starter_Bundle_V1.0b#LED',
172
+  init: function() {
173
+    this.setColour(190);
174
+    this.appendDummyInput()
175
+        .appendField("Serial LCD")
176
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/6/6a/LCD1.jpg/400px-LCD1.jpg", 64, 64))
177
+        .appendField("PIN#")
178
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
179
+    this.appendDummyInput()
180
+        .setAlign(Blockly.ALIGN_RIGHT)
181
+        .appendField("Power")
182
+        .appendField(new Blockly.FieldDropdown([["ON", "ON"], ["OFF", "OFF"]]), "STAT");
183
+    this.setPreviousStatement(true, null);
184
+    this.setNextStatement(true, null);
185
+    this.setTooltip('Turn LCD power on/off');
186
+  }
187
+};
188
+
189
+//scroll left/right/no scroll/blink/noblink
190
+Blockly.Blocks['grove_serial_lcd_effect'] = {
191
+  helpUrl: 'http://www.seeedstudio.com/wiki/index.php?title=GROVE_-_Starter_Bundle_V1.0b#LED',
192
+  init: function() {
193
+    this.setColour(190);
194
+    this.appendDummyInput()
195
+        .appendField("Serial LCD")
196
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/6/6a/LCD1.jpg/400px-LCD1.jpg", 64, 64))
197
+        .appendField("PIN#")
198
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
199
+    this.appendDummyInput()
200
+        .setAlign(Blockly.ALIGN_RIGHT)
201
+        .appendField("Effect")
202
+        .appendField(new Blockly.FieldDropdown([["Scroll Left", "LEFT"], ["Scroll Right", "RIGHT"], ["Scroll Auto", "AUTO"]]), "STAT");
203
+    this.setPreviousStatement(true, null);
204
+    this.setNextStatement(true, null);
205
+    this.setTooltip('Turn LCD power on/off');
206
+  }
207
+};
208
+
209
+Blockly.Blocks['grove_sound_sensor'] = {
210
+  helpUrl: 'http://www.seeedstudio.com/wiki/Grove_-_Sound_Sensor',
211
+  init: function() {
212
+    this.setColour(10);
213
+    this.appendDummyInput()
214
+        .appendField("Sound Sensor")
215
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/e/e3/Twig-Sound-sensor.jpg/400px-Twig-Sound-sensor.jpg", 64, 64))
216
+        .appendField("PIN#")
217
+        .appendField(new Blockly.FieldDropdown(profile.default.analog), "PIN")
218
+    this.setOutput(true, 'Number');
219
+    this.setTooltip('Detect the sound strength of the environment');
220
+  }
221
+};
222
+
223
+Blockly.Blocks['grove_pir_motion_sensor'] = {
224
+  helpUrl: 'http://www.seeedstudio.com/wiki/Grove_-_PIR_Motion_Sensor',
225
+  init: function() {
226
+    this.setColour(190);
227
+    this.appendDummyInput()
228
+        .appendField("PIR Motion Sensor")
229
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/f/fd/Twig-PIR_Motion_Sensor.jpg/400px-Twig-PIR_Motion_Sensor.jpg", 64, 64))
230
+        .appendField("PIN#")
231
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN")
232
+    this.setOutput(true, 'Number');
233
+    this.setTooltip('When anyone moves in it\'s detecting range, the sensor outputs HIGH.');
234
+  }
235
+};
236
+
237
+Blockly.Blocks['grove_line_finder'] = {
238
+  helpUrl: 'http://www.seeedstudio.com/wiki/Grove_-_Line_Finder',
239
+  init: function() {
240
+    this.setColour(190);
241
+    this.appendDummyInput()
242
+        .appendField("Line Finder")
243
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/8/82/Grovelinefinder1.jpg/400px-Grovelinefinder1.jpg", 64, 64))
244
+	      .appendField("PIN#")
245
+	      .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN");
246
+    this.setOutput(true, 'Boolean');
247
+    this.setTooltip('Output digital signal so the robot can reliably follow a black line on a white background');
248
+  }
249
+};
250
+
251
+Blockly.Blocks['grove_ultrasonic_ranger'] = {
252
+  helpUrl: 'http://www.seeedstudio.com/wiki/Grove_-_Ultrasonic_Ranger',
253
+  init: function() {
254
+    this.setColour(190);
255
+    this.appendDummyInput()
256
+	      .appendField("Ultrasonic Ranger")
257
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/b/b0/Twig_-_Ultrasonic_Ranger2.jpg/200px-Twig_-_Ultrasonic_Ranger2.jpg", 64, 64))
258
+	      .appendField("PIN#")
259
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN")
260
+        .appendField("unit")
261
+        .appendField(new Blockly.FieldDropdown([["cm", "cm"],  ["inch", "inch"]]), "UNIT");
262
+    this.setOutput(true, 'Boolean');
263
+    this.setTooltip('Non-contact distance measurement module');
264
+  }
265
+};
266
+
267
+Blockly.Blocks['grove_motor_shield'] = {
268
+  helpUrl: 'http://www.seeedstudio.com/wiki/Motor_Shield',
269
+  init: function() {
270
+    this.setColour(190);
271
+    this.appendDummyInput()
272
+        .appendField("Motor")
273
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/4/4d/Smotoshield2.jpg/400px-Smotoshield2.jpg", 64, 64))
274
+        .appendField(new Blockly.FieldDropdown([["Stop", "stop"], ["Forward", "forward"], ["Right", "right"], ["Left", "left"], ["Backward", "backward"]]), "DIRECTION");
275
+    /*this.appendValueInput("SPEED", 'Number')
276
+        .setCheck('Number')
277
+        .setAlign(Blockly.ALIGN_RIGHT)
278
+        .appendField("Speed");*/
279
+    this.setPreviousStatement(true, null);
280
+    this.setNextStatement(true, null);
281
+    this.setTooltip('Drive two brushed DC motors');
282
+  }
283
+};
284
+
285
+Blockly.Blocks['grove_thumb_joystick'] = {
286
+  helpUrl: 'http://www.seeedstudio.com/wiki/Grove_-_Thumb_Joystick',
287
+  init: function() {
288
+    this.setColour(10);
289
+    this.appendDummyInput()
290
+	.appendField("Thumb Joystick")
291
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/images/thumb/e/e0/Twig_-_Thumb_Joystick_v0.9b.jpg/200px-Twig_-_Thumb_Joystick_v0.9b.jpg", 64, 64))
292
+	.appendField("PIN#")
293
+        .appendField(new Blockly.FieldDropdown(profile.default.analog), "PIN")
294
+        .appendField("axis")
295
+        .appendField(new Blockly.FieldDropdown([["x", "x"],  ["y", "y"]]), "AXIS");
296
+    this.setOutput(true, 'Number');
297
+this.setTooltip('output two analog values(200~800) representing two directions');
298
+  }
299
+};
300
+
301
+Blockly.Blocks['grove_rgb_led'] = {
302
+  helpUrl: 'http://www.seeedstudio.com/wiki/index.php?title=Twig_-_Chainable_RGB_LED',
303
+  init: function() {
304
+    this.setColour(190);
305
+    this.appendDummyInput()
306
+  .appendField("Chainable RGB LED")
307
+        .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/depot/images/product/chanbalelednb1.jpg", 64, 64))
308
+  .appendField("PIN#")
309
+        .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN")
310
+    this.appendDummyInput("COLOR0")
311
+        .setAlign(Blockly.ALIGN_RIGHT)
312
+        .appendField("Color 1")
313
+        .appendField(new Blockly.FieldColour("#00ff00"), "RGB0");
314
+    this.setMutator(new Blockly.Mutator(['grove_rgb_led_item']));
315
+    this.setPreviousStatement(true, null);
316
+    this.setNextStatement(true, null);
317
+    this.setTooltip('256 color LED, currently Chainable feature is not support');
318
+    this.itemCount_ = 1;
319
+  },
320
+  mutationToDom: function() {
321
+    var container = document.createElement('mutation');
322
+    container.setAttribute('items', this.itemCount_);
323
+    for (var x = 0; x < this.itemCount_; x++) {
324
+      var colour_rgb = this.getFieldValue('RGB0');
325
+      //alert(colour_rgb);
326
+      container.setAttribute('RGB' + x, colour_rgb);
327
+    }
328
+    return container;
329
+  },
330
+  domToMutation: function(xmlElement) {
331
+    for (var x = 0; x < this.itemCount_; x++) {
332
+      this.removeInput('COLOR' + x);
333
+    }
334
+    this.itemCount_ = window.parseInt(xmlElement.getAttribute('items'), 10);
335
+    for (var x = 0; x < this.itemCount_; x++) {
336
+      var color = window.parseInt(xmlElement.getAttribute('RGB'+x), "#00ff00");
337
+      var input = this.appendDummyInput('COLOR' + x);
338
+      //if (x == 0) {
339
+        input.setAlign(Blockly.ALIGN_RIGHT)
340
+             .appendField("Color "+(x+1))
341
+             .appendField(new Blockly.FieldColour(color), "RGB" + x);
342
+      //}
343
+    }
344
+    if (this.itemCount_ == 0) {
345
+      this.appendDummyInput('COLOR0')
346
+          .setAlign(Blockly.ALIGN_RIGHT)
347
+          .appendField("Color 1")
348
+          .appendField(new Blockly.FieldColour("#00ff00"), "RGB0");
349
+    }
350
+  },
351
+  decompose: function(workspace) {
352
+    var containerBlock = Blockly.Block.obtain(workspace,
353
+                                              'grove_rgb_led_container');
354
+    containerBlock.initSvg();
355
+    var connection = containerBlock.getInput('STACK').connection;
356
+    for (var x = 0; x < this.itemCount_; x++) {
357
+      var itemBlock = Blockly.Block.obtain(workspace, 'grove_rgb_led_item');
358
+      itemBlock.initSvg();
359
+      connection.connect(itemBlock.previousConnection);
360
+      connection = itemBlock.nextConnection;
361
+    }
362
+    return containerBlock;
363
+  },
364
+  compose: function(containerBlock) {
365
+    // Disconnect all input blocks and remove all inputs.
366
+    if (this.itemCount_ == 0) {
367
+      this.removeInput('COLOR0');
368
+    } else {
369
+      for (var x = this.itemCount_ - 1; x >= 0; x--) {
370
+        //console.log("cnt:"+x);
371
+        this.removeInput('COLOR' + x);
372
+      }
373
+    }
374
+    /*var top;
375
+    if(this.itemCount_ > 0){
376
+      top = this.itemCount_-1;
377
+    } else {
378
+      top = 0;
379
+    }
380
+    console.log("top:"+top);*/
381
+    this.itemCount_ = 0;
382
+    // Rebuild the block's inputs.
383
+    var itemBlock = containerBlock.getInputTargetBlock('STACK');
384
+    while (itemBlock) {
385
+      var colour_rgb = this.getFieldValue('RGB' + this.itemCount_);
386
+      if(colour_rgb==null){
387
+          colour_rgb = "00ff00";
388
+      }
389
+      //console.log("blk:"+this.itemCount_);
390
+      /*if(top>this.itemCount_){
391
+        this.removeInput('COLOR' + this.itemCount_);
392
+      }*/
393
+      var input = this.appendDummyInput('COLOR' + this.itemCount_);
394
+      //if (this.itemCount_ == 0) {
395
+        input.setAlign(Blockly.ALIGN_RIGHT)
396
+             .appendField("Color " + (this.itemCount_+1))
397
+             .appendField(new Blockly.FieldColour(colour_rgb), "RGB" + this.itemCount_);
398
+      //}
399
+      // Reconnect any child blocks.
400
+      if (itemBlock.valueConnection_) {
401
+        input.connection.connect(itemBlock.valueConnection_);
402
+      }
403
+      this.itemCount_++;
404
+      itemBlock = itemBlock.nextConnection &&
405
+          itemBlock.nextConnection.targetBlock();
406
+    }
407
+    if (this.itemCount_ == 0) {
408
+      this.appendDummyInput('COLOR0')
409
+          .setAlign(Blockly.ALIGN_RIGHT)
410
+          .appendField("Color 1")
411
+          .appendField(new Blockly.FieldColour("#00ff00"), "RGB0");
412
+    }
413
+  }
414
+  /*saveConnections: function(containerBlock) {
415
+    // Store a pointer to any connected child blocks.
416
+    var itemBlock = containerBlock.getInputTargetBlock('STACK');
417
+    var x = 0;
418
+    while (itemBlock) {
419
+      var input = this.getInput('COLOR' + x);
420
+      itemBlock.valueConnection_ = input && input.connection.targetConnection;
421
+      x++;
422
+      itemBlock = itemBlock.nextConnection &&
423
+          itemBlock.nextConnection.targetBlock();
424
+    }
425
+  }*/
426
+};
427
+
428
+Blockly.Blocks['grove_rgb_led_container'] = {
429
+  // Container.
430
+  init: function() {
431
+    this.setColour(190);
432
+    this.appendDummyInput()
433
+        .appendField("Container");
434
+    this.appendStatementInput('STACK');
435
+    this.setTooltip("Add, remove items to reconfigure this chain");
436
+    this.contextMenu = false;
437
+  }
438
+};
439
+
440
+Blockly.Blocks['grove_rgb_led_item'] = {
441
+  // Add items.
442
+  init: function() {
443
+    this.setColour(190);
444
+    this.appendDummyInput()
445
+        .appendField("Item");
446
+    this.setPreviousStatement(true);
447
+    this.setNextStatement(true);
448
+    this.setTooltip("Add an item to the chain");
449
+    this.contextMenu = false;
450
+  }
451
+};
452
+
453
+Blockly.Blocks['grove_bluetooth_slave'] = {
454
+  category: 'Network',
455
+  helpUrl: 'http://www.seeedstudio.com/wiki/Grove_-_Serial_Bluetooth',
456
+  init: function() {
457
+    this.setColour(190);
458
+    this.appendDummyInput()
459
+      .appendField("Bluetooth Slave")
460
+      .appendField(new Blockly.FieldImage("http://www.seeedstudio.com/wiki/File:Twigbt00.jpg", 64, 64))
461
+      .appendField("PIN#")
462
+      .appendField(new Blockly.FieldDropdown(profile.default.digital), "PIN")
463
+    this.appendDummyInput()
464
+      .setAlign(Blockly.ALIGN_RIGHT)
465
+      .appendField("Name")
466
+      .appendField(new Blockly.FieldTextInput('blocklyduino'), 'NAME');
467
+    this.appendDummyInput()
468
+      .setAlign(Blockly.ALIGN_RIGHT)
469
+      .appendField("Pincode")
470
+      .appendField(new Blockly.FieldTextInput('0000'), 'PINCODE');
471
+    this.appendStatementInput("RCV")
472
+      .setAlign(Blockly.ALIGN_RIGHT)
473
+      .appendField("Receive");
474
+    this.appendStatementInput("SNT")
475
+      .setAlign(Blockly.ALIGN_RIGHT)
476
+      .appendField("Send");
477
+    this.setPreviousStatement(true, null);
478
+    this.setNextStatement(true, null);
479
+    this.setTooltip('Bluetooth V2.0+EDR slave. Support single slave per board');
480
+  }
481
+};
482
+//http://www.seeedstudio.com/wiki/File:Twig-Temp%26Humi.jpg
483
+//http://www.seeedstudio.com/wiki/Grove-_Temperature_and_Humidity_Sensor
484
+
485
+//http://www.seeedstudio.com/wiki/Grove_-_125KHz_RFID_Reader
486
+
487
+/*
488
+void setup()
489
+{
490
+	pinMode( 3 , OUTPUT);
491
+	pinMode( 1 , INPUT);
492
+}
493
+
494
+void loop()
495
+{
496
+	if (digitalRead( 1))
497
+	{
498
+		digitalWrite( 3 , HIGH);
499
+	}
500
+	else
501
+	{
502
+		digitalWrite( 1 , LOW);
503
+	}
504
+}
505
+*/

+ 681 - 0
src/blockly/blocks/lists.js

@@ -0,0 +1,681 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview List blocks for Blockly.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Blocks.lists');
28
+
29
+goog.require('Blockly.Blocks');
30
+
31
+
32
+/**
33
+ * Common HSV hue for all blocks in this category.
34
+ */
35
+Blockly.Blocks.lists.HUE = 260;
36
+
37
+Blockly.Blocks['lists_create_empty'] = {
38
+  /**
39
+   * Block for creating an empty list.
40
+   * @this Blockly.Block
41
+   */
42
+  init: function() {
43
+    this.setHelpUrl(Blockly.Msg.LISTS_CREATE_EMPTY_HELPURL);
44
+    this.setColour(Blockly.Blocks.lists.HUE);
45
+    this.setOutput(true, 'Array');
46
+    this.appendDummyInput()
47
+        .appendField(Blockly.Msg.LISTS_CREATE_EMPTY_TITLE);
48
+    this.setTooltip(Blockly.Msg.LISTS_CREATE_EMPTY_TOOLTIP);
49
+  }
50
+};
51
+
52
+Blockly.Blocks['lists_create_with'] = {
53
+  /**
54
+   * Block for creating a list with any number of elements of any type.
55
+   * @this Blockly.Block
56
+   */
57
+  init: function() {
58
+    this.setHelpUrl(Blockly.Msg.LISTS_CREATE_WITH_HELPURL);
59
+    this.setColour(Blockly.Blocks.lists.HUE);
60
+    this.itemCount_ = 3;
61
+    this.updateShape_();
62
+    this.setOutput(true, 'Array');
63
+    this.setMutator(new Blockly.Mutator(['lists_create_with_item']));
64
+    this.setTooltip(Blockly.Msg.LISTS_CREATE_WITH_TOOLTIP);
65
+  },
66
+  /**
67
+   * Create XML to represent list inputs.
68
+   * @return {Element} XML storage element.
69
+   * @this Blockly.Block
70
+   */
71
+  mutationToDom: function() {
72
+    var container = document.createElement('mutation');
73
+    container.setAttribute('items', this.itemCount_);
74
+    return container;
75
+  },
76
+  /**
77
+   * Parse XML to restore the list inputs.
78
+   * @param {!Element} xmlElement XML storage element.
79
+   * @this Blockly.Block
80
+   */
81
+  domToMutation: function(xmlElement) {
82
+    this.itemCount_ = parseInt(xmlElement.getAttribute('items'), 10);
83
+    this.updateShape_();
84
+  },
85
+  /**
86
+   * Populate the mutator's dialog with this block's components.
87
+   * @param {!Blockly.Workspace} workspace Mutator's workspace.
88
+   * @return {!Blockly.Block} Root block in mutator.
89
+   * @this Blockly.Block
90
+   */
91
+  decompose: function(workspace) {
92
+    var containerBlock =
93
+        Blockly.Block.obtain(workspace, 'lists_create_with_container');
94
+    containerBlock.initSvg();
95
+    var connection = containerBlock.getInput('STACK').connection;
96
+    for (var i = 0; i < this.itemCount_; i++) {
97
+      var itemBlock = Blockly.Block.obtain(workspace, 'lists_create_with_item');
98
+      itemBlock.initSvg();
99
+      connection.connect(itemBlock.previousConnection);
100
+      connection = itemBlock.nextConnection;
101
+    }
102
+    return containerBlock;
103
+  },
104
+  /**
105
+   * Reconfigure this block based on the mutator dialog's components.
106
+   * @param {!Blockly.Block} containerBlock Root block in mutator.
107
+   * @this Blockly.Block
108
+   */
109
+  compose: function(containerBlock) {
110
+    var itemBlock = containerBlock.getInputTargetBlock('STACK');
111
+    // Count number of inputs.
112
+    var connections = [];
113
+    var i = 0;
114
+    while (itemBlock) {
115
+      connections[i] = itemBlock.valueConnection_;
116
+      itemBlock = itemBlock.nextConnection &&
117
+          itemBlock.nextConnection.targetBlock();
118
+      i++;
119
+    }
120
+    this.itemCount_ = i;
121
+    this.updateShape_();
122
+    // Reconnect any child blocks.
123
+    for (var i = 0; i < this.itemCount_; i++) {
124
+      if (connections[i]) {
125
+        this.getInput('ADD' + i).connection.connect(connections[i]);
126
+      }
127
+    }
128
+  },
129
+  /**
130
+   * Store pointers to any connected child blocks.
131
+   * @param {!Blockly.Block} containerBlock Root block in mutator.
132
+   * @this Blockly.Block
133
+   */
134
+  saveConnections: function(containerBlock) {
135
+    var itemBlock = containerBlock.getInputTargetBlock('STACK');
136
+    var i = 0;
137
+    while (itemBlock) {
138
+      var input = this.getInput('ADD' + i);
139
+      itemBlock.valueConnection_ = input && input.connection.targetConnection;
140
+      i++;
141
+      itemBlock = itemBlock.nextConnection &&
142
+          itemBlock.nextConnection.targetBlock();
143
+    }
144
+  },
145
+  /**
146
+   * Modify this block to have the correct number of inputs.
147
+   * @private
148
+   * @this Blockly.Block
149
+   */
150
+  updateShape_: function() {
151
+    // Delete everything.
152
+    if (this.getInput('EMPTY')) {
153
+      this.removeInput('EMPTY');
154
+    } else {
155
+      var i = 0;
156
+      while (this.getInput('ADD' + i)) {
157
+        this.removeInput('ADD' + i);
158
+        i++;
159
+      }
160
+    }
161
+    // Rebuild block.
162
+    if (this.itemCount_ == 0) {
163
+      this.appendDummyInput('EMPTY')
164
+          .appendField(Blockly.Msg.LISTS_CREATE_EMPTY_TITLE);
165
+    } else {
166
+      for (var i = 0; i < this.itemCount_; i++) {
167
+        var input = this.appendValueInput('ADD' + i);
168
+        if (i == 0) {
169
+          input.appendField(Blockly.Msg.LISTS_CREATE_WITH_INPUT_WITH);
170
+        }
171
+      }
172
+    }
173
+  }
174
+};
175
+
176
+Blockly.Blocks['lists_create_with_container'] = {
177
+  /**
178
+   * Mutator block for list container.
179
+   * @this Blockly.Block
180
+   */
181
+  init: function() {
182
+    this.setColour(Blockly.Blocks.lists.HUE);
183
+    this.appendDummyInput()
184
+        .appendField(Blockly.Msg.LISTS_CREATE_WITH_CONTAINER_TITLE_ADD);
185
+    this.appendStatementInput('STACK');
186
+    this.setTooltip(Blockly.Msg.LISTS_CREATE_WITH_CONTAINER_TOOLTIP);
187
+    this.contextMenu = false;
188
+  }
189
+};
190
+
191
+Blockly.Blocks['lists_create_with_item'] = {
192
+  /**
193
+   * Mutator bolck for adding items.
194
+   * @this Blockly.Block
195
+   */
196
+  init: function() {
197
+    this.setColour(Blockly.Blocks.lists.HUE);
198
+    this.appendDummyInput()
199
+        .appendField(Blockly.Msg.LISTS_CREATE_WITH_ITEM_TITLE);
200
+    this.setPreviousStatement(true);
201
+    this.setNextStatement(true);
202
+    this.setTooltip(Blockly.Msg.LISTS_CREATE_WITH_ITEM_TOOLTIP);
203
+    this.contextMenu = false;
204
+  }
205
+};
206
+
207
+Blockly.Blocks['lists_repeat'] = {
208
+  /**
209
+   * Block for creating a list with one element repeated.
210
+   * @this Blockly.Block
211
+   */
212
+  init: function() {
213
+    this.jsonInit({
214
+      "message0": Blockly.Msg.LISTS_REPEAT_TITLE,
215
+      "args0": [
216
+        {
217
+          "type": "input_value",
218
+          "name": "ITEM"
219
+        },
220
+        {
221
+          "type": "input_value",
222
+          "name": "NUM",
223
+          "check": "Number"
224
+        }
225
+      ],
226
+      "output": "Array",
227
+      "colour": Blockly.Blocks.lists.HUE,
228
+      "tooltip": Blockly.Msg.LISTS_REPEAT_TOOLTIP,
229
+      "helpUrl": Blockly.Msg.LISTS_REPEAT_HELPURL
230
+    });
231
+  }
232
+};
233
+
234
+Blockly.Blocks['lists_length'] = {
235
+  /**
236
+   * Block for list length.
237
+   * @this Blockly.Block
238
+   */
239
+  init: function() {
240
+    this.jsonInit({
241
+      "message0": Blockly.Msg.LISTS_LENGTH_TITLE,
242
+      "args0": [
243
+        {
244
+          "type": "input_value",
245
+          "name": "VALUE",
246
+          "check": ['String', 'Array']
247
+        }
248
+      ],
249
+      "output": 'Number',
250
+      "colour": Blockly.Blocks.lists.HUE,
251
+      "tooltip": Blockly.Msg.LISTS_LENGTH_TOOLTIP,
252
+      "helpUrl": Blockly.Msg.LISTS_LENGTH_HELPURL
253
+    });
254
+  }
255
+};
256
+
257
+Blockly.Blocks['lists_isEmpty'] = {
258
+  /**
259
+   * Block for is the list empty?
260
+   * @this Blockly.Block
261
+   */
262
+  init: function() {
263
+    this.jsonInit({
264
+      "message0": Blockly.Msg.LISTS_ISEMPTY_TITLE,
265
+      "args0": [
266
+        {
267
+          "type": "input_value",
268
+          "name": "VALUE",
269
+          "check": ['String', 'Array']
270
+        }
271
+      ],
272
+      "output": 'Boolean',
273
+      "colour": Blockly.Blocks.lists.HUE,
274
+      "tooltip": Blockly.Msg.LISTS_ISEMPTY_TOOLTIP,
275
+      "helpUrl": Blockly.Msg.LISTS_ISEMPTY_HELPURL
276
+    });
277
+  }
278
+};
279
+
280
+Blockly.Blocks['lists_indexOf'] = {
281
+  /**
282
+   * Block for finding an item in the list.
283
+   * @this Blockly.Block
284
+   */
285
+  init: function() {
286
+    var OPERATORS =
287
+        [[Blockly.Msg.LISTS_INDEX_OF_FIRST, 'FIRST'],
288
+         [Blockly.Msg.LISTS_INDEX_OF_LAST, 'LAST']];
289
+    this.setHelpUrl(Blockly.Msg.LISTS_INDEX_OF_HELPURL);
290
+    this.setColour(Blockly.Blocks.lists.HUE);
291
+    this.setOutput(true, 'Number');
292
+    this.appendValueInput('VALUE')
293
+        .setCheck('Array')
294
+        .appendField(Blockly.Msg.LISTS_INDEX_OF_INPUT_IN_LIST);
295
+    this.appendValueInput('FIND')
296
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'END');
297
+    this.setInputsInline(true);
298
+    this.setTooltip(Blockly.Msg.LISTS_INDEX_OF_TOOLTIP);
299
+  }
300
+};
301
+
302
+Blockly.Blocks['lists_getIndex'] = {
303
+  /**
304
+   * Block for getting element at index.
305
+   * @this Blockly.Block
306
+   */
307
+  init: function() {
308
+    var MODE =
309
+        [[Blockly.Msg.LISTS_GET_INDEX_GET, 'GET'],
310
+         [Blockly.Msg.LISTS_GET_INDEX_GET_REMOVE, 'GET_REMOVE'],
311
+         [Blockly.Msg.LISTS_GET_INDEX_REMOVE, 'REMOVE']];
312
+    this.WHERE_OPTIONS =
313
+        [[Blockly.Msg.LISTS_GET_INDEX_FROM_START, 'FROM_START'],
314
+         [Blockly.Msg.LISTS_GET_INDEX_FROM_END, 'FROM_END'],
315
+         [Blockly.Msg.LISTS_GET_INDEX_FIRST, 'FIRST'],
316
+         [Blockly.Msg.LISTS_GET_INDEX_LAST, 'LAST'],
317
+         [Blockly.Msg.LISTS_GET_INDEX_RANDOM, 'RANDOM']];
318
+    this.setHelpUrl(Blockly.Msg.LISTS_GET_INDEX_HELPURL);
319
+    this.setColour(Blockly.Blocks.lists.HUE);
320
+    var modeMenu = new Blockly.FieldDropdown(MODE, function(value) {
321
+      var isStatement = (value == 'REMOVE');
322
+      this.sourceBlock_.updateStatement_(isStatement);
323
+    });
324
+    this.appendValueInput('VALUE')
325
+        .setCheck('Array')
326
+        .appendField(Blockly.Msg.LISTS_GET_INDEX_INPUT_IN_LIST);
327
+    this.appendDummyInput()
328
+        .appendField(modeMenu, 'MODE')
329
+        .appendField('', 'SPACE');
330
+    this.appendDummyInput('AT');
331
+    if (Blockly.Msg.LISTS_GET_INDEX_TAIL) {
332
+      this.appendDummyInput('TAIL')
333
+          .appendField(Blockly.Msg.LISTS_GET_INDEX_TAIL);
334
+    }
335
+    this.setInputsInline(true);
336
+    this.setOutput(true);
337
+    this.updateAt_(true);
338
+    // Assign 'this' to a variable for use in the tooltip closure below.
339
+    var thisBlock = this;
340
+    this.setTooltip(function() {
341
+      var combo = thisBlock.getFieldValue('MODE') + '_' +
342
+          thisBlock.getFieldValue('WHERE');
343
+      return Blockly.Msg['LISTS_GET_INDEX_TOOLTIP_' + combo];
344
+    });
345
+  },
346
+  /**
347
+   * Create XML to represent whether the block is a statement or a value.
348
+   * Also represent whether there is an 'AT' input.
349
+   * @return {Element} XML storage element.
350
+   * @this Blockly.Block
351
+   */
352
+  mutationToDom: function() {
353
+    var container = document.createElement('mutation');
354
+    var isStatement = !this.outputConnection;
355
+    container.setAttribute('statement', isStatement);
356
+    var isAt = this.getInput('AT').type == Blockly.INPUT_VALUE;
357
+    container.setAttribute('at', isAt);
358
+    return container;
359
+  },
360
+  /**
361
+   * Parse XML to restore the 'AT' input.
362
+   * @param {!Element} xmlElement XML storage element.
363
+   * @this Blockly.Block
364
+   */
365
+  domToMutation: function(xmlElement) {
366
+    // Note: Until January 2013 this block did not have mutations,
367
+    // so 'statement' defaults to false and 'at' defaults to true.
368
+    var isStatement = (xmlElement.getAttribute('statement') == 'true');
369
+    this.updateStatement_(isStatement);
370
+    var isAt = (xmlElement.getAttribute('at') != 'false');
371
+    this.updateAt_(isAt);
372
+  },
373
+  /**
374
+   * Switch between a value block and a statement block.
375
+   * @param {boolean} newStatement True if the block should be a statement.
376
+   *     False if the block should be a value.
377
+   * @private
378
+   * @this Blockly.Block
379
+   */
380
+  updateStatement_: function(newStatement) {
381
+    var oldStatement = !this.outputConnection;
382
+    if (newStatement != oldStatement) {
383
+      this.unplug(true, true);
384
+      if (newStatement) {
385
+        this.setOutput(false);
386
+        this.setPreviousStatement(true);
387
+        this.setNextStatement(true);
388
+      } else {
389
+        this.setPreviousStatement(false);
390
+        this.setNextStatement(false);
391
+        this.setOutput(true);
392
+      }
393
+    }
394
+  },
395
+  /**
396
+   * Create or delete an input for the numeric index.
397
+   * @param {boolean} isAt True if the input should exist.
398
+   * @private
399
+   * @this Blockly.Block
400
+   */
401
+  updateAt_: function(isAt) {
402
+    // Destroy old 'AT' and 'ORDINAL' inputs.
403
+    this.removeInput('AT');
404
+    this.removeInput('ORDINAL', true);
405
+    // Create either a value 'AT' input or a dummy input.
406
+    if (isAt) {
407
+      this.appendValueInput('AT').setCheck('Number');
408
+      if (Blockly.Msg.ORDINAL_NUMBER_SUFFIX) {
409
+        this.appendDummyInput('ORDINAL')
410
+            .appendField(Blockly.Msg.ORDINAL_NUMBER_SUFFIX);
411
+      }
412
+    } else {
413
+      this.appendDummyInput('AT');
414
+    }
415
+    var menu = new Blockly.FieldDropdown(this.WHERE_OPTIONS, function(value) {
416
+      var newAt = (value == 'FROM_START') || (value == 'FROM_END');
417
+      // The 'isAt' variable is available due to this function being a closure.
418
+      if (newAt != isAt) {
419
+        var block = this.sourceBlock_;
420
+        block.updateAt_(newAt);
421
+        // This menu has been destroyed and replaced.  Update the replacement.
422
+        block.setFieldValue(value, 'WHERE');
423
+        return null;
424
+      }
425
+      return undefined;
426
+    });
427
+    this.getInput('AT').appendField(menu, 'WHERE');
428
+    if (Blockly.Msg.LISTS_GET_INDEX_TAIL) {
429
+      this.moveInputBefore('TAIL', null);
430
+    }
431
+  }
432
+};
433
+
434
+Blockly.Blocks['lists_setIndex'] = {
435
+  /**
436
+   * Block for setting the element at index.
437
+   * @this Blockly.Block
438
+   */
439
+  init: function() {
440
+    var MODE =
441
+        [[Blockly.Msg.LISTS_SET_INDEX_SET, 'SET'],
442
+         [Blockly.Msg.LISTS_SET_INDEX_INSERT, 'INSERT']];
443
+    this.WHERE_OPTIONS =
444
+        [[Blockly.Msg.LISTS_GET_INDEX_FROM_START, 'FROM_START'],
445
+         [Blockly.Msg.LISTS_GET_INDEX_FROM_END, 'FROM_END'],
446
+         [Blockly.Msg.LISTS_GET_INDEX_FIRST, 'FIRST'],
447
+         [Blockly.Msg.LISTS_GET_INDEX_LAST, 'LAST'],
448
+         [Blockly.Msg.LISTS_GET_INDEX_RANDOM, 'RANDOM']];
449
+    this.setHelpUrl(Blockly.Msg.LISTS_SET_INDEX_HELPURL);
450
+    this.setColour(Blockly.Blocks.lists.HUE);
451
+    this.appendValueInput('LIST')
452
+        .setCheck('Array')
453
+        .appendField(Blockly.Msg.LISTS_SET_INDEX_INPUT_IN_LIST);
454
+    this.appendDummyInput()
455
+        .appendField(new Blockly.FieldDropdown(MODE), 'MODE')
456
+        .appendField('', 'SPACE');
457
+    this.appendDummyInput('AT');
458
+    this.appendValueInput('TO')
459
+        .appendField(Blockly.Msg.LISTS_SET_INDEX_INPUT_TO);
460
+    this.setInputsInline(true);
461
+    this.setPreviousStatement(true);
462
+    this.setNextStatement(true);
463
+    this.setTooltip(Blockly.Msg.LISTS_SET_INDEX_TOOLTIP);
464
+    this.updateAt_(true);
465
+    // Assign 'this' to a variable for use in the tooltip closure below.
466
+    var thisBlock = this;
467
+    this.setTooltip(function() {
468
+      var combo = thisBlock.getFieldValue('MODE') + '_' +
469
+          thisBlock.getFieldValue('WHERE');
470
+      return Blockly.Msg['LISTS_SET_INDEX_TOOLTIP_' + combo];
471
+    });
472
+  },
473
+  /**
474
+   * Create XML to represent whether there is an 'AT' input.
475
+   * @return {Element} XML storage element.
476
+   * @this Blockly.Block
477
+   */
478
+  mutationToDom: function() {
479
+    var container = document.createElement('mutation');
480
+    var isAt = this.getInput('AT').type == Blockly.INPUT_VALUE;
481
+    container.setAttribute('at', isAt);
482
+    return container;
483
+  },
484
+  /**
485
+   * Parse XML to restore the 'AT' input.
486
+   * @param {!Element} xmlElement XML storage element.
487
+   * @this Blockly.Block
488
+   */
489
+  domToMutation: function(xmlElement) {
490
+    // Note: Until January 2013 this block did not have mutations,
491
+    // so 'at' defaults to true.
492
+    var isAt = (xmlElement.getAttribute('at') != 'false');
493
+    this.updateAt_(isAt);
494
+  },
495
+  /**
496
+   * Create or delete an input for the numeric index.
497
+   * @param {boolean} isAt True if the input should exist.
498
+   * @private
499
+   * @this Blockly.Block
500
+   */
501
+  updateAt_: function(isAt) {
502
+    // Destroy old 'AT' and 'ORDINAL' input.
503
+    this.removeInput('AT');
504
+    this.removeInput('ORDINAL', true);
505
+    // Create either a value 'AT' input or a dummy input.
506
+    if (isAt) {
507
+      this.appendValueInput('AT').setCheck('Number');
508
+      if (Blockly.Msg.ORDINAL_NUMBER_SUFFIX) {
509
+        this.appendDummyInput('ORDINAL')
510
+            .appendField(Blockly.Msg.ORDINAL_NUMBER_SUFFIX);
511
+      }
512
+    } else {
513
+      this.appendDummyInput('AT');
514
+    }
515
+    var menu = new Blockly.FieldDropdown(this.WHERE_OPTIONS, function(value) {
516
+      var newAt = (value == 'FROM_START') || (value == 'FROM_END');
517
+      // The 'isAt' variable is available due to this function being a closure.
518
+      if (newAt != isAt) {
519
+        var block = this.sourceBlock_;
520
+        block.updateAt_(newAt);
521
+        // This menu has been destroyed and replaced.  Update the replacement.
522
+        block.setFieldValue(value, 'WHERE');
523
+        return null;
524
+      }
525
+      return undefined;
526
+    });
527
+    this.moveInputBefore('AT', 'TO');
528
+    if (this.getInput('ORDINAL')) {
529
+      this.moveInputBefore('ORDINAL', 'TO');
530
+    }
531
+
532
+    this.getInput('AT').appendField(menu, 'WHERE');
533
+  }
534
+};
535
+
536
+Blockly.Blocks['lists_getSublist'] = {
537
+  /**
538
+   * Block for getting sublist.
539
+   * @this Blockly.Block
540
+   */
541
+  init: function() {
542
+    this['WHERE_OPTIONS_1'] =
543
+        [[Blockly.Msg.LISTS_GET_SUBLIST_START_FROM_START, 'FROM_START'],
544
+         [Blockly.Msg.LISTS_GET_SUBLIST_START_FROM_END, 'FROM_END'],
545
+         [Blockly.Msg.LISTS_GET_SUBLIST_START_FIRST, 'FIRST']];
546
+    this['WHERE_OPTIONS_2'] =
547
+        [[Blockly.Msg.LISTS_GET_SUBLIST_END_FROM_START, 'FROM_START'],
548
+         [Blockly.Msg.LISTS_GET_SUBLIST_END_FROM_END, 'FROM_END'],
549
+         [Blockly.Msg.LISTS_GET_SUBLIST_END_LAST, 'LAST']];
550
+    this.setHelpUrl(Blockly.Msg.LISTS_GET_SUBLIST_HELPURL);
551
+    this.setColour(Blockly.Blocks.lists.HUE);
552
+    this.appendValueInput('LIST')
553
+        .setCheck('Array')
554
+        .appendField(Blockly.Msg.LISTS_GET_SUBLIST_INPUT_IN_LIST);
555
+    this.appendDummyInput('AT1');
556
+    this.appendDummyInput('AT2');
557
+    if (Blockly.Msg.LISTS_GET_SUBLIST_TAIL) {
558
+      this.appendDummyInput('TAIL')
559
+          .appendField(Blockly.Msg.LISTS_GET_SUBLIST_TAIL);
560
+    }
561
+    this.setInputsInline(true);
562
+    this.setOutput(true, 'Array');
563
+    this.updateAt_(1, true);
564
+    this.updateAt_(2, true);
565
+    this.setTooltip(Blockly.Msg.LISTS_GET_SUBLIST_TOOLTIP);
566
+  },
567
+  /**
568
+   * Create XML to represent whether there are 'AT' inputs.
569
+   * @return {Element} XML storage element.
570
+   * @this Blockly.Block
571
+   */
572
+  mutationToDom: function() {
573
+    var container = document.createElement('mutation');
574
+    var isAt1 = this.getInput('AT1').type == Blockly.INPUT_VALUE;
575
+    container.setAttribute('at1', isAt1);
576
+    var isAt2 = this.getInput('AT2').type == Blockly.INPUT_VALUE;
577
+    container.setAttribute('at2', isAt2);
578
+    return container;
579
+  },
580
+  /**
581
+   * Parse XML to restore the 'AT' inputs.
582
+   * @param {!Element} xmlElement XML storage element.
583
+   * @this Blockly.Block
584
+   */
585
+  domToMutation: function(xmlElement) {
586
+    var isAt1 = (xmlElement.getAttribute('at1') == 'true');
587
+    var isAt2 = (xmlElement.getAttribute('at2') == 'true');
588
+    this.updateAt_(1, isAt1);
589
+    this.updateAt_(2, isAt2);
590
+  },
591
+  /**
592
+   * Create or delete an input for a numeric index.
593
+   * This block has two such inputs, independant of each other.
594
+   * @param {number} n Specify first or second input (1 or 2).
595
+   * @param {boolean} isAt True if the input should exist.
596
+   * @private
597
+   * @this Blockly.Block
598
+   */
599
+  updateAt_: function(n, isAt) {
600
+    // Create or delete an input for the numeric index.
601
+    // Destroy old 'AT' and 'ORDINAL' inputs.
602
+    this.removeInput('AT' + n);
603
+    this.removeInput('ORDINAL' + n, true);
604
+    // Create either a value 'AT' input or a dummy input.
605
+    if (isAt) {
606
+      this.appendValueInput('AT' + n).setCheck('Number');
607
+      if (Blockly.Msg.ORDINAL_NUMBER_SUFFIX) {
608
+        this.appendDummyInput('ORDINAL' + n)
609
+            .appendField(Blockly.Msg.ORDINAL_NUMBER_SUFFIX);
610
+      }
611
+    } else {
612
+      this.appendDummyInput('AT' + n);
613
+    }
614
+    var menu = new Blockly.FieldDropdown(this['WHERE_OPTIONS_' + n],
615
+        function(value) {
616
+      var newAt = (value == 'FROM_START') || (value == 'FROM_END');
617
+      // The 'isAt' variable is available due to this function being a closure.
618
+      if (newAt != isAt) {
619
+        var block = this.sourceBlock_;
620
+        block.updateAt_(n, newAt);
621
+        // This menu has been destroyed and replaced.  Update the replacement.
622
+        block.setFieldValue(value, 'WHERE' + n);
623
+        return null;
624
+      }
625
+      return undefined;
626
+    });
627
+    this.getInput('AT' + n)
628
+        .appendField(menu, 'WHERE' + n);
629
+    if (n == 1) {
630
+      this.moveInputBefore('AT1', 'AT2');
631
+      if (this.getInput('ORDINAL1')) {
632
+        this.moveInputBefore('ORDINAL1', 'AT2');
633
+      }
634
+    }
635
+    if (Blockly.Msg.LISTS_GET_SUBLIST_TAIL) {
636
+      this.moveInputBefore('TAIL', null);
637
+    }
638
+  }
639
+};
640
+
641
+Blockly.Blocks['lists_split'] = {
642
+  /**
643
+   * Block for splitting text into a list, or joining a list into text.
644
+   * @this Blockly.Block
645
+   */
646
+  init: function() {
647
+    // Assign 'this' to a variable for use in the closures below.
648
+    var thisBlock = this;
649
+    var dropdown = new Blockly.FieldDropdown(
650
+        [[Blockly.Msg.LISTS_SPLIT_LIST_FROM_TEXT, 'SPLIT'],
651
+         [Blockly.Msg.LISTS_SPLIT_TEXT_FROM_LIST, 'JOIN']],
652
+        function(newOp) {
653
+          if (newOp == 'SPLIT') {
654
+            thisBlock.outputConnection.setCheck('Array');
655
+            thisBlock.getInput('INPUT').setCheck('String');
656
+          } else {
657
+            thisBlock.outputConnection.setCheck('String');
658
+            thisBlock.getInput('INPUT').setCheck('Array');
659
+          }
660
+        });
661
+    this.setHelpUrl(Blockly.Msg.LISTS_SPLIT_HELPURL);
662
+    this.setColour(Blockly.Blocks.lists.HUE);
663
+    this.appendValueInput('INPUT')
664
+        .setCheck('String')
665
+        .appendField(dropdown, 'MODE');
666
+    this.appendValueInput('DELIM')
667
+        .setCheck('String')
668
+        .appendField(Blockly.Msg.LISTS_SPLIT_WITH_DELIMITER);
669
+    this.setInputsInline(true);
670
+    this.setOutput(true, 'Array');
671
+    this.setTooltip(function() {
672
+      var mode = thisBlock.getFieldValue('MODE');
673
+      if (mode == 'SPLIT') {
674
+        return Blockly.Msg.LISTS_SPLIT_TOOLTIP_SPLIT;
675
+      } else if (mode == 'JOIN') {
676
+        return Blockly.Msg.LISTS_SPLIT_TOOLTIP_JOIN;
677
+      }
678
+      throw 'Unknown mode: ' + mode;
679
+    });
680
+  }
681
+};

+ 464 - 0
src/blockly/blocks/logic.js

@@ -0,0 +1,464 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Logic blocks for Blockly.
23
+ * @author q.neutron@gmail.com (Quynh Neutron)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Blocks.logic');
28
+
29
+goog.require('Blockly.Blocks');
30
+
31
+
32
+/**
33
+ * Common HSV hue for all blocks in this category.
34
+ */
35
+Blockly.Blocks.logic.HUE = 210;
36
+
37
+Blockly.Blocks['controls_if'] = {
38
+  /**
39
+   * Block for if/elseif/else condition.
40
+   * @this Blockly.Block
41
+   */
42
+  init: function() {
43
+    this.setHelpUrl(Blockly.Msg.CONTROLS_IF_HELPURL);
44
+    this.setColour(Blockly.Blocks.logic.HUE);
45
+    this.appendValueInput('IF0')
46
+        .setCheck('Boolean')
47
+        .appendField(Blockly.Msg.CONTROLS_IF_MSG_IF);
48
+    this.appendStatementInput('DO0')
49
+        .appendField(Blockly.Msg.CONTROLS_IF_MSG_THEN);
50
+    this.setPreviousStatement(true);
51
+    this.setNextStatement(true);
52
+    this.setMutator(new Blockly.Mutator(['controls_if_elseif',
53
+                                         'controls_if_else']));
54
+    // Assign 'this' to a variable for use in the tooltip closure below.
55
+    var thisBlock = this;
56
+    this.setTooltip(function() {
57
+      if (!thisBlock.elseifCount_ && !thisBlock.elseCount_) {
58
+        return Blockly.Msg.CONTROLS_IF_TOOLTIP_1;
59
+      } else if (!thisBlock.elseifCount_ && thisBlock.elseCount_) {
60
+        return Blockly.Msg.CONTROLS_IF_TOOLTIP_2;
61
+      } else if (thisBlock.elseifCount_ && !thisBlock.elseCount_) {
62
+        return Blockly.Msg.CONTROLS_IF_TOOLTIP_3;
63
+      } else if (thisBlock.elseifCount_ && thisBlock.elseCount_) {
64
+        return Blockly.Msg.CONTROLS_IF_TOOLTIP_4;
65
+      }
66
+      return '';
67
+    });
68
+    this.elseifCount_ = 0;
69
+    this.elseCount_ = 0;
70
+  },
71
+  /**
72
+   * Create XML to represent the number of else-if and else inputs.
73
+   * @return {Element} XML storage element.
74
+   * @this Blockly.Block
75
+   */
76
+  mutationToDom: function() {
77
+    if (!this.elseifCount_ && !this.elseCount_) {
78
+      return null;
79
+    }
80
+    var container = document.createElement('mutation');
81
+    if (this.elseifCount_) {
82
+      container.setAttribute('elseif', this.elseifCount_);
83
+    }
84
+    if (this.elseCount_) {
85
+      container.setAttribute('else', 1);
86
+    }
87
+    return container;
88
+  },
89
+  /**
90
+   * Parse XML to restore the else-if and else inputs.
91
+   * @param {!Element} xmlElement XML storage element.
92
+   * @this Blockly.Block
93
+   */
94
+  domToMutation: function(xmlElement) {
95
+    this.elseifCount_ = parseInt(xmlElement.getAttribute('elseif'), 10) || 0;
96
+    this.elseCount_ = parseInt(xmlElement.getAttribute('else'), 10) || 0;
97
+    for (var i = 1; i <= this.elseifCount_; i++) {
98
+      this.appendValueInput('IF' + i)
99
+          .setCheck('Boolean')
100
+          .appendField(Blockly.Msg.CONTROLS_IF_MSG_ELSEIF);
101
+      this.appendStatementInput('DO' + i)
102
+          .appendField(Blockly.Msg.CONTROLS_IF_MSG_THEN);
103
+    }
104
+    if (this.elseCount_) {
105
+      this.appendStatementInput('ELSE')
106
+          .appendField(Blockly.Msg.CONTROLS_IF_MSG_ELSE);
107
+    }
108
+  },
109
+  /**
110
+   * Populate the mutator's dialog with this block's components.
111
+   * @param {!Blockly.Workspace} workspace Mutator's workspace.
112
+   * @return {!Blockly.Block} Root block in mutator.
113
+   * @this Blockly.Block
114
+   */
115
+  decompose: function(workspace) {
116
+    var containerBlock = Blockly.Block.obtain(workspace, 'controls_if_if');
117
+    containerBlock.initSvg();
118
+    var connection = containerBlock.getInput('STACK').connection;
119
+    for (var i = 1; i <= this.elseifCount_; i++) {
120
+      var elseifBlock = Blockly.Block.obtain(workspace, 'controls_if_elseif');
121
+      elseifBlock.initSvg();
122
+      connection.connect(elseifBlock.previousConnection);
123
+      connection = elseifBlock.nextConnection;
124
+    }
125
+    if (this.elseCount_) {
126
+      var elseBlock = Blockly.Block.obtain(workspace, 'controls_if_else');
127
+      elseBlock.initSvg();
128
+      connection.connect(elseBlock.previousConnection);
129
+    }
130
+    return containerBlock;
131
+  },
132
+  /**
133
+   * Reconfigure this block based on the mutator dialog's components.
134
+   * @param {!Blockly.Block} containerBlock Root block in mutator.
135
+   * @this Blockly.Block
136
+   */
137
+  compose: function(containerBlock) {
138
+    // Disconnect the else input blocks and remove the inputs.
139
+    if (this.elseCount_) {
140
+      this.removeInput('ELSE');
141
+    }
142
+    this.elseCount_ = 0;
143
+    // Disconnect all the elseif input blocks and remove the inputs.
144
+    for (var i = this.elseifCount_; i > 0; i--) {
145
+      this.removeInput('IF' + i);
146
+      this.removeInput('DO' + i);
147
+    }
148
+    this.elseifCount_ = 0;
149
+    // Rebuild the block's optional inputs.
150
+    var clauseBlock = containerBlock.getInputTargetBlock('STACK');
151
+    while (clauseBlock) {
152
+      switch (clauseBlock.type) {
153
+        case 'controls_if_elseif':
154
+          this.elseifCount_++;
155
+          var ifInput = this.appendValueInput('IF' + this.elseifCount_)
156
+              .setCheck('Boolean')
157
+              .appendField(Blockly.Msg.CONTROLS_IF_MSG_ELSEIF);
158
+          var doInput = this.appendStatementInput('DO' + this.elseifCount_);
159
+          doInput.appendField(Blockly.Msg.CONTROLS_IF_MSG_THEN);
160
+          // Reconnect any child blocks.
161
+          if (clauseBlock.valueConnection_) {
162
+            ifInput.connection.connect(clauseBlock.valueConnection_);
163
+          }
164
+          if (clauseBlock.statementConnection_) {
165
+            doInput.connection.connect(clauseBlock.statementConnection_);
166
+          }
167
+          break;
168
+        case 'controls_if_else':
169
+          this.elseCount_++;
170
+          var elseInput = this.appendStatementInput('ELSE');
171
+          elseInput.appendField(Blockly.Msg.CONTROLS_IF_MSG_ELSE);
172
+          // Reconnect any child blocks.
173
+          if (clauseBlock.statementConnection_) {
174
+            elseInput.connection.connect(clauseBlock.statementConnection_);
175
+          }
176
+          break;
177
+        default:
178
+          throw 'Unknown block type.';
179
+      }
180
+      clauseBlock = clauseBlock.nextConnection &&
181
+          clauseBlock.nextConnection.targetBlock();
182
+    }
183
+  },
184
+  /**
185
+   * Store pointers to any connected child blocks.
186
+   * @param {!Blockly.Block} containerBlock Root block in mutator.
187
+   * @this Blockly.Block
188
+   */
189
+  saveConnections: function(containerBlock) {
190
+    var clauseBlock = containerBlock.getInputTargetBlock('STACK');
191
+    var i = 1;
192
+    while (clauseBlock) {
193
+      switch (clauseBlock.type) {
194
+        case 'controls_if_elseif':
195
+          var inputIf = this.getInput('IF' + i);
196
+          var inputDo = this.getInput('DO' + i);
197
+          clauseBlock.valueConnection_ =
198
+              inputIf && inputIf.connection.targetConnection;
199
+          clauseBlock.statementConnection_ =
200
+              inputDo && inputDo.connection.targetConnection;
201
+          i++;
202
+          break;
203
+        case 'controls_if_else':
204
+          var inputDo = this.getInput('ELSE');
205
+          clauseBlock.statementConnection_ =
206
+              inputDo && inputDo.connection.targetConnection;
207
+          break;
208
+        default:
209
+          throw 'Unknown block type.';
210
+      }
211
+      clauseBlock = clauseBlock.nextConnection &&
212
+          clauseBlock.nextConnection.targetBlock();
213
+    }
214
+  }
215
+};
216
+
217
+Blockly.Blocks['controls_if_if'] = {
218
+  /**
219
+   * Mutator block for if container.
220
+   * @this Blockly.Block
221
+   */
222
+  init: function() {
223
+    this.setColour(Blockly.Blocks.logic.HUE);
224
+    this.appendDummyInput()
225
+        .appendField(Blockly.Msg.CONTROLS_IF_IF_TITLE_IF);
226
+    this.appendStatementInput('STACK');
227
+    this.setTooltip(Blockly.Msg.CONTROLS_IF_IF_TOOLTIP);
228
+    this.contextMenu = false;
229
+  }
230
+};
231
+
232
+Blockly.Blocks['controls_if_elseif'] = {
233
+  /**
234
+   * Mutator bolck for else-if condition.
235
+   * @this Blockly.Block
236
+   */
237
+  init: function() {
238
+    this.setColour(Blockly.Blocks.logic.HUE);
239
+    this.appendDummyInput()
240
+        .appendField(Blockly.Msg.CONTROLS_IF_ELSEIF_TITLE_ELSEIF);
241
+    this.setPreviousStatement(true);
242
+    this.setNextStatement(true);
243
+    this.setTooltip(Blockly.Msg.CONTROLS_IF_ELSEIF_TOOLTIP);
244
+    this.contextMenu = false;
245
+  }
246
+};
247
+
248
+Blockly.Blocks['controls_if_else'] = {
249
+  /**
250
+   * Mutator block for else condition.
251
+   * @this Blockly.Block
252
+   */
253
+  init: function() {
254
+    this.setColour(Blockly.Blocks.logic.HUE);
255
+    this.appendDummyInput()
256
+        .appendField(Blockly.Msg.CONTROLS_IF_ELSE_TITLE_ELSE);
257
+    this.setPreviousStatement(true);
258
+    this.setTooltip(Blockly.Msg.CONTROLS_IF_ELSE_TOOLTIP);
259
+    this.contextMenu = false;
260
+  }
261
+};
262
+
263
+Blockly.Blocks['logic_compare'] = {
264
+  /**
265
+   * Block for comparison operator.
266
+   * @this Blockly.Block
267
+   */
268
+  init: function() {
269
+    var OPERATORS = this.RTL ? [
270
+          ['=', 'EQ'],
271
+          ['\u2260', 'NEQ'],
272
+          ['>', 'LT'],
273
+          ['\u2265', 'LTE'],
274
+          ['<', 'GT'],
275
+          ['\u2264', 'GTE']
276
+        ] : [
277
+          ['=', 'EQ'],
278
+          ['\u2260', 'NEQ'],
279
+          ['<', 'LT'],
280
+          ['\u2264', 'LTE'],
281
+          ['>', 'GT'],
282
+          ['\u2265', 'GTE']
283
+        ];
284
+    this.setHelpUrl(Blockly.Msg.LOGIC_COMPARE_HELPURL);
285
+    this.setColour(Blockly.Blocks.logic.HUE);
286
+    this.setOutput(true, 'Boolean');
287
+    this.appendValueInput('A');
288
+    this.appendValueInput('B')
289
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'OP');
290
+    this.setInputsInline(true);
291
+    // Assign 'this' to a variable for use in the tooltip closure below.
292
+    var thisBlock = this;
293
+    this.setTooltip(function() {
294
+      var op = thisBlock.getFieldValue('OP');
295
+      var TOOLTIPS = {
296
+        'EQ': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_EQ,
297
+        'NEQ': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_NEQ,
298
+        'LT': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_LT,
299
+        'LTE': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_LTE,
300
+        'GT': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_GT,
301
+        'GTE': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_GTE
302
+      };
303
+      return TOOLTIPS[op];
304
+    });
305
+    this.prevBlocks_ = [null, null];
306
+  },
307
+  /**
308
+   * Called whenever anything on the workspace changes.
309
+   * Prevent mismatched types from being compared.
310
+   * @this Blockly.Block
311
+   */
312
+  onchange: function() {
313
+    var blockA = this.getInputTargetBlock('A');
314
+    var blockB = this.getInputTargetBlock('B');
315
+    // Disconnect blocks that existed prior to this change if they don't match.
316
+    if (blockA && blockB &&
317
+        !blockA.outputConnection.checkType_(blockB.outputConnection)) {
318
+      // Mismatch between two inputs.  Disconnect previous and bump it away.
319
+      for (var i = 0; i < this.prevBlocks_.length; i++) {
320
+        var block = this.prevBlocks_[i];
321
+        if (block === blockA || block === blockB) {
322
+          block.setParent(null);
323
+          block.bumpNeighbours_();
324
+        }
325
+      }
326
+    }
327
+    this.prevBlocks_[0] = blockA;
328
+    this.prevBlocks_[1] = blockB;
329
+  }
330
+};
331
+
332
+Blockly.Blocks['logic_operation'] = {
333
+  /**
334
+   * Block for logical operations: 'and', 'or'.
335
+   * @this Blockly.Block
336
+   */
337
+  init: function() {
338
+    var OPERATORS =
339
+        [[Blockly.Msg.LOGIC_OPERATION_AND, 'AND'],
340
+         [Blockly.Msg.LOGIC_OPERATION_OR, 'OR']];
341
+    this.setHelpUrl(Blockly.Msg.LOGIC_OPERATION_HELPURL);
342
+    this.setColour(Blockly.Blocks.logic.HUE);
343
+    this.setOutput(true, 'Boolean');
344
+    this.appendValueInput('A')
345
+        .setCheck('Boolean');
346
+    this.appendValueInput('B')
347
+        .setCheck('Boolean')
348
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'OP');
349
+    this.setInputsInline(true);
350
+    // Assign 'this' to a variable for use in the tooltip closure below.
351
+    var thisBlock = this;
352
+    this.setTooltip(function() {
353
+      var op = thisBlock.getFieldValue('OP');
354
+      var TOOLTIPS = {
355
+        'AND': Blockly.Msg.LOGIC_OPERATION_TOOLTIP_AND,
356
+        'OR': Blockly.Msg.LOGIC_OPERATION_TOOLTIP_OR
357
+      };
358
+      return TOOLTIPS[op];
359
+    });
360
+  }
361
+};
362
+
363
+Blockly.Blocks['logic_negate'] = {
364
+  /**
365
+   * Block for negation.
366
+   * @this Blockly.Block
367
+   */
368
+  init: function() {
369
+    this.jsonInit({
370
+      "message0": Blockly.Msg.LOGIC_NEGATE_TITLE,
371
+      "args0": [
372
+        {
373
+          "type": "input_value",
374
+          "name": "BOOL",
375
+          "check": "Boolean"
376
+        }
377
+      ],
378
+      "output": "Boolean",
379
+      "colour": Blockly.Blocks.logic.HUE,
380
+      "tooltip": Blockly.Msg.LOGIC_NEGATE_TOOLTIP,
381
+      "helpUrl": Blockly.Msg.LOGIC_NEGATE_HELPURL
382
+    });
383
+  }
384
+};
385
+
386
+Blockly.Blocks['logic_boolean'] = {
387
+  /**
388
+   * Block for boolean data type: true and false.
389
+   * @this Blockly.Block
390
+   */
391
+  init: function() {
392
+    var BOOLEANS =
393
+        [[Blockly.Msg.LOGIC_BOOLEAN_TRUE, 'TRUE'],
394
+         [Blockly.Msg.LOGIC_BOOLEAN_FALSE, 'FALSE']];
395
+    this.setHelpUrl(Blockly.Msg.LOGIC_BOOLEAN_HELPURL);
396
+    this.setColour(Blockly.Blocks.logic.HUE);
397
+    this.setOutput(true, 'Boolean');
398
+    this.appendDummyInput()
399
+        .appendField(new Blockly.FieldDropdown(BOOLEANS), 'BOOL');
400
+    this.setTooltip(Blockly.Msg.LOGIC_BOOLEAN_TOOLTIP);
401
+  }
402
+};
403
+
404
+Blockly.Blocks['logic_null'] = {
405
+  /**
406
+   * Block for null data type.
407
+   * @this Blockly.Block
408
+   */
409
+  init: function() {
410
+    this.setHelpUrl(Blockly.Msg.LOGIC_NULL_HELPURL);
411
+    this.setColour(Blockly.Blocks.logic.HUE);
412
+    this.setOutput(true);
413
+    this.appendDummyInput()
414
+        .appendField(Blockly.Msg.LOGIC_NULL);
415
+    this.setTooltip(Blockly.Msg.LOGIC_NULL_TOOLTIP);
416
+  }
417
+};
418
+
419
+Blockly.Blocks['logic_ternary'] = {
420
+  /**
421
+   * Block for ternary operator.
422
+   * @this Blockly.Block
423
+   */
424
+  init: function() {
425
+    this.setHelpUrl(Blockly.Msg.LOGIC_TERNARY_HELPURL);
426
+    this.setColour(Blockly.Blocks.logic.HUE);
427
+    this.appendValueInput('IF')
428
+        .setCheck('Boolean')
429
+        .appendField(Blockly.Msg.LOGIC_TERNARY_CONDITION);
430
+    this.appendValueInput('THEN')
431
+        .appendField(Blockly.Msg.LOGIC_TERNARY_IF_TRUE);
432
+    this.appendValueInput('ELSE')
433
+        .appendField(Blockly.Msg.LOGIC_TERNARY_IF_FALSE);
434
+    this.setOutput(true);
435
+    this.setTooltip(Blockly.Msg.LOGIC_TERNARY_TOOLTIP);
436
+    this.prevParentConnection_ = null;
437
+  },
438
+  /**
439
+   * Called whenever anything on the workspace changes.
440
+   * Prevent mismatched types.
441
+   * @this Blockly.Block
442
+   */
443
+  onchange: function() {
444
+    var blockA = this.getInputTargetBlock('THEN');
445
+    var blockB = this.getInputTargetBlock('ELSE');
446
+    var parentConnection = this.outputConnection.targetConnection;
447
+    // Disconnect blocks that existed prior to this change if they don't match.
448
+    if ((blockA || blockB) && parentConnection) {
449
+      for (var i = 0; i < 2; i++) {
450
+        var block = (i == 1) ? blockA : blockB;
451
+        if (block && !block.outputConnection.checkType_(parentConnection)) {
452
+          if (parentConnection === this.prevParentConnection_) {
453
+            this.setParent(null);
454
+            parentConnection.sourceBlock_.bumpNeighbours_();
455
+          } else {
456
+            block.setParent(null);
457
+            block.bumpNeighbours_();
458
+          }
459
+        }
460
+      }
461
+    }
462
+    this.prevParentConnection_ = parentConnection;
463
+  }
464
+};

+ 318 - 0
src/blockly/blocks/loops.js

@@ -0,0 +1,318 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Loop blocks for Blockly.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Blocks.loops');
28
+
29
+goog.require('Blockly.Blocks');
30
+
31
+
32
+/**
33
+ * Common HSV hue for all blocks in this category.
34
+ */
35
+Blockly.Blocks.loops.HUE = 120;
36
+
37
+Blockly.Blocks['controls_repeat'] = {
38
+  /**
39
+   * Block for repeat n times (internal number).
40
+   * @this Blockly.Block
41
+   */
42
+  init: function() {
43
+    this.jsonInit({
44
+      "message0": Blockly.Msg.CONTROLS_REPEAT_TITLE,
45
+      "args0": [
46
+        {
47
+          "type": "field_input",
48
+          "name": "TIMES",
49
+          "text": "10"
50
+        }
51
+      ],
52
+      "previousStatement": null,
53
+      "nextStatement": null,
54
+      "colour": Blockly.Blocks.loops.HUE,
55
+      "tooltip": Blockly.Msg.CONTROLS_REPEAT_TOOLTIP,
56
+      "helpUrl": Blockly.Msg.CONTROLS_REPEAT_HELPURL
57
+    });
58
+    this.appendStatementInput('DO')
59
+        .appendField(Blockly.Msg.CONTROLS_REPEAT_INPUT_DO);
60
+    this.getField('TIMES').setChangeHandler(
61
+        Blockly.FieldTextInput.nonnegativeIntegerValidator);
62
+  }
63
+};
64
+
65
+Blockly.Blocks['controls_repeat_ext'] = {
66
+  /**
67
+   * Block for repeat n times (external number).
68
+   * @this Blockly.Block
69
+   */
70
+  init: function() {
71
+    this.jsonInit({
72
+      "message0": Blockly.Msg.CONTROLS_REPEAT_TITLE,
73
+      "args0": [
74
+        {
75
+          "type": "input_value",
76
+          "name": "TIMES",
77
+          "check": "Number"
78
+        }
79
+      ],
80
+      "previousStatement": null,
81
+      "nextStatement": null,
82
+      "colour": Blockly.Blocks.loops.HUE,
83
+      "tooltip": Blockly.Msg.CONTROLS_REPEAT_TOOLTIP,
84
+      "helpUrl": Blockly.Msg.CONTROLS_REPEAT_HELPURL
85
+    });
86
+    this.appendStatementInput('DO')
87
+        .appendField(Blockly.Msg.CONTROLS_REPEAT_INPUT_DO);
88
+  }
89
+};
90
+
91
+Blockly.Blocks['controls_whileUntil'] = {
92
+  /**
93
+   * Block for 'do while/until' loop.
94
+   * @this Blockly.Block
95
+   */
96
+  init: function() {
97
+    var OPERATORS =
98
+        [[Blockly.Msg.CONTROLS_WHILEUNTIL_OPERATOR_WHILE, 'WHILE'],
99
+         [Blockly.Msg.CONTROLS_WHILEUNTIL_OPERATOR_UNTIL, 'UNTIL']];
100
+    this.setHelpUrl(Blockly.Msg.CONTROLS_WHILEUNTIL_HELPURL);
101
+    this.setColour(Blockly.Blocks.loops.HUE);
102
+    this.appendValueInput('BOOL')
103
+        .setCheck('Boolean')
104
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'MODE');
105
+    this.appendStatementInput('DO')
106
+        .appendField(Blockly.Msg.CONTROLS_WHILEUNTIL_INPUT_DO);
107
+    this.setPreviousStatement(true);
108
+    this.setNextStatement(true);
109
+    // Assign 'this' to a variable for use in the tooltip closure below.
110
+    var thisBlock = this;
111
+    this.setTooltip(function() {
112
+      var op = thisBlock.getFieldValue('MODE');
113
+      var TOOLTIPS = {
114
+        'WHILE': Blockly.Msg.CONTROLS_WHILEUNTIL_TOOLTIP_WHILE,
115
+        'UNTIL': Blockly.Msg.CONTROLS_WHILEUNTIL_TOOLTIP_UNTIL
116
+      };
117
+      return TOOLTIPS[op];
118
+    });
119
+  }
120
+};
121
+
122
+Blockly.Blocks['controls_for'] = {
123
+  /**
124
+   * Block for 'for' loop.
125
+   * @this Blockly.Block
126
+   */
127
+  init: function() {
128
+    this.jsonInit({
129
+      "message0": Blockly.Msg.CONTROLS_FOR_TITLE,
130
+      "args0": [
131
+        {
132
+          "type": "field_variable",
133
+          "name": "VAR",
134
+          "variable": null
135
+        },
136
+        {
137
+          "type": "input_value",
138
+          "name": "FROM",
139
+          "check": "Number",
140
+          "align": "RIGHT"
141
+        },
142
+        {
143
+          "type": "input_value",
144
+          "name": "TO",
145
+          "check": "Number",
146
+          "align": "RIGHT"
147
+        },
148
+        {
149
+          "type": "input_value",
150
+          "name": "BY",
151
+          "check": "Number",
152
+          "align": "RIGHT"
153
+        }
154
+      ],
155
+      "inputsInline": true,
156
+      "previousStatement": null,
157
+      "nextStatement": null,
158
+      "colour": Blockly.Blocks.loops.HUE,
159
+      "helpUrl": Blockly.Msg.CONTROLS_FOR_HELPURL
160
+    });
161
+    this.appendStatementInput('DO')
162
+        .appendField(Blockly.Msg.CONTROLS_FOR_INPUT_DO);
163
+    // Assign 'this' to a variable for use in the tooltip closure below.
164
+    var thisBlock = this;
165
+    this.setTooltip(function() {
166
+      return Blockly.Msg.CONTROLS_FOR_TOOLTIP.replace('%1',
167
+          thisBlock.getFieldValue('VAR'));
168
+    });
169
+  },
170
+  /**
171
+   * Return all variables referenced by this block.
172
+   * @return {!Array.<string>} List of variable names.
173
+   * @this Blockly.Block
174
+   */
175
+  getVars: function() {
176
+    return [this.getFieldValue('VAR')];
177
+  },
178
+  /**
179
+   * Notification that a variable is renaming.
180
+   * If the name matches one of this block's variables, rename it.
181
+   * @param {string} oldName Previous name of variable.
182
+   * @param {string} newName Renamed variable.
183
+   * @this Blockly.Block
184
+   */
185
+  renameVar: function(oldName, newName) {
186
+    if (Blockly.Names.equals(oldName, this.getFieldValue('VAR'))) {
187
+      this.setFieldValue(newName, 'VAR');
188
+    }
189
+  },
190
+  /**
191
+   * Add menu option to create getter block for loop variable.
192
+   * @param {!Array} options List of menu options to add to.
193
+   * @this Blockly.Block
194
+   */
195
+  customContextMenu: function(options) {
196
+    if (!this.isCollapsed()) {
197
+      var option = {enabled: true};
198
+      var name = this.getFieldValue('VAR');
199
+      option.text = Blockly.Msg.VARIABLES_SET_CREATE_GET.replace('%1', name);
200
+      var xmlField = goog.dom.createDom('field', null, name);
201
+      xmlField.setAttribute('name', 'VAR');
202
+      var xmlBlock = goog.dom.createDom('block', null, xmlField);
203
+      xmlBlock.setAttribute('type', 'variables_get');
204
+      option.callback = Blockly.ContextMenu.callbackFactory(this, xmlBlock);
205
+      options.push(option);
206
+    }
207
+  }
208
+};
209
+
210
+Blockly.Blocks['controls_forEach'] = {
211
+  /**
212
+   * Block for 'for each' loop.
213
+   * @this Blockly.Block
214
+   */
215
+  init: function() {
216
+    this.jsonInit({
217
+      "message0": Blockly.Msg.CONTROLS_FOREACH_TITLE,
218
+      "args0": [
219
+        {
220
+          "type": "field_variable",
221
+          "name": "VAR",
222
+          "variable": null
223
+        },
224
+        {
225
+          "type": "input_value",
226
+          "name": "LIST",
227
+          "check": "Array"
228
+        }
229
+      ],
230
+      "previousStatement": null,
231
+      "nextStatement": null,
232
+      "colour": Blockly.Blocks.loops.HUE,
233
+      "helpUrl": Blockly.Msg.CONTROLS_FOREACH_HELPURL
234
+    });
235
+    this.appendStatementInput('DO')
236
+        .appendField(Blockly.Msg.CONTROLS_FOREACH_INPUT_DO);
237
+    // Assign 'this' to a variable for use in the tooltip closure below.
238
+    var thisBlock = this;
239
+    this.setTooltip(function() {
240
+      return Blockly.Msg.CONTROLS_FOREACH_TOOLTIP.replace('%1',
241
+          thisBlock.getFieldValue('VAR'));
242
+    });
243
+  },
244
+  /**
245
+   * Return all variables referenced by this block.
246
+   * @return {!Array.<string>} List of variable names.
247
+   * @this Blockly.Block
248
+   */
249
+  getVars: function() {
250
+    return [this.getFieldValue('VAR')];
251
+  },
252
+  /**
253
+   * Notification that a variable is renaming.
254
+   * If the name matches one of this block's variables, rename it.
255
+   * @param {string} oldName Previous name of variable.
256
+   * @param {string} newName Renamed variable.
257
+   * @this Blockly.Block
258
+   */
259
+  renameVar: function(oldName, newName) {
260
+    if (Blockly.Names.equals(oldName, this.getFieldValue('VAR'))) {
261
+      this.setFieldValue(newName, 'VAR');
262
+    }
263
+  },
264
+  customContextMenu: Blockly.Blocks['controls_for'].customContextMenu
265
+};
266
+
267
+Blockly.Blocks['controls_flow_statements'] = {
268
+  /**
269
+   * Block for flow statements: continue, break.
270
+   * @this Blockly.Block
271
+   */
272
+  init: function() {
273
+    var OPERATORS =
274
+        [[Blockly.Msg.CONTROLS_FLOW_STATEMENTS_OPERATOR_BREAK, 'BREAK'],
275
+         [Blockly.Msg.CONTROLS_FLOW_STATEMENTS_OPERATOR_CONTINUE, 'CONTINUE']];
276
+    this.setHelpUrl(Blockly.Msg.CONTROLS_FLOW_STATEMENTS_HELPURL);
277
+    this.setColour(Blockly.Blocks.loops.HUE);
278
+    this.appendDummyInput()
279
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'FLOW');
280
+    this.setPreviousStatement(true);
281
+    // Assign 'this' to a variable for use in the tooltip closure below.
282
+    var thisBlock = this;
283
+    this.setTooltip(function() {
284
+      var op = thisBlock.getFieldValue('FLOW');
285
+      var TOOLTIPS = {
286
+        'BREAK': Blockly.Msg.CONTROLS_FLOW_STATEMENTS_TOOLTIP_BREAK,
287
+        'CONTINUE': Blockly.Msg.CONTROLS_FLOW_STATEMENTS_TOOLTIP_CONTINUE
288
+      };
289
+      return TOOLTIPS[op];
290
+    });
291
+  },
292
+  /**
293
+   * Called whenever anything on the workspace changes.
294
+   * Add warning if this flow block is not nested inside a loop.
295
+   * @this Blockly.Block
296
+   */
297
+  onchange: function() {
298
+    var legal = false;
299
+    // Is the block nested in a loop?
300
+    var block = this;
301
+    do {
302
+      if (block.type == 'controls_repeat' ||
303
+          block.type == 'controls_repeat_ext' ||
304
+          block.type == 'controls_forEach' ||
305
+          block.type == 'controls_for' ||
306
+          block.type == 'controls_whileUntil') {
307
+        legal = true;
308
+        break;
309
+      }
310
+      block = block.getSurroundParent();
311
+    } while (block);
312
+    if (legal) {
313
+      this.setWarningText(null);
314
+    } else {
315
+      this.setWarningText(Blockly.Msg.CONTROLS_FLOW_STATEMENTS_WARNING);
316
+    }
317
+  }
318
+};

+ 481 - 0
src/blockly/blocks/math.js

@@ -0,0 +1,481 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Math blocks for Blockly.
23
+ * @author q.neutron@gmail.com (Quynh Neutron)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Blocks.math');
28
+
29
+goog.require('Blockly.Blocks');
30
+
31
+
32
+/**
33
+ * Common HSV hue for all blocks in this category.
34
+ */
35
+Blockly.Blocks.math.HUE = 230;
36
+
37
+Blockly.Blocks['math_number'] = {
38
+  /**
39
+   * Block for numeric value.
40
+   * @this Blockly.Block
41
+   */
42
+  init: function() {
43
+    this.setHelpUrl(Blockly.Msg.MATH_NUMBER_HELPURL);
44
+    this.setColour(Blockly.Blocks.math.HUE);
45
+    this.appendDummyInput()
46
+        .appendField(new Blockly.FieldTextInput('0',
47
+        Blockly.FieldTextInput.numberValidator), 'NUM');
48
+    this.setOutput(true, 'Number');
49
+    this.setTooltip(Blockly.Msg.MATH_NUMBER_TOOLTIP);
50
+  }
51
+};
52
+
53
+Blockly.Blocks['math_arithmetic'] = {
54
+  /**
55
+   * Block for basic arithmetic operator.
56
+   * @this Blockly.Block
57
+   */
58
+  init: function() {
59
+    var OPERATORS =
60
+        [[Blockly.Msg.MATH_ADDITION_SYMBOL, 'ADD'],
61
+         [Blockly.Msg.MATH_SUBTRACTION_SYMBOL, 'MINUS'],
62
+         [Blockly.Msg.MATH_MULTIPLICATION_SYMBOL, 'MULTIPLY'],
63
+         [Blockly.Msg.MATH_DIVISION_SYMBOL, 'DIVIDE'],
64
+         [Blockly.Msg.MATH_POWER_SYMBOL, 'POWER']];
65
+    this.setHelpUrl(Blockly.Msg.MATH_ARITHMETIC_HELPURL);
66
+    this.setColour(Blockly.Blocks.math.HUE);
67
+    this.setOutput(true, 'Number');
68
+    this.appendValueInput('A')
69
+        .setCheck('Number');
70
+    this.appendValueInput('B')
71
+        .setCheck('Number')
72
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'OP');
73
+    this.setInputsInline(true);
74
+    // Assign 'this' to a variable for use in the tooltip closure below.
75
+    var thisBlock = this;
76
+    this.setTooltip(function() {
77
+      var mode = thisBlock.getFieldValue('OP');
78
+      var TOOLTIPS = {
79
+        'ADD': Blockly.Msg.MATH_ARITHMETIC_TOOLTIP_ADD,
80
+        'MINUS': Blockly.Msg.MATH_ARITHMETIC_TOOLTIP_MINUS,
81
+        'MULTIPLY': Blockly.Msg.MATH_ARITHMETIC_TOOLTIP_MULTIPLY,
82
+        'DIVIDE': Blockly.Msg.MATH_ARITHMETIC_TOOLTIP_DIVIDE,
83
+        'POWER': Blockly.Msg.MATH_ARITHMETIC_TOOLTIP_POWER
84
+      };
85
+      return TOOLTIPS[mode];
86
+    });
87
+  }
88
+};
89
+
90
+Blockly.Blocks['math_single'] = {
91
+  /**
92
+   * Block for advanced math operators with single operand.
93
+   * @this Blockly.Block
94
+   */
95
+  init: function() {
96
+    var OPERATORS =
97
+        [[Blockly.Msg.MATH_SINGLE_OP_ROOT, 'ROOT'],
98
+         [Blockly.Msg.MATH_SINGLE_OP_ABSOLUTE, 'ABS'],
99
+         ['-', 'NEG'],
100
+         ['ln', 'LN'],
101
+         ['log10', 'LOG10'],
102
+         ['e^', 'EXP'],
103
+         ['10^', 'POW10']];
104
+    this.setHelpUrl(Blockly.Msg.MATH_SINGLE_HELPURL);
105
+    this.setColour(Blockly.Blocks.math.HUE);
106
+    this.setOutput(true, 'Number');
107
+    this.appendValueInput('NUM')
108
+        .setCheck('Number')
109
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'OP');
110
+    // Assign 'this' to a variable for use in the tooltip closure below.
111
+    var thisBlock = this;
112
+    this.setTooltip(function() {
113
+      var mode = thisBlock.getFieldValue('OP');
114
+      var TOOLTIPS = {
115
+        'ROOT': Blockly.Msg.MATH_SINGLE_TOOLTIP_ROOT,
116
+        'ABS': Blockly.Msg.MATH_SINGLE_TOOLTIP_ABS,
117
+        'NEG': Blockly.Msg.MATH_SINGLE_TOOLTIP_NEG,
118
+        'LN': Blockly.Msg.MATH_SINGLE_TOOLTIP_LN,
119
+        'LOG10': Blockly.Msg.MATH_SINGLE_TOOLTIP_LOG10,
120
+        'EXP': Blockly.Msg.MATH_SINGLE_TOOLTIP_EXP,
121
+        'POW10': Blockly.Msg.MATH_SINGLE_TOOLTIP_POW10
122
+      };
123
+      return TOOLTIPS[mode];
124
+    });
125
+  }
126
+};
127
+
128
+Blockly.Blocks['math_trig'] = {
129
+  /**
130
+   * Block for trigonometry operators.
131
+   * @this Blockly.Block
132
+   */
133
+  init: function() {
134
+    var OPERATORS =
135
+        [[Blockly.Msg.MATH_TRIG_SIN, 'SIN'],
136
+         [Blockly.Msg.MATH_TRIG_COS, 'COS'],
137
+         [Blockly.Msg.MATH_TRIG_TAN, 'TAN'],
138
+         [Blockly.Msg.MATH_TRIG_ASIN, 'ASIN'],
139
+         [Blockly.Msg.MATH_TRIG_ACOS, 'ACOS'],
140
+         [Blockly.Msg.MATH_TRIG_ATAN, 'ATAN']];
141
+    this.setHelpUrl(Blockly.Msg.MATH_TRIG_HELPURL);
142
+    this.setColour(Blockly.Blocks.math.HUE);
143
+    this.setOutput(true, 'Number');
144
+    this.appendValueInput('NUM')
145
+        .setCheck('Number')
146
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'OP');
147
+    // Assign 'this' to a variable for use in the tooltip closure below.
148
+    var thisBlock = this;
149
+    this.setTooltip(function() {
150
+      var mode = thisBlock.getFieldValue('OP');
151
+      var TOOLTIPS = {
152
+        'SIN': Blockly.Msg.MATH_TRIG_TOOLTIP_SIN,
153
+        'COS': Blockly.Msg.MATH_TRIG_TOOLTIP_COS,
154
+        'TAN': Blockly.Msg.MATH_TRIG_TOOLTIP_TAN,
155
+        'ASIN': Blockly.Msg.MATH_TRIG_TOOLTIP_ASIN,
156
+        'ACOS': Blockly.Msg.MATH_TRIG_TOOLTIP_ACOS,
157
+        'ATAN': Blockly.Msg.MATH_TRIG_TOOLTIP_ATAN
158
+      };
159
+      return TOOLTIPS[mode];
160
+    });
161
+  }
162
+};
163
+
164
+Blockly.Blocks['math_constant'] = {
165
+  /**
166
+   * Block for constants: PI, E, the Golden Ratio, sqrt(2), 1/sqrt(2), INFINITY.
167
+   * @this Blockly.Block
168
+   */
169
+  init: function() {
170
+    var CONSTANTS =
171
+        [['\u03c0', 'PI'],
172
+         ['e', 'E'],
173
+         ['\u03c6', 'GOLDEN_RATIO'],
174
+         ['sqrt(2)', 'SQRT2'],
175
+         ['sqrt(\u00bd)', 'SQRT1_2'],
176
+         ['\u221e', 'INFINITY']];
177
+    this.setHelpUrl(Blockly.Msg.MATH_CONSTANT_HELPURL);
178
+    this.setColour(Blockly.Blocks.math.HUE);
179
+    this.setOutput(true, 'Number');
180
+    this.appendDummyInput()
181
+        .appendField(new Blockly.FieldDropdown(CONSTANTS), 'CONSTANT');
182
+    this.setTooltip(Blockly.Msg.MATH_CONSTANT_TOOLTIP);
183
+  }
184
+};
185
+
186
+Blockly.Blocks['math_number_property'] = {
187
+  /**
188
+   * Block for checking if a number is even, odd, prime, whole, positive,
189
+   * negative or if it is divisible by certain number.
190
+   * @this Blockly.Block
191
+   */
192
+  init: function() {
193
+    var PROPERTIES =
194
+        [[Blockly.Msg.MATH_IS_EVEN, 'EVEN'],
195
+         [Blockly.Msg.MATH_IS_ODD, 'ODD'],
196
+         [Blockly.Msg.MATH_IS_PRIME, 'PRIME'],
197
+         [Blockly.Msg.MATH_IS_WHOLE, 'WHOLE'],
198
+         [Blockly.Msg.MATH_IS_POSITIVE, 'POSITIVE'],
199
+         [Blockly.Msg.MATH_IS_NEGATIVE, 'NEGATIVE'],
200
+         [Blockly.Msg.MATH_IS_DIVISIBLE_BY, 'DIVISIBLE_BY']];
201
+    this.setColour(Blockly.Blocks.math.HUE);
202
+    this.appendValueInput('NUMBER_TO_CHECK')
203
+        .setCheck('Number');
204
+    var dropdown = new Blockly.FieldDropdown(PROPERTIES, function(option) {
205
+      var divisorInput = (option == 'DIVISIBLE_BY');
206
+      this.sourceBlock_.updateShape_(divisorInput);
207
+    });
208
+    this.appendDummyInput()
209
+        .appendField(dropdown, 'PROPERTY');
210
+    this.setInputsInline(true);
211
+    this.setOutput(true, 'Boolean');
212
+    this.setTooltip(Blockly.Msg.MATH_IS_TOOLTIP);
213
+  },
214
+  /**
215
+   * Create XML to represent whether the 'divisorInput' should be present.
216
+   * @return {Element} XML storage element.
217
+   * @this Blockly.Block
218
+   */
219
+  mutationToDom: function() {
220
+    var container = document.createElement('mutation');
221
+    var divisorInput = (this.getFieldValue('PROPERTY') == 'DIVISIBLE_BY');
222
+    container.setAttribute('divisor_input', divisorInput);
223
+    return container;
224
+  },
225
+  /**
226
+   * Parse XML to restore the 'divisorInput'.
227
+   * @param {!Element} xmlElement XML storage element.
228
+   * @this Blockly.Block
229
+   */
230
+  domToMutation: function(xmlElement) {
231
+    var divisorInput = (xmlElement.getAttribute('divisor_input') == 'true');
232
+    this.updateShape_(divisorInput);
233
+  },
234
+  /**
235
+   * Modify this block to have (or not have) an input for 'is divisible by'.
236
+   * @param {boolean} divisorInput True if this block has a divisor input.
237
+   * @private
238
+   * @this Blockly.Block
239
+   */
240
+  updateShape_: function(divisorInput) {
241
+    // Add or remove a Value Input.
242
+    var inputExists = this.getInput('DIVISOR');
243
+    if (divisorInput) {
244
+      if (!inputExists) {
245
+        this.appendValueInput('DIVISOR')
246
+            .setCheck('Number');
247
+      }
248
+    } else if (inputExists) {
249
+      this.removeInput('DIVISOR');
250
+    }
251
+  }
252
+};
253
+
254
+Blockly.Blocks['math_change'] = {
255
+  /**
256
+   * Block for adding to a variable in place.
257
+   * @this Blockly.Block
258
+   */
259
+  init: function() {
260
+    this.jsonInit({
261
+      "message0": Blockly.Msg.MATH_CHANGE_TITLE,
262
+      "args0": [
263
+        {
264
+          "type": "field_variable",
265
+          "name": "VAR",
266
+          "variable": Blockly.Msg.MATH_CHANGE_TITLE_ITEM
267
+        },
268
+        {
269
+          "type": "input_value",
270
+          "name": "DELTA",
271
+          "check": "Number"
272
+        }
273
+      ],
274
+      "previousStatement": null,
275
+      "nextStatement": null,
276
+      "colour": Blockly.Blocks.math.HUE,
277
+      "helpUrl": Blockly.Msg.MATH_CHANGE_HELPURL
278
+    });
279
+    // Assign 'this' to a variable for use in the tooltip closure below.
280
+    var thisBlock = this;
281
+    this.setTooltip(function() {
282
+      return Blockly.Msg.MATH_CHANGE_TOOLTIP.replace('%1',
283
+          thisBlock.getFieldValue('VAR'));
284
+    });
285
+  },
286
+  /**
287
+   * Return all variables referenced by this block.
288
+   * @return {!Array.<string>} List of variable names.
289
+   * @this Blockly.Block
290
+   */
291
+  getVars: function() {
292
+    return [this.getFieldValue('VAR')];
293
+  },
294
+  /**
295
+   * Notification that a variable is renaming.
296
+   * If the name matches one of this block's variables, rename it.
297
+   * @param {string} oldName Previous name of variable.
298
+   * @param {string} newName Renamed variable.
299
+   * @this Blockly.Block
300
+   */
301
+  renameVar: function(oldName, newName) {
302
+    if (Blockly.Names.equals(oldName, this.getFieldValue('VAR'))) {
303
+      this.setFieldValue(newName, 'VAR');
304
+    }
305
+  }
306
+};
307
+
308
+Blockly.Blocks['math_round'] = {
309
+  /**
310
+   * Block for rounding functions.
311
+   * @this Blockly.Block
312
+   */
313
+  init: function() {
314
+    var OPERATORS =
315
+        [[Blockly.Msg.MATH_ROUND_OPERATOR_ROUND, 'ROUND'],
316
+         [Blockly.Msg.MATH_ROUND_OPERATOR_ROUNDUP, 'ROUNDUP'],
317
+         [Blockly.Msg.MATH_ROUND_OPERATOR_ROUNDDOWN, 'ROUNDDOWN']];
318
+    this.setHelpUrl(Blockly.Msg.MATH_ROUND_HELPURL);
319
+    this.setColour(Blockly.Blocks.math.HUE);
320
+    this.setOutput(true, 'Number');
321
+    this.appendValueInput('NUM')
322
+        .setCheck('Number')
323
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'OP');
324
+    this.setTooltip(Blockly.Msg.MATH_ROUND_TOOLTIP);
325
+  }
326
+};
327
+
328
+Blockly.Blocks['math_on_list'] = {
329
+  /**
330
+   * Block for evaluating a list of numbers to return sum, average, min, max,
331
+   * etc.  Some functions also work on text (min, max, mode, median).
332
+   * @this Blockly.Block
333
+   */
334
+  init: function() {
335
+    var OPERATORS =
336
+        [[Blockly.Msg.MATH_ONLIST_OPERATOR_SUM, 'SUM'],
337
+         [Blockly.Msg.MATH_ONLIST_OPERATOR_MIN, 'MIN'],
338
+         [Blockly.Msg.MATH_ONLIST_OPERATOR_MAX, 'MAX'],
339
+         [Blockly.Msg.MATH_ONLIST_OPERATOR_AVERAGE, 'AVERAGE'],
340
+         [Blockly.Msg.MATH_ONLIST_OPERATOR_MEDIAN, 'MEDIAN'],
341
+         [Blockly.Msg.MATH_ONLIST_OPERATOR_MODE, 'MODE'],
342
+         [Blockly.Msg.MATH_ONLIST_OPERATOR_STD_DEV, 'STD_DEV'],
343
+         [Blockly.Msg.MATH_ONLIST_OPERATOR_RANDOM, 'RANDOM']];
344
+    // Assign 'this' to a variable for use in the closures below.
345
+    var thisBlock = this;
346
+    this.setHelpUrl(Blockly.Msg.MATH_ONLIST_HELPURL);
347
+    this.setColour(Blockly.Blocks.math.HUE);
348
+    this.setOutput(true, 'Number');
349
+    var dropdown = new Blockly.FieldDropdown(OPERATORS, function(newOp) {
350
+      if (newOp == 'MODE') {
351
+        thisBlock.outputConnection.setCheck('Array');
352
+      } else {
353
+        thisBlock.outputConnection.setCheck('Number');
354
+      }
355
+    });
356
+    this.appendValueInput('LIST')
357
+        .setCheck('Array')
358
+        .appendField(dropdown, 'OP');
359
+    this.setTooltip(function() {
360
+      var mode = thisBlock.getFieldValue('OP');
361
+      var TOOLTIPS = {
362
+        'SUM': Blockly.Msg.MATH_ONLIST_TOOLTIP_SUM,
363
+        'MIN': Blockly.Msg.MATH_ONLIST_TOOLTIP_MIN,
364
+        'MAX': Blockly.Msg.MATH_ONLIST_TOOLTIP_MAX,
365
+        'AVERAGE': Blockly.Msg.MATH_ONLIST_TOOLTIP_AVERAGE,
366
+        'MEDIAN': Blockly.Msg.MATH_ONLIST_TOOLTIP_MEDIAN,
367
+        'MODE': Blockly.Msg.MATH_ONLIST_TOOLTIP_MODE,
368
+        'STD_DEV': Blockly.Msg.MATH_ONLIST_TOOLTIP_STD_DEV,
369
+        'RANDOM': Blockly.Msg.MATH_ONLIST_TOOLTIP_RANDOM
370
+      };
371
+      return TOOLTIPS[mode];
372
+    });
373
+  }
374
+};
375
+
376
+Blockly.Blocks['math_modulo'] = {
377
+  /**
378
+   * Block for remainder of a division.
379
+   * @this Blockly.Block
380
+   */
381
+  init: function() {
382
+    this.jsonInit({
383
+      "message0": Blockly.Msg.MATH_MODULO_TITLE,
384
+      "args0": [
385
+        {
386
+          "type": "input_value",
387
+          "name": "DIVIDEND",
388
+          "check": "Number"
389
+        },
390
+        {
391
+          "type": "input_value",
392
+          "name": "DIVISOR",
393
+          "check": "Number"
394
+        }
395
+      ],
396
+      "inputsInline": true,
397
+      "output": "Number",
398
+      "colour": Blockly.Blocks.math.HUE,
399
+      "tooltip": Blockly.Msg.MATH_MODULO_TOOLTIP,
400
+      "helpUrl": Blockly.Msg.MATH_MODULO_HELPURL
401
+    });
402
+  }
403
+};
404
+
405
+Blockly.Blocks['math_constrain'] = {
406
+  /**
407
+   * Block for constraining a number between two limits.
408
+   * @this Blockly.Block
409
+   */
410
+  init: function() {
411
+    this.jsonInit({
412
+      "message0": Blockly.Msg.MATH_CONSTRAIN_TITLE,
413
+      "args0": [
414
+        {
415
+          "type": "input_value",
416
+          "name": "VALUE",
417
+          "check": "Number"
418
+        },
419
+        {
420
+          "type": "input_value",
421
+          "name": "LOW",
422
+          "check": "Number"
423
+        },
424
+        {
425
+          "type": "input_value",
426
+          "name": "HIGH",
427
+          "check": "Number"
428
+        }
429
+      ],
430
+      "inputsInline": true,
431
+      "output": "Number",
432
+      "colour": Blockly.Blocks.math.HUE,
433
+      "tooltip": Blockly.Msg.MATH_CONSTRAIN_TOOLTIP,
434
+      "helpUrl": Blockly.Msg.MATH_CONSTRAIN_HELPURL
435
+    });
436
+  }
437
+};
438
+
439
+Blockly.Blocks['math_random_int'] = {
440
+  /**
441
+   * Block for random integer between [X] and [Y].
442
+   * @this Blockly.Block
443
+   */
444
+  init: function() {
445
+    this.jsonInit({
446
+      "message0": Blockly.Msg.MATH_RANDOM_INT_TITLE,
447
+      "args0": [
448
+        {
449
+          "type": "input_value",
450
+          "name": "FROM",
451
+          "check": "Number"
452
+        },
453
+        {
454
+          "type": "input_value",
455
+          "name": "TO",
456
+          "check": "Number"
457
+        }
458
+      ],
459
+      "inputsInline": true,
460
+      "output": "Number",
461
+      "colour": Blockly.Blocks.math.HUE,
462
+      "tooltip": Blockly.Msg.MATH_RANDOM_INT_TOOLTIP,
463
+      "helpUrl": Blockly.Msg.MATH_RANDOM_INT_HELPURL
464
+    });
465
+  }
466
+};
467
+
468
+Blockly.Blocks['math_random_float'] = {
469
+  /**
470
+   * Block for random fraction between 0 and 1.
471
+   * @this Blockly.Block
472
+   */
473
+  init: function() {
474
+    this.setHelpUrl(Blockly.Msg.MATH_RANDOM_FLOAT_HELPURL);
475
+    this.setColour(Blockly.Blocks.math.HUE);
476
+    this.setOutput(true, 'Number');
477
+    this.appendDummyInput()
478
+        .appendField(Blockly.Msg.MATH_RANDOM_FLOAT_TITLE_RANDOM);
479
+    this.setTooltip(Blockly.Msg.MATH_RANDOM_FLOAT_TOOLTIP);
480
+  }
481
+};

+ 767 - 0
src/blockly/blocks/procedures.js

@@ -0,0 +1,767 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Procedure blocks for Blockly.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Blocks.procedures');
28
+
29
+goog.require('Blockly.Blocks');
30
+
31
+
32
+/**
33
+ * Common HSV hue for all blocks in this category.
34
+ */
35
+Blockly.Blocks.procedures.HUE = 290;
36
+
37
+Blockly.Blocks['procedures_defnoreturn'] = {
38
+  /**
39
+   * Block for defining a procedure with no return value.
40
+   * @this Blockly.Block
41
+   */
42
+  init: function() {
43
+    this.setHelpUrl(Blockly.Msg.PROCEDURES_DEFNORETURN_HELPURL);
44
+    this.setColour(Blockly.Blocks.procedures.HUE);
45
+    var name = Blockly.Procedures.findLegalName(
46
+        Blockly.Msg.PROCEDURES_DEFNORETURN_PROCEDURE, this);
47
+    var nameField = new Blockly.FieldTextInput(name,
48
+        Blockly.Procedures.rename);
49
+    nameField.setSpellcheck(false);
50
+    this.appendDummyInput()
51
+        .appendField(Blockly.Msg.PROCEDURES_DEFNORETURN_TITLE)
52
+        .appendField(nameField, 'NAME')
53
+        .appendField('', 'PARAMS');
54
+    this.setMutator(new Blockly.Mutator(['procedures_mutatorarg']));
55
+    this.setTooltip(Blockly.Msg.PROCEDURES_DEFNORETURN_TOOLTIP);
56
+    this.arguments_ = [];
57
+    this.setStatements_(true);
58
+    this.statementConnection_ = null;
59
+  },
60
+  /**
61
+   * Add or remove the statement block from this function definition.
62
+   * @param {boolean} hasStatements True if a statement block is needed.
63
+   * @this Blockly.Block
64
+   */
65
+  setStatements_: function(hasStatements) {
66
+    if (this.hasStatements_ === hasStatements) {
67
+      return;
68
+    }
69
+    if (hasStatements) {
70
+      this.appendStatementInput('STACK')
71
+          .appendField(Blockly.Msg.PROCEDURES_DEFNORETURN_DO);
72
+      if (this.getInput('RETURN')) {
73
+        this.moveInputBefore('STACK', 'RETURN');
74
+      }
75
+    } else {
76
+      this.removeInput('STACK', true);
77
+    }
78
+    this.hasStatements_ = hasStatements;
79
+  },
80
+  /**
81
+   * Update the display of parameters for this procedure definition block.
82
+   * Display a warning if there are duplicately named parameters.
83
+   * @private
84
+   * @this Blockly.Block
85
+   */
86
+  updateParams_: function() {
87
+    // Check for duplicated arguments.
88
+    var badArg = false;
89
+    var hash = {};
90
+    for (var i = 0; i < this.arguments_.length; i++) {
91
+      if (hash['arg_' + this.arguments_[i].toLowerCase()]) {
92
+        badArg = true;
93
+        break;
94
+      }
95
+      hash['arg_' + this.arguments_[i].toLowerCase()] = true;
96
+    }
97
+    if (badArg) {
98
+      this.setWarningText(Blockly.Msg.PROCEDURES_DEF_DUPLICATE_WARNING);
99
+    } else {
100
+      this.setWarningText(null);
101
+    }
102
+    // Merge the arguments into a human-readable list.
103
+    var paramString = '';
104
+    if (this.arguments_.length) {
105
+      paramString = Blockly.Msg.PROCEDURES_BEFORE_PARAMS +
106
+          ' ' + this.arguments_.join(', ');
107
+    }
108
+    this.setFieldValue(paramString, 'PARAMS');
109
+  },
110
+  /**
111
+   * Create XML to represent the argument inputs.
112
+   * @return {Element} XML storage element.
113
+   * @this Blockly.Block
114
+   */
115
+  mutationToDom: function() {
116
+    var container = document.createElement('mutation');
117
+    for (var i = 0; i < this.arguments_.length; i++) {
118
+      var parameter = document.createElement('arg');
119
+      parameter.setAttribute('name', this.arguments_[i]);
120
+      container.appendChild(parameter);
121
+    }
122
+
123
+    // Save whether the statement input is visible.
124
+    if (!this.hasStatements_) {
125
+      container.setAttribute('statements', 'false');
126
+    }
127
+    return container;
128
+  },
129
+  /**
130
+   * Parse XML to restore the argument inputs.
131
+   * @param {!Element} xmlElement XML storage element.
132
+   * @this Blockly.Block
133
+   */
134
+  domToMutation: function(xmlElement) {
135
+    this.arguments_ = [];
136
+    for (var i = 0, childNode; childNode = xmlElement.childNodes[i]; i++) {
137
+      if (childNode.nodeName.toLowerCase() == 'arg') {
138
+        this.arguments_.push(childNode.getAttribute('name'));
139
+      }
140
+    }
141
+    this.updateParams_();
142
+
143
+    // Show or hide the statement input.
144
+    this.setStatements_(xmlElement.getAttribute('statements') !== 'false');
145
+  },
146
+  /**
147
+   * Populate the mutator's dialog with this block's components.
148
+   * @param {!Blockly.Workspace} workspace Mutator's workspace.
149
+   * @return {!Blockly.Block} Root block in mutator.
150
+   * @this Blockly.Block
151
+   */
152
+  decompose: function(workspace) {
153
+    var containerBlock = Blockly.Block.obtain(workspace,
154
+                                              'procedures_mutatorcontainer');
155
+    containerBlock.initSvg();
156
+
157
+    // Check/uncheck the allow statement box.
158
+    if (this.getInput('RETURN')) {
159
+      containerBlock.setFieldValue(this.hasStatements_ ? 'TRUE' : 'FALSE',
160
+                                   'STATEMENTS');
161
+    } else {
162
+      containerBlock.getInput('STATEMENT_INPUT').setVisible(false);
163
+    }
164
+
165
+    // Parameter list.
166
+    var connection = containerBlock.getInput('STACK').connection;
167
+    for (var i = 0; i < this.arguments_.length; i++) {
168
+      var paramBlock = Blockly.Block.obtain(workspace, 'procedures_mutatorarg');
169
+      paramBlock.initSvg();
170
+      paramBlock.setFieldValue(this.arguments_[i], 'NAME');
171
+      // Store the old location.
172
+      paramBlock.oldLocation = i;
173
+      connection.connect(paramBlock.previousConnection);
174
+      connection = paramBlock.nextConnection;
175
+    }
176
+    // Initialize procedure's callers with blank IDs.
177
+    Blockly.Procedures.mutateCallers(this.getFieldValue('NAME'),
178
+                                     this.workspace, this.arguments_, null);
179
+    return containerBlock;
180
+  },
181
+  /**
182
+   * Reconfigure this block based on the mutator dialog's components.
183
+   * @param {!Blockly.Block} containerBlock Root block in mutator.
184
+   * @this Blockly.Block
185
+   */
186
+  compose: function(containerBlock) {
187
+    // Parameter list.
188
+    this.arguments_ = [];
189
+    this.paramIds_ = [];
190
+    var paramBlock = containerBlock.getInputTargetBlock('STACK');
191
+    while (paramBlock) {
192
+      this.arguments_.push(paramBlock.getFieldValue('NAME'));
193
+      this.paramIds_.push(paramBlock.id);
194
+      paramBlock = paramBlock.nextConnection &&
195
+          paramBlock.nextConnection.targetBlock();
196
+    }
197
+    this.updateParams_();
198
+    Blockly.Procedures.mutateCallers(this.getFieldValue('NAME'),
199
+        this.workspace, this.arguments_, this.paramIds_);
200
+
201
+    // Show/hide the statement input.
202
+    var hasStatements = containerBlock.getFieldValue('STATEMENTS');
203
+    if (hasStatements !== null) {
204
+      hasStatements = hasStatements == 'TRUE';
205
+      if (this.hasStatements_ != hasStatements) {
206
+        if (hasStatements) {
207
+          this.setStatements_(true);
208
+          // Restore the stack, if one was saved.
209
+          var stackConnection = this.getInput('STACK').connection;
210
+          if (stackConnection.targetConnection ||
211
+              !this.statementConnection_ ||
212
+              this.statementConnection_.targetConnection ||
213
+              this.statementConnection_.sourceBlock_.workspace !=
214
+              this.workspace) {
215
+            // Block no longer exists or has been attached elsewhere.
216
+            this.statementConnection_ = null;
217
+          } else {
218
+            stackConnection.connect(this.statementConnection_);
219
+          }
220
+        } else {
221
+          // Save the stack, then disconnect it.
222
+          var stackConnection = this.getInput('STACK').connection;
223
+          this.statementConnection_ = stackConnection.targetConnection;
224
+          if (this.statementConnection_) {
225
+            var stackBlock = stackConnection.targetBlock();
226
+            stackBlock.setParent(null);
227
+            stackBlock.bumpNeighbours_();
228
+          }
229
+          this.setStatements_(false);
230
+        }
231
+      }
232
+    }
233
+  },
234
+  /**
235
+   * Dispose of any callers.
236
+   * @this Blockly.Block
237
+   */
238
+  dispose: function() {
239
+    var name = this.getFieldValue('NAME');
240
+    Blockly.Procedures.disposeCallers(name, this.workspace);
241
+    // Call parent's destructor.
242
+    this.constructor.prototype.dispose.apply(this, arguments);
243
+  },
244
+  /**
245
+   * Return the signature of this procedure definition.
246
+   * @return {!Array} Tuple containing three elements:
247
+   *     - the name of the defined procedure,
248
+   *     - a list of all its arguments,
249
+   *     - that it DOES NOT have a return value.
250
+   * @this Blockly.Block
251
+   */
252
+  getProcedureDef: function() {
253
+    return [this.getFieldValue('NAME'), this.arguments_, false];
254
+  },
255
+  /**
256
+   * Return all variables referenced by this block.
257
+   * @return {!Array.<string>} List of variable names.
258
+   * @this Blockly.Block
259
+   */
260
+  getVars: function() {
261
+    return this.arguments_;
262
+  },
263
+  /**
264
+   * Notification that a variable is renaming.
265
+   * If the name matches one of this block's variables, rename it.
266
+   * @param {string} oldName Previous name of variable.
267
+   * @param {string} newName Renamed variable.
268
+   * @this Blockly.Block
269
+   */
270
+  renameVar: function(oldName, newName) {
271
+    var change = false;
272
+    for (var i = 0; i < this.arguments_.length; i++) {
273
+      if (Blockly.Names.equals(oldName, this.arguments_[i])) {
274
+        this.arguments_[i] = newName;
275
+        change = true;
276
+      }
277
+    }
278
+    if (change) {
279
+      this.updateParams_();
280
+      // Update the mutator's variables if the mutator is open.
281
+      if (this.mutator.isVisible()) {
282
+        var blocks = this.mutator.workspace_.getAllBlocks();
283
+        for (var i = 0, block; block = blocks[i]; i++) {
284
+          if (block.type == 'procedures_mutatorarg' &&
285
+              Blockly.Names.equals(oldName, block.getFieldValue('NAME'))) {
286
+            block.setFieldValue(newName, 'NAME');
287
+          }
288
+        }
289
+      }
290
+    }
291
+  },
292
+  /**
293
+   * Add custom menu options to this block's context menu.
294
+   * @param {!Array} options List of menu options to add to.
295
+   * @this Blockly.Block
296
+   */
297
+  customContextMenu: function(options) {
298
+    // Add option to create caller.
299
+    var option = {enabled: true};
300
+    var name = this.getFieldValue('NAME');
301
+    option.text = Blockly.Msg.PROCEDURES_CREATE_DO.replace('%1', name);
302
+    var xmlMutation = goog.dom.createDom('mutation');
303
+    xmlMutation.setAttribute('name', name);
304
+    for (var i = 0; i < this.arguments_.length; i++) {
305
+      var xmlArg = goog.dom.createDom('arg');
306
+      xmlArg.setAttribute('name', this.arguments_[i]);
307
+      xmlMutation.appendChild(xmlArg);
308
+    }
309
+    var xmlBlock = goog.dom.createDom('block', null, xmlMutation);
310
+    xmlBlock.setAttribute('type', this.callType_);
311
+    option.callback = Blockly.ContextMenu.callbackFactory(this, xmlBlock);
312
+    options.push(option);
313
+
314
+    // Add options to create getters for each parameter.
315
+    if (!this.isCollapsed()) {
316
+      for (var i = 0; i < this.arguments_.length; i++) {
317
+        var option = {enabled: true};
318
+        var name = this.arguments_[i];
319
+        option.text = Blockly.Msg.VARIABLES_SET_CREATE_GET.replace('%1', name);
320
+        var xmlField = goog.dom.createDom('field', null, name);
321
+        xmlField.setAttribute('name', 'VAR');
322
+        var xmlBlock = goog.dom.createDom('block', null, xmlField);
323
+        xmlBlock.setAttribute('type', 'variables_get');
324
+        option.callback = Blockly.ContextMenu.callbackFactory(this, xmlBlock);
325
+        options.push(option);
326
+      }
327
+    }
328
+  },
329
+  callType_: 'procedures_callnoreturn'
330
+};
331
+
332
+Blockly.Blocks['procedures_defreturn'] = {
333
+  /**
334
+   * Block for defining a procedure with a return value.
335
+   * @this Blockly.Block
336
+   */
337
+  init: function() {
338
+    this.setHelpUrl(Blockly.Msg.PROCEDURES_DEFRETURN_HELPURL);
339
+    this.setColour(Blockly.Blocks.procedures.HUE);
340
+    var name = Blockly.Procedures.findLegalName(
341
+        Blockly.Msg.PROCEDURES_DEFRETURN_PROCEDURE, this);
342
+    var nameField = new Blockly.FieldTextInput(name,
343
+        Blockly.Procedures.rename);
344
+    nameField.setSpellcheck(false);
345
+    this.appendDummyInput()
346
+        .appendField(Blockly.Msg.PROCEDURES_DEFRETURN_TITLE)
347
+        .appendField(nameField, 'NAME')
348
+        .appendField('', 'PARAMS');
349
+    this.appendValueInput('RETURN')
350
+        .setAlign(Blockly.ALIGN_RIGHT)
351
+        .appendField(Blockly.Msg.PROCEDURES_DEFRETURN_RETURN)
352
+        .setCheck(["Number", "Boolean"]);
353
+    this.setMutator(new Blockly.Mutator(['procedures_mutatorarg']));
354
+    this.setTooltip(Blockly.Msg.PROCEDURES_DEFRETURN_TOOLTIP);
355
+    this.arguments_ = [];
356
+    this.setStatements_(true);
357
+    this.statementConnection_ = null;
358
+  },
359
+  setStatements_: Blockly.Blocks['procedures_defnoreturn'].setStatements_,
360
+  updateParams_: Blockly.Blocks['procedures_defnoreturn'].updateParams_,
361
+  mutationToDom: Blockly.Blocks['procedures_defnoreturn'].mutationToDom,
362
+  domToMutation: Blockly.Blocks['procedures_defnoreturn'].domToMutation,
363
+  decompose: Blockly.Blocks['procedures_defnoreturn'].decompose,
364
+  compose: Blockly.Blocks['procedures_defnoreturn'].compose,
365
+  dispose: Blockly.Blocks['procedures_defnoreturn'].dispose,
366
+  /**
367
+   * Return the signature of this procedure definition.
368
+   * @return {!Array} Tuple containing three elements:
369
+   *     - the name of the defined procedure,
370
+   *     - a list of all its arguments,
371
+   *     - that it DOES have a return value.
372
+   * @this Blockly.Block
373
+   */
374
+  getProcedureDef: function() {
375
+    return [this.getFieldValue('NAME'), this.arguments_, true];
376
+  },
377
+  getVars: Blockly.Blocks['procedures_defnoreturn'].getVars,
378
+  renameVar: Blockly.Blocks['procedures_defnoreturn'].renameVar,
379
+  customContextMenu: Blockly.Blocks['procedures_defnoreturn'].customContextMenu,
380
+  callType_: 'procedures_callreturn'
381
+};
382
+
383
+Blockly.Blocks['procedures_mutatorcontainer'] = {
384
+  /**
385
+   * Mutator block for procedure container.
386
+   * @this Blockly.Block
387
+   */
388
+  init: function() {
389
+    this.setColour(Blockly.Blocks.procedures.HUE);
390
+    this.appendDummyInput()
391
+        .appendField(Blockly.Msg.PROCEDURES_MUTATORCONTAINER_TITLE);
392
+    this.appendStatementInput('STACK');
393
+    this.appendDummyInput('STATEMENT_INPUT')
394
+        .appendField(Blockly.Msg.PROCEDURES_ALLOW_STATEMENTS)
395
+        .appendField(new Blockly.FieldCheckbox('TRUE'), 'STATEMENTS');
396
+    this.setTooltip(Blockly.Msg.PROCEDURES_MUTATORCONTAINER_TOOLTIP);
397
+    this.contextMenu = false;
398
+  }
399
+};
400
+
401
+Blockly.Blocks['procedures_mutatorarg'] = {
402
+  /**
403
+   * Mutator block for procedure argument.
404
+   * @this Blockly.Block
405
+   */
406
+  init: function() {
407
+    this.setColour(Blockly.Blocks.procedures.HUE);
408
+    this.appendDummyInput()
409
+        .appendField(Blockly.Msg.PROCEDURES_MUTATORARG_TITLE)
410
+        .appendField(new Blockly.FieldTextInput('x', this.validator_), 'NAME');
411
+    this.setPreviousStatement(true);
412
+    this.setNextStatement(true);
413
+    this.setTooltip(Blockly.Msg.PROCEDURES_MUTATORARG_TOOLTIP);
414
+    this.contextMenu = false;
415
+  },
416
+  /**
417
+   * Obtain a valid name for the procedure.
418
+   * Merge runs of whitespace.  Strip leading and trailing whitespace.
419
+   * Beyond this, all names are legal.
420
+   * @param {string} newVar User-supplied name.
421
+   * @return {?string} Valid name, or null if a name was not specified.
422
+   * @private
423
+   * @this Blockly.Block
424
+   */
425
+  validator_: function(newVar) {
426
+    newVar = newVar.replace(/[\s\xa0]+/g, ' ').replace(/^ | $/g, '');
427
+    return newVar || null;
428
+  }
429
+};
430
+
431
+Blockly.Blocks['procedures_callnoreturn'] = {
432
+  /**
433
+   * Block for calling a procedure with no return value.
434
+   * @this Blockly.Block
435
+   */
436
+  init: function() {
437
+    this.setHelpUrl(Blockly.Msg.PROCEDURES_CALLNORETURN_HELPURL);
438
+    this.setColour(Blockly.Blocks.procedures.HUE);
439
+    this.appendDummyInput('TOPROW')
440
+        .appendField(Blockly.Msg.PROCEDURES_CALLNORETURN_CALL)
441
+        .appendField('', 'NAME');
442
+    this.setPreviousStatement(true);
443
+    this.setNextStatement(true);
444
+    // Tooltip is set in domToMutation.
445
+    this.arguments_ = [];
446
+    this.quarkConnections_ = {};
447
+    this.quarkArguments_ = null;
448
+  },
449
+  /**
450
+   * Returns the name of the procedure this block calls.
451
+   * @return {string} Procedure name.
452
+   * @this Blockly.Block
453
+   */
454
+  getProcedureCall: function() {
455
+    // The NAME field is guaranteed to exist, null will never be returned.
456
+    return /** @type {string} */ (this.getFieldValue('NAME'));
457
+  },
458
+  /**
459
+   * Notification that a procedure is renaming.
460
+   * If the name matches this block's procedure, rename it.
461
+   * @param {string} oldName Previous name of procedure.
462
+   * @param {string} newName Renamed procedure.
463
+   * @this Blockly.Block
464
+   */
465
+  renameProcedure: function(oldName, newName) {
466
+    if (Blockly.Names.equals(oldName, this.getProcedureCall())) {
467
+      this.setFieldValue(newName, 'NAME');
468
+      this.setTooltip(
469
+          (this.outputConnection ? Blockly.Msg.PROCEDURES_CALLRETURN_TOOLTIP :
470
+           Blockly.Msg.PROCEDURES_CALLNORETURN_TOOLTIP)
471
+          .replace('%1', newName));
472
+    }
473
+  },
474
+  /**
475
+   * Notification that the procedure's parameters have changed.
476
+   * @param {!Array.<string>} paramNames New param names, e.g. ['x', 'y', 'z'].
477
+   * @param {!Array.<string>} paramIds IDs of params (consistent for each
478
+   *     parameter through the life of a mutator, regardless of param renaming),
479
+   *     e.g. ['piua', 'f8b_', 'oi.o'].
480
+   * @this Blockly.Block
481
+   */
482
+  setProcedureParameters: function(paramNames, paramIds) {
483
+    // Data structures:
484
+    // this.arguments = ['x', 'y']
485
+    //     Existing param names.
486
+    // this.quarkConnections_ {piua: null, f8b_: Blockly.Connection}
487
+    //     Look-up of paramIds to connections plugged into the call block.
488
+    // this.quarkArguments_ = ['piua', 'f8b_']
489
+    //     Existing param IDs.
490
+    // Note that quarkConnections_ may include IDs that no longer exist, but
491
+    // which might reappear if a param is reattached in the mutator.
492
+    if (!paramIds) {
493
+      // Reset the quarks (a mutator is about to open).
494
+      this.quarkConnections_ = {};
495
+      this.quarkArguments_ = null;
496
+      return;
497
+    }
498
+    if (goog.array.equals(this.arguments_, paramNames)) {
499
+      // No change.
500
+      this.quarkArguments_ = paramIds;
501
+      return;
502
+    }
503
+    this.setCollapsed(false);
504
+    if (paramIds.length != paramNames.length) {
505
+      throw 'Error: paramNames and paramIds must be the same length.';
506
+    }
507
+    if (!this.quarkArguments_) {
508
+      // Initialize tracking for this block.
509
+      this.quarkConnections_ = {};
510
+      if (paramNames.join('\n') == this.arguments_.join('\n')) {
511
+        // No change to the parameters, allow quarkConnections_ to be
512
+        // populated with the existing connections.
513
+        this.quarkArguments_ = paramIds;
514
+      } else {
515
+        this.quarkArguments_ = [];
516
+      }
517
+    }
518
+    // Switch off rendering while the block is rebuilt.
519
+    var savedRendered = this.rendered;
520
+    this.rendered = false;
521
+    // Update the quarkConnections_ with existing connections.
522
+    for (var i = this.arguments_.length - 1; i >= 0; i--) {
523
+      var input = this.getInput('ARG' + i);
524
+      if (input) {
525
+        var connection = input.connection.targetConnection;
526
+        this.quarkConnections_[this.quarkArguments_[i]] = connection;
527
+        // Disconnect all argument blocks and remove all inputs.
528
+        this.removeInput('ARG' + i);
529
+      }
530
+    }
531
+    // Rebuild the block's arguments.
532
+    this.arguments_ = [].concat(paramNames);
533
+    this.renderArgs_();
534
+    this.quarkArguments_ = paramIds;
535
+    // Reconnect any child blocks.
536
+    if (this.quarkArguments_) {
537
+      for (var i = 0; i < this.arguments_.length; i++) {
538
+        var input = this.getInput('ARG' + i);
539
+        var quarkName = this.quarkArguments_[i];
540
+        if (quarkName in this.quarkConnections_) {
541
+          var connection = this.quarkConnections_[quarkName];
542
+          if (!connection || connection.targetConnection ||
543
+              connection.sourceBlock_.workspace != this.workspace) {
544
+            // Block no longer exists or has been attached elsewhere.
545
+            delete this.quarkConnections_[quarkName];
546
+          } else {
547
+            input.connection.connect(connection);
548
+          }
549
+        }
550
+      }
551
+    }
552
+    // Restore rendering and show the changes.
553
+    this.rendered = savedRendered;
554
+    if (this.rendered) {
555
+      this.render();
556
+    }
557
+  },
558
+  /**
559
+   * Render the arguments.
560
+   * @this Blockly.Block
561
+   * @private
562
+   */
563
+  renderArgs_: function() {
564
+    for (var i = 0; i < this.arguments_.length; i++) {
565
+      var input = this.appendValueInput('ARG' + i)
566
+          .setAlign(Blockly.ALIGN_RIGHT)
567
+          .appendField(this.arguments_[i]);
568
+      input.init();
569
+    }
570
+    // Add 'with:' if there are parameters.
571
+    var input = this.getInput('TOPROW');
572
+    if (input) {
573
+      if (this.arguments_.length) {
574
+        if (!this.getField('WITH')) {
575
+          input.appendField(Blockly.Msg.PROCEDURES_CALL_BEFORE_PARAMS, 'WITH');
576
+          input.init();
577
+        }
578
+      } else {
579
+        if (this.getField('WITH')) {
580
+          input.removeField('WITH');
581
+        }
582
+      }
583
+    }
584
+  },
585
+  /**
586
+   * Create XML to represent the (non-editable) name and arguments.
587
+   * @return {Element} XML storage element.
588
+   * @this Blockly.Block
589
+   */
590
+  mutationToDom: function() {
591
+    var container = document.createElement('mutation');
592
+    container.setAttribute('name', this.getProcedureCall());
593
+    for (var i = 0; i < this.arguments_.length; i++) {
594
+      var parameter = document.createElement('arg');
595
+      parameter.setAttribute('name', this.arguments_[i]);
596
+      container.appendChild(parameter);
597
+    }
598
+    return container;
599
+  },
600
+  /**
601
+   * Parse XML to restore the (non-editable) name and parameters.
602
+   * @param {!Element} xmlElement XML storage element.
603
+   * @this Blockly.Block
604
+   */
605
+  domToMutation: function(xmlElement) {
606
+    var name = xmlElement.getAttribute('name');
607
+    this.setFieldValue(name, 'NAME');
608
+    this.setTooltip(
609
+        (this.outputConnection ? Blockly.Msg.PROCEDURES_CALLRETURN_TOOLTIP :
610
+         Blockly.Msg.PROCEDURES_CALLNORETURN_TOOLTIP).replace('%1', name));
611
+    var def = Blockly.Procedures.getDefinition(name, this.workspace);
612
+    if (def && def.mutator && def.mutator.isVisible()) {
613
+      // Initialize caller with the mutator's IDs.
614
+      this.setProcedureParameters(def.arguments_, def.paramIds_);
615
+    } else {
616
+      var args = [];
617
+      for (var i = 0, childNode; childNode = xmlElement.childNodes[i]; i++) {
618
+        if (childNode.nodeName.toLowerCase() == 'arg') {
619
+          args.push(childNode.getAttribute('name'));
620
+        }
621
+      }
622
+      // For the second argument (paramIds) use the arguments list as a dummy
623
+      // list.
624
+      this.setProcedureParameters(args, args);
625
+    }
626
+  },
627
+  /**
628
+   * Notification that a variable is renaming.
629
+   * If the name matches one of this block's variables, rename it.
630
+   * @param {string} oldName Previous name of variable.
631
+   * @param {string} newName Renamed variable.
632
+   * @this Blockly.Block
633
+   */
634
+  renameVar: function(oldName, newName) {
635
+    for (var i = 0; i < this.arguments_.length; i++) {
636
+      if (Blockly.Names.equals(oldName, this.arguments_[i])) {
637
+        this.arguments_[i] = newName;
638
+        this.getInput('ARG' + i).fieldRow[0].setText(newName);
639
+      }
640
+    }
641
+  },
642
+  /**
643
+   * Add menu option to find the definition block for this call.
644
+   * @param {!Array} options List of menu options to add to.
645
+   * @this Blockly.Block
646
+   */
647
+  customContextMenu: function(options) {
648
+    var option = {enabled: true};
649
+    option.text = Blockly.Msg.PROCEDURES_HIGHLIGHT_DEF;
650
+    var name = this.getProcedureCall();
651
+    var workspace = this.workspace;
652
+    option.callback = function() {
653
+      var def = Blockly.Procedures.getDefinition(name, workspace);
654
+      def && def.select();
655
+    };
656
+    options.push(option);
657
+  }
658
+};
659
+
660
+Blockly.Blocks['procedures_callreturn'] = {
661
+  /**
662
+   * Block for calling a procedure with a return value.
663
+   * @this Blockly.Block
664
+   */
665
+  init: function() {
666
+    this.setHelpUrl(Blockly.Msg.PROCEDURES_CALLRETURN_HELPURL);
667
+    this.setColour(Blockly.Blocks.procedures.HUE);
668
+    this.appendDummyInput('TOPROW')
669
+        .appendField(Blockly.Msg.PROCEDURES_CALLRETURN_CALL)
670
+        .appendField('', 'NAME');
671
+    this.setOutput(true);
672
+    // Tooltip is set in domToMutation.
673
+    this.arguments_ = [];
674
+    this.quarkConnections_ = {};
675
+    this.quarkArguments_ = null;
676
+  },
677
+  getProcedureCall: Blockly.Blocks['procedures_callnoreturn'].getProcedureCall,
678
+  renameProcedure: Blockly.Blocks['procedures_callnoreturn'].renameProcedure,
679
+  setProcedureParameters:
680
+      Blockly.Blocks['procedures_callnoreturn'].setProcedureParameters,
681
+  renderArgs_: Blockly.Blocks['procedures_callnoreturn'].renderArgs_,
682
+  mutationToDom: Blockly.Blocks['procedures_callnoreturn'].mutationToDom,
683
+  domToMutation: Blockly.Blocks['procedures_callnoreturn'].domToMutation,
684
+  renameVar: Blockly.Blocks['procedures_callnoreturn'].renameVar,
685
+  customContextMenu: Blockly.Blocks['procedures_callnoreturn'].customContextMenu
686
+};
687
+
688
+Blockly.Blocks['procedures_ifreturn'] = {
689
+  /**
690
+   * Block for conditionally returning a value from a procedure.
691
+   * @this Blockly.Block
692
+   */
693
+  init: function() {
694
+    this.setHelpUrl('http://c2.com/cgi/wiki?GuardClause');
695
+    this.setColour(Blockly.Blocks.procedures.HUE);
696
+    this.appendValueInput('CONDITION')
697
+        .setCheck('Boolean')
698
+        .appendField(Blockly.Msg.CONTROLS_IF_MSG_IF);
699
+    this.appendValueInput('VALUE')
700
+        .appendField(Blockly.Msg.PROCEDURES_DEFRETURN_RETURN);
701
+    this.setInputsInline(true);
702
+    this.setPreviousStatement(true);
703
+    this.setNextStatement(true);
704
+    this.setTooltip(Blockly.Msg.PROCEDURES_IFRETURN_TOOLTIP);
705
+    this.hasReturnValue_ = true;
706
+  },
707
+  /**
708
+   * Create XML to represent whether this block has a return value.
709
+   * @return {Element} XML storage element.
710
+   * @this Blockly.Block
711
+   */
712
+  mutationToDom: function() {
713
+    var container = document.createElement('mutation');
714
+    container.setAttribute('value', Number(this.hasReturnValue_));
715
+    return container;
716
+  },
717
+  /**
718
+   * Parse XML to restore whether this block has a return value.
719
+   * @param {!Element} xmlElement XML storage element.
720
+   * @this Blockly.Block
721
+   */
722
+  domToMutation: function(xmlElement) {
723
+    var value = xmlElement.getAttribute('value');
724
+    this.hasReturnValue_ = (value == 1);
725
+    if (!this.hasReturnValue_) {
726
+      this.removeInput('VALUE');
727
+      this.appendDummyInput('VALUE')
728
+        .appendField(Blockly.Msg.PROCEDURES_DEFRETURN_RETURN);
729
+    }
730
+  },
731
+  /**
732
+   * Called whenever anything on the workspace changes.
733
+   * Add warning if this flow block is not nested inside a loop.
734
+   * @this Blockly.Block
735
+   */
736
+  onchange: function() {
737
+    var legal = false;
738
+    // Is the block nested in a procedure?
739
+    var block = this;
740
+    do {
741
+      if (block.type == 'procedures_defnoreturn' ||
742
+          block.type == 'procedures_defreturn') {
743
+        legal = true;
744
+        break;
745
+      }
746
+      block = block.getSurroundParent();
747
+    } while (block);
748
+    if (legal) {
749
+      // If needed, toggle whether this block has a return value.
750
+      if (block.type == 'procedures_defnoreturn' && this.hasReturnValue_) {
751
+        this.removeInput('VALUE');
752
+        this.appendDummyInput('VALUE')
753
+          .appendField(Blockly.Msg.PROCEDURES_DEFRETURN_RETURN);
754
+        this.hasReturnValue_ = false;
755
+      } else if (block.type == 'procedures_defreturn' &&
756
+                 !this.hasReturnValue_) {
757
+        this.removeInput('VALUE');
758
+        this.appendValueInput('VALUE')
759
+          .appendField(Blockly.Msg.PROCEDURES_DEFRETURN_RETURN);
760
+        this.hasReturnValue_ = true;
761
+      }
762
+      this.setWarningText(null);
763
+    } else {
764
+      this.setWarningText(Blockly.Msg.PROCEDURES_IFRETURN_WARNING);
765
+    }
766
+  }
767
+};

+ 664 - 0
src/blockly/blocks/text.js

@@ -0,0 +1,664 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Text blocks for Blockly.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Blocks.texts');
28
+
29
+goog.require('Blockly.Blocks');
30
+
31
+
32
+/**
33
+ * Common HSV hue for all blocks in this category.
34
+ */
35
+Blockly.Blocks.texts.HUE = 160;
36
+
37
+Blockly.Blocks['text'] = {
38
+  /**
39
+   * Block for text value.
40
+   * @this Blockly.Block
41
+   */
42
+  init: function() {
43
+    this.setHelpUrl(Blockly.Msg.TEXT_TEXT_HELPURL);
44
+    this.setColour(Blockly.Blocks.texts.HUE);
45
+    this.appendDummyInput()
46
+        .appendField(this.newQuote_(true))
47
+        .appendField(new Blockly.FieldTextInput(''), 'TEXT')
48
+        .appendField(this.newQuote_(false));
49
+    this.setOutput(true, 'String');
50
+    this.setTooltip(Blockly.Msg.TEXT_TEXT_TOOLTIP);
51
+  },
52
+  /**
53
+   * Create an image of an open or closed quote.
54
+   * @param {boolean} open True if open quote, false if closed.
55
+   * @return {!Blockly.FieldImage} The field image of the quote.
56
+   * @this Blockly.Block
57
+   * @private
58
+   */
59
+  newQuote_: function(open) {
60
+    if (open == this.RTL) {
61
+      var file = '';
62
+    } else {
63
+      var file = '';
64
+    }
65
+    return new Blockly.FieldImage(file, 12, 12, '"');
66
+  }
67
+};
68
+
69
+Blockly.Blocks['text_join'] = {
70
+  /**
71
+   * Block for creating a string made up of any number of elements of any type.
72
+   * @this Blockly.Block
73
+   */
74
+  init: function() {
75
+    this.setHelpUrl(Blockly.Msg.TEXT_JOIN_HELPURL);
76
+    this.setColour(Blockly.Blocks.texts.HUE);
77
+    this.itemCount_ = 2;
78
+    this.updateShape_();
79
+    this.setOutput(true, 'String');
80
+    this.setMutator(new Blockly.Mutator(['text_create_join_item']));
81
+    this.setTooltip(Blockly.Msg.TEXT_JOIN_TOOLTIP);
82
+  },
83
+  /**
84
+   * Create XML to represent number of text inputs.
85
+   * @return {Element} XML storage element.
86
+   * @this Blockly.Block
87
+   */
88
+  mutationToDom: function() {
89
+    var container = document.createElement('mutation');
90
+    container.setAttribute('items', this.itemCount_);
91
+    return container;
92
+  },
93
+  /**
94
+   * Parse XML to restore the text inputs.
95
+   * @param {!Element} xmlElement XML storage element.
96
+   * @this Blockly.Block
97
+   */
98
+  domToMutation: function(xmlElement) {
99
+    this.itemCount_ = parseInt(xmlElement.getAttribute('items'), 10);
100
+    this.updateShape_();
101
+  },
102
+  /**
103
+   * Populate the mutator's dialog with this block's components.
104
+   * @param {!Blockly.Workspace} workspace Mutator's workspace.
105
+   * @return {!Blockly.Block} Root block in mutator.
106
+   * @this Blockly.Block
107
+   */
108
+  decompose: function(workspace) {
109
+    var containerBlock = Blockly.Block.obtain(workspace,
110
+                                           'text_create_join_container');
111
+    containerBlock.initSvg();
112
+    var connection = containerBlock.getInput('STACK').connection;
113
+    for (var i = 0; i < this.itemCount_; i++) {
114
+      var itemBlock = Blockly.Block.obtain(workspace, 'text_create_join_item');
115
+      itemBlock.initSvg();
116
+      connection.connect(itemBlock.previousConnection);
117
+      connection = itemBlock.nextConnection;
118
+    }
119
+    return containerBlock;
120
+  },
121
+  /**
122
+   * Reconfigure this block based on the mutator dialog's components.
123
+   * @param {!Blockly.Block} containerBlock Root block in mutator.
124
+   * @this Blockly.Block
125
+   */
126
+  compose: function(containerBlock) {
127
+    var itemBlock = containerBlock.getInputTargetBlock('STACK');
128
+    // Count number of inputs.
129
+    var connections = [];
130
+    var i = 0;
131
+    while (itemBlock) {
132
+      connections[i] = itemBlock.valueConnection_;
133
+      itemBlock = itemBlock.nextConnection &&
134
+          itemBlock.nextConnection.targetBlock();
135
+      i++;
136
+    }
137
+    this.itemCount_ = i;
138
+    this.updateShape_();
139
+    // Reconnect any child blocks.
140
+    for (var i = 0; i < this.itemCount_; i++) {
141
+      if (connections[i]) {
142
+        this.getInput('ADD' + i).connection.connect(connections[i]);
143
+      }
144
+    }
145
+  },
146
+  /**
147
+   * Store pointers to any connected child blocks.
148
+   * @param {!Blockly.Block} containerBlock Root block in mutator.
149
+   * @this Blockly.Block
150
+   */
151
+  saveConnections: function(containerBlock) {
152
+    var itemBlock = containerBlock.getInputTargetBlock('STACK');
153
+    var i = 0;
154
+    while (itemBlock) {
155
+      var input = this.getInput('ADD' + i);
156
+      itemBlock.valueConnection_ = input && input.connection.targetConnection;
157
+      i++;
158
+      itemBlock = itemBlock.nextConnection &&
159
+          itemBlock.nextConnection.targetBlock();
160
+    }
161
+  },
162
+  /**
163
+   * Modify this block to have the correct number of inputs.
164
+   * @private
165
+   * @this Blockly.Block
166
+   */
167
+  updateShape_: function() {
168
+    // Delete everything.
169
+    if (this.getInput('EMPTY')) {
170
+      this.removeInput('EMPTY');
171
+    } else {
172
+      var i = 0;
173
+      while (this.getInput('ADD' + i)) {
174
+        this.removeInput('ADD' + i);
175
+        i++;
176
+      }
177
+    }
178
+    // Rebuild block.
179
+    if (this.itemCount_ == 0) {
180
+      this.appendDummyInput('EMPTY')
181
+          .appendField(this.newQuote_(true))
182
+          .appendField(this.newQuote_(false));
183
+    } else {
184
+      for (var i = 0; i < this.itemCount_; i++) {
185
+        var input = this.appendValueInput('ADD' + i);
186
+        if (i == 0) {
187
+          input.appendField(Blockly.Msg.TEXT_JOIN_TITLE_CREATEWITH);
188
+        }
189
+      }
190
+    }
191
+  },
192
+  newQuote_: Blockly.Blocks['text'].newQuote_
193
+};
194
+
195
+Blockly.Blocks['text_create_join_container'] = {
196
+  /**
197
+   * Mutator block for container.
198
+   * @this Blockly.Block
199
+   */
200
+  init: function() {
201
+    this.setColour(Blockly.Blocks.texts.HUE);
202
+    this.appendDummyInput()
203
+        .appendField(Blockly.Msg.TEXT_CREATE_JOIN_TITLE_JOIN);
204
+    this.appendStatementInput('STACK');
205
+    this.setTooltip(Blockly.Msg.TEXT_CREATE_JOIN_TOOLTIP);
206
+    this.contextMenu = false;
207
+  }
208
+};
209
+
210
+Blockly.Blocks['text_create_join_item'] = {
211
+  /**
212
+   * Mutator block for add items.
213
+   * @this Blockly.Block
214
+   */
215
+  init: function() {
216
+    this.setColour(Blockly.Blocks.texts.HUE);
217
+    this.appendDummyInput()
218
+        .appendField(Blockly.Msg.TEXT_CREATE_JOIN_ITEM_TITLE_ITEM);
219
+    this.setPreviousStatement(true);
220
+    this.setNextStatement(true);
221
+    this.setTooltip(Blockly.Msg.TEXT_CREATE_JOIN_ITEM_TOOLTIP);
222
+    this.contextMenu = false;
223
+  }
224
+};
225
+
226
+Blockly.Blocks['text_append'] = {
227
+  /**
228
+   * Block for appending to a variable in place.
229
+   * @this Blockly.Block
230
+   */
231
+  init: function() {
232
+    this.setHelpUrl(Blockly.Msg.TEXT_APPEND_HELPURL);
233
+    this.setColour(Blockly.Blocks.texts.HUE);
234
+    this.appendValueInput('TEXT')
235
+        .appendField(Blockly.Msg.TEXT_APPEND_TO)
236
+        .appendField(new Blockly.FieldVariable(
237
+        Blockly.Msg.TEXT_APPEND_VARIABLE), 'VAR')
238
+        .appendField(Blockly.Msg.TEXT_APPEND_APPENDTEXT);
239
+    this.setPreviousStatement(true);
240
+    this.setNextStatement(true);
241
+    // Assign 'this' to a variable for use in the tooltip closure below.
242
+    var thisBlock = this;
243
+    this.setTooltip(function() {
244
+      return Blockly.Msg.TEXT_APPEND_TOOLTIP.replace('%1',
245
+          thisBlock.getFieldValue('VAR'));
246
+    });
247
+  },
248
+  /**
249
+   * Return all variables referenced by this block.
250
+   * @return {!Array.<string>} List of variable names.
251
+   * @this Blockly.Block
252
+   */
253
+  getVars: function() {
254
+    return [this.getFieldValue('VAR')];
255
+  },
256
+  /**
257
+   * Notification that a variable is renaming.
258
+   * If the name matches one of this block's variables, rename it.
259
+   * @param {string} oldName Previous name of variable.
260
+   * @param {string} newName Renamed variable.
261
+   * @this Blockly.Block
262
+   */
263
+  renameVar: function(oldName, newName) {
264
+    if (Blockly.Names.equals(oldName, this.getFieldValue('VAR'))) {
265
+      this.setFieldValue(newName, 'VAR');
266
+    }
267
+  }
268
+};
269
+
270
+Blockly.Blocks['text_length'] = {
271
+  /**
272
+   * Block for string length.
273
+   * @this Blockly.Block
274
+   */
275
+  init: function() {
276
+    this.jsonInit({
277
+      "message0": Blockly.Msg.TEXT_LENGTH_TITLE,
278
+      "args0": [
279
+        {
280
+          "type": "input_value",
281
+          "name": "VALUE",
282
+          "check": ['String', 'Array']
283
+        }
284
+      ],
285
+      "output": 'Number',
286
+      "colour": Blockly.Blocks.texts.HUE,
287
+      "tooltip": Blockly.Msg.TEXT_LENGTH_TOOLTIP,
288
+      "helpUrl": Blockly.Msg.TEXT_LENGTH_HELPURL
289
+    });
290
+  }
291
+};
292
+
293
+Blockly.Blocks['text_isEmpty'] = {
294
+  /**
295
+   * Block for is the string null?
296
+   * @this Blockly.Block
297
+   */
298
+  init: function() {
299
+    this.jsonInit({
300
+      "message0": Blockly.Msg.TEXT_ISEMPTY_TITLE,
301
+      "args0": [
302
+        {
303
+          "type": "input_value",
304
+          "name": "VALUE",
305
+          "check": ['String', 'Array']
306
+        }
307
+      ],
308
+      "output": 'Boolean',
309
+      "colour": Blockly.Blocks.texts.HUE,
310
+      "tooltip": Blockly.Msg.TEXT_ISEMPTY_TOOLTIP,
311
+      "helpUrl": Blockly.Msg.TEXT_ISEMPTY_HELPURL
312
+    });
313
+  }
314
+};
315
+
316
+Blockly.Blocks['text_indexOf'] = {
317
+  /**
318
+   * Block for finding a substring in the text.
319
+   * @this Blockly.Block
320
+   */
321
+  init: function() {
322
+    var OPERATORS =
323
+        [[Blockly.Msg.TEXT_INDEXOF_OPERATOR_FIRST, 'FIRST'],
324
+         [Blockly.Msg.TEXT_INDEXOF_OPERATOR_LAST, 'LAST']];
325
+    this.setHelpUrl(Blockly.Msg.TEXT_INDEXOF_HELPURL);
326
+    this.setColour(Blockly.Blocks.texts.HUE);
327
+    this.setOutput(true, 'Number');
328
+    this.appendValueInput('VALUE')
329
+        .setCheck('String')
330
+        .appendField(Blockly.Msg.TEXT_INDEXOF_INPUT_INTEXT);
331
+    this.appendValueInput('FIND')
332
+        .setCheck('String')
333
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'END');
334
+    if (Blockly.Msg.TEXT_INDEXOF_TAIL) {
335
+      this.appendDummyInput().appendField(Blockly.Msg.TEXT_INDEXOF_TAIL);
336
+    }
337
+    this.setInputsInline(true);
338
+    this.setTooltip(Blockly.Msg.TEXT_INDEXOF_TOOLTIP);
339
+  }
340
+};
341
+
342
+Blockly.Blocks['text_charAt'] = {
343
+  /**
344
+   * Block for getting a character from the string.
345
+   * @this Blockly.Block
346
+   */
347
+  init: function() {
348
+    this.WHERE_OPTIONS =
349
+        [[Blockly.Msg.TEXT_CHARAT_FROM_START, 'FROM_START'],
350
+         [Blockly.Msg.TEXT_CHARAT_FROM_END, 'FROM_END'],
351
+         [Blockly.Msg.TEXT_CHARAT_FIRST, 'FIRST'],
352
+         [Blockly.Msg.TEXT_CHARAT_LAST, 'LAST'],
353
+         [Blockly.Msg.TEXT_CHARAT_RANDOM, 'RANDOM']];
354
+    this.setHelpUrl(Blockly.Msg.TEXT_CHARAT_HELPURL);
355
+    this.setColour(Blockly.Blocks.texts.HUE);
356
+    this.setOutput(true, 'String');
357
+    this.appendValueInput('VALUE')
358
+        .setCheck('String')
359
+        .appendField(Blockly.Msg.TEXT_CHARAT_INPUT_INTEXT);
360
+    this.appendDummyInput('AT');
361
+    this.setInputsInline(true);
362
+    this.updateAt_(true);
363
+    this.setTooltip(Blockly.Msg.TEXT_CHARAT_TOOLTIP);
364
+  },
365
+  /**
366
+   * Create XML to represent whether there is an 'AT' input.
367
+   * @return {Element} XML storage element.
368
+   * @this Blockly.Block
369
+   */
370
+  mutationToDom: function() {
371
+    var container = document.createElement('mutation');
372
+    var isAt = this.getInput('AT').type == Blockly.INPUT_VALUE;
373
+    container.setAttribute('at', isAt);
374
+    return container;
375
+  },
376
+  /**
377
+   * Parse XML to restore the 'AT' input.
378
+   * @param {!Element} xmlElement XML storage element.
379
+   * @this Blockly.Block
380
+   */
381
+  domToMutation: function(xmlElement) {
382
+    // Note: Until January 2013 this block did not have mutations,
383
+    // so 'at' defaults to true.
384
+    var isAt = (xmlElement.getAttribute('at') != 'false');
385
+    this.updateAt_(isAt);
386
+  },
387
+  /**
388
+   * Create or delete an input for the numeric index.
389
+   * @param {boolean} isAt True if the input should exist.
390
+   * @private
391
+   * @this Blockly.Block
392
+   */
393
+  updateAt_: function(isAt) {
394
+    // Destroy old 'AT' and 'ORDINAL' inputs.
395
+    this.removeInput('AT');
396
+    this.removeInput('ORDINAL', true);
397
+    // Create either a value 'AT' input or a dummy input.
398
+    if (isAt) {
399
+      this.appendValueInput('AT').setCheck('Number');
400
+      if (Blockly.Msg.ORDINAL_NUMBER_SUFFIX) {
401
+        this.appendDummyInput('ORDINAL')
402
+            .appendField(Blockly.Msg.ORDINAL_NUMBER_SUFFIX);
403
+      }
404
+    } else {
405
+      this.appendDummyInput('AT');
406
+    }
407
+    if (Blockly.Msg.TEXT_CHARAT_TAIL) {
408
+      this.removeInput('TAIL', true);
409
+      this.appendDummyInput('TAIL')
410
+          .appendField(Blockly.Msg.TEXT_CHARAT_TAIL);
411
+    }
412
+    var menu = new Blockly.FieldDropdown(this.WHERE_OPTIONS, function(value) {
413
+      var newAt = (value == 'FROM_START') || (value == 'FROM_END');
414
+      // The 'isAt' variable is available due to this function being a closure.
415
+      if (newAt != isAt) {
416
+        var block = this.sourceBlock_;
417
+        block.updateAt_(newAt);
418
+        // This menu has been destroyed and replaced.  Update the replacement.
419
+        block.setFieldValue(value, 'WHERE');
420
+        return null;
421
+      }
422
+      return undefined;
423
+    });
424
+    this.getInput('AT').appendField(menu, 'WHERE');
425
+  }
426
+};
427
+
428
+Blockly.Blocks['text_getSubstring'] = {
429
+  /**
430
+   * Block for getting substring.
431
+   * @this Blockly.Block
432
+   */
433
+  init: function() {
434
+    this['WHERE_OPTIONS_1'] =
435
+        [[Blockly.Msg.TEXT_GET_SUBSTRING_START_FROM_START, 'FROM_START'],
436
+         [Blockly.Msg.TEXT_GET_SUBSTRING_START_FROM_END, 'FROM_END'],
437
+         [Blockly.Msg.TEXT_GET_SUBSTRING_START_FIRST, 'FIRST']];
438
+    this['WHERE_OPTIONS_2'] =
439
+        [[Blockly.Msg.TEXT_GET_SUBSTRING_END_FROM_START, 'FROM_START'],
440
+         [Blockly.Msg.TEXT_GET_SUBSTRING_END_FROM_END, 'FROM_END'],
441
+         [Blockly.Msg.TEXT_GET_SUBSTRING_END_LAST, 'LAST']];
442
+    this.setHelpUrl(Blockly.Msg.TEXT_GET_SUBSTRING_HELPURL);
443
+    this.setColour(Blockly.Blocks.texts.HUE);
444
+    this.appendValueInput('STRING')
445
+        .setCheck('String')
446
+        .appendField(Blockly.Msg.TEXT_GET_SUBSTRING_INPUT_IN_TEXT);
447
+    this.appendDummyInput('AT1');
448
+    this.appendDummyInput('AT2');
449
+    if (Blockly.Msg.TEXT_GET_SUBSTRING_TAIL) {
450
+      this.appendDummyInput('TAIL')
451
+          .appendField(Blockly.Msg.TEXT_GET_SUBSTRING_TAIL);
452
+    }
453
+    this.setInputsInline(true);
454
+    this.setOutput(true, 'String');
455
+    this.updateAt_(1, true);
456
+    this.updateAt_(2, true);
457
+    this.setTooltip(Blockly.Msg.TEXT_GET_SUBSTRING_TOOLTIP);
458
+  },
459
+  /**
460
+   * Create XML to represent whether there are 'AT' inputs.
461
+   * @return {Element} XML storage element.
462
+   * @this Blockly.Block
463
+   */
464
+  mutationToDom: function() {
465
+    var container = document.createElement('mutation');
466
+    var isAt1 = this.getInput('AT1').type == Blockly.INPUT_VALUE;
467
+    container.setAttribute('at1', isAt1);
468
+    var isAt2 = this.getInput('AT2').type == Blockly.INPUT_VALUE;
469
+    container.setAttribute('at2', isAt2);
470
+    return container;
471
+  },
472
+  /**
473
+   * Parse XML to restore the 'AT' inputs.
474
+   * @param {!Element} xmlElement XML storage element.
475
+   * @this Blockly.Block
476
+   */
477
+  domToMutation: function(xmlElement) {
478
+    var isAt1 = (xmlElement.getAttribute('at1') == 'true');
479
+    var isAt2 = (xmlElement.getAttribute('at2') == 'true');
480
+    this.updateAt_(1, isAt1);
481
+    this.updateAt_(2, isAt2);
482
+  },
483
+  /**
484
+   * Create or delete an input for a numeric index.
485
+   * This block has two such inputs, independant of each other.
486
+   * @param {number} n Specify first or second input (1 or 2).
487
+   * @param {boolean} isAt True if the input should exist.
488
+   * @private
489
+   * @this Blockly.Block
490
+   */
491
+  updateAt_: function(n, isAt) {
492
+    // Create or delete an input for the numeric index.
493
+    // Destroy old 'AT' and 'ORDINAL' inputs.
494
+    this.removeInput('AT' + n);
495
+    this.removeInput('ORDINAL' + n, true);
496
+    // Create either a value 'AT' input or a dummy input.
497
+    if (isAt) {
498
+      this.appendValueInput('AT' + n).setCheck('Number');
499
+      if (Blockly.Msg.ORDINAL_NUMBER_SUFFIX) {
500
+        this.appendDummyInput('ORDINAL' + n)
501
+            .appendField(Blockly.Msg.ORDINAL_NUMBER_SUFFIX);
502
+      }
503
+    } else {
504
+      this.appendDummyInput('AT' + n);
505
+    }
506
+    // Move tail, if present, to end of block.
507
+    if (n == 2 && Blockly.Msg.TEXT_GET_SUBSTRING_TAIL) {
508
+      this.removeInput('TAIL', true);
509
+      this.appendDummyInput('TAIL')
510
+          .appendField(Blockly.Msg.TEXT_GET_SUBSTRING_TAIL);
511
+    }
512
+    var menu = new Blockly.FieldDropdown(this['WHERE_OPTIONS_' + n],
513
+        function(value) {
514
+      var newAt = (value == 'FROM_START') || (value == 'FROM_END');
515
+      // The 'isAt' variable is available due to this function being a closure.
516
+      if (newAt != isAt) {
517
+        var block = this.sourceBlock_;
518
+        block.updateAt_(n, newAt);
519
+        // This menu has been destroyed and replaced.  Update the replacement.
520
+        block.setFieldValue(value, 'WHERE' + n);
521
+        return null;
522
+      }
523
+      return undefined;
524
+    });
525
+    this.getInput('AT' + n)
526
+        .appendField(menu, 'WHERE' + n);
527
+    if (n == 1) {
528
+      this.moveInputBefore('AT1', 'AT2');
529
+    }
530
+  }
531
+};
532
+
533
+Blockly.Blocks['text_changeCase'] = {
534
+  /**
535
+   * Block for changing capitalization.
536
+   * @this Blockly.Block
537
+   */
538
+  init: function() {
539
+    var OPERATORS =
540
+        [[Blockly.Msg.TEXT_CHANGECASE_OPERATOR_UPPERCASE, 'UPPERCASE'],
541
+         [Blockly.Msg.TEXT_CHANGECASE_OPERATOR_LOWERCASE, 'LOWERCASE'],
542
+         [Blockly.Msg.TEXT_CHANGECASE_OPERATOR_TITLECASE, 'TITLECASE']];
543
+    this.setHelpUrl(Blockly.Msg.TEXT_CHANGECASE_HELPURL);
544
+    this.setColour(Blockly.Blocks.texts.HUE);
545
+    this.appendValueInput('TEXT')
546
+        .setCheck('String')
547
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'CASE');
548
+    this.setOutput(true, 'String');
549
+    this.setTooltip(Blockly.Msg.TEXT_CHANGECASE_TOOLTIP);
550
+  }
551
+};
552
+
553
+Blockly.Blocks['text_trim'] = {
554
+  /**
555
+   * Block for trimming spaces.
556
+   * @this Blockly.Block
557
+   */
558
+  init: function() {
559
+    var OPERATORS =
560
+        [[Blockly.Msg.TEXT_TRIM_OPERATOR_BOTH, 'BOTH'],
561
+         [Blockly.Msg.TEXT_TRIM_OPERATOR_LEFT, 'LEFT'],
562
+         [Blockly.Msg.TEXT_TRIM_OPERATOR_RIGHT, 'RIGHT']];
563
+    this.setHelpUrl(Blockly.Msg.TEXT_TRIM_HELPURL);
564
+    this.setColour(Blockly.Blocks.texts.HUE);
565
+    this.appendValueInput('TEXT')
566
+        .setCheck('String')
567
+        .appendField(new Blockly.FieldDropdown(OPERATORS), 'MODE');
568
+    this.setOutput(true, 'String');
569
+    this.setTooltip(Blockly.Msg.TEXT_TRIM_TOOLTIP);
570
+  }
571
+};
572
+
573
+Blockly.Blocks['text_print'] = {
574
+  /**
575
+   * Block for print statement.
576
+   * @this Blockly.Block
577
+   */
578
+  init: function() {
579
+    this.jsonInit({
580
+      "message0": Blockly.Msg.TEXT_PRINT_TITLE,
581
+      "args0": [
582
+        {
583
+          "type": "input_value",
584
+          "name": "TEXT"
585
+        }
586
+      ],
587
+      "previousStatement": null,
588
+      "nextStatement": null,
589
+      "colour": Blockly.Blocks.texts.HUE,
590
+      "tooltip": Blockly.Msg.TEXT_PRINT_TOOLTIP,
591
+      "helpUrl": Blockly.Msg.TEXT_PRINT_HELPURL
592
+    });
593
+  }
594
+};
595
+
596
+Blockly.Blocks['text_prompt'] = {
597
+  /**
598
+   * Block for prompt function (internal message).
599
+   * @this Blockly.Block
600
+   */
601
+  init: function() {
602
+    var TYPES =
603
+        [[Blockly.Msg.TEXT_PROMPT_TYPE_TEXT, 'TEXT'],
604
+         [Blockly.Msg.TEXT_PROMPT_TYPE_NUMBER, 'NUMBER']];
605
+    // Assign 'this' to a variable for use in the closure below.
606
+    var thisBlock = this;
607
+    this.setHelpUrl(Blockly.Msg.TEXT_PROMPT_HELPURL);
608
+    this.setColour(Blockly.Blocks.texts.HUE);
609
+    var dropdown = new Blockly.FieldDropdown(TYPES, function(newOp) {
610
+      if (newOp == 'NUMBER') {
611
+        thisBlock.changeOutput('Number');
612
+      } else {
613
+        thisBlock.changeOutput('String');
614
+      }
615
+    });
616
+    this.appendDummyInput()
617
+        .appendField(dropdown, 'TYPE')
618
+        .appendField(this.newQuote_(true))
619
+        .appendField(new Blockly.FieldTextInput(''), 'TEXT')
620
+        .appendField(this.newQuote_(false));
621
+    this.setOutput(true, 'String');
622
+    // Assign 'this' to a variable for use in the tooltip closure below.
623
+    var thisBlock = this;
624
+    this.setTooltip(function() {
625
+      return (thisBlock.getFieldValue('TYPE') == 'TEXT') ?
626
+          Blockly.Msg.TEXT_PROMPT_TOOLTIP_TEXT :
627
+          Blockly.Msg.TEXT_PROMPT_TOOLTIP_NUMBER;
628
+    });
629
+  },
630
+  newQuote_: Blockly.Blocks['text'].newQuote_
631
+};
632
+
633
+Blockly.Blocks['text_prompt_ext'] = {
634
+  /**
635
+   * Block for prompt function (external message).
636
+   * @this Blockly.Block
637
+   */
638
+  init: function() {
639
+    var TYPES =
640
+        [[Blockly.Msg.TEXT_PROMPT_TYPE_TEXT, 'TEXT'],
641
+         [Blockly.Msg.TEXT_PROMPT_TYPE_NUMBER, 'NUMBER']];
642
+    // Assign 'this' to a variable for use in the closure below.
643
+    var thisBlock = this;
644
+    this.setHelpUrl(Blockly.Msg.TEXT_PROMPT_HELPURL);
645
+    this.setColour(Blockly.Blocks.texts.HUE);
646
+    var dropdown = new Blockly.FieldDropdown(TYPES, function(newOp) {
647
+      if (newOp == 'NUMBER') {
648
+        thisBlock.changeOutput('Number');
649
+      } else {
650
+        thisBlock.changeOutput('String');
651
+      }
652
+    });
653
+    this.appendValueInput('TEXT')
654
+        .appendField(dropdown, 'TYPE');
655
+    this.setOutput(true, 'String');
656
+    // Assign 'this' to a variable for use in the tooltip closure below.
657
+    var thisBlock = this;
658
+    this.setTooltip(function() {
659
+      return (thisBlock.getFieldValue('TYPE') == 'TEXT') ?
660
+          Blockly.Msg.TEXT_PROMPT_TOOLTIP_TEXT :
661
+          Blockly.Msg.TEXT_PROMPT_TOOLTIP_NUMBER;
662
+    });
663
+  }
664
+};

+ 140 - 0
src/blockly/blocks/variables.js

@@ -0,0 +1,140 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Variable blocks for Blockly.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Blocks.variables');
28
+
29
+goog.require('Blockly.Blocks');
30
+
31
+
32
+/**
33
+ * Common HSV hue for all blocks in this category.
34
+ */
35
+Blockly.Blocks.variables.HUE = 330;
36
+
37
+Blockly.Blocks['variables_get'] = {
38
+  /**
39
+   * Block for variable getter.
40
+   * @this Blockly.Block
41
+   */
42
+  init: function() {
43
+    this.setHelpUrl(Blockly.Msg.VARIABLES_GET_HELPURL);
44
+    this.setColour(Blockly.Blocks.variables.HUE);
45
+    this.appendDummyInput()
46
+        .appendField(new Blockly.FieldVariable(
47
+        Blockly.Msg.VARIABLES_DEFAULT_NAME), 'VAR');
48
+    this.setOutput(true);
49
+    this.setTooltip(Blockly.Msg.VARIABLES_GET_TOOLTIP);
50
+    this.contextMenuMsg_ = Blockly.Msg.VARIABLES_GET_CREATE_SET;
51
+  },
52
+  /**
53
+   * Return all variables referenced by this block.
54
+   * @return {!Array.<string>} List of variable names.
55
+   * @this Blockly.Block
56
+   */
57
+  getVars: function() {
58
+    return [this.getFieldValue('VAR')];
59
+  },
60
+  /**
61
+   * Notification that a variable is renaming.
62
+   * If the name matches one of this block's variables, rename it.
63
+   * @param {string} oldName Previous name of variable.
64
+   * @param {string} newName Renamed variable.
65
+   * @this Blockly.Block
66
+   */
67
+  renameVar: function(oldName, newName) {
68
+    if (Blockly.Names.equals(oldName, this.getFieldValue('VAR'))) {
69
+      this.setFieldValue(newName, 'VAR');
70
+    }
71
+  },
72
+  contextMenuType_: 'variables_set',
73
+  /**
74
+   * Add menu option to create getter/setter block for this setter/getter.
75
+   * @param {!Array} options List of menu options to add to.
76
+   * @this Blockly.Block
77
+   */
78
+  customContextMenu: function(options) {
79
+    var option = {enabled: true};
80
+    var name = this.getFieldValue('VAR');
81
+    option.text = this.contextMenuMsg_.replace('%1', name);
82
+    var xmlField = goog.dom.createDom('field', null, name);
83
+    xmlField.setAttribute('name', 'VAR');
84
+    var xmlBlock = goog.dom.createDom('block', null, xmlField);
85
+    xmlBlock.setAttribute('type', this.contextMenuType_);
86
+    option.callback = Blockly.ContextMenu.callbackFactory(this, xmlBlock);
87
+    options.push(option);
88
+  }
89
+};
90
+
91
+Blockly.Blocks['variables_set'] = {
92
+  /**
93
+   * Block for variable setter.
94
+   * @this Blockly.Block
95
+   */
96
+  init: function() {
97
+    this.jsonInit({
98
+      "message0": Blockly.Msg.VARIABLES_SET,
99
+      "args0": [
100
+        {
101
+          "type": "field_variable",
102
+          "name": "VAR",
103
+          "variable": Blockly.Msg.VARIABLES_DEFAULT_NAME
104
+        },
105
+        {
106
+          "type": "input_value",
107
+          "name": "VALUE"
108
+        }
109
+      ],
110
+      "previousStatement": null,
111
+      "nextStatement": null,
112
+      "colour": Blockly.Blocks.variables.HUE,
113
+      "tooltip": Blockly.Msg.VARIABLES_SET_TOOLTIP,
114
+      "helpUrl": Blockly.Msg.VARIABLES_SET_HELPURL
115
+    });
116
+    this.contextMenuMsg_ = Blockly.Msg.VARIABLES_SET_CREATE_GET;
117
+  },
118
+  /**
119
+   * Return all variables referenced by this block.
120
+   * @return {!Array.<string>} List of variable names.
121
+   * @this Blockly.Block
122
+   */
123
+  getVars: function() {
124
+    return [this.getFieldValue('VAR')];
125
+  },
126
+  /**
127
+   * Notification that a variable is renaming.
128
+   * If the name matches one of this block's variables, rename it.
129
+   * @param {string} oldName Previous name of variable.
130
+   * @param {string} newName Renamed variable.
131
+   * @this Blockly.Block
132
+   */
133
+  renameVar: function(oldName, newName) {
134
+    if (Blockly.Names.equals(oldName, this.getFieldValue('VAR'))) {
135
+      this.setFieldValue(newName, 'VAR');
136
+    }
137
+  },
138
+  contextMenuType_: 'variables_get',
139
+  customContextMenu: Blockly.Blocks['variables_get'].customContextMenu
140
+};

File diff suppressed because it is too large
+ 204 - 0
src/blockly/blocks_compressed.js


+ 440 - 0
src/blockly/build.py

@@ -0,0 +1,440 @@
1
+#!/usr/bin/python2.7
2
+# Compresses the core Blockly files into a single JavaScript file.
3
+#
4
+# Copyright 2012 Google Inc.
5
+# https://developers.google.com/blockly/
6
+#
7
+# Licensed under the Apache License, Version 2.0 (the "License");
8
+# you may not use this file except in compliance with the License.
9
+# You may obtain a copy of the License at
10
+#
11
+#   http://www.apache.org/licenses/LICENSE-2.0
12
+#
13
+# Unless required by applicable law or agreed to in writing, software
14
+# distributed under the License is distributed on an "AS IS" BASIS,
15
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+# See the License for the specific language governing permissions and
17
+# limitations under the License.
18
+
19
+# This script generates two versions of Blockly's core files:
20
+#   blockly_compressed.js
21
+#   blockly_uncompressed.js
22
+# The compressed file is a concatenation of all of Blockly's core files which
23
+# have been run through Google's Closure Compiler.  This is done using the
24
+# online API (which takes a few seconds and requires an Internet connection).
25
+# The uncompressed file is a script that loads in each of Blockly's core files
26
+# one by one.  This takes much longer for a browser to load, but is useful
27
+# when debugging code since line numbers are meaningful and variables haven't
28
+# been renamed.  The uncompressed file also allows for a faster developement
29
+# cycle since there is no need to rebuild or recompile, just reload.
30
+#
31
+# This script also generates:
32
+#   blocks_compressed.js: The compressed Blockly language blocks.
33
+#   javascript_compressed.js: The compressed Javascript generator.
34
+#   python_compressed.js: The compressed Python generator.
35
+#   dart_compressed.js: The compressed Dart generator.
36
+#   msg/js/<LANG>.js for every language <LANG> defined in msg/js/<LANG>.json.
37
+
38
+import sys
39
+if sys.version_info[0] != 2:
40
+  raise Exception("Blockly build only compatible with Python 2.x.\n"
41
+                  "You are using: " + sys.version)
42
+
43
+import errno, glob, httplib, json, os, re, subprocess, threading, urllib
44
+
45
+
46
+def import_path(fullpath):
47
+  """Import a file with full path specification.
48
+  Allows one to import from any directory, something __import__ does not do.
49
+
50
+  Args:
51
+      fullpath:  Path and filename of import.
52
+
53
+  Returns:
54
+      An imported module.
55
+  """
56
+  path, filename = os.path.split(fullpath)
57
+  filename, ext = os.path.splitext(filename)
58
+  sys.path.append(path)
59
+  module = __import__(filename)
60
+  reload(module)  # Might be out of date.
61
+  del sys.path[-1]
62
+  return module
63
+
64
+
65
+HEADER = ("// Do not edit this file; automatically generated by build.py.\n"
66
+          "'use strict';\n")
67
+
68
+
69
+class Gen_uncompressed(threading.Thread):
70
+  """Generate a JavaScript file that loads Blockly's raw files.
71
+  Runs in a separate thread.
72
+  """
73
+  def __init__(self, search_paths):
74
+    threading.Thread.__init__(self)
75
+    self.search_paths = search_paths
76
+
77
+  def run(self):
78
+    target_filename = "blockly_uncompressed.js"
79
+    f = open(target_filename, "w")
80
+    f.write(HEADER)
81
+    f.write("""
82
+// 'this' is 'window' in a browser, or 'global' in node.js.
83
+this.BLOCKLY_DIR = (function() {
84
+  // Find name of current directory.
85
+  var scripts = document.getElementsByTagName('script');
86
+  var re = new RegExp('(.+)[\/]blockly_uncompressed\.js$');
87
+  for (var x = 0, script; script = scripts[x]; x++) {
88
+    var match = re.exec(script.src);
89
+    if (match) {
90
+      return match[1];
91
+    }
92
+  }
93
+  alert('Could not detect Blockly\\'s directory name.');
94
+  return '';
95
+})();
96
+
97
+this.BLOCKLY_BOOT = function() {
98
+// Execute after Closure has loaded.
99
+if (!this.goog) {
100
+  alert('Error: Closure not found.  Read this:\\n' +
101
+        'developers.google.com/blockly/hacking/closure');
102
+}
103
+
104
+// Build map of all dependencies (used and unused).
105
+var dir = this.BLOCKLY_DIR.match(/[^\\/]+$/)[0];
106
+""")
107
+    add_dependency = []
108
+    base_path = calcdeps.FindClosureBasePath(self.search_paths)
109
+    for dep in calcdeps.BuildDependenciesFromFiles(self.search_paths):
110
+      add_dependency.append(calcdeps.GetDepsLine(dep, base_path))
111
+    add_dependency = "\n".join(add_dependency)
112
+    # Find the Blockly directory name and replace it with a JS variable.
113
+    # This allows blockly_uncompressed.js to be compiled on one computer and be
114
+    # used on another, even if the directory name differs.
115
+    m = re.search("[\\/]([^\\/]+)[\\/]core[\\/]blockly.js", add_dependency)
116
+    add_dependency = re.sub("([\\/])" + re.escape(m.group(1)) +
117
+        "([\\/]core[\\/])", '\\1" + dir + "\\2', add_dependency)
118
+    f.write(add_dependency + "\n")
119
+
120
+    provides = []
121
+    for dep in calcdeps.BuildDependenciesFromFiles(self.search_paths):
122
+      if not dep.filename.startswith(os.pardir + os.sep):  # "../"
123
+        provides.extend(dep.provides)
124
+    provides.sort()
125
+    f.write("\n")
126
+    f.write("// Load Blockly.\n")
127
+    for provide in provides:
128
+      f.write("goog.require('%s');\n" % provide)
129
+
130
+    f.write("""
131
+delete this.BLOCKLY_DIR;
132
+delete this.BLOCKLY_BOOT;
133
+};
134
+
135
+if (typeof DOMParser == 'undefined' && typeof require == 'function') {
136
+  // Node.js needs DOMParser loaded separately.
137
+  var DOMParser = require('xmldom').DOMParser;
138
+}
139
+
140
+// Delete any existing Closure (e.g. Soy's nogoog_shim).
141
+document.write('<script>var goog = undefined;</script>');
142
+// Load fresh Closure Library.
143
+document.write('<script src="' + this.BLOCKLY_DIR +
144
+    '/../closure-library/closure/goog/base.js"></script>');
145
+document.write('<script>this.BLOCKLY_BOOT()</script>');
146
+""")
147
+    f.close()
148
+    print("SUCCESS: " + target_filename)
149
+
150
+
151
+class Gen_compressed(threading.Thread):
152
+  """Generate a JavaScript file that contains all of Blockly's core and all
153
+  required parts of Closure, compiled together.
154
+  Uses the Closure Compiler's online API.
155
+  Runs in a separate thread.
156
+  """
157
+  def __init__(self, search_paths):
158
+    threading.Thread.__init__(self)
159
+    self.search_paths = search_paths
160
+
161
+  def run(self):
162
+    self.gen_core()
163
+    self.gen_blocks()
164
+    self.gen_generator("arduino")
165
+
166
+  def gen_core(self):
167
+    target_filename = "blockly_compressed.js"
168
+    # Define the parameters for the POST request.
169
+    params = [
170
+        ("compilation_level", "SIMPLE_OPTIMIZATIONS"),
171
+        ("use_closure_library", "true"),
172
+        ("output_format", "json"),
173
+        ("output_info", "compiled_code"),
174
+        ("output_info", "warnings"),
175
+        ("output_info", "errors"),
176
+        ("output_info", "statistics"),
177
+      ]
178
+
179
+    # Read in all the source files.
180
+    filenames = calcdeps.CalculateDependencies(self.search_paths,
181
+        [os.path.join("core", "blockly.js")])
182
+    for filename in filenames:
183
+      # Filter out the Closure files (the compiler will add them).
184
+      if filename.startswith(os.pardir + os.sep):  # '../'
185
+        continue
186
+      f = open(filename)
187
+      params.append(("js_code", "".join(f.readlines())))
188
+      f.close()
189
+
190
+    self.do_compile(params, target_filename, filenames, "")
191
+
192
+  def gen_blocks(self):
193
+    target_filename = "blocks_compressed.js"
194
+    # Define the parameters for the POST request.
195
+    params = [
196
+        ("compilation_level", "SIMPLE_OPTIMIZATIONS"),
197
+        ("output_format", "json"),
198
+        ("output_info", "compiled_code"),
199
+        ("output_info", "warnings"),
200
+        ("output_info", "errors"),
201
+        ("output_info", "statistics"),
202
+      ]
203
+
204
+    # Read in all the source files.
205
+    # Add Blockly.Blocks to be compatible with the compiler.
206
+    params.append(("js_code", "goog.provide('Blockly.Blocks');"))
207
+    filenames = glob.glob(os.path.join("blocks", "*.js"))
208
+    for filename in filenames:
209
+      f = open(filename)
210
+      params.append(("js_code", "".join(f.readlines())))
211
+      f.close()
212
+
213
+    # Remove Blockly.Blocks to be compatible with Blockly.
214
+    remove = "var Blockly={Blocks:{}};"
215
+    self.do_compile(params, target_filename, filenames, remove)
216
+
217
+  def gen_generator(self, language):
218
+    target_filename = language + "_compressed.js"
219
+    # Define the parameters for the POST request.
220
+    params = [
221
+        ("compilation_level", "SIMPLE_OPTIMIZATIONS"),
222
+        ("output_format", "json"),
223
+        ("output_info", "compiled_code"),
224
+        ("output_info", "warnings"),
225
+        ("output_info", "errors"),
226
+        ("output_info", "statistics"),
227
+      ]
228
+
229
+    # Read in all the source files.
230
+    # Add Blockly.Generator to be compatible with the compiler.
231
+    params.append(("js_code", "goog.provide('Blockly.Generator');"))
232
+    filenames = glob.glob(
233
+        os.path.join("generators", language, "*.js"))
234
+    filenames.insert(0, os.path.join("generators", language + ".js"))
235
+    for filename in filenames:
236
+      f = open(filename)
237
+      params.append(("js_code", "".join(f.readlines())))
238
+      f.close()
239
+    filenames.insert(0, "[goog.provide]")
240
+
241
+    # Remove Blockly.Generator to be compatible with Blockly.
242
+    remove = "var Blockly={Generator:{}};"
243
+    self.do_compile(params, target_filename, filenames, remove)
244
+
245
+  def do_compile(self, params, target_filename, filenames, remove):
246
+    # Send the request to Google.
247
+    headers = {"Content-type": "application/x-www-form-urlencoded"}
248
+    conn = httplib.HTTPConnection("closure-compiler.appspot.com")
249
+    conn.request("POST", "/compile", urllib.urlencode(params), headers)
250
+    response = conn.getresponse()
251
+    json_str = response.read()
252
+    conn.close()
253
+
254
+    # Parse the JSON response.
255
+    json_data = json.loads(json_str)
256
+
257
+    def file_lookup(name):
258
+      if not name.startswith("Input_"):
259
+        return "???"
260
+      n = int(name[6:]) - 1
261
+      return filenames[n]
262
+
263
+    if json_data.has_key("serverErrors"):
264
+      errors = json_data["serverErrors"]
265
+      for error in errors:
266
+        print("SERVER ERROR: %s" % target_filename)
267
+        print(error["error"])
268
+    elif json_data.has_key("errors"):
269
+      errors = json_data["errors"]
270
+      for error in errors:
271
+        print("FATAL ERROR")
272
+        print(error["error"])
273
+        if error["file"]:
274
+          print("%s at line %d:" % (
275
+              file_lookup(error["file"]), error["lineno"]))
276
+          print(error["line"])
277
+          print((" " * error["charno"]) + "^")
278
+        sys.exit(1)
279
+    else:
280
+      if json_data.has_key("warnings"):
281
+        warnings = json_data["warnings"]
282
+        for warning in warnings:
283
+          print("WARNING")
284
+          print(warning["warning"])
285
+          if warning["file"]:
286
+            print("%s at line %d:" % (
287
+                file_lookup(warning["file"]), warning["lineno"]))
288
+            print(warning["line"])
289
+            print((" " * warning["charno"]) + "^")
290
+        print()
291
+
292
+      if not json_data.has_key("compiledCode"):
293
+        print("FATAL ERROR: Compiler did not return compiledCode.")
294
+        sys.exit(1)
295
+
296
+      code = HEADER + "\n" + json_data["compiledCode"]
297
+      code = code.replace(remove, "")
298
+
299
+      # Trim down Google's Apache licences.
300
+      LICENSE = re.compile("""/\\*
301
+
302
+ [\w ]+
303
+
304
+ (Copyright \\d+ Google Inc.)
305
+ https://developers.google.com/blockly/
306
+
307
+ Licensed under the Apache License, Version 2.0 \(the "License"\);
308
+ you may not use this file except in compliance with the License.
309
+ You may obtain a copy of the License at
310
+
311
+   http://www.apache.org/licenses/LICENSE-2.0
312
+
313
+ Unless required by applicable law or agreed to in writing, software
314
+ distributed under the License is distributed on an "AS IS" BASIS,
315
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
316
+ See the License for the specific language governing permissions and
317
+ limitations under the License.
318
+\\*/""")
319
+      code = re.sub(LICENSE, r"\n// \1  Apache License 2.0", code)
320
+
321
+      stats = json_data["statistics"]
322
+      original_b = stats["originalSize"]
323
+      compressed_b = stats["compressedSize"]
324
+      if original_b > 0 and compressed_b > 0:
325
+        f = open(target_filename, "w")
326
+        f.write(code)
327
+        f.close()
328
+
329
+        original_kb = int(original_b / 1024 + 0.5)
330
+        compressed_kb = int(compressed_b / 1024 + 0.5)
331
+        ratio = int(float(compressed_b) / float(original_b) * 100 + 0.5)
332
+        print("SUCCESS: " + target_filename)
333
+        print("Size changed from %d KB to %d KB (%d%%)." % (
334
+            original_kb, compressed_kb, ratio))
335
+      else:
336
+        print("UNKNOWN ERROR")
337
+
338
+
339
+class Gen_langfiles(threading.Thread):
340
+  """Generate JavaScript file for each natural language supported.
341
+
342
+  Runs in a separate thread.
343
+  """
344
+
345
+  def __init__(self):
346
+    threading.Thread.__init__(self)
347
+
348
+  def _rebuild(self, srcs, dests):
349
+    # Determine whether any of the files in srcs is newer than any in dests.
350
+    try:
351
+      return (max(os.path.getmtime(src) for src in srcs) >
352
+              min(os.path.getmtime(dest) for dest in dests))
353
+    except OSError as e:
354
+      # Was a file not found?
355
+      if e.errno == errno.ENOENT:
356
+        # If it was a source file, we can't proceed.
357
+        if e.filename in srcs:
358
+          print("Source file missing: " + e.filename)
359
+          sys.exit(1)
360
+        else:
361
+          # If a destination file was missing, rebuild.
362
+          return True
363
+      else:
364
+        print("Error checking file creation times: " + e)
365
+
366
+  def run(self):
367
+    # The files msg/json/{en,qqq,synonyms}.json depend on msg/messages.js.
368
+    if self._rebuild([os.path.join("msg", "messages.js")],
369
+                     [os.path.join("msg", "json", f) for f in
370
+                      ["en.json", "qqq.json", "synonyms.json"]]):
371
+      try:
372
+        subprocess.check_call([
373
+            "python",
374
+            os.path.join("i18n", "js_to_json.py"),
375
+            "--input_file", "msg/messages.js",
376
+            "--output_dir", "msg/json/",
377
+            "--quiet"])
378
+      except (subprocess.CalledProcessError, OSError) as e:
379
+        # Documentation for subprocess.check_call says that CalledProcessError
380
+        # will be raised on failure, but I found that OSError is also possible.
381
+        print("Error running i18n/js_to_json.py: ", e)
382
+        sys.exit(1)
383
+
384
+    # Checking whether it is necessary to rebuild the js files would be a lot of
385
+    # work since we would have to compare each <lang>.json file with each
386
+    # <lang>.js file.  Rebuilding is easy and cheap, so just go ahead and do it.
387
+    try:
388
+      # Use create_messages.py to create .js files from .json files.
389
+      cmd = [
390
+          "python",
391
+          os.path.join("i18n", "create_messages.py"),
392
+          "--source_lang_file", os.path.join("msg", "json", "en.json"),
393
+          "--source_synonym_file", os.path.join("msg", "json", "synonyms.json"),
394
+          "--key_file", os.path.join("msg", "json", "keys.json"),
395
+          "--output_dir", os.path.join("msg", "js"),
396
+          "--quiet"]
397
+      json_files = glob.glob(os.path.join("msg", "json", "*.json"))
398
+      json_files = [file for file in json_files if not
399
+                    (file.endswith(("keys.json", "synonyms.json", "qqq.json")))]
400
+      cmd.extend(json_files)
401
+      subprocess.check_call(cmd)
402
+    except (subprocess.CalledProcessError, OSError) as e:
403
+      print("Error running i18n/create_messages.py: ", e)
404
+      sys.exit(1)
405
+
406
+    # Output list of .js files created.
407
+    for f in json_files:
408
+      # This assumes the path to the current directory does not contain "json".
409
+      f = f.replace("json", "js")
410
+      if os.path.isfile(f):
411
+        print("SUCCESS: " + f)
412
+      else:
413
+        print("FAILED to create " + f)
414
+
415
+
416
+if __name__ == "__main__":
417
+  try:
418
+    calcdeps = import_path(os.path.join(
419
+        os.path.pardir, "closure-library", "closure", "bin", "calcdeps.py"))
420
+  except ImportError:
421
+    if os.path.isdir(os.path.join(os.path.pardir, "closure-library-read-only")):
422
+      # Dir got renamed when Closure moved from Google Code to GitHub in 2014.
423
+      print("Error: Closure directory needs to be renamed from"
424
+            "'closure-library-read-only' to 'closure-library'.\n"
425
+            "Please rename this directory.")
426
+    else:
427
+      print("""Error: Closure not found.  Read this:
428
+https://developers.google.com/blockly/hacking/closure""")
429
+    sys.exit(1)
430
+  search_paths = calcdeps.ExpandDirectories(
431
+      ["core", os.path.join(os.path.pardir, "closure-library")])
432
+
433
+  # Run both tasks in parallel threads.
434
+  # Uncompressed is limited by processor speed.
435
+  # Compressed is limited by network and server speed.
436
+  Gen_uncompressed(search_paths).start()
437
+  Gen_compressed(search_paths).start()
438
+
439
+  # This is run locally in a separate thread.
440
+  Gen_langfiles().start()

File diff suppressed because it is too large
+ 1258 - 0
src/blockly/core/block.js


File diff suppressed because it is too large
+ 2247 - 0
src/blockly/core/block_svg.js


+ 652 - 0
src/blockly/core/blockly.js

@@ -0,0 +1,652 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2011 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Core JavaScript library for Blockly.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+// Top level object for Blockly.
28
+goog.provide('Blockly');
29
+
30
+goog.require('Blockly.BlockSvg');
31
+goog.require('Blockly.FieldAngle');
32
+goog.require('Blockly.FieldCheckbox');
33
+goog.require('Blockly.FieldColour');
34
+// Date picker commented out since it increases footprint by 60%.
35
+// Add it only if you need it.
36
+//goog.require('Blockly.FieldDate');
37
+goog.require('Blockly.FieldDropdown');
38
+goog.require('Blockly.FieldImage');
39
+goog.require('Blockly.FieldTextInput');
40
+goog.require('Blockly.FieldVariable');
41
+goog.require('Blockly.Generator');
42
+goog.require('Blockly.Msg');
43
+goog.require('Blockly.Procedures');
44
+// Realtime is currently badly broken.  Stub it out.
45
+//goog.require('Blockly.Realtime');
46
+Blockly.Realtime = {
47
+  isEnabled: function() {return false;},
48
+  blockChanged: function() {},
49
+  doCommand: function(cmdThunk) {cmdThunk();}
50
+};
51
+goog.require('Blockly.Toolbox');
52
+goog.require('Blockly.WidgetDiv');
53
+goog.require('Blockly.WorkspaceSvg');
54
+goog.require('Blockly.inject');
55
+goog.require('Blockly.utils');
56
+goog.require('goog.color');
57
+goog.require('goog.userAgent');
58
+
59
+
60
+// Turn off debugging when compiled.
61
+var CLOSURE_DEFINES = {'goog.DEBUG': false};
62
+
63
+/**
64
+ * Required name space for SVG elements.
65
+ * @const
66
+ */
67
+Blockly.SVG_NS = 'http://www.w3.org/2000/svg';
68
+/**
69
+ * Required name space for HTML elements.
70
+ * @const
71
+ */
72
+Blockly.HTML_NS = 'http://www.w3.org/1999/xhtml';
73
+
74
+/**
75
+ * The richness of block colours, regardless of the hue.
76
+ * Must be in the range of 0 (inclusive) to 1 (exclusive).
77
+ */
78
+Blockly.HSV_SATURATION = 0.45;
79
+/**
80
+ * The intensity of block colours, regardless of the hue.
81
+ * Must be in the range of 0 (inclusive) to 1 (exclusive).
82
+ */
83
+Blockly.HSV_VALUE = 0.65;
84
+
85
+/**
86
+ * Sprited icons and images.
87
+ */
88
+Blockly.SPRITE = {
89
+  width: 96,
90
+  height: 124,
91
+  url: 'sprites.png'
92
+};
93
+
94
+/**
95
+ * Convert a hue (HSV model) into an RGB hex triplet.
96
+ * @param {number} hue Hue on a colour wheel (0-360).
97
+ * @return {string} RGB code, e.g. '#5ba65b'.
98
+ */
99
+Blockly.makeColour = function(hue) {
100
+  return goog.color.hsvToHex(hue, Blockly.HSV_SATURATION,
101
+      Blockly.HSV_VALUE * 255);
102
+};
103
+
104
+/**
105
+ * ENUM for a right-facing value input.  E.g. 'set item to' or 'return'.
106
+ * @const
107
+ */
108
+Blockly.INPUT_VALUE = 1;
109
+/**
110
+ * ENUM for a left-facing value output.  E.g. 'random fraction'.
111
+ * @const
112
+ */
113
+Blockly.OUTPUT_VALUE = 2;
114
+/**
115
+ * ENUM for a down-facing block stack.  E.g. 'if-do' or 'else'.
116
+ * @const
117
+ */
118
+Blockly.NEXT_STATEMENT = 3;
119
+/**
120
+ * ENUM for an up-facing block stack.  E.g. 'break out of loop'.
121
+ * @const
122
+ */
123
+Blockly.PREVIOUS_STATEMENT = 4;
124
+/**
125
+ * ENUM for an dummy input.  Used to add field(s) with no input.
126
+ * @const
127
+ */
128
+Blockly.DUMMY_INPUT = 5;
129
+
130
+/**
131
+ * ENUM for left alignment.
132
+ * @const
133
+ */
134
+Blockly.ALIGN_LEFT = -1;
135
+/**
136
+ * ENUM for centre alignment.
137
+ * @const
138
+ */
139
+Blockly.ALIGN_CENTRE = 0;
140
+/**
141
+ * ENUM for right alignment.
142
+ * @const
143
+ */
144
+Blockly.ALIGN_RIGHT = 1;
145
+
146
+/**
147
+ * Lookup table for determining the opposite type of a connection.
148
+ * @const
149
+ */
150
+Blockly.OPPOSITE_TYPE = [];
151
+Blockly.OPPOSITE_TYPE[Blockly.INPUT_VALUE] = Blockly.OUTPUT_VALUE;
152
+Blockly.OPPOSITE_TYPE[Blockly.OUTPUT_VALUE] = Blockly.INPUT_VALUE;
153
+Blockly.OPPOSITE_TYPE[Blockly.NEXT_STATEMENT] = Blockly.PREVIOUS_STATEMENT;
154
+Blockly.OPPOSITE_TYPE[Blockly.PREVIOUS_STATEMENT] = Blockly.NEXT_STATEMENT;
155
+
156
+/**
157
+ * Currently selected block.
158
+ * @type {Blockly.Block}
159
+ */
160
+Blockly.selected = null;
161
+
162
+/**
163
+ * Currently highlighted connection (during a drag).
164
+ * @type {Blockly.Connection}
165
+ * @private
166
+ */
167
+Blockly.highlightedConnection_ = null;
168
+
169
+/**
170
+ * Connection on dragged block that matches the highlighted connection.
171
+ * @type {Blockly.Connection}
172
+ * @private
173
+ */
174
+Blockly.localConnection_ = null;
175
+
176
+/**
177
+ * Number of pixels the mouse must move before a drag starts.
178
+ */
179
+Blockly.DRAG_RADIUS = 5;
180
+
181
+/**
182
+ * Maximum misalignment between connections for them to snap together.
183
+ */
184
+Blockly.SNAP_RADIUS = 20;
185
+
186
+/**
187
+ * Delay in ms between trigger and bumping unconnected block out of alignment.
188
+ */
189
+Blockly.BUMP_DELAY = 250;
190
+
191
+/**
192
+ * Number of characters to truncate a collapsed block to.
193
+ */
194
+Blockly.COLLAPSE_CHARS = 30;
195
+
196
+/**
197
+ * Length in ms for a touch to become a long press.
198
+ */
199
+Blockly.LONGPRESS = 750;
200
+
201
+/**
202
+ * The main workspace most recently used.
203
+ * Set by Blockly.WorkspaceSvg.prototype.markFocused
204
+ * @type {Blockly.Workspace}
205
+ */
206
+Blockly.mainWorkspace = null;
207
+
208
+/**
209
+ * Contents of the local clipboard.
210
+ * @type {Element}
211
+ * @private
212
+ */
213
+Blockly.clipboardXml_ = null;
214
+
215
+/**
216
+ * Source of the local clipboard.
217
+ * @type {Blockly.WorkspaceSvg}
218
+ * @private
219
+ */
220
+Blockly.clipboardSource_ = null;
221
+
222
+/**
223
+ * Is the mouse dragging a block?
224
+ * 0 - No drag operation.
225
+ * 1 - Still inside the sticky DRAG_RADIUS.
226
+ * 2 - Freely draggable.
227
+ * @private
228
+ */
229
+Blockly.dragMode_ = 0;
230
+
231
+/**
232
+ * Wrapper function called when a touch mouseUp occurs during a drag operation.
233
+ * @type {Array.<!Array>}
234
+ * @private
235
+ */
236
+Blockly.onTouchUpWrapper_ = null;
237
+
238
+/**
239
+ * Returns the dimensions of the specified SVG image.
240
+ * @param {!Element} svg SVG image.
241
+ * @return {!Object} Contains width and height properties.
242
+ */
243
+Blockly.svgSize = function(svg) {
244
+  return {width: svg.cachedWidth_,
245
+          height: svg.cachedHeight_};
246
+};
247
+
248
+/**
249
+ * Size the SVG image to completely fill its container.
250
+ * Record the height/width of the SVG image.
251
+ * @param {!Blockly.WorkspaceSvg} workspace Any workspace in the SVG.
252
+ */
253
+Blockly.svgResize = function(workspace) {
254
+  var mainWorkspace = workspace;
255
+  while (mainWorkspace.options.parentWorkspace) {
256
+    mainWorkspace = mainWorkspace.options.parentWorkspace;
257
+  }
258
+  var svg = mainWorkspace.options.svg;
259
+  var div = svg.parentNode;
260
+  if (!div) {
261
+    // Workspace deteted, or something.
262
+    return;
263
+  }
264
+  var width = div.offsetWidth;
265
+  var height = div.offsetHeight;
266
+  if (svg.cachedWidth_ != width) {
267
+    svg.setAttribute('width', width + 'px');
268
+    svg.cachedWidth_ = width;
269
+  }
270
+  if (svg.cachedHeight_ != height) {
271
+    svg.setAttribute('height', height + 'px');
272
+    svg.cachedHeight_ = height;
273
+  }
274
+  mainWorkspace.resize();
275
+};
276
+
277
+/**
278
+ * Handle a mouse-up anywhere on the page.
279
+ * @param {!Event} e Mouse up event.
280
+ * @private
281
+ */
282
+Blockly.onMouseUp_ = function(e) {
283
+  var workspace = Blockly.getMainWorkspace();
284
+  Blockly.Css.setCursor(Blockly.Css.Cursor.OPEN);
285
+  workspace.isScrolling = false;
286
+
287
+  // Unbind the touch event if it exists.
288
+  if (Blockly.onTouchUpWrapper_) {
289
+    Blockly.unbindEvent_(Blockly.onTouchUpWrapper_);
290
+    Blockly.onTouchUpWrapper_ = null;
291
+  }
292
+  if (Blockly.onMouseMoveWrapper_) {
293
+    Blockly.unbindEvent_(Blockly.onMouseMoveWrapper_);
294
+    Blockly.onMouseMoveWrapper_ = null;
295
+  }
296
+};
297
+
298
+/**
299
+ * Handle a mouse-move on SVG drawing surface.
300
+ * @param {!Event} e Mouse move event.
301
+ * @private
302
+ */
303
+Blockly.onMouseMove_ = function(e) {
304
+  if (e.touches && e.touches.length >= 2) {
305
+    return;  // Multi-touch gestures won't have e.clientX.
306
+  }
307
+  var workspace = Blockly.getMainWorkspace();
308
+  if (workspace.isScrolling) {
309
+    Blockly.removeAllRanges();
310
+    var dx = e.clientX - workspace.startDragMouseX;
311
+    var dy = e.clientY - workspace.startDragMouseY;
312
+    var metrics = workspace.startDragMetrics;
313
+    var x = workspace.startScrollX + dx;
314
+    var y = workspace.startScrollY + dy;
315
+    x = Math.min(x, -metrics.contentLeft);
316
+    y = Math.min(y, -metrics.contentTop);
317
+    x = Math.max(x, metrics.viewWidth - metrics.contentLeft -
318
+                 metrics.contentWidth);
319
+    y = Math.max(y, metrics.viewHeight - metrics.contentTop -
320
+                 metrics.contentHeight);
321
+
322
+    // Move the scrollbars and the page will scroll automatically.
323
+    workspace.scrollbar.set(-x - metrics.contentLeft,
324
+                            -y - metrics.contentTop);
325
+    // Cancel the long-press if the drag has moved too far.
326
+    if (Math.sqrt(dx * dx + dy * dy) > Blockly.DRAG_RADIUS) {
327
+      Blockly.longStop_();
328
+    }
329
+    e.stopPropagation();
330
+  }
331
+};
332
+
333
+/**
334
+ * Handle a key-down on SVG drawing surface.
335
+ * @param {!Event} e Key down event.
336
+ * @private
337
+ */
338
+Blockly.onKeyDown_ = function(e) {
339
+  if (Blockly.isTargetInput_(e)) {
340
+    // When focused on an HTML text input widget, don't trap any keys.
341
+    return;
342
+  }
343
+  var deleteBlock = false;
344
+  if (e.keyCode == 27) {
345
+    // Pressing esc closes the context menu.
346
+    Blockly.hideChaff();
347
+  } else if (e.keyCode == 8 || e.keyCode == 46) {
348
+    // Delete or backspace.
349
+    try {
350
+      if (Blockly.selected && Blockly.selected.isDeletable()) {
351
+        deleteBlock = true;
352
+      }
353
+    } finally {
354
+      // Stop the browser from going back to the previous page.
355
+      // Use a finally so that any error in delete code above doesn't disappear
356
+      // from the console when the page rolls back.
357
+      e.preventDefault();
358
+    }
359
+  } else if (e.altKey || e.ctrlKey || e.metaKey) {
360
+    if (Blockly.selected &&
361
+        Blockly.selected.isDeletable() && Blockly.selected.isMovable()) {
362
+      if (e.keyCode == 67) {
363
+        // 'c' for copy.
364
+        Blockly.hideChaff();
365
+        Blockly.copy_(Blockly.selected);
366
+      } else if (e.keyCode == 88) {
367
+        // 'x' for cut.
368
+        Blockly.copy_(Blockly.selected);
369
+        deleteBlock = true;
370
+      }
371
+    }
372
+    if (e.keyCode == 86) {
373
+      // 'v' for paste.
374
+      if (Blockly.clipboardXml_) {
375
+        Blockly.clipboardSource_.paste(Blockly.clipboardXml_);
376
+      }
377
+    }
378
+  }
379
+  if (deleteBlock) {
380
+    // Common code for delete and cut.
381
+    Blockly.hideChaff();
382
+    var heal = Blockly.dragMode_ != 2;
383
+    Blockly.selected.dispose(heal, true);
384
+    if (Blockly.highlightedConnection_) {
385
+      Blockly.highlightedConnection_.unhighlight();
386
+      Blockly.highlightedConnection_ = null;
387
+    }
388
+  }
389
+};
390
+
391
+/**
392
+ * Stop binding to the global mouseup and mousemove events.
393
+ * @private
394
+ */
395
+Blockly.terminateDrag_ = function() {
396
+  Blockly.BlockSvg.terminateDrag_();
397
+  Blockly.Flyout.terminateDrag_();
398
+};
399
+
400
+/**
401
+ * PID of queued long-press task.
402
+ * @private
403
+ */
404
+Blockly.longPid_ = 0;
405
+
406
+/**
407
+ * Context menus on touch devices are activated using a long-press.
408
+ * Unfortunately the contextmenu touch event is currently (2015) only suported
409
+ * by Chrome.  This function is fired on any touchstart event, queues a task,
410
+ * which after about a second opens the context menu.  The tasks is killed
411
+ * if the touch event terminates early.
412
+ * @param {!Event} e Touch start event.
413
+ * @param {!Blockly.Block|!Blockly.WorkspaceSvg} uiObject The block or workspace
414
+ *   under the touchstart event.
415
+ * @private
416
+ */
417
+Blockly.longStart_ = function(e, uiObject) {
418
+  Blockly.longStop_();
419
+  Blockly.longPid_ = setTimeout(function() {
420
+      e.button = 2;  // Simulate a right button click.
421
+      uiObject.onMouseDown_(e);
422
+    }, Blockly.LONGPRESS);
423
+};
424
+
425
+/**
426
+ * Nope, that's not a long-press.  Either touchend or touchcancel was fired,
427
+ * or a drag hath begun.  Kill the queued long-press task.
428
+ * @private
429
+ */
430
+Blockly.longStop_ = function() {
431
+  if (Blockly.longPid_) {
432
+    clearTimeout(Blockly.longPid_);
433
+    Blockly.longPid_ = 0;
434
+  }
435
+};
436
+
437
+/**
438
+ * Copy a block onto the local clipboard.
439
+ * @param {!Blockly.Block} block Block to be copied.
440
+ * @private
441
+ */
442
+Blockly.copy_ = function(block) {
443
+  var xmlBlock = Blockly.Xml.blockToDom_(block);
444
+  if (Blockly.dragMode_ != 2) {
445
+    Blockly.Xml.deleteNext(xmlBlock);
446
+  }
447
+  // Encode start position in XML.
448
+  var xy = block.getRelativeToSurfaceXY();
449
+  xmlBlock.setAttribute('x', block.RTL ? -xy.x : xy.x);
450
+  xmlBlock.setAttribute('y', xy.y);
451
+  Blockly.clipboardXml_ = xmlBlock;
452
+  Blockly.clipboardSource_ = block.workspace;
453
+};
454
+
455
+/**
456
+ * Duplicate this block and its children.
457
+ * @param {!Blockly.Block} block Block to be copied.
458
+ * @private
459
+ */
460
+Blockly.duplicate_ = function(block) {
461
+  // Save the clipboard.
462
+  var clipboardXml = Blockly.clipboardXml_;
463
+  var clipboardSource = Blockly.clipboardSource_;
464
+
465
+  // Create a duplicate via a copy/paste operation.
466
+  Blockly.copy_(block);
467
+  block.workspace.paste(Blockly.clipboardXml_);
468
+
469
+  // Restore the clipboard.
470
+  Blockly.clipboardXml_ = clipboardXml;
471
+  Blockly.clipboardSource_ = clipboardSource;
472
+};
473
+
474
+/**
475
+ * Cancel the native context menu, unless the focus is on an HTML input widget.
476
+ * @param {!Event} e Mouse down event.
477
+ * @private
478
+ */
479
+Blockly.onContextMenu_ = function(e) {
480
+  if (!Blockly.isTargetInput_(e)) {
481
+    // When focused on an HTML text input widget, don't cancel the context menu.
482
+    e.preventDefault();
483
+  }
484
+};
485
+
486
+/**
487
+ * Close tooltips, context menus, dropdown selections, etc.
488
+ * @param {boolean=} opt_allowToolbox If true, don't close the toolbox.
489
+ */
490
+Blockly.hideChaff = function(opt_allowToolbox) {
491
+  Blockly.Tooltip.hide();
492
+  Blockly.WidgetDiv.hide();
493
+  if (!opt_allowToolbox) {
494
+    var workspace = Blockly.getMainWorkspace();
495
+    if (workspace.toolbox_ &&
496
+        workspace.toolbox_.flyout_ &&
497
+        workspace.toolbox_.flyout_.autoClose) {
498
+      workspace.toolbox_.clearSelection();
499
+    }
500
+  }
501
+};
502
+
503
+/**
504
+ * Return an object with all the metrics required to size scrollbars for the
505
+ * main workspace.  The following properties are computed:
506
+ * .viewHeight: Height of the visible rectangle,
507
+ * .viewWidth: Width of the visible rectangle,
508
+ * .contentHeight: Height of the contents,
509
+ * .contentWidth: Width of the content,
510
+ * .viewTop: Offset of top edge of visible rectangle from parent,
511
+ * .viewLeft: Offset of left edge of visible rectangle from parent,
512
+ * .contentTop: Offset of the top-most content from the y=0 coordinate,
513
+ * .contentLeft: Offset of the left-most content from the x=0 coordinate.
514
+ * .absoluteTop: Top-edge of view.
515
+ * .absoluteLeft: Left-edge of view.
516
+ * @return {Object} Contains size and position metrics of main workspace.
517
+ * @private
518
+ * @this Blockly.WorkspaceSvg
519
+ */
520
+Blockly.getMainWorkspaceMetrics_ = function() {
521
+  var svgSize = Blockly.svgSize(this.options.svg);
522
+  if (this.toolbox_) {
523
+    svgSize.width -= this.toolbox_.width;
524
+  }
525
+  // Set the margin to match the flyout's margin so that the workspace does
526
+  // not jump as blocks are added.
527
+  var MARGIN = Blockly.Flyout.prototype.CORNER_RADIUS - 1;
528
+  var viewWidth = svgSize.width - MARGIN;
529
+  var viewHeight = svgSize.height - MARGIN;
530
+  try {
531
+    var blockBox = this.getCanvas().getBBox();
532
+  } catch (e) {
533
+    // Firefox has trouble with hidden elements (Bug 528969).
534
+    return null;
535
+  }
536
+  // Fix scale.
537
+  var contentWidth = blockBox.width * this.scale;
538
+  var contentHeight = blockBox.height * this.scale;
539
+  var contentX = blockBox.x * this.scale;
540
+  var contentY = blockBox.y * this.scale;
541
+  if (this.scrollbar) {
542
+    // Add a border around the content that is at least half a screenful wide.
543
+    // Ensure border is wide enough that blocks can scroll over entire screen.
544
+    var leftEdge = Math.min(contentX - viewWidth / 2,
545
+                            contentX + contentWidth - viewWidth);
546
+    var rightEdge = Math.max(contentX + contentWidth + viewWidth / 2,
547
+                             contentX + viewWidth);
548
+    var topEdge = Math.min(contentY - viewHeight / 2,
549
+                           contentY + contentHeight - viewHeight);
550
+    var bottomEdge = Math.max(contentY + contentHeight + viewHeight / 2,
551
+                              contentY + viewHeight);
552
+  } else {
553
+    var leftEdge = blockBox.x;
554
+    var rightEdge = leftEdge + blockBox.width;
555
+    var topEdge = blockBox.y;
556
+    var bottomEdge = topEdge + blockBox.height;
557
+  }
558
+  var absoluteLeft = 0;
559
+  if (!this.RTL && this.toolbox_) {
560
+    absoluteLeft = this.toolbox_.width;
561
+  }
562
+  var metrics = {
563
+    viewHeight: svgSize.height,
564
+    viewWidth: svgSize.width,
565
+    contentHeight: bottomEdge - topEdge,
566
+    contentWidth: rightEdge - leftEdge,
567
+    viewTop: -this.scrollY,
568
+    viewLeft: -this.scrollX,
569
+    contentTop: topEdge,
570
+    contentLeft: leftEdge,
571
+    absoluteTop: 0,
572
+    absoluteLeft: absoluteLeft
573
+  };
574
+  return metrics;
575
+};
576
+
577
+/**
578
+ * Sets the X/Y translations of the main workspace to match the scrollbars.
579
+ * @param {!Object} xyRatio Contains an x and/or y property which is a float
580
+ *     between 0 and 1 specifying the degree of scrolling.
581
+ * @private
582
+ * @this Blockly.WorkspaceSvg
583
+ */
584
+Blockly.setMainWorkspaceMetrics_ = function(xyRatio) {
585
+  if (!this.scrollbar) {
586
+    throw 'Attempt to set main workspace scroll without scrollbars.';
587
+  }
588
+  var metrics = this.getMetrics();
589
+  if (goog.isNumber(xyRatio.x)) {
590
+    this.scrollX = -metrics.contentWidth * xyRatio.x - metrics.contentLeft;
591
+  }
592
+  if (goog.isNumber(xyRatio.y)) {
593
+    this.scrollY = -metrics.contentHeight * xyRatio.y - metrics.contentTop;
594
+  }
595
+  var x = this.scrollX + metrics.absoluteLeft;
596
+  var y = this.scrollY + metrics.absoluteTop;
597
+  this.translate(x, y);
598
+  if (this.options.gridPattern) {
599
+    this.options.gridPattern.setAttribute('x', x);
600
+    this.options.gridPattern.setAttribute('y', y);
601
+    if (goog.userAgent.IE) {
602
+      // IE doesn't notice that the x/y offsets have changed.  Force an update.
603
+      this.updateGridPattern_();
604
+    }
605
+  }
606
+};
607
+
608
+/**
609
+ * Execute a command.  Generally, a command is the result of a user action
610
+ * e.g., a click, drag or context menu selection.  Calling the cmdThunk function
611
+ * through doCommand() allows us to capture information that can be used for
612
+ * capabilities like undo (which is supported by the realtime collaboration
613
+ * feature).
614
+ * @param {function()} cmdThunk A function representing the command execution.
615
+ */
616
+Blockly.doCommand = function(cmdThunk) {
617
+  if (Blockly.Realtime.isEnabled) {
618
+    Blockly.Realtime.doCommand(cmdThunk);
619
+  } else {
620
+    cmdThunk();
621
+  }
622
+};
623
+
624
+/**
625
+ * When something in Blockly's workspace changes, call a function.
626
+ * @param {!Function} func Function to call.
627
+ * @return {!Array.<!Array>} Opaque data that can be passed to
628
+ *     removeChangeListener.
629
+ * @deprecated April 2015
630
+ */
631
+Blockly.addChangeListener = function(func) {
632
+  // Backwards compatability from before there could be multiple workspaces.
633
+  console.warn('Deprecated call to Blockly.addChangeListener, ' +
634
+               'use workspace.addChangeListener instead.');
635
+  return Blockly.getMainWorkspace().addChangeListener(func);
636
+};
637
+
638
+/**
639
+ * Returns the main workspace.  Returns the last used main workspace (based on
640
+ * focus).
641
+ * @return {!Blockly.Workspace} The main workspace.
642
+ */
643
+Blockly.getMainWorkspace = function() {
644
+  return Blockly.mainWorkspace;
645
+};
646
+
647
+// Export symbols that would otherwise be renamed by Closure compiler.
648
+if (!goog.global['Blockly']) {
649
+  goog.global['Blockly'] = {};
650
+}
651
+goog.global['Blockly']['getMainWorkspace'] = Blockly.getMainWorkspace;
652
+goog.global['Blockly']['addChangeListener'] = Blockly.addChangeListener;

+ 48 - 0
src/blockly/core/blocks.js

@@ -0,0 +1,48 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2013 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Name space for the Blocks singleton.
23
+ * @author spertus@google.com (Ellen Spertus)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Blocks');
28
+
29
+
30
+/**
31
+ * Unique ID counter for created blocks.
32
+ * @private
33
+ */
34
+Blockly.Blocks.uidCounter_ = 0;
35
+
36
+/**
37
+ * Generate a unique ID.  This will be locally or globally unique, depending on
38
+ * whether we are in single user or realtime collaborative mode.
39
+ * @return {string}
40
+ */
41
+Blockly.Blocks.genUid = function() {
42
+  var uid = (++Blockly.Blocks.uidCounter_).toString();
43
+  if (Blockly.Realtime.isEnabled()) {
44
+    return Blockly.Realtime.genUid(uid);
45
+  } else {
46
+    return uid;
47
+  }
48
+};

+ 578 - 0
src/blockly/core/bubble.js

@@ -0,0 +1,578 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Object representing a UI bubble.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Bubble');
28
+
29
+goog.require('Blockly.Workspace');
30
+goog.require('goog.dom');
31
+goog.require('goog.math');
32
+goog.require('goog.userAgent');
33
+
34
+
35
+/**
36
+ * Class for UI bubble.
37
+ * @param {!Blockly.Workspace} workspace The workspace on which to draw the
38
+ *     bubble.
39
+ * @param {!Element} content SVG content for the bubble.
40
+ * @param {Element} shape SVG element to avoid eclipsing.
41
+ * @param {number} anchorX Absolute horizontal position of bubbles anchor point.
42
+ * @param {number} anchorY Absolute vertical position of bubbles anchor point.
43
+ * @param {?number} bubbleWidth Width of bubble, or null if not resizable.
44
+ * @param {?number} bubbleHeight Height of bubble, or null if not resizable.
45
+ * @constructor
46
+ */
47
+Blockly.Bubble = function(workspace, content, shape,
48
+                          anchorX, anchorY,
49
+                          bubbleWidth, bubbleHeight) {
50
+  this.workspace_ = workspace;
51
+  this.content_ = content;
52
+  this.shape_ = shape;
53
+
54
+  var angle = Blockly.Bubble.ARROW_ANGLE;
55
+  if (this.workspace_.RTL) {
56
+    angle = -angle;
57
+  }
58
+  this.arrow_radians_ = goog.math.toRadians(angle);
59
+
60
+  var canvas = workspace.getBubbleCanvas();
61
+  canvas.appendChild(this.createDom_(content, !!(bubbleWidth && bubbleHeight)));
62
+
63
+  this.setAnchorLocation(anchorX, anchorY);
64
+  if (!bubbleWidth || !bubbleHeight) {
65
+    var bBox = /** @type {SVGLocatable} */ (this.content_).getBBox();
66
+    bubbleWidth = bBox.width + 2 * Blockly.Bubble.BORDER_WIDTH;
67
+    bubbleHeight = bBox.height + 2 * Blockly.Bubble.BORDER_WIDTH;
68
+  }
69
+  this.setBubbleSize(bubbleWidth, bubbleHeight);
70
+
71
+  // Render the bubble.
72
+  this.positionBubble_();
73
+  this.renderArrow_();
74
+  this.rendered_ = true;
75
+
76
+  if (!workspace.options.readOnly) {
77
+    Blockly.bindEvent_(this.bubbleBack_, 'mousedown', this,
78
+                       this.bubbleMouseDown_);
79
+    if (this.resizeGroup_) {
80
+      Blockly.bindEvent_(this.resizeGroup_, 'mousedown', this,
81
+                         this.resizeMouseDown_);
82
+    }
83
+  }
84
+};
85
+
86
+/**
87
+ * Width of the border around the bubble.
88
+ */
89
+Blockly.Bubble.BORDER_WIDTH = 6;
90
+
91
+/**
92
+ * Determines the thickness of the base of the arrow in relation to the size
93
+ * of the bubble.  Higher numbers result in thinner arrows.
94
+ */
95
+Blockly.Bubble.ARROW_THICKNESS = 10;
96
+
97
+/**
98
+ * The number of degrees that the arrow bends counter-clockwise.
99
+ */
100
+Blockly.Bubble.ARROW_ANGLE = 20;
101
+
102
+/**
103
+ * The sharpness of the arrow's bend.  Higher numbers result in smoother arrows.
104
+ */
105
+Blockly.Bubble.ARROW_BEND = 4;
106
+
107
+/**
108
+ * Distance between arrow point and anchor point.
109
+ */
110
+Blockly.Bubble.ANCHOR_RADIUS = 8;
111
+
112
+/**
113
+ * Wrapper function called when a mouseUp occurs during a drag operation.
114
+ * @type {Array.<!Array>}
115
+ * @private
116
+ */
117
+Blockly.Bubble.onMouseUpWrapper_ = null;
118
+
119
+/**
120
+ * Wrapper function called when a mouseMove occurs during a drag operation.
121
+ * @type {Array.<!Array>}
122
+ * @private
123
+ */
124
+Blockly.Bubble.onMouseMoveWrapper_ = null;
125
+
126
+/**
127
+ * Stop binding to the global mouseup and mousemove events.
128
+ * @private
129
+ */
130
+Blockly.Bubble.unbindDragEvents_ = function() {
131
+  if (Blockly.Bubble.onMouseUpWrapper_) {
132
+    Blockly.unbindEvent_(Blockly.Bubble.onMouseUpWrapper_);
133
+    Blockly.Bubble.onMouseUpWrapper_ = null;
134
+  }
135
+  if (Blockly.Bubble.onMouseMoveWrapper_) {
136
+    Blockly.unbindEvent_(Blockly.Bubble.onMouseMoveWrapper_);
137
+    Blockly.Bubble.onMouseMoveWrapper_ = null;
138
+  }
139
+};
140
+
141
+/**
142
+ * Flag to stop incremental rendering during construction.
143
+ * @private
144
+ */
145
+Blockly.Bubble.prototype.rendered_ = false;
146
+
147
+/**
148
+ * Absolute X coordinate of anchor point.
149
+ * @private
150
+ */
151
+Blockly.Bubble.prototype.anchorX_ = 0;
152
+
153
+/**
154
+ * Absolute Y coordinate of anchor point.
155
+ * @private
156
+ */
157
+Blockly.Bubble.prototype.anchorY_ = 0;
158
+
159
+/**
160
+ * Relative X coordinate of bubble with respect to the anchor's centre.
161
+ * In RTL mode the initial value is negated.
162
+ * @private
163
+ */
164
+Blockly.Bubble.prototype.relativeLeft_ = 0;
165
+
166
+/**
167
+ * Relative Y coordinate of bubble with respect to the anchor's centre.
168
+ * @private
169
+ */
170
+Blockly.Bubble.prototype.relativeTop_ = 0;
171
+
172
+/**
173
+ * Width of bubble.
174
+ * @private
175
+ */
176
+Blockly.Bubble.prototype.width_ = 0;
177
+
178
+/**
179
+ * Height of bubble.
180
+ * @private
181
+ */
182
+Blockly.Bubble.prototype.height_ = 0;
183
+
184
+/**
185
+ * Automatically position and reposition the bubble.
186
+ * @private
187
+ */
188
+Blockly.Bubble.prototype.autoLayout_ = true;
189
+
190
+/**
191
+ * Create the bubble's DOM.
192
+ * @param {!Element} content SVG content for the bubble.
193
+ * @param {boolean} hasResize Add diagonal resize gripper if true.
194
+ * @return {!Element} The bubble's SVG group.
195
+ * @private
196
+ */
197
+Blockly.Bubble.prototype.createDom_ = function(content, hasResize) {
198
+  /* Create the bubble.  Here's the markup that will be generated:
199
+  <g>
200
+    <g filter="url(#blocklyEmbossFilter837493)">
201
+      <path d="... Z" />
202
+      <rect class="blocklyDraggable" rx="8" ry="8" width="180" height="180"/>
203
+    </g>
204
+    <g transform="translate(165, 165)" class="blocklyResizeSE">
205
+      <polygon points="0,15 15,15 15,0"/>
206
+      <line class="blocklyResizeLine" x1="5" y1="14" x2="14" y2="5"/>
207
+      <line class="blocklyResizeLine" x1="10" y1="14" x2="14" y2="10"/>
208
+    </g>
209
+    [...content goes here...]
210
+  </g>
211
+  */
212
+  this.bubbleGroup_ = Blockly.createSvgElement('g', {}, null);
213
+  var filter =
214
+      {'filter': 'url(#' + this.workspace_.options.embossFilterId + ')'};
215
+  if (goog.userAgent.getUserAgentString().indexOf('JavaFX') != -1) {
216
+    // Multiple reports that JavaFX can't handle filters.  UserAgent:
217
+    // Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.44
218
+    //     (KHTML, like Gecko) JavaFX/8.0 Safari/537.44
219
+    // https://github.com/google/blockly/issues/99
220
+    filter = {};
221
+  }
222
+  var bubbleEmboss = Blockly.createSvgElement('g',
223
+      filter, this.bubbleGroup_);
224
+  this.bubbleArrow_ = Blockly.createSvgElement('path', {}, bubbleEmboss);
225
+  this.bubbleBack_ = Blockly.createSvgElement('rect',
226
+      {'class': 'blocklyDraggable', 'x': 0, 'y': 0,
227
+      'rx': Blockly.Bubble.BORDER_WIDTH, 'ry': Blockly.Bubble.BORDER_WIDTH},
228
+      bubbleEmboss);
229
+  if (hasResize) {
230
+    this.resizeGroup_ = Blockly.createSvgElement('g',
231
+        {'class': this.workspace_.RTL ?
232
+                  'blocklyResizeSW' : 'blocklyResizeSE'},
233
+        this.bubbleGroup_);
234
+    var resizeSize = 2 * Blockly.Bubble.BORDER_WIDTH;
235
+    Blockly.createSvgElement('polygon',
236
+        {'points': '0,x x,x x,0'.replace(/x/g, resizeSize.toString())},
237
+        this.resizeGroup_);
238
+    Blockly.createSvgElement('line',
239
+        {'class': 'blocklyResizeLine',
240
+        'x1': resizeSize / 3, 'y1': resizeSize - 1,
241
+        'x2': resizeSize - 1, 'y2': resizeSize / 3}, this.resizeGroup_);
242
+    Blockly.createSvgElement('line',
243
+        {'class': 'blocklyResizeLine',
244
+        'x1': resizeSize * 2 / 3, 'y1': resizeSize - 1,
245
+        'x2': resizeSize - 1, 'y2': resizeSize * 2 / 3}, this.resizeGroup_);
246
+  } else {
247
+    this.resizeGroup_ = null;
248
+  }
249
+  this.bubbleGroup_.appendChild(content);
250
+  return this.bubbleGroup_;
251
+};
252
+
253
+/**
254
+ * Handle a mouse-down on bubble's border.
255
+ * @param {!Event} e Mouse down event.
256
+ * @private
257
+ */
258
+Blockly.Bubble.prototype.bubbleMouseDown_ = function(e) {
259
+  this.promote_();
260
+  Blockly.Bubble.unbindDragEvents_();
261
+  if (Blockly.isRightButton(e)) {
262
+    // No right-click.
263
+    e.stopPropagation();
264
+    return;
265
+  } else if (Blockly.isTargetInput_(e)) {
266
+    // When focused on an HTML text input widget, don't trap any events.
267
+    return;
268
+  }
269
+  // Left-click (or middle click)
270
+  Blockly.Css.setCursor(Blockly.Css.Cursor.CLOSED);
271
+
272
+  this.workspace_.startDrag(e,
273
+      this.workspace_.RTL ? -this.relativeLeft_ : this.relativeLeft_,
274
+      this.relativeTop_);
275
+
276
+  Blockly.Bubble.onMouseUpWrapper_ = Blockly.bindEvent_(document,
277
+      'mouseup', this, Blockly.Bubble.unbindDragEvents_);
278
+  Blockly.Bubble.onMouseMoveWrapper_ = Blockly.bindEvent_(document,
279
+      'mousemove', this, this.bubbleMouseMove_);
280
+  Blockly.hideChaff();
281
+  // This event has been handled.  No need to bubble up to the document.
282
+  e.stopPropagation();
283
+};
284
+
285
+/**
286
+ * Drag this bubble to follow the mouse.
287
+ * @param {!Event} e Mouse move event.
288
+ * @private
289
+ */
290
+Blockly.Bubble.prototype.bubbleMouseMove_ = function(e) {
291
+  this.autoLayout_ = false;
292
+  var newXY = this.workspace_.moveDrag(e);
293
+  this.relativeLeft_ = this.workspace_.RTL ? -newXY.x : newXY.x;
294
+  this.relativeTop_ = newXY.y;
295
+  this.positionBubble_();
296
+  this.renderArrow_();
297
+};
298
+
299
+/**
300
+ * Handle a mouse-down on bubble's resize corner.
301
+ * @param {!Event} e Mouse down event.
302
+ * @private
303
+ */
304
+Blockly.Bubble.prototype.resizeMouseDown_ = function(e) {
305
+  this.promote_();
306
+  Blockly.Bubble.unbindDragEvents_();
307
+  if (Blockly.isRightButton(e)) {
308
+    // No right-click.
309
+    e.stopPropagation();
310
+    return;
311
+  }
312
+  // Left-click (or middle click)
313
+  Blockly.Css.setCursor(Blockly.Css.Cursor.CLOSED);
314
+
315
+  this.workspace_.startDrag(e,
316
+      this.workspace_.RTL ? -this.width_ : this.width_, this.height_);
317
+
318
+  Blockly.Bubble.onMouseUpWrapper_ = Blockly.bindEvent_(document,
319
+      'mouseup', this, Blockly.Bubble.unbindDragEvents_);
320
+  Blockly.Bubble.onMouseMoveWrapper_ = Blockly.bindEvent_(document,
321
+      'mousemove', this, this.resizeMouseMove_);
322
+  Blockly.hideChaff();
323
+  // This event has been handled.  No need to bubble up to the document.
324
+  e.stopPropagation();
325
+};
326
+
327
+/**
328
+ * Resize this bubble to follow the mouse.
329
+ * @param {!Event} e Mouse move event.
330
+ * @private
331
+ */
332
+Blockly.Bubble.prototype.resizeMouseMove_ = function(e) {
333
+  this.autoLayout_ = false;
334
+  var newXY = this.workspace_.moveDrag(e);
335
+  this.setBubbleSize(this.workspace_.RTL ? -newXY.x : newXY.x, newXY.y);
336
+  if (this.workspace_.RTL) {
337
+    // RTL requires the bubble to move its left edge.
338
+    this.positionBubble_();
339
+  }
340
+};
341
+
342
+/**
343
+ * Register a function as a callback event for when the bubble is resized.
344
+ * @param {Object} thisObject The value of 'this' in the callback.
345
+ * @param {!Function} callback The function to call on resize.
346
+ */
347
+Blockly.Bubble.prototype.registerResizeEvent = function(thisObject, callback) {
348
+  Blockly.bindEvent_(this.bubbleGroup_, 'resize', thisObject, callback);
349
+};
350
+
351
+/**
352
+ * Move this bubble to the top of the stack.
353
+ * @private
354
+ */
355
+Blockly.Bubble.prototype.promote_ = function() {
356
+  var svgGroup = this.bubbleGroup_.parentNode;
357
+  svgGroup.appendChild(this.bubbleGroup_);
358
+};
359
+
360
+/**
361
+ * Notification that the anchor has moved.
362
+ * Update the arrow and bubble accordingly.
363
+ * @param {number} x Absolute horizontal location.
364
+ * @param {number} y Absolute vertical location.
365
+ */
366
+Blockly.Bubble.prototype.setAnchorLocation = function(x, y) {
367
+  this.anchorX_ = x;
368
+  this.anchorY_ = y;
369
+  if (this.rendered_) {
370
+    this.positionBubble_();
371
+  }
372
+};
373
+
374
+/**
375
+ * Position the bubble so that it does not fall off-screen.
376
+ * @private
377
+ */
378
+Blockly.Bubble.prototype.layoutBubble_ = function() {
379
+  // Compute the preferred bubble location.
380
+  var relativeLeft = -this.width_ / 4;
381
+  var relativeTop = -this.height_ - Blockly.BlockSvg.MIN_BLOCK_Y;
382
+  // Prevent the bubble from being off-screen.
383
+  var metrics = this.workspace_.getMetrics();
384
+  metrics.viewWidth /= this.workspace_.scale;
385
+  metrics.viewLeft /= this.workspace_.scale;
386
+  if (this.workspace_.RTL) {
387
+    if (this.anchorX_ - metrics.viewLeft - relativeLeft - this.width_ <
388
+        Blockly.Scrollbar.scrollbarThickness) {
389
+      // Slide the bubble right until it is onscreen.
390
+      relativeLeft = this.anchorX_ - metrics.viewLeft - this.width_ -
391
+        Blockly.Scrollbar.scrollbarThickness;
392
+    } else if (this.anchorX_ - metrics.viewLeft - relativeLeft >
393
+               metrics.viewWidth) {
394
+      // Slide the bubble left until it is onscreen.
395
+      relativeLeft = this.anchorX_ - metrics.viewLeft - metrics.viewWidth;
396
+    }
397
+  } else {
398
+    if (this.anchorX_ + relativeLeft < metrics.viewLeft) {
399
+      // Slide the bubble right until it is onscreen.
400
+      relativeLeft = metrics.viewLeft - this.anchorX_;
401
+    } else if (metrics.viewLeft + metrics.viewWidth <
402
+        this.anchorX_ + relativeLeft + this.width_ +
403
+        Blockly.BlockSvg.SEP_SPACE_X +
404
+        Blockly.Scrollbar.scrollbarThickness) {
405
+      // Slide the bubble left until it is onscreen.
406
+      relativeLeft = metrics.viewLeft + metrics.viewWidth - this.anchorX_ -
407
+          this.width_ - Blockly.Scrollbar.scrollbarThickness;
408
+    }
409
+  }
410
+  if (this.anchorY_ + relativeTop < metrics.viewTop) {
411
+    // Slide the bubble below the block.
412
+    var bBox = /** @type {SVGLocatable} */ (this.shape_).getBBox();
413
+    relativeTop = bBox.height;
414
+  }
415
+  this.relativeLeft_ = relativeLeft;
416
+  this.relativeTop_ = relativeTop;
417
+};
418
+
419
+/**
420
+ * Move the bubble to a location relative to the anchor's centre.
421
+ * @private
422
+ */
423
+Blockly.Bubble.prototype.positionBubble_ = function() {
424
+  var left;
425
+  if (this.workspace_.RTL) {
426
+    left = this.anchorX_ - this.relativeLeft_ - this.width_;
427
+  } else {
428
+    left = this.anchorX_ + this.relativeLeft_;
429
+  }
430
+  var top = this.relativeTop_ + this.anchorY_;
431
+  this.bubbleGroup_.setAttribute('transform',
432
+      'translate(' + left + ',' + top + ')');
433
+};
434
+
435
+/**
436
+ * Get the dimensions of this bubble.
437
+ * @return {!Object} Object with width and height properties.
438
+ */
439
+Blockly.Bubble.prototype.getBubbleSize = function() {
440
+  return {width: this.width_, height: this.height_};
441
+};
442
+
443
+/**
444
+ * Size this bubble.
445
+ * @param {number} width Width of the bubble.
446
+ * @param {number} height Height of the bubble.
447
+ */
448
+Blockly.Bubble.prototype.setBubbleSize = function(width, height) {
449
+  var doubleBorderWidth = 2 * Blockly.Bubble.BORDER_WIDTH;
450
+  // Minimum size of a bubble.
451
+  width = Math.max(width, doubleBorderWidth + 45);
452
+  height = Math.max(height, doubleBorderWidth + 20);
453
+  this.width_ = width;
454
+  this.height_ = height;
455
+  this.bubbleBack_.setAttribute('width', width);
456
+  this.bubbleBack_.setAttribute('height', height);
457
+  if (this.resizeGroup_) {
458
+    if (this.workspace_.RTL) {
459
+      // Mirror the resize group.
460
+      var resizeSize = 2 * Blockly.Bubble.BORDER_WIDTH;
461
+      this.resizeGroup_.setAttribute('transform', 'translate(' +
462
+          resizeSize + ',' + (height - doubleBorderWidth) + ') scale(-1 1)');
463
+    } else {
464
+      this.resizeGroup_.setAttribute('transform', 'translate(' +
465
+          (width - doubleBorderWidth) + ',' +
466
+          (height - doubleBorderWidth) + ')');
467
+    }
468
+  }
469
+  if (this.rendered_) {
470
+    if (this.autoLayout_) {
471
+      this.layoutBubble_();
472
+    }
473
+    this.positionBubble_();
474
+    this.renderArrow_();
475
+  }
476
+  // Fire an event to allow the contents to resize.
477
+  Blockly.fireUiEvent(this.bubbleGroup_, 'resize');
478
+};
479
+
480
+/**
481
+ * Draw the arrow between the bubble and the origin.
482
+ * @private
483
+ */
484
+Blockly.Bubble.prototype.renderArrow_ = function() {
485
+  var steps = [];
486
+  // Find the relative coordinates of the center of the bubble.
487
+  var relBubbleX = this.width_ / 2;
488
+  var relBubbleY = this.height_ / 2;
489
+  // Find the relative coordinates of the center of the anchor.
490
+  var relAnchorX = -this.relativeLeft_;
491
+  var relAnchorY = -this.relativeTop_;
492
+  if (relBubbleX == relAnchorX && relBubbleY == relAnchorY) {
493
+    // Null case.  Bubble is directly on top of the anchor.
494
+    // Short circuit this rather than wade through divide by zeros.
495
+    steps.push('M ' + relBubbleX + ',' + relBubbleY);
496
+  } else {
497
+    // Compute the angle of the arrow's line.
498
+    var rise = relAnchorY - relBubbleY;
499
+    var run = relAnchorX - relBubbleX;
500
+    if (this.workspace_.RTL) {
501
+      run *= -1;
502
+    }
503
+    var hypotenuse = Math.sqrt(rise * rise + run * run);
504
+    var angle = Math.acos(run / hypotenuse);
505
+    if (rise < 0) {
506
+      angle = 2 * Math.PI - angle;
507
+    }
508
+    // Compute a line perpendicular to the arrow.
509
+    var rightAngle = angle + Math.PI / 2;
510
+    if (rightAngle > Math.PI * 2) {
511
+      rightAngle -= Math.PI * 2;
512
+    }
513
+    var rightRise = Math.sin(rightAngle);
514
+    var rightRun = Math.cos(rightAngle);
515
+
516
+    // Calculate the thickness of the base of the arrow.
517
+    var bubbleSize = this.getBubbleSize();
518
+    var thickness = (bubbleSize.width + bubbleSize.height) /
519
+                    Blockly.Bubble.ARROW_THICKNESS;
520
+    thickness = Math.min(thickness, bubbleSize.width, bubbleSize.height) / 2;
521
+
522
+    // Back the tip of the arrow off of the anchor.
523
+    var backoffRatio = 1 - Blockly.Bubble.ANCHOR_RADIUS / hypotenuse;
524
+    relAnchorX = relBubbleX + backoffRatio * run;
525
+    relAnchorY = relBubbleY + backoffRatio * rise;
526
+
527
+    // Coordinates for the base of the arrow.
528
+    var baseX1 = relBubbleX + thickness * rightRun;
529
+    var baseY1 = relBubbleY + thickness * rightRise;
530
+    var baseX2 = relBubbleX - thickness * rightRun;
531
+    var baseY2 = relBubbleY - thickness * rightRise;
532
+
533
+    // Distortion to curve the arrow.
534
+    var swirlAngle = angle + this.arrow_radians_;
535
+    if (swirlAngle > Math.PI * 2) {
536
+      swirlAngle -= Math.PI * 2;
537
+    }
538
+    var swirlRise = Math.sin(swirlAngle) *
539
+        hypotenuse / Blockly.Bubble.ARROW_BEND;
540
+    var swirlRun = Math.cos(swirlAngle) *
541
+        hypotenuse / Blockly.Bubble.ARROW_BEND;
542
+
543
+    steps.push('M' + baseX1 + ',' + baseY1);
544
+    steps.push('C' + (baseX1 + swirlRun) + ',' + (baseY1 + swirlRise) +
545
+               ' ' + relAnchorX + ',' + relAnchorY +
546
+               ' ' + relAnchorX + ',' + relAnchorY);
547
+    steps.push('C' + relAnchorX + ',' + relAnchorY +
548
+               ' ' + (baseX2 + swirlRun) + ',' + (baseY2 + swirlRise) +
549
+               ' ' + baseX2 + ',' + baseY2);
550
+  }
551
+  steps.push('z');
552
+  this.bubbleArrow_.setAttribute('d', steps.join(' '));
553
+};
554
+
555
+/**
556
+ * Change the colour of a bubble.
557
+ * @param {string} hexColour Hex code of colour.
558
+ */
559
+Blockly.Bubble.prototype.setColour = function(hexColour) {
560
+  this.bubbleBack_.setAttribute('fill', hexColour);
561
+  this.bubbleArrow_.setAttribute('fill', hexColour);
562
+};
563
+
564
+/**
565
+ * Dispose of this bubble.
566
+ */
567
+Blockly.Bubble.prototype.dispose = function() {
568
+  Blockly.Bubble.unbindDragEvents_();
569
+  // Dispose of and unlink the bubble.
570
+  goog.dom.removeNode(this.bubbleGroup_);
571
+  this.bubbleGroup_ = null;
572
+  this.bubbleArrow_ = null;
573
+  this.bubbleBack_ = null;
574
+  this.resizeGroup_ = null;
575
+  this.workspace_ = null;
576
+  this.content_ = null;
577
+  this.shape_ = null;
578
+};

File diff suppressed because it is too large
+ 241 - 0
src/blockly/core/comment.js


+ 925 - 0
src/blockly/core/connection.js

@@ -0,0 +1,925 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2011 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Components for creating connections between blocks.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Connection');
28
+goog.provide('Blockly.ConnectionDB');
29
+
30
+goog.require('goog.dom');
31
+
32
+
33
+/**
34
+ * Class for a connection between blocks.
35
+ * @param {!Blockly.Block} source The block establishing this connection.
36
+ * @param {number} type The type of the connection.
37
+ * @constructor
38
+ */
39
+Blockly.Connection = function(source, type) {
40
+  /** @type {!Blockly.Block} */
41
+  this.sourceBlock_ = source;
42
+  /** @type {number} */
43
+  this.type = type;
44
+  // Shortcut for the databases for this connection's workspace.
45
+  if (source.workspace.connectionDBList) {
46
+    this.db_ = source.workspace.connectionDBList[type];
47
+    this.dbOpposite_ =
48
+        source.workspace.connectionDBList[Blockly.OPPOSITE_TYPE[type]];
49
+    this.hidden_ = !this.db_;
50
+  }
51
+};
52
+
53
+/**
54
+ * Connection this connection connects to.  Null if not connected.
55
+ * @type {Blockly.Connection}
56
+ */
57
+Blockly.Connection.prototype.targetConnection = null;
58
+
59
+/**
60
+ * List of compatible value types.  Null if all types are compatible.
61
+ * @type {Array}
62
+ * @private
63
+ */
64
+Blockly.Connection.prototype.check_ = null;
65
+
66
+/**
67
+ * DOM representation of a shadow block, or null if none.
68
+ * @type {Element}
69
+ * @private
70
+ */
71
+Blockly.Connection.prototype.shadowDom_ = null;
72
+
73
+/**
74
+ * Horizontal location of this connection.
75
+ * @type {number}
76
+ * @private
77
+ */
78
+Blockly.Connection.prototype.x_ = 0;
79
+
80
+/**
81
+ * Vertical location of this connection.
82
+ * @type {number}
83
+ * @private
84
+ */
85
+Blockly.Connection.prototype.y_ = 0;
86
+
87
+/**
88
+ * Has this connection been added to the connection database?
89
+ * @type {boolean}
90
+ * @private
91
+ */
92
+Blockly.Connection.prototype.inDB_ = false;
93
+
94
+/**
95
+ * Connection database for connections of this type on the current workspace.
96
+ * @type {Blockly.ConnectionDB}
97
+ * @private
98
+ */
99
+Blockly.Connection.prototype.db_ = null;
100
+
101
+/**
102
+ * Connection database for connections compatible with this type on the
103
+ * current workspace.
104
+ * @type {Blockly.ConnectionDB}
105
+ * @private
106
+ */
107
+Blockly.Connection.prototype.dbOpposite_ = null;
108
+
109
+/**
110
+ * Whether this connections is hidden (not tracked in a database) or not.
111
+ * @type {boolean}
112
+ * @private
113
+ */
114
+Blockly.Connection.prototype.hidden_ = null;
115
+
116
+/**
117
+ * Sever all links to this connection (not including from the source object).
118
+ */
119
+Blockly.Connection.prototype.dispose = function() {
120
+  if (this.targetConnection) {
121
+    throw 'Disconnect connection before disposing of it.';
122
+  }
123
+  if (this.inDB_) {
124
+    this.db_.removeConnection_(this);
125
+  }
126
+  if (Blockly.highlightedConnection_ == this) {
127
+    Blockly.highlightedConnection_ = null;
128
+  }
129
+  if (Blockly.localConnection_ == this) {
130
+    Blockly.localConnection_ = null;
131
+  }
132
+  this.db_ = null;
133
+  this.dbOpposite_ = null;
134
+};
135
+
136
+/**
137
+ * Does the connection belong to a superior block (higher in the source stack)?
138
+ * @return {boolean} True if connection faces down or right.
139
+ */
140
+Blockly.Connection.prototype.isSuperior = function() {
141
+  return this.type == Blockly.INPUT_VALUE ||
142
+      this.type == Blockly.NEXT_STATEMENT;
143
+};
144
+
145
+/**
146
+ * Connect this connection to another connection.
147
+ * @param {!Blockly.Connection} otherConnection Connection to connect to.
148
+ */
149
+Blockly.Connection.prototype.connect = function(otherConnection) {
150
+  if (this.sourceBlock_ == otherConnection.sourceBlock_) {
151
+    throw 'Attempted to connect a block to itself.';
152
+  }
153
+  if (this.sourceBlock_.workspace !== otherConnection.sourceBlock_.workspace) {
154
+    throw 'Blocks are on different workspaces.';
155
+  }
156
+  if (Blockly.OPPOSITE_TYPE[this.type] != otherConnection.type) {
157
+    throw 'Attempt to connect incompatible types.';
158
+  }
159
+  if (this.type == Blockly.INPUT_VALUE || this.type == Blockly.OUTPUT_VALUE) {
160
+    if (this.targetConnection) {
161
+      // Can't make a value connection if male block is already connected.
162
+      throw 'Source connection already connected (value).';
163
+    } else if (otherConnection.targetConnection) {
164
+      // Record and disable the shadow so that it does not respawn here.
165
+      var shadowDom = otherConnection.getShadowDom();
166
+      otherConnection.setShadowDom(null);
167
+      // If female block is already connected, disconnect and bump the male.
168
+      var orphanBlock = otherConnection.targetBlock();
169
+      orphanBlock.setParent(null);
170
+      if (orphanBlock.isShadow()) {
171
+        otherConnection.setShadowDom(Blockly.Xml.blockToDom_(orphanBlock));
172
+        orphanBlock.dispose();
173
+      } else {
174
+        if (!orphanBlock.outputConnection) {
175
+          throw 'Orphan block does not have an output connection.';
176
+        }
177
+        // Attempt to reattach the orphan at the end of the newly inserted
178
+        // block.  Since this block may be a row, walk down to the end.
179
+        var newBlock = this.sourceBlock_;
180
+        var connection;
181
+        while (connection = Blockly.Connection.singleConnection_(
182
+            /** @type {!Blockly.Block} */ (newBlock), orphanBlock)) {
183
+          // '=' is intentional in line above.
184
+          newBlock = connection.targetBlock();
185
+          if (!newBlock || newBlock.isShadow()) {
186
+            orphanBlock.outputConnection.connect(connection);
187
+            orphanBlock = null;
188
+            break;
189
+          }
190
+        }
191
+        if (orphanBlock) {
192
+          // Unable to reattach orphan.  Bump it off to the side.
193
+          setTimeout(function() {
194
+                orphanBlock.outputConnection.bumpAwayFrom_(otherConnection);
195
+              }, Blockly.BUMP_DELAY);
196
+        }
197
+        // Restore the shadow.
198
+        otherConnection.setShadowDom(shadowDom);
199
+      }
200
+    }
201
+  } else {
202
+    if (this.targetConnection) {
203
+      throw 'Source connection already connected (block).';
204
+    } else if (otherConnection.targetConnection) {
205
+      // Statement blocks may be inserted into the middle of a stack.
206
+      if (this.type != Blockly.PREVIOUS_STATEMENT) {
207
+        throw 'Can only do a mid-stack connection with the top of a block.';
208
+      }
209
+      // Split the stack.
210
+      var orphanBlock = otherConnection.targetBlock();
211
+      orphanBlock.setParent(null);
212
+      if (!orphanBlock.previousConnection) {
213
+        throw 'Orphan block does not have a previous connection.';
214
+      }
215
+      // Attempt to reattach the orphan at the bottom of the newly inserted
216
+      // block.  Since this block may be a stack, walk down to the end.
217
+      var newBlock = this.sourceBlock_;
218
+      while (newBlock.nextConnection) {
219
+        if (newBlock.nextConnection.targetConnection) {
220
+          newBlock = newBlock.getNextBlock();
221
+        } else {
222
+          if (orphanBlock.previousConnection.checkType_(
223
+              newBlock.nextConnection)) {
224
+            newBlock.nextConnection.connect(orphanBlock.previousConnection);
225
+            orphanBlock = null;
226
+          }
227
+          break;
228
+        }
229
+      }
230
+      if (orphanBlock) {
231
+        // Unable to reattach orphan.  Bump it off to the side.
232
+        setTimeout(function() {
233
+              orphanBlock.previousConnection.bumpAwayFrom_(otherConnection);
234
+            }, Blockly.BUMP_DELAY);
235
+      }
236
+    }
237
+  }
238
+
239
+  // Determine which block is superior (higher in the source stack).
240
+  var parentBlock, childBlock;
241
+  if (this.isSuperior()) {
242
+    // Superior block.
243
+    parentBlock = this.sourceBlock_;
244
+    childBlock = otherConnection.sourceBlock_;
245
+  } else {
246
+    // Inferior block.
247
+    parentBlock = otherConnection.sourceBlock_;
248
+    childBlock = this.sourceBlock_;
249
+  }
250
+
251
+  // Establish the connections.
252
+  this.targetConnection = otherConnection;
253
+  otherConnection.targetConnection = this;
254
+
255
+  // Demote the inferior block so that one is a child of the superior one.
256
+  childBlock.setParent(parentBlock);
257
+
258
+  if (parentBlock.rendered) {
259
+    parentBlock.updateDisabled();
260
+  }
261
+  if (childBlock.rendered) {
262
+    childBlock.updateDisabled();
263
+  }
264
+  if (parentBlock.rendered && childBlock.rendered) {
265
+    if (this.type == Blockly.NEXT_STATEMENT ||
266
+        this.type == Blockly.PREVIOUS_STATEMENT) {
267
+      // Child block may need to square off its corners if it is in a stack.
268
+      // Rendering a child will render its parent.
269
+      childBlock.render();
270
+    } else {
271
+      // Child block does not change shape.  Rendering the parent node will
272
+      // move its connected children into position.
273
+      parentBlock.render();
274
+    }
275
+  }
276
+};
277
+
278
+/**
279
+ * Does the given block have one and only one connection point that will accept
280
+ * an orphaned block?
281
+ * @param {!Blockly.Block} block The superior block.
282
+ * @param {!Blockly.Block} orphanBlock The inferior block.
283
+ * @return {Blockly.Connection} The suitable connection point on 'block',
284
+ *     or null.
285
+ * @private
286
+ */
287
+Blockly.Connection.singleConnection_ = function(block, orphanBlock) {
288
+  var connection = false;
289
+  for (var i = 0; i < block.inputList.length; i++) {
290
+    var thisConnection = block.inputList[i].connection;
291
+    if (thisConnection && thisConnection.type == Blockly.INPUT_VALUE &&
292
+        orphanBlock.outputConnection.checkType_(thisConnection)) {
293
+      if (connection) {
294
+        return null;  // More than one connection.
295
+      }
296
+      connection = thisConnection;
297
+    }
298
+  }
299
+  return connection;
300
+};
301
+
302
+/**
303
+ * Disconnect this connection.
304
+ */
305
+Blockly.Connection.prototype.disconnect = function() {
306
+  var otherConnection = this.targetConnection;
307
+  if (!otherConnection) {
308
+    throw 'Source connection not connected.';
309
+  } else if (otherConnection.targetConnection != this) {
310
+    throw 'Target connection not connected to source connection.';
311
+  }
312
+  otherConnection.targetConnection = null;
313
+  this.targetConnection = null;
314
+
315
+  // Rerender the parent so that it may reflow.
316
+  var parentBlock, childBlock, parentConnection;
317
+  if (this.isSuperior()) {
318
+    // Superior block.
319
+    parentBlock = this.sourceBlock_;
320
+    childBlock = otherConnection.sourceBlock_;
321
+    parentConnection = this;
322
+  } else {
323
+    // Inferior block.
324
+    parentBlock = otherConnection.sourceBlock_;
325
+    childBlock = this.sourceBlock_;
326
+    parentConnection = otherConnection;
327
+  }
328
+  var shadow = parentConnection.getShadowDom();
329
+  if (parentBlock.workspace && !childBlock.isShadow() && shadow) {
330
+    // Respawn the shadow block.
331
+    var blockShadow =
332
+        Blockly.Xml.domToBlock(parentBlock.workspace, shadow);
333
+    if (blockShadow.outputConnection) {
334
+      parentConnection.connect(blockShadow.outputConnection);
335
+    } else if (blockShadow.previousConnection) {
336
+      parentConnection.connect(blockShadow.previousConnection);
337
+    } else {
338
+      throw 'Child block does not have output or previous statement.';
339
+    }
340
+    blockShadow.initSvg();
341
+    blockShadow.render(false);
342
+  }
343
+  if (parentBlock.rendered) {
344
+    parentBlock.render();
345
+  }
346
+  if (childBlock.rendered) {
347
+    childBlock.updateDisabled();
348
+    childBlock.render();
349
+  }
350
+};
351
+
352
+/**
353
+ * Returns the block that this connection connects to.
354
+ * @return {Blockly.Block} The connected block or null if none is connected.
355
+ */
356
+Blockly.Connection.prototype.targetBlock = function() {
357
+  if (this.targetConnection) {
358
+    return this.targetConnection.sourceBlock_;
359
+  }
360
+  return null;
361
+};
362
+
363
+/**
364
+ * Move the block(s) belonging to the connection to a point where they don't
365
+ * visually interfere with the specified connection.
366
+ * @param {!Blockly.Connection} staticConnection The connection to move away
367
+ *     from.
368
+ * @private
369
+ */
370
+Blockly.Connection.prototype.bumpAwayFrom_ = function(staticConnection) {
371
+  if (Blockly.dragMode_ != 0) {
372
+    // Don't move blocks around while the user is doing the same.
373
+    return;
374
+  }
375
+  // Move the root block.
376
+  var rootBlock = this.sourceBlock_.getRootBlock();
377
+  if (rootBlock.isInFlyout) {
378
+    // Don't move blocks around in a flyout.
379
+    return;
380
+  }
381
+  var reverse = false;
382
+  if (!rootBlock.isMovable()) {
383
+    // Can't bump an uneditable block away.
384
+    // Check to see if the other block is movable.
385
+    rootBlock = staticConnection.sourceBlock_.getRootBlock();
386
+    if (!rootBlock.isMovable()) {
387
+      return;
388
+    }
389
+    // Swap the connections and move the 'static' connection instead.
390
+    staticConnection = this;
391
+    reverse = true;
392
+  }
393
+  // Raise it to the top for extra visibility.
394
+  rootBlock.getSvgRoot().parentNode.appendChild(rootBlock.getSvgRoot());
395
+  var dx = (staticConnection.x_ + Blockly.SNAP_RADIUS) - this.x_;
396
+  var dy = (staticConnection.y_ + Blockly.SNAP_RADIUS) - this.y_;
397
+  if (reverse) {
398
+    // When reversing a bump due to an uneditable block, bump up.
399
+    dy = -dy;
400
+  }
401
+  if (rootBlock.RTL) {
402
+    dx = -dx;
403
+  }
404
+  rootBlock.moveBy(dx, dy);
405
+};
406
+
407
+/**
408
+ * Change the connection's coordinates.
409
+ * @param {number} x New absolute x coordinate.
410
+ * @param {number} y New absolute y coordinate.
411
+ */
412
+Blockly.Connection.prototype.moveTo = function(x, y) {
413
+  // Remove it from its old location in the database (if already present)
414
+  if (this.inDB_) {
415
+    this.db_.removeConnection_(this);
416
+  }
417
+  this.x_ = x;
418
+  this.y_ = y;
419
+  // Insert it into its new location in the database.
420
+  if (!this.hidden_) {
421
+    this.db_.addConnection_(this);
422
+  }
423
+};
424
+
425
+/**
426
+ * Change the connection's coordinates.
427
+ * @param {number} dx Change to x coordinate.
428
+ * @param {number} dy Change to y coordinate.
429
+ */
430
+Blockly.Connection.prototype.moveBy = function(dx, dy) {
431
+  this.moveTo(this.x_ + dx, this.y_ + dy);
432
+};
433
+
434
+/**
435
+ * Add highlighting around this connection.
436
+ */
437
+Blockly.Connection.prototype.highlight = function() {
438
+  var steps;
439
+  if (this.type == Blockly.INPUT_VALUE || this.type == Blockly.OUTPUT_VALUE) {
440
+    var tabWidth = this.sourceBlock_.RTL ? -Blockly.BlockSvg.TAB_WIDTH :
441
+        Blockly.BlockSvg.TAB_WIDTH;
442
+    steps = 'm 0,0 v 5 c 0,10 ' + -tabWidth + ',-8 ' + -tabWidth + ',7.5 s ' +
443
+            tabWidth + ',-2.5 ' + tabWidth + ',7.5 v 5';
444
+  } else {
445
+    if (this.sourceBlock_.RTL) {
446
+      steps = 'm 20,0 h -5 ' + Blockly.BlockSvg.NOTCH_PATH_RIGHT + ' h -5';
447
+    } else {
448
+      steps = 'm -20,0 h 5 ' + Blockly.BlockSvg.NOTCH_PATH_LEFT + ' h 5';
449
+    }
450
+  }
451
+  var xy = this.sourceBlock_.getRelativeToSurfaceXY();
452
+  var x = this.x_ - xy.x;
453
+  var y = this.y_ - xy.y;
454
+  Blockly.Connection.highlightedPath_ = Blockly.createSvgElement('path',
455
+      {'class': 'blocklyHighlightedConnectionPath',
456
+       'd': steps,
457
+       transform: 'translate(' + x + ',' + y + ')'},
458
+      this.sourceBlock_.getSvgRoot());
459
+};
460
+
461
+/**
462
+ * Remove the highlighting around this connection.
463
+ */
464
+Blockly.Connection.prototype.unhighlight = function() {
465
+  goog.dom.removeNode(Blockly.Connection.highlightedPath_);
466
+  delete Blockly.Connection.highlightedPath_;
467
+};
468
+
469
+/**
470
+ * Move the blocks on either side of this connection right next to each other.
471
+ * @private
472
+ */
473
+Blockly.Connection.prototype.tighten_ = function() {
474
+  var dx = this.targetConnection.x_ - this.x_;
475
+  var dy = this.targetConnection.y_ - this.y_;
476
+  if (dx != 0 || dy != 0) {
477
+    var block = this.targetBlock();
478
+    var svgRoot = block.getSvgRoot();
479
+    if (!svgRoot) {
480
+      throw 'block is not rendered.';
481
+    }
482
+    var xy = Blockly.getRelativeXY_(svgRoot);
483
+    block.getSvgRoot().setAttribute('transform',
484
+        'translate(' + (xy.x - dx) + ',' + (xy.y - dy) + ')');
485
+    block.moveConnections_(-dx, -dy);
486
+  }
487
+};
488
+
489
+/**
490
+ * Find the closest compatible connection to this connection.
491
+ * @param {number} maxLimit The maximum radius to another connection.
492
+ * @param {number} dx Horizontal offset between this connection's location
493
+ *     in the database and the current location (as a result of dragging).
494
+ * @param {number} dy Vertical offset between this connection's location
495
+ *     in the database and the current location (as a result of dragging).
496
+ * @return {!Object} Contains two properties: 'connection' which is either
497
+ *     another connection or null, and 'radius' which is the distance.
498
+ */
499
+Blockly.Connection.prototype.closest = function(maxLimit, dx, dy) {
500
+  if (this.targetConnection) {
501
+    // Don't offer to connect to a connection that's already connected.
502
+    return {connection: null, radius: maxLimit};
503
+  }
504
+  // Determine the opposite type of connection.
505
+  var db = this.dbOpposite_;
506
+
507
+  // Since this connection is probably being dragged, add the delta.
508
+  var currentX = this.x_ + dx;
509
+  var currentY = this.y_ + dy;
510
+
511
+  // Binary search to find the closest y location.
512
+  var pointerMin = 0;
513
+  var pointerMax = db.length - 2;
514
+  var pointerMid = pointerMax;
515
+  while (pointerMin < pointerMid) {
516
+    if (db[pointerMid].y_ < currentY) {
517
+      pointerMin = pointerMid;
518
+    } else {
519
+      pointerMax = pointerMid;
520
+    }
521
+    pointerMid = Math.floor((pointerMin + pointerMax) / 2);
522
+  }
523
+
524
+  // Walk forward and back on the y axis looking for the closest x,y point.
525
+  pointerMin = pointerMid;
526
+  pointerMax = pointerMid;
527
+  var closestConnection = null;
528
+  var sourceBlock = this.sourceBlock_;
529
+  var thisConnection = this;
530
+  if (db.length) {
531
+    while (pointerMin >= 0 && checkConnection_(pointerMin)) {
532
+      pointerMin--;
533
+    }
534
+    do {
535
+      pointerMax++;
536
+    } while (pointerMax < db.length && checkConnection_(pointerMax));
537
+  }
538
+
539
+  /**
540
+   * Computes if the current connection is within the allowed radius of another
541
+   * connection.
542
+   * This function is a closure and has access to outside variables.
543
+   * @param {number} yIndex The other connection's index in the database.
544
+   * @return {boolean} True if the search needs to continue: either the current
545
+   *     connection's vertical distance from the other connection is less than
546
+   *     the allowed radius, or if the connection is not compatible.
547
+   * @private
548
+   */
549
+  function checkConnection_(yIndex) {
550
+    var connection = db[yIndex];
551
+    if (connection.type == Blockly.OUTPUT_VALUE ||
552
+        connection.type == Blockly.PREVIOUS_STATEMENT) {
553
+      // Don't offer to connect an already connected left (male) value plug to
554
+      // an available right (female) value plug.  Don't offer to connect the
555
+      // bottom of a statement block to one that's already connected.
556
+      if (connection.targetConnection) {
557
+        return true;
558
+      }
559
+    }
560
+    // Offering to connect the top of a statement block to an already connected
561
+    // connection is ok, we'll just insert it into the stack.
562
+
563
+    // Offering to connect the left (male) of a value block to an already
564
+    // connected value pair is ok, we'll splice it in.
565
+    // However, don't offer to splice into an unmovable block.
566
+    if (connection.type == Blockly.INPUT_VALUE &&
567
+        connection.targetConnection &&
568
+        !connection.targetBlock().isMovable() &&
569
+        !connection.targetBlock().isShadow()) {
570
+      return true;
571
+    }
572
+
573
+    // Do type checking.
574
+    if (!thisConnection.checkType_(connection)) {
575
+      return true;
576
+    }
577
+
578
+    // Don't let blocks try to connect to themselves or ones they nest.
579
+    var targetSourceBlock = connection.sourceBlock_;
580
+    do {
581
+      if (sourceBlock == targetSourceBlock) {
582
+        return true;
583
+      }
584
+      targetSourceBlock = targetSourceBlock.getParent();
585
+    } while (targetSourceBlock);
586
+
587
+    // Only connections within the maxLimit radius.
588
+    var dx = currentX - connection.x_;
589
+    var dy = currentY - connection.y_;
590
+    var r = Math.sqrt(dx * dx + dy * dy);
591
+    if (r <= maxLimit) {
592
+      closestConnection = connection;
593
+      maxLimit = r;
594
+    }
595
+    return Math.abs(dy) < maxLimit;
596
+  }
597
+  return {connection: closestConnection, radius: maxLimit};
598
+};
599
+
600
+/**
601
+ * Is this connection compatible with another connection with respect to the
602
+ * value type system.  E.g. square_root("Hello") is not compatible.
603
+ * @param {!Blockly.Connection} otherConnection Connection to compare against.
604
+ * @return {boolean} True if the connections share a type.
605
+ * @private
606
+ */
607
+Blockly.Connection.prototype.checkType_ = function(otherConnection) {
608
+  // Don't split a connection where both sides are immovable.
609
+  var thisTargetBlock = this.targetBlock();
610
+  if (thisTargetBlock && !thisTargetBlock.isMovable() &&
611
+      !this.sourceBlock_.isMovable()) {
612
+    return false;
613
+  }
614
+  var otherTargetBlock = otherConnection.targetBlock();
615
+  if (otherTargetBlock && !otherTargetBlock.isMovable() &&
616
+      !otherConnection.sourceBlock_.isMovable()) {
617
+    return false;
618
+  }
619
+  if (!this.check_ || !otherConnection.check_) {
620
+    // One or both sides are promiscuous enough that anything will fit.
621
+    return true;
622
+  }
623
+  // Find any intersection in the check lists.
624
+  for (var i = 0; i < this.check_.length; i++) {
625
+    if (otherConnection.check_.indexOf(this.check_[i]) != -1) {
626
+      return true;
627
+    }
628
+  }
629
+  // No intersection.
630
+  return false;
631
+};
632
+
633
+/**
634
+ * Change a connection's compatibility.
635
+ * @param {*} check Compatible value type or list of value types.
636
+ *     Null if all types are compatible.
637
+ * @return {!Blockly.Connection} The connection being modified
638
+ *     (to allow chaining).
639
+ */
640
+Blockly.Connection.prototype.setCheck = function(check) {
641
+  if (check) {
642
+    // Ensure that check is in an array.
643
+    if (!goog.isArray(check)) {
644
+      check = [check];
645
+    }
646
+    this.check_ = check;
647
+    // The new value type may not be compatible with the existing connection.
648
+    if (this.targetConnection && !this.checkType_(this.targetConnection)) {
649
+      if (this.isSuperior()) {
650
+        this.targetBlock().setParent(null);
651
+      } else {
652
+        this.sourceBlock_.setParent(null);
653
+      }
654
+      // Bump away.
655
+      this.sourceBlock_.bumpNeighbours_();
656
+    }
657
+  } else {
658
+    this.check_ = null;
659
+  }
660
+  return this;
661
+};
662
+
663
+/**
664
+ * Change a connection's shadow block.
665
+ * @param {Element} shadow DOM representation of a block or null.
666
+ */
667
+Blockly.Connection.prototype.setShadowDom = function(shadow) {
668
+  this.shadowDom_ = shadow;
669
+};
670
+
671
+/**
672
+ * Return a connection's shadow block.
673
+ * @return {Element} shadow DOM representation of a block or null.
674
+ */
675
+Blockly.Connection.prototype.getShadowDom = function() {
676
+  return this.shadowDom_;
677
+};
678
+
679
+/**
680
+ * Find all nearby compatible connections to this connection.
681
+ * Type checking does not apply, since this function is used for bumping.
682
+ * @param {number} maxLimit The maximum radius to another connection.
683
+ * @return {!Array.<Blockly.Connection>} List of connections.
684
+ * @private
685
+ */
686
+Blockly.Connection.prototype.neighbours_ = function(maxLimit) {
687
+  // Determine the opposite type of connection.
688
+  var db = this.dbOpposite_;
689
+
690
+  var currentX = this.x_;
691
+  var currentY = this.y_;
692
+
693
+  // Binary search to find the closest y location.
694
+  var pointerMin = 0;
695
+  var pointerMax = db.length - 2;
696
+  var pointerMid = pointerMax;
697
+  while (pointerMin < pointerMid) {
698
+    if (db[pointerMid].y_ < currentY) {
699
+      pointerMin = pointerMid;
700
+    } else {
701
+      pointerMax = pointerMid;
702
+    }
703
+    pointerMid = Math.floor((pointerMin + pointerMax) / 2);
704
+  }
705
+
706
+  // Walk forward and back on the y axis looking for the closest x,y point.
707
+  pointerMin = pointerMid;
708
+  pointerMax = pointerMid;
709
+  var neighbours = [];
710
+  var sourceBlock = this.sourceBlock_;
711
+  if (db.length) {
712
+    while (pointerMin >= 0 && checkConnection_(pointerMin)) {
713
+      pointerMin--;
714
+    }
715
+    do {
716
+      pointerMax++;
717
+    } while (pointerMax < db.length && checkConnection_(pointerMax));
718
+  }
719
+
720
+  /**
721
+   * Computes if the current connection is within the allowed radius of another
722
+   * connection.
723
+   * This function is a closure and has access to outside variables.
724
+   * @param {number} yIndex The other connection's index in the database.
725
+   * @return {boolean} True if the current connection's vertical distance from
726
+   *     the other connection is less than the allowed radius.
727
+   */
728
+  function checkConnection_(yIndex) {
729
+    var dx = currentX - db[yIndex].x_;
730
+    var dy = currentY - db[yIndex].y_;
731
+    var r = Math.sqrt(dx * dx + dy * dy);
732
+    if (r <= maxLimit) {
733
+      neighbours.push(db[yIndex]);
734
+    }
735
+    return dy < maxLimit;
736
+  }
737
+  return neighbours;
738
+};
739
+
740
+/**
741
+ * Set whether this connections is hidden (not tracked in a database) or not.
742
+ * @param {boolean} hidden True if connection is hidden.
743
+ */
744
+Blockly.Connection.prototype.setHidden = function(hidden) {
745
+  this.hidden_ = hidden;
746
+  if (hidden && this.inDB_) {
747
+    this.db_.removeConnection_(this);
748
+  } else if (!hidden && !this.inDB_) {
749
+    this.db_.addConnection_(this);
750
+  }
751
+};
752
+
753
+/**
754
+ * Hide this connection, as well as all down-stream connections on any block
755
+ * attached to this connection.  This happens when a block is collapsed.
756
+ * Also hides down-stream comments.
757
+ */
758
+Blockly.Connection.prototype.hideAll = function() {
759
+  this.setHidden(true);
760
+  if (this.targetConnection) {
761
+    var blocks = this.targetBlock().getDescendants();
762
+    for (var b = 0; b < blocks.length; b++) {
763
+      var block = blocks[b];
764
+      // Hide all connections of all children.
765
+      var connections = block.getConnections_(true);
766
+      for (var c = 0; c < connections.length; c++) {
767
+        connections[c].setHidden(true);
768
+      }
769
+      // Close all bubbles of all children.
770
+      var icons = block.getIcons();
771
+      for (var i = 0; i < icons.length; i++) {
772
+        icons[i].setVisible(false);
773
+      }
774
+    }
775
+  }
776
+};
777
+
778
+/**
779
+ * Unhide this connection, as well as all down-stream connections on any block
780
+ * attached to this connection.  This happens when a block is expanded.
781
+ * Also unhides down-stream comments.
782
+ * @return {!Array.<!Blockly.Block>} List of blocks to render.
783
+ */
784
+Blockly.Connection.prototype.unhideAll = function() {
785
+  this.setHidden(false);
786
+  // All blocks that need unhiding must be unhidden before any rendering takes
787
+  // place, since rendering requires knowing the dimensions of lower blocks.
788
+  // Also, since rendering a block renders all its parents, we only need to
789
+  // render the leaf nodes.
790
+  var renderList = [];
791
+  if (this.type != Blockly.INPUT_VALUE && this.type != Blockly.NEXT_STATEMENT) {
792
+    // Only spider down.
793
+    return renderList;
794
+  }
795
+  var block = this.targetBlock();
796
+  if (block) {
797
+    var connections;
798
+    if (block.isCollapsed()) {
799
+      // This block should only be partially revealed since it is collapsed.
800
+      connections = [];
801
+      block.outputConnection && connections.push(block.outputConnection);
802
+      block.nextConnection && connections.push(block.nextConnection);
803
+      block.previousConnection && connections.push(block.previousConnection);
804
+    } else {
805
+      // Show all connections of this block.
806
+      connections = block.getConnections_(true);
807
+    }
808
+    for (var c = 0; c < connections.length; c++) {
809
+      renderList.push.apply(renderList, connections[c].unhideAll());
810
+    }
811
+    if (renderList.length == 0) {
812
+      // Leaf block.
813
+      renderList[0] = block;
814
+    }
815
+  }
816
+  return renderList;
817
+};
818
+
819
+
820
+/**
821
+ * Database of connections.
822
+ * Connections are stored in order of their vertical component.  This way
823
+ * connections in an area may be looked up quickly using a binary search.
824
+ * @constructor
825
+ */
826
+Blockly.ConnectionDB = function() {
827
+};
828
+
829
+Blockly.ConnectionDB.prototype = new Array();
830
+/**
831
+ * Don't inherit the constructor from Array.
832
+ * @type {!Function}
833
+ */
834
+Blockly.ConnectionDB.constructor = Blockly.ConnectionDB;
835
+
836
+/**
837
+ * Add a connection to the database.  Must not already exist in DB.
838
+ * @param {!Blockly.Connection} connection The connection to be added.
839
+ * @private
840
+ */
841
+Blockly.ConnectionDB.prototype.addConnection_ = function(connection) {
842
+  if (connection.inDB_) {
843
+    throw 'Connection already in database.';
844
+  }
845
+  if (connection.sourceBlock_.isInFlyout) {
846
+    // Don't bother maintaining a database of connections in a flyout.
847
+    return;
848
+  }
849
+  // Insert connection using binary search.
850
+  var pointerMin = 0;
851
+  var pointerMax = this.length;
852
+  while (pointerMin < pointerMax) {
853
+    var pointerMid = Math.floor((pointerMin + pointerMax) / 2);
854
+    if (this[pointerMid].y_ < connection.y_) {
855
+      pointerMin = pointerMid + 1;
856
+    } else if (this[pointerMid].y_ > connection.y_) {
857
+      pointerMax = pointerMid;
858
+    } else {
859
+      pointerMin = pointerMid;
860
+      break;
861
+    }
862
+  }
863
+  this.splice(pointerMin, 0, connection);
864
+  connection.inDB_ = true;
865
+};
866
+
867
+/**
868
+ * Remove a connection from the database.  Must already exist in DB.
869
+ * @param {!Blockly.Connection} connection The connection to be removed.
870
+ * @private
871
+ */
872
+Blockly.ConnectionDB.prototype.removeConnection_ = function(connection) {
873
+  if (!connection.inDB_) {
874
+    throw 'Connection not in database.';
875
+  }
876
+  connection.inDB_ = false;
877
+  // Find the connection using a binary search.
878
+  // About 10% faster than a linear search using indexOf.
879
+  var pointerMin = 0;
880
+  var pointerMax = this.length - 2;
881
+  var pointerMid = pointerMax;
882
+  while (pointerMin < pointerMid) {
883
+    if (this[pointerMid].y_ < connection.y_) {
884
+      pointerMin = pointerMid;
885
+    } else {
886
+      pointerMax = pointerMid;
887
+    }
888
+    pointerMid = Math.floor((pointerMin + pointerMax) / 2);
889
+  }
890
+
891
+  // Walk forward and back on the y axis looking for the connection.
892
+  // When found, splice it out of the array.
893
+  pointerMin = pointerMid;
894
+  pointerMax = pointerMid;
895
+  while (pointerMin >= 0 && this[pointerMin].y_ == connection.y_) {
896
+    if (this[pointerMin] == connection) {
897
+      this.splice(pointerMin, 1);
898
+      return;
899
+    }
900
+    pointerMin--;
901
+  }
902
+  do {
903
+    if (this[pointerMax] == connection) {
904
+      this.splice(pointerMax, 1);
905
+      return;
906
+    }
907
+    pointerMax++;
908
+  } while (pointerMax < this.length &&
909
+           this[pointerMax].y_ == connection.y_);
910
+  throw 'Unable to find connection in connectionDB.';
911
+};
912
+
913
+/**
914
+ * Initialize a set of connection DBs for a specified workspace.
915
+ * @param {!Blockly.Workspace} workspace The workspace this DB is for.
916
+ */
917
+Blockly.ConnectionDB.init = function(workspace) {
918
+  // Create four databases, one for each connection type.
919
+  var dbList = [];
920
+  dbList[Blockly.INPUT_VALUE] = new Blockly.ConnectionDB();
921
+  dbList[Blockly.OUTPUT_VALUE] = new Blockly.ConnectionDB();
922
+  dbList[Blockly.NEXT_STATEMENT] = new Blockly.ConnectionDB();
923
+  dbList[Blockly.PREVIOUS_STATEMENT] = new Blockly.ConnectionDB();
924
+  workspace.connectionDBList = dbList;
925
+};

+ 141 - 0
src/blockly/core/contextmenu.js

@@ -0,0 +1,141 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2011 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Functionality for the right-click context menus.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.ContextMenu');
28
+
29
+goog.require('goog.dom');
30
+goog.require('goog.events');
31
+goog.require('goog.style');
32
+goog.require('goog.ui.Menu');
33
+goog.require('goog.ui.MenuItem');
34
+
35
+
36
+/**
37
+ * Which block is the context menu attached to?
38
+ * @type {Blockly.Block}
39
+ */
40
+Blockly.ContextMenu.currentBlock = null;
41
+
42
+/**
43
+ * Construct the menu based on the list of options and show the menu.
44
+ * @param {!Event} e Mouse event.
45
+ * @param {!Array.<!Object>} options Array of menu options.
46
+ * @param {boolean} rtl True if RTL, false if LTR.
47
+ */
48
+Blockly.ContextMenu.show = function(e, options, rtl) {
49
+  Blockly.WidgetDiv.show(Blockly.ContextMenu, rtl, null);
50
+  if (!options.length) {
51
+    Blockly.ContextMenu.hide();
52
+    return;
53
+  }
54
+  /* Here's what one option object looks like:
55
+    {text: 'Make It So',
56
+     enabled: true,
57
+     callback: Blockly.MakeItSo}
58
+  */
59
+  var menu = new goog.ui.Menu();
60
+  menu.setRightToLeft(rtl);
61
+  for (var x = 0, option; option = options[x]; x++) {
62
+    var menuItem = new goog.ui.MenuItem(option.text);
63
+    menuItem.setRightToLeft(rtl);
64
+    menu.addChild(menuItem, true);
65
+    menuItem.setEnabled(option.enabled);
66
+    if (option.enabled) {
67
+      var evtHandlerCapturer = function(callback) {
68
+        return function() { Blockly.doCommand(callback); };
69
+      };
70
+      goog.events.listen(menuItem, goog.ui.Component.EventType.ACTION,
71
+                         evtHandlerCapturer(option.callback));
72
+    }
73
+  }
74
+  goog.events.listen(menu, goog.ui.Component.EventType.ACTION,
75
+                     Blockly.ContextMenu.hide);
76
+  // Record windowSize and scrollOffset before adding menu.
77
+  var windowSize = goog.dom.getViewportSize();
78
+  var scrollOffset = goog.style.getViewportPageOffset(document);
79
+  var div = Blockly.WidgetDiv.DIV;
80
+  menu.render(div);
81
+  var menuDom = menu.getElement();
82
+  Blockly.addClass_(menuDom, 'blocklyContextMenu');
83
+  // Record menuSize after adding menu.
84
+  var menuSize = goog.style.getSize(menuDom);
85
+
86
+  // Position the menu.
87
+  var x = e.clientX + scrollOffset.x;
88
+  var y = e.clientY + scrollOffset.y;
89
+  // Flip menu vertically if off the bottom.
90
+  if (e.clientY + menuSize.height >= windowSize.height) {
91
+    y -= menuSize.height;
92
+  }
93
+  // Flip menu horizontally if off the edge.
94
+  if (rtl) {
95
+    if (menuSize.width >= e.clientX) {
96
+      x += menuSize.width;
97
+    }
98
+  } else {
99
+    if (e.clientX + menuSize.width >= windowSize.width) {
100
+      x -= menuSize.width;
101
+    }
102
+  }
103
+  Blockly.WidgetDiv.position(x, y, windowSize, scrollOffset, rtl);
104
+
105
+  menu.setAllowAutoFocus(true);
106
+  // 1ms delay is required for focusing on context menus because some other
107
+  // mouse event is still waiting in the queue and clears focus.
108
+  setTimeout(function() {menuDom.focus();}, 1);
109
+  Blockly.ContextMenu.currentBlock = null;  // May be set by Blockly.Block.
110
+};
111
+
112
+/**
113
+ * Hide the context menu.
114
+ */
115
+Blockly.ContextMenu.hide = function() {
116
+  Blockly.WidgetDiv.hideIfOwner(Blockly.ContextMenu);
117
+  Blockly.ContextMenu.currentBlock = null;
118
+};
119
+
120
+/**
121
+ * Create a callback function that creates and configures a block,
122
+ *   then places the new block next to the original.
123
+ * @param {!Blockly.Block} block Original block.
124
+ * @param {!Element} xml XML representation of new block.
125
+ * @return {!Function} Function that creates a block.
126
+ */
127
+Blockly.ContextMenu.callbackFactory = function(block, xml) {
128
+  return function() {
129
+    var newBlock = Blockly.Xml.domToBlock(block.workspace, xml);
130
+    // Move the new block next to the old block.
131
+    var xy = block.getRelativeToSurfaceXY();
132
+    if (block.RTL) {
133
+      xy.x -= Blockly.SNAP_RADIUS;
134
+    } else {
135
+      xy.x += Blockly.SNAP_RADIUS;
136
+    }
137
+    xy.y += Blockly.SNAP_RADIUS * 2;
138
+    newBlock.moveBy(xy.x, xy.y);
139
+    newBlock.select();
140
+  };
141
+};

+ 732 - 0
src/blockly/core/css.js

@@ -0,0 +1,732 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2013 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Inject Blockly's CSS synchronously.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Css');
28
+
29
+
30
+/**
31
+ * List of cursors.
32
+ * @enum {string}
33
+ */
34
+Blockly.Css.Cursor = {
35
+  OPEN: 'handopen',
36
+  CLOSED: 'handclosed',
37
+  DELETE: 'handdelete'
38
+};
39
+
40
+/**
41
+ * Current cursor (cached value).
42
+ * @type {string}
43
+ * @private
44
+ */
45
+Blockly.Css.currentCursor_ = '';
46
+
47
+/**
48
+ * Large stylesheet added by Blockly.Css.inject.
49
+ * @type {Element}
50
+ * @private
51
+ */
52
+Blockly.Css.styleSheet_ = null;
53
+
54
+/**
55
+ * Path to media directory, with any trailing slash removed.
56
+ * @type {string}
57
+ * @private
58
+ */
59
+Blockly.Css.mediaPath_ = '';
60
+
61
+/**
62
+ * Inject the CSS into the DOM.  This is preferable over using a regular CSS
63
+ * file since:
64
+ * a) It loads synchronously and doesn't force a redraw later.
65
+ * b) It speeds up loading by not blocking on a separate HTTP transfer.
66
+ * c) The CSS content may be made dynamic depending on init options.
67
+ * @param {boolean} hasCss If false, don't inject CSS
68
+ *     (providing CSS becomes the document's responsibility).
69
+ * @param {string} pathToMedia Path from page to the Blockly media directory.
70
+ */
71
+Blockly.Css.inject = function(hasCss, pathToMedia) {
72
+  // Only inject the CSS once.
73
+  if (Blockly.Css.styleSheet_) {
74
+    return;
75
+  }
76
+  // Placeholder for cursor rule.  Must be first rule (index 0).
77
+  var text = '.blocklyDraggable {}\n';
78
+  if (hasCss) {
79
+    text += Blockly.Css.CONTENT.join('\n');
80
+    if (Blockly.FieldDate) {
81
+      text += Blockly.FieldDate.CSS.join('\n');
82
+    }
83
+  }
84
+  // Strip off any trailing slash (either Unix or Windows).
85
+  Blockly.Css.mediaPath_ = pathToMedia.replace(/[\\\/]$/, '');
86
+  text = text.replace(/<<<PATH>>>/g, Blockly.Css.mediaPath_);
87
+  // Inject CSS tag.
88
+  var cssNode = document.createElement('style');
89
+  document.head.appendChild(cssNode);
90
+  var cssTextNode = document.createTextNode(text);
91
+  cssNode.appendChild(cssTextNode);
92
+  Blockly.Css.styleSheet_ = cssNode.sheet;
93
+  Blockly.Css.setCursor(Blockly.Css.Cursor.OPEN);
94
+};
95
+
96
+/**
97
+ * Set the cursor to be displayed when over something draggable.
98
+ * @param {Blockly.Css.Cursor} cursor Enum.
99
+ */
100
+Blockly.Css.setCursor = function(cursor) {
101
+  if (Blockly.Css.currentCursor_ == cursor) {
102
+    return;
103
+  }
104
+  Blockly.Css.currentCursor_ = cursor;
105
+  var url = 'url(' + Blockly.Css.mediaPath_ + '/' + cursor + '.cur), auto';
106
+  // There are potentially hundreds of draggable objects.  Changing their style
107
+  // properties individually is too slow, so change the CSS rule instead.
108
+  var rule = '.blocklyDraggable {\n  cursor: ' + url + ';\n}\n';
109
+  Blockly.Css.styleSheet_.deleteRule(0);
110
+  Blockly.Css.styleSheet_.insertRule(rule, 0);
111
+  // There is probably only one toolbox, so just change its style property.
112
+  var toolboxen = document.getElementsByClassName('blocklyToolboxDiv');
113
+  for (var i = 0, toolbox; toolbox = toolboxen[i]; i++) {
114
+    if (cursor == Blockly.Css.Cursor.DELETE) {
115
+      toolbox.style.cursor = url;
116
+    } else {
117
+      toolbox.style.cursor = '';
118
+    }
119
+  }
120
+  // Set cursor on the whole document, so that rapid movements
121
+  // don't result in cursor changing to an arrow momentarily.
122
+  var html = document.body.parentNode;
123
+  if (cursor == Blockly.Css.Cursor.OPEN) {
124
+    html.style.cursor = '';
125
+  } else {
126
+    html.style.cursor = url;
127
+  }
128
+};
129
+
130
+/**
131
+ * Array making up the CSS content for Blockly.
132
+ */
133
+Blockly.Css.CONTENT = [
134
+  '.blocklySvg {',
135
+    'background-color: #fff;',
136
+    'outline: none;',
137
+    'overflow: hidden;',  /* IE overflows by default. */
138
+  '}',
139
+
140
+  '.blocklyWidgetDiv {',
141
+    'display: none;',
142
+    'position: absolute;',
143
+    'z-index: 999;',
144
+  '}',
145
+
146
+  '.blocklyTooltipDiv {',
147
+    'background-color: #ffffc7;',
148
+    'border: 1px solid #ddc;',
149
+    'box-shadow: 4px 4px 20px 1px rgba(0,0,0,.15);',
150
+    'color: #000;',
151
+    'display: none;',
152
+    'font-family: sans-serif;',
153
+    'font-size: 9pt;',
154
+    'opacity: 0.9;',
155
+    'padding: 2px;',
156
+    'position: absolute;',
157
+    'z-index: 1000;',
158
+  '}',
159
+
160
+  '.blocklyResizeSE {',
161
+    'cursor: se-resize;',
162
+    'fill: #aaa;',
163
+  '}',
164
+
165
+  '.blocklyResizeSW {',
166
+    'cursor: sw-resize;',
167
+    'fill: #aaa;',
168
+  '}',
169
+
170
+  '.blocklyResizeLine {',
171
+    'stroke: #888;',
172
+    'stroke-width: 1;',
173
+  '}',
174
+
175
+  '.blocklyHighlightedConnectionPath {',
176
+    'fill: none;',
177
+    'stroke: #fc3;',
178
+    'stroke-width: 4px;',
179
+  '}',
180
+
181
+  '.blocklyPathLight {',
182
+    'fill: none;',
183
+    'stroke-linecap: round;',
184
+    'stroke-width: 1;',
185
+  '}',
186
+
187
+  '.blocklySelected>.blocklyPath {',
188
+    'stroke: #fc3;',
189
+    'stroke-width: 3px;',
190
+  '}',
191
+
192
+  '.blocklySelected>.blocklyPathLight {',
193
+    'display: none;',
194
+  '}',
195
+
196
+  '.blocklyDragging>.blocklyPath,',
197
+  '.blocklyDragging>.blocklyPathLight {',
198
+    'fill-opacity: .8;',
199
+    'stroke-opacity: .8;',
200
+  '}',
201
+
202
+  '.blocklyDragging>.blocklyPathDark {',
203
+    'display: none;',
204
+  '}',
205
+
206
+  '.blocklyDisabled>.blocklyPath {',
207
+    'fill-opacity: .5;',
208
+    'stroke-opacity: .5;',
209
+  '}',
210
+
211
+  '.blocklyDisabled>.blocklyPathLight,',
212
+  '.blocklyDisabled>.blocklyPathDark {',
213
+    'display: none;',
214
+  '}',
215
+
216
+  '.blocklyText {',
217
+    'cursor: default;',
218
+    'fill: #fff;',
219
+    'font-family: sans-serif;',
220
+    'font-size: 11pt;',
221
+  '}',
222
+
223
+  '.blocklyNonEditableText>text {',
224
+    'pointer-events: none;',
225
+  '}',
226
+
227
+  '.blocklyNonEditableText>rect,',
228
+  '.blocklyEditableText>rect {',
229
+    'fill: #fff;',
230
+    'fill-opacity: .6;',
231
+  '}',
232
+
233
+  '.blocklyNonEditableText>text,',
234
+  '.blocklyEditableText>text {',
235
+    'fill: #000;',
236
+  '}',
237
+
238
+  '.blocklyEditableText:hover>rect {',
239
+    'stroke: #fff;',
240
+    'stroke-width: 2;',
241
+  '}',
242
+
243
+  '.blocklyBubbleText {',
244
+    'fill: #000;',
245
+  '}',
246
+
247
+  /*
248
+    Don't allow users to select text.  It gets annoying when trying to
249
+    drag a block and selected text moves instead.
250
+  */
251
+  '.blocklySvg text {',
252
+    'user-select: none;',
253
+    '-moz-user-select: none;',
254
+    '-webkit-user-select: none;',
255
+    'cursor: inherit;',
256
+  '}',
257
+
258
+  '.blocklyHidden {',
259
+    'display: none;',
260
+  '}',
261
+
262
+  '.blocklyFieldDropdown:not(.blocklyHidden) {',
263
+    'display: block;',
264
+  '}',
265
+
266
+  '.blocklyIconGroup {',
267
+    'cursor: default;',
268
+  '}',
269
+
270
+  '.blocklyIconGroup:not(:hover),',
271
+  '.blocklyIconGroupReadonly {',
272
+    'opacity: .6;',
273
+  '}',
274
+
275
+  '.blocklyMinimalBody {',
276
+    'margin: 0;',
277
+    'padding: 0;',
278
+  '}',
279
+
280
+  '.blocklyCommentTextarea {',
281
+    'background-color: #ffc;',
282
+    'border: 0;',
283
+    'margin: 0;',
284
+    'padding: 2px;',
285
+    'resize: none;',
286
+  '}',
287
+
288
+  '.blocklyHtmlInput {',
289
+    'border: none;',
290
+    'border-radius: 4px;',
291
+    'font-family: sans-serif;',
292
+    'height: 100%;',
293
+    'margin: 0;',
294
+    'outline: none;',
295
+    'padding: 0 1px;',
296
+    'width: 100%',
297
+  '}',
298
+
299
+  '.blocklyMainBackground {',
300
+    'stroke-width: 1;',
301
+    'stroke: #c6c6c6;',  /* Equates to #ddd due to border being off-pixel. */
302
+  '}',
303
+
304
+  '.blocklyMutatorBackground {',
305
+    'fill: #fff;',
306
+    'stroke: #ddd;',
307
+    'stroke-width: 1;',
308
+  '}',
309
+
310
+  '.blocklyFlyoutBackground {',
311
+    'fill: #ddd;',
312
+    'fill-opacity: .8;',
313
+  '}',
314
+
315
+  '.blocklyScrollbarBackground {',
316
+    'opacity: 0;',
317
+  '}',
318
+
319
+  '.blocklyScrollbarKnob {',
320
+    'fill: #ccc;',
321
+  '}',
322
+
323
+  '.blocklyScrollbarBackground:hover+.blocklyScrollbarKnob,',
324
+  '.blocklyScrollbarKnob:hover {',
325
+    'fill: #bbb;',
326
+  '}',
327
+
328
+  '.blocklyZoom>image {',
329
+    'opacity: .4;',
330
+  '}',
331
+
332
+  '.blocklyZoom>image:hover {',
333
+    'opacity: .6;',
334
+  '}',
335
+
336
+  '.blocklyZoom>image:active {',
337
+    'opacity: .8;',
338
+  '}',
339
+
340
+  /* Darken flyout scrollbars due to being on a grey background. */
341
+  /* By contrast, workspace scrollbars are on a white background. */
342
+  '.blocklyFlyout .blocklyScrollbarKnob {',
343
+    'fill: #bbb;',
344
+  '}',
345
+
346
+  '.blocklyFlyout .blocklyScrollbarBackground:hover+.blocklyScrollbarKnob,',
347
+  '.blocklyFlyout .blocklyScrollbarKnob:hover {',
348
+    'fill: #aaa;',
349
+  '}',
350
+
351
+  '.blocklyInvalidInput {',
352
+    'background: #faa;',
353
+  '}',
354
+
355
+  '.blocklyAngleCircle {',
356
+    'stroke: #444;',
357
+    'stroke-width: 1;',
358
+    'fill: #ddd;',
359
+    'fill-opacity: .8;',
360
+  '}',
361
+
362
+  '.blocklyAngleMarks {',
363
+    'stroke: #444;',
364
+    'stroke-width: 1;',
365
+  '}',
366
+
367
+  '.blocklyAngleGauge {',
368
+    'fill: #f88;',
369
+    'fill-opacity: .8;  ',
370
+  '}',
371
+
372
+  '.blocklyAngleLine {',
373
+    'stroke: #f00;',
374
+    'stroke-width: 2;',
375
+    'stroke-linecap: round;',
376
+  '}',
377
+
378
+  '.blocklyContextMenu {',
379
+    'border-radius: 4px;',
380
+  '}',
381
+
382
+  '.blocklyDropdownMenu {',
383
+    'padding: 0 !important;',
384
+  '}',
385
+
386
+  /* Override the default Closure URL. */
387
+  '.blocklyWidgetDiv .goog-option-selected .goog-menuitem-checkbox,',
388
+  '.blocklyWidgetDiv .goog-option-selected .goog-menuitem-icon {',
389
+    'background: url(<<<PATH>>>/sprites.png) no-repeat -48px -16px !important;',
390
+  '}',
391
+
392
+  /* Category tree in Toolbox. */
393
+  '.blocklyToolboxDiv {',
394
+    'background-color: #ddd;',
395
+    'overflow-x: visible;',
396
+    'overflow-y: auto;',
397
+    'position: absolute;',
398
+  '}',
399
+
400
+  '.blocklyTreeRoot {',
401
+    'padding: 4px 0;',
402
+  '}',
403
+
404
+  '.blocklyTreeRoot:focus {',
405
+    'outline: none;',
406
+  '}',
407
+
408
+  '.blocklyTreeRow {',
409
+    'height: 22px;',
410
+    'line-height: 22px;',
411
+    'margin-bottom: 3px;',
412
+    'padding-right: 8px;',
413
+    'white-space: nowrap;',
414
+  '}',
415
+
416
+  '.blocklyToolboxDiv[dir="RTL"] .blocklyTreeRow {',
417
+    'margin-left: 8px;',
418
+  '}',
419
+
420
+  '.blocklyTreeRow:not(.blocklyTreeSelected):hover {',
421
+    'background-color: #e4e4e4;',
422
+  '}',
423
+
424
+  '.blocklyTreeSeparator {',
425
+    'border-bottom: solid #e5e5e5 1px;',
426
+    'height: 0px;',
427
+    'margin: 5px 0;',
428
+  '}',
429
+
430
+  '.blocklyTreeIcon {',
431
+    'background-image: url(<<<PATH>>>/sprites.png);',
432
+    'height: 16px;',
433
+    'vertical-align: middle;',
434
+    'width: 16px;',
435
+  '}',
436
+
437
+  '.blocklyTreeIconClosedLtr {',
438
+    'background-position: -32px -1px;',
439
+  '}',
440
+
441
+  '.blocklyTreeIconClosedRtl {',
442
+    'background-position: 0px -1px;',
443
+  '}',
444
+
445
+  '.blocklyTreeIconOpen {',
446
+    'background-position: -16px -1px;',
447
+  '}',
448
+
449
+  '.blocklyTreeSelected>.blocklyTreeIconClosedLtr {',
450
+    'background-position: -32px -17px;',
451
+  '}',
452
+
453
+  '.blocklyTreeSelected>.blocklyTreeIconClosedRtl {',
454
+    'background-position: 0px -17px;',
455
+  '}',
456
+
457
+  '.blocklyTreeSelected>.blocklyTreeIconOpen {',
458
+    'background-position: -16px -17px;',
459
+  '}',
460
+
461
+  '.blocklyTreeIconNone,',
462
+  '.blocklyTreeSelected>.blocklyTreeIconNone {',
463
+    'background-position: -48px -1px;',
464
+  '}',
465
+
466
+  '.blocklyTreeLabel {',
467
+    'cursor: default;',
468
+    'font-family: sans-serif;',
469
+    'font-size: 16px;',
470
+    'padding: 0 3px;',
471
+    'vertical-align: middle;',
472
+  '}',
473
+
474
+  '.blocklyTreeSelected .blocklyTreeLabel {',
475
+    'color: #fff;',
476
+  '}',
477
+
478
+  /* Copied from: goog/css/colorpicker-simplegrid.css */
479
+  /*
480
+   * Copyright 2007 The Closure Library Authors. All Rights Reserved.
481
+   *
482
+   * Use of this source code is governed by the Apache License, Version 2.0.
483
+   * See the COPYING file for details.
484
+   */
485
+
486
+  /* Author: pupius@google.com (Daniel Pupius) */
487
+
488
+  /*
489
+    Styles to make the colorpicker look like the old gmail color picker
490
+    NOTE: without CSS scoping this will override styles defined in palette.css
491
+  */
492
+  '.blocklyWidgetDiv .goog-palette {',
493
+    'outline: none;',
494
+    'cursor: default;',
495
+  '}',
496
+
497
+  '.blocklyWidgetDiv .goog-palette-table {',
498
+    'border: 1px solid #666;',
499
+    'border-collapse: collapse;',
500
+  '}',
501
+
502
+  '.blocklyWidgetDiv .goog-palette-cell {',
503
+    'height: 13px;',
504
+    'width: 15px;',
505
+    'margin: 0;',
506
+    'border: 0;',
507
+    'text-align: center;',
508
+    'vertical-align: middle;',
509
+    'border-right: 1px solid #666;',
510
+    'font-size: 1px;',
511
+  '}',
512
+
513
+  '.blocklyWidgetDiv .goog-palette-colorswatch {',
514
+    'position: relative;',
515
+    'height: 13px;',
516
+    'width: 15px;',
517
+    'border: 1px solid #666;',
518
+  '}',
519
+
520
+  '.blocklyWidgetDiv .goog-palette-cell-hover .goog-palette-colorswatch {',
521
+    'border: 1px solid #FFF;',
522
+  '}',
523
+
524
+  '.blocklyWidgetDiv .goog-palette-cell-selected .goog-palette-colorswatch {',
525
+    'border: 1px solid #000;',
526
+    'color: #fff;',
527
+  '}',
528
+
529
+  /* Copied from: goog/css/menu.css */
530
+  /*
531
+   * Copyright 2009 The Closure Library Authors. All Rights Reserved.
532
+   *
533
+   * Use of this source code is governed by the Apache License, Version 2.0.
534
+   * See the COPYING file for details.
535
+   */
536
+
537
+  /**
538
+   * Standard styling for menus created by goog.ui.MenuRenderer.
539
+   *
540
+   * @author attila@google.com (Attila Bodis)
541
+   */
542
+
543
+  '.blocklyWidgetDiv .goog-menu {',
544
+    'background: #fff;',
545
+    'border-color: #ccc #666 #666 #ccc;',
546
+    'border-style: solid;',
547
+    'border-width: 1px;',
548
+    'cursor: default;',
549
+    'font: normal 13px Arial, sans-serif;',
550
+    'margin: 0;',
551
+    'outline: none;',
552
+    'padding: 4px 0;',
553
+    'position: absolute;',
554
+    'overflow-y: auto;',
555
+    'overflow-x: hidden;',
556
+    'max-height: 100%;',
557
+    'z-index: 20000;',  /* Arbitrary, but some apps depend on it... */
558
+  '}',
559
+
560
+  /* Copied from: goog/css/menuitem.css */
561
+  /*
562
+   * Copyright 2009 The Closure Library Authors. All Rights Reserved.
563
+   *
564
+   * Use of this source code is governed by the Apache License, Version 2.0.
565
+   * See the COPYING file for details.
566
+   */
567
+
568
+  /**
569
+   * Standard styling for menus created by goog.ui.MenuItemRenderer.
570
+   *
571
+   * @author attila@google.com (Attila Bodis)
572
+   */
573
+
574
+  /**
575
+   * State: resting.
576
+   *
577
+   * NOTE(mleibman,chrishenry):
578
+   * The RTL support in Closure is provided via two mechanisms -- "rtl" CSS
579
+   * classes and BiDi flipping done by the CSS compiler.  Closure supports RTL
580
+   * with or without the use of the CSS compiler.  In order for them not
581
+   * to conflict with each other, the "rtl" CSS classes need to have the #noflip
582
+   * annotation.  The non-rtl counterparts should ideally have them as well, but,
583
+   * since .goog-menuitem existed without .goog-menuitem-rtl for so long before
584
+   * being added, there is a risk of people having templates where they are not
585
+   * rendering the .goog-menuitem-rtl class when in RTL and instead rely solely
586
+   * on the BiDi flipping by the CSS compiler.  That's why we're not adding the
587
+   * #noflip to .goog-menuitem.
588
+   */
589
+  '.blocklyWidgetDiv .goog-menuitem {',
590
+    'color: #000;',
591
+    'font: normal 13px Arial, sans-serif;',
592
+    'list-style: none;',
593
+    'margin: 0;',
594
+     /* 28px on the left for icon or checkbox; 7em on the right for shortcut. */
595
+    'padding: 4px 7em 4px 28px;',
596
+    'white-space: nowrap;',
597
+  '}',
598
+
599
+  /* BiDi override for the resting state. */
600
+  /* #noflip */
601
+  '.blocklyWidgetDiv .goog-menuitem.goog-menuitem-rtl {',
602
+     /* Flip left/right padding for BiDi. */
603
+    'padding-left: 7em;',
604
+    'padding-right: 28px;',
605
+  '}',
606
+
607
+  /* If a menu doesn't have checkable items or items with icons, remove padding. */
608
+  '.blocklyWidgetDiv .goog-menu-nocheckbox .goog-menuitem,',
609
+  '.blocklyWidgetDiv .goog-menu-noicon .goog-menuitem {',
610
+    'padding-left: 12px;',
611
+  '}',
612
+
613
+  /*
614
+   * If a menu doesn't have items with shortcuts, leave just enough room for
615
+   * submenu arrows, if they are rendered.
616
+   */
617
+  '.blocklyWidgetDiv .goog-menu-noaccel .goog-menuitem {',
618
+    'padding-right: 20px;',
619
+  '}',
620
+
621
+  '.blocklyWidgetDiv .goog-menuitem-content {',
622
+    'color: #000;',
623
+    'font: normal 13px Arial, sans-serif;',
624
+  '}',
625
+
626
+  /* State: disabled. */
627
+  '.blocklyWidgetDiv .goog-menuitem-disabled .goog-menuitem-accel,',
628
+  '.blocklyWidgetDiv .goog-menuitem-disabled .goog-menuitem-content {',
629
+    'color: #ccc !important;',
630
+  '}',
631
+
632
+  '.blocklyWidgetDiv .goog-menuitem-disabled .goog-menuitem-icon {',
633
+    'opacity: 0.3;',
634
+    '-moz-opacity: 0.3;',
635
+    'filter: alpha(opacity=30);',
636
+  '}',
637
+
638
+  /* State: hover. */
639
+  '.blocklyWidgetDiv .goog-menuitem-highlight,',
640
+  '.blocklyWidgetDiv .goog-menuitem-hover {',
641
+    'background-color: #d6e9f8;',
642
+     /* Use an explicit top and bottom border so that the selection is visible',
643
+      * in high contrast mode. */
644
+    'border-color: #d6e9f8;',
645
+    'border-style: dotted;',
646
+    'border-width: 1px 0;',
647
+    'padding-bottom: 3px;',
648
+    'padding-top: 3px;',
649
+  '}',
650
+
651
+  /* State: selected/checked. */
652
+  '.blocklyWidgetDiv .goog-menuitem-checkbox,',
653
+  '.blocklyWidgetDiv .goog-menuitem-icon {',
654
+    'background-repeat: no-repeat;',
655
+    'height: 16px;',
656
+    'left: 6px;',
657
+    'position: absolute;',
658
+    'right: auto;',
659
+    'vertical-align: middle;',
660
+    'width: 16px;',
661
+  '}',
662
+
663
+  /* BiDi override for the selected/checked state. */
664
+  /* #noflip */
665
+  '.blocklyWidgetDiv .goog-menuitem-rtl .goog-menuitem-checkbox,',
666
+  '.blocklyWidgetDiv .goog-menuitem-rtl .goog-menuitem-icon {',
667
+     /* Flip left/right positioning. */
668
+    'left: auto;',
669
+    'right: 6px;',
670
+  '}',
671
+
672
+  '.blocklyWidgetDiv .goog-option-selected .goog-menuitem-checkbox,',
673
+  '.blocklyWidgetDiv .goog-option-selected .goog-menuitem-icon {',
674
+     /* Client apps may override the URL at which they serve the sprite. */
675
+    'background: url(//ssl.gstatic.com/editor/editortoolbar.png) no-repeat -512px 0;',
676
+  '}',
677
+
678
+  /* Keyboard shortcut ("accelerator") style. */
679
+  '.blocklyWidgetDiv .goog-menuitem-accel {',
680
+    'color: #999;',
681
+     /* Keyboard shortcuts are untranslated; always left-to-right. */
682
+     /* #noflip */
683
+    'direction: ltr;',
684
+    'left: auto;',
685
+    'padding: 0 6px;',
686
+    'position: absolute;',
687
+    'right: 0;',
688
+    'text-align: right;',
689
+  '}',
690
+
691
+  /* BiDi override for shortcut style. */
692
+  /* #noflip */
693
+  '.blocklyWidgetDiv .goog-menuitem-rtl .goog-menuitem-accel {',
694
+     /* Flip left/right positioning and text alignment. */
695
+    'left: 0;',
696
+    'right: auto;',
697
+    'text-align: left;',
698
+  '}',
699
+
700
+  /* Mnemonic styles. */
701
+  '.blocklyWidgetDiv .goog-menuitem-mnemonic-hint {',
702
+    'text-decoration: underline;',
703
+  '}',
704
+
705
+  '.blocklyWidgetDiv .goog-menuitem-mnemonic-separator {',
706
+    'color: #999;',
707
+    'font-size: 12px;',
708
+    'padding-left: 4px;',
709
+  '}',
710
+
711
+  /* Copied from: goog/css/menuseparator.css */
712
+  /*
713
+   * Copyright 2009 The Closure Library Authors. All Rights Reserved.
714
+   *
715
+   * Use of this source code is governed by the Apache License, Version 2.0.
716
+   * See the COPYING file for details.
717
+   */
718
+
719
+  /**
720
+   * Standard styling for menus created by goog.ui.MenuSeparatorRenderer.
721
+   *
722
+   * @author attila@google.com (Attila Bodis)
723
+   */
724
+
725
+  '.blocklyWidgetDiv .goog-menuseparator {',
726
+    'border-top: 1px solid #ccc;',
727
+    'margin: 4px 0;',
728
+    'padding: 0;',
729
+  '}',
730
+
731
+  ''
732
+];

+ 414 - 0
src/blockly/core/field.js

@@ -0,0 +1,414 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Input field.  Used for editable titles, variables, etc.
23
+ * This is an abstract class that defines the UI on the block.  Actual
24
+ * instances would be Blockly.FieldTextInput, Blockly.FieldDropdown, etc.
25
+ * @author fraser@google.com (Neil Fraser)
26
+ */
27
+'use strict';
28
+
29
+goog.provide('Blockly.Field');
30
+
31
+goog.require('goog.asserts');
32
+goog.require('goog.dom');
33
+goog.require('goog.math.Size');
34
+goog.require('goog.style');
35
+goog.require('goog.userAgent');
36
+
37
+
38
+/**
39
+ * Class for an editable field.
40
+ * @param {string} text The initial content of the field.
41
+ * @constructor
42
+ */
43
+Blockly.Field = function(text) {
44
+  this.size_ = new goog.math.Size(0, 25);
45
+  this.setText(text);
46
+};
47
+
48
+/**
49
+ * Temporary cache of text widths.
50
+ * @type {Object}
51
+ * @private
52
+ */
53
+Blockly.Field.cacheWidths_ = null;
54
+
55
+/**
56
+ * Number of current references to cache.
57
+ * @type {number}
58
+ * @private
59
+ */
60
+Blockly.Field.cacheReference_ = 0;
61
+
62
+/**
63
+ * Maximum characters of text to display before adding an ellipsis.
64
+ */
65
+Blockly.Field.prototype.maxDisplayLength = 50;
66
+
67
+/**
68
+ * Block this field is attached to.  Starts as null, then in set in init.
69
+ * @private
70
+ */
71
+Blockly.Field.prototype.sourceBlock_ = null;
72
+
73
+/**
74
+ * Is the field visible, or hidden due to the block being collapsed?
75
+ * @private
76
+ */
77
+Blockly.Field.prototype.visible_ = true;
78
+
79
+/**
80
+ * Change handler called when user edits an editable field.
81
+ * @private
82
+ */
83
+Blockly.Field.prototype.changeHandler_ = null;
84
+
85
+/**
86
+ * Non-breaking space.
87
+ */
88
+Blockly.Field.NBSP = '\u00A0';
89
+
90
+/**
91
+ * Editable fields are saved by the XML renderer, non-editable fields are not.
92
+ */
93
+Blockly.Field.prototype.EDITABLE = true;
94
+
95
+/**
96
+ * Install this field on a block.
97
+ * @param {!Blockly.Block} block The block containing this field.
98
+ */
99
+Blockly.Field.prototype.init = function(block) {
100
+  if (this.sourceBlock_) {
101
+    // Field has already been initialized once.
102
+    return;
103
+  }
104
+  this.sourceBlock_ = block;
105
+  // Build the DOM.
106
+  this.fieldGroup_ = Blockly.createSvgElement('g', {}, null);
107
+  if (!this.visible_) {
108
+    this.fieldGroup_.style.display = 'none';
109
+  }
110
+  this.borderRect_ = Blockly.createSvgElement('rect',
111
+      {'rx': 4,
112
+       'ry': 4,
113
+       'x': -Blockly.BlockSvg.SEP_SPACE_X / 2,
114
+       'y': 0,
115
+       'height': 16}, this.fieldGroup_, this.sourceBlock_.workspace);
116
+  /** @type {!Element} */
117
+  this.textElement_ = Blockly.createSvgElement('text',
118
+      {'class': 'blocklyText', 'y': this.size_.height - 12.5},
119
+      this.fieldGroup_);
120
+
121
+  this.updateEditable();
122
+  block.getSvgRoot().appendChild(this.fieldGroup_);
123
+  this.mouseUpWrapper_ =
124
+      Blockly.bindEvent_(this.fieldGroup_, 'mouseup', this, this.onMouseUp_);
125
+  // Force a render.
126
+  this.updateTextNode_();
127
+};
128
+
129
+/**
130
+ * Dispose of all DOM objects belonging to this editable field.
131
+ */
132
+Blockly.Field.prototype.dispose = function() {
133
+  if (this.mouseUpWrapper_) {
134
+    Blockly.unbindEvent_(this.mouseUpWrapper_);
135
+    this.mouseUpWrapper_ = null;
136
+  }
137
+  this.sourceBlock_ = null;
138
+  goog.dom.removeNode(this.fieldGroup_);
139
+  this.fieldGroup_ = null;
140
+  this.textElement_ = null;
141
+  this.borderRect_ = null;
142
+  this.changeHandler_ = null;
143
+};
144
+
145
+/**
146
+ * Add or remove the UI indicating if this field is editable or not.
147
+ */
148
+Blockly.Field.prototype.updateEditable = function() {
149
+  if (!this.EDITABLE || !this.sourceBlock_) {
150
+    return;
151
+  }
152
+  if (this.sourceBlock_.isEditable()) {
153
+    Blockly.addClass_(/** @type {!Element} */ (this.fieldGroup_),
154
+                      'blocklyEditableText');
155
+    Blockly.removeClass_(/** @type {!Element} */ (this.fieldGroup_),
156
+                         'blocklyNoNEditableText');
157
+    this.fieldGroup_.style.cursor = this.CURSOR;
158
+  } else {
159
+    Blockly.addClass_(/** @type {!Element} */ (this.fieldGroup_),
160
+                      'blocklyNonEditableText');
161
+    Blockly.removeClass_(/** @type {!Element} */ (this.fieldGroup_),
162
+                         'blocklyEditableText');
163
+    this.fieldGroup_.style.cursor = '';
164
+  }
165
+};
166
+
167
+/**
168
+ * Gets whether this editable field is visible or not.
169
+ * @return {boolean} True if visible.
170
+ */
171
+Blockly.Field.prototype.isVisible = function() {
172
+  return this.visible_;
173
+};
174
+
175
+/**
176
+ * Sets whether this editable field is visible or not.
177
+ * @param {boolean} visible True if visible.
178
+ */
179
+Blockly.Field.prototype.setVisible = function(visible) {
180
+  if (this.visible_ == visible) {
181
+    return;
182
+  }
183
+  this.visible_ = visible;
184
+  var root = this.getSvgRoot();
185
+  if (root) {
186
+    root.style.display = visible ? 'block' : 'none';
187
+    this.render_();
188
+  }
189
+};
190
+
191
+/**
192
+ * Sets a new change handler for editable fields.
193
+ * @param {Function} handler New change handler, or null.
194
+ */
195
+Blockly.Field.prototype.setChangeHandler = function(handler) {
196
+  this.changeHandler_ = handler;
197
+};
198
+
199
+/**
200
+ * Gets the group element for this editable field.
201
+ * Used for measuring the size and for positioning.
202
+ * @return {!Element} The group element.
203
+ */
204
+Blockly.Field.prototype.getSvgRoot = function() {
205
+  return /** @type {!Element} */ (this.fieldGroup_);
206
+};
207
+
208
+/**
209
+ * Draws the border with the correct width.
210
+ * Saves the computed width in a property.
211
+ * @private
212
+ */
213
+Blockly.Field.prototype.render_ = function() {
214
+  if (this.visible_ && this.textElement_) {
215
+    var key = this.textElement_.textContent + '\n' +
216
+        this.textElement_.className.baseVal;
217
+    if (Blockly.Field.cacheWidths_ && Blockly.Field.cacheWidths_[key]) {
218
+      var width = Blockly.Field.cacheWidths_[key];
219
+    } else {
220
+      try {
221
+        var width = this.textElement_.getComputedTextLength();
222
+      } catch (e) {
223
+        // MSIE 11 is known to throw "Unexpected call to method or property
224
+        // access." if Blockly is hidden.
225
+        var width = this.textElement_.textContent.length * 8;
226
+      }
227
+      if (Blockly.Field.cacheWidths_) {
228
+        Blockly.Field.cacheWidths_[key] = width;
229
+      }
230
+    }
231
+    if (this.borderRect_) {
232
+      this.borderRect_.setAttribute('width',
233
+          width + Blockly.BlockSvg.SEP_SPACE_X);
234
+    }
235
+  } else {
236
+    var width = 0;
237
+  }
238
+  this.size_.width = width;
239
+};
240
+
241
+/**
242
+ * Start caching field widths.  Every call to this function MUST also call
243
+ * stopCache.  Caches must not survive between execution threads.
244
+ * @type {Object}
245
+ * @private
246
+ */
247
+Blockly.Field.startCache = function() {
248
+  Blockly.Field.cacheReference_++;
249
+  if (!Blockly.Field.cacheWidths_) {
250
+    Blockly.Field.cacheWidths_ = {};
251
+  }
252
+};
253
+
254
+/**
255
+ * Stop caching field widths.  Unless caching was already on when the
256
+ * corresponding call to startCache was made.
257
+ * @type {number}
258
+ * @private
259
+ */
260
+Blockly.Field.stopCache = function() {
261
+  Blockly.Field.cacheReference_--;
262
+  if (!Blockly.Field.cacheReference_) {
263
+    Blockly.Field.cacheWidths_ = null;
264
+  }
265
+};
266
+
267
+/**
268
+ * Returns the height and width of the field.
269
+ * @return {!goog.math.Size} Height and width.
270
+ */
271
+Blockly.Field.prototype.getSize = function() {
272
+  if (!this.size_.width) {
273
+    this.render_();
274
+  }
275
+  return this.size_;
276
+};
277
+
278
+/**
279
+ * Returns the height and width of the field,
280
+ * accounting for the workspace scaling.
281
+ * @return {!goog.math.Size} Height and width.
282
+ */
283
+Blockly.Field.prototype.getScaledBBox_ = function() {
284
+  var bBox = this.borderRect_.getBBox();
285
+  // Create new object, as getBBox can return an uneditable SVGRect in IE.
286
+  return new goog.math.Size(bBox.width * this.sourceBlock_.workspace.scale,
287
+                            bBox.height * this.sourceBlock_.workspace.scale);
288
+};
289
+
290
+/**
291
+ * Get the text from this field.
292
+ * @return {string} Current text.
293
+ */
294
+Blockly.Field.prototype.getText = function() {
295
+  return this.text_;
296
+};
297
+
298
+/**
299
+ * Set the text in this field.  Trigger a rerender of the source block.
300
+ * @param {*} text New text.
301
+ */
302
+Blockly.Field.prototype.setText = function(text) {
303
+  if (text === null) {
304
+    // No change if null.
305
+    return;
306
+  }
307
+  text = String(text);
308
+  if (text === this.text_) {
309
+    // No change.
310
+    return;
311
+  }
312
+  this.text_ = text;
313
+  this.updateTextNode_();
314
+
315
+  if (this.sourceBlock_ && this.sourceBlock_.rendered) {
316
+    this.sourceBlock_.render();
317
+    this.sourceBlock_.bumpNeighbours_();
318
+    this.sourceBlock_.workspace.fireChangeEvent();
319
+  }
320
+};
321
+
322
+/**
323
+ * Update the text node of this field to display the current text.
324
+ * @private
325
+ */
326
+Blockly.Field.prototype.updateTextNode_ = function() {
327
+  if (!this.textElement_) {
328
+    // Not rendered yet.
329
+    return;
330
+  }
331
+  var text = this.text_;
332
+  if (text.length > this.maxDisplayLength) {
333
+    // Truncate displayed string and add an ellipsis ('...').
334
+    text = text.substring(0, this.maxDisplayLength - 2) + '\u2026';
335
+  }
336
+  // Empty the text element.
337
+  goog.dom.removeChildren(/** @type {!Element} */ (this.textElement_));
338
+  // Replace whitespace with non-breaking spaces so the text doesn't collapse.
339
+  text = text.replace(/\s/g, Blockly.Field.NBSP);
340
+  if (this.sourceBlock_.RTL && text) {
341
+    // The SVG is LTR, force text to be RTL.
342
+    text += '\u200F';
343
+  }
344
+  if (!text) {
345
+    // Prevent the field from disappearing if empty.
346
+    text = Blockly.Field.NBSP;
347
+  }
348
+  var textNode = document.createTextNode(text);
349
+  this.textElement_.appendChild(textNode);
350
+
351
+  // Cached width is obsolete.  Clear it.
352
+  this.size_.width = 0;
353
+};
354
+
355
+/**
356
+ * By default there is no difference between the human-readable text and
357
+ * the language-neutral values.  Subclasses (such as dropdown) may define this.
358
+ * @return {string} Current text.
359
+ */
360
+Blockly.Field.prototype.getValue = function() {
361
+  return this.getText();
362
+};
363
+
364
+/**
365
+ * By default there is no difference between the human-readable text and
366
+ * the language-neutral values.  Subclasses (such as dropdown) may define this.
367
+ * @param {string} text New text.
368
+ */
369
+Blockly.Field.prototype.setValue = function(text) {
370
+  this.setText(text);
371
+};
372
+
373
+/**
374
+ * Handle a mouse up event on an editable field.
375
+ * @param {!Event} e Mouse up event.
376
+ * @private
377
+ */
378
+Blockly.Field.prototype.onMouseUp_ = function(e) {
379
+  if ((goog.userAgent.IPHONE || goog.userAgent.IPAD) &&
380
+      !goog.userAgent.isVersionOrHigher('537.51.2') &&
381
+      e.layerX !== 0 && e.layerY !== 0) {
382
+    // Old iOS spawns a bogus event on the next touch after a 'prompt()' edit.
383
+    // Unlike the real events, these have a layerX and layerY set.
384
+    return;
385
+  } else if (Blockly.isRightButton(e)) {
386
+    // Right-click.
387
+    return;
388
+  } else if (Blockly.dragMode_ == 2) {
389
+    // Drag operation is concluding.  Don't open the editor.
390
+    return;
391
+  } else if (this.sourceBlock_.isEditable()) {
392
+    // Non-abstract sub-classes must define a showEditor_ method.
393
+    this.showEditor_();
394
+  }
395
+};
396
+
397
+/**
398
+ * Change the tooltip text for this field.
399
+ * @param {string|!Element} newTip Text for tooltip or a parent element to
400
+ *     link to for its tooltip.
401
+ */
402
+Blockly.Field.prototype.setTooltip = function(newTip) {
403
+  // Non-abstract sub-classes may wish to implement this.  See FieldLabel.
404
+};
405
+
406
+/**
407
+ * Return the absolute coordinates of the top-left corner of this field.
408
+ * The origin (0,0) is the top-left corner of the page body.
409
+ * @return {{!goog.math.Coordinate}} Object with .x and .y properties.
410
+ * @private
411
+ */
412
+Blockly.Field.prototype.getAbsoluteXY_ = function() {
413
+  return goog.style.getPageOffset(this.borderRect_);
414
+};

+ 279 - 0
src/blockly/core/field_angle.js

@@ -0,0 +1,279 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2013 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Angle input field.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.FieldAngle');
28
+
29
+goog.require('Blockly.FieldTextInput');
30
+goog.require('goog.math');
31
+goog.require('goog.userAgent');
32
+
33
+
34
+/**
35
+ * Class for an editable angle field.
36
+ * @param {string} text The initial content of the field.
37
+ * @param {Function=} opt_changeHandler An optional function that is called
38
+ *     to validate any constraints on what the user entered.  Takes the new
39
+ *     text as an argument and returns the accepted text or null to abort
40
+ *     the change.
41
+ * @extends {Blockly.FieldTextInput}
42
+ * @constructor
43
+ */
44
+Blockly.FieldAngle = function(text, opt_changeHandler) {
45
+  // Add degree symbol: "360°" (LTR) or "°360" (RTL)
46
+  this.symbol_ = Blockly.createSvgElement('tspan', {}, null);
47
+  this.symbol_.appendChild(document.createTextNode('\u00B0'));
48
+
49
+  Blockly.FieldAngle.superClass_.constructor.call(this, text, null);
50
+  this.setChangeHandler(opt_changeHandler);
51
+};
52
+goog.inherits(Blockly.FieldAngle, Blockly.FieldTextInput);
53
+
54
+/**
55
+ * Sets a new change handler for angle field.
56
+ * @param {Function} handler New change handler, or null.
57
+ */
58
+Blockly.FieldAngle.prototype.setChangeHandler = function(handler) {
59
+  var wrappedHandler;
60
+  if (handler) {
61
+    // Wrap the user's change handler together with the angle validator.
62
+    wrappedHandler = function(value) {
63
+      var v1 = handler.call(this, value);
64
+      if (v1 === null) {
65
+        var v2 = v1;
66
+      } else {
67
+        if (v1 === undefined) {
68
+          v1 = value;
69
+        }
70
+        var v2 = Blockly.FieldAngle.angleValidator.call(this, v1);
71
+        if (v2 !== undefined) {
72
+          v2 = v1;
73
+        }
74
+      }
75
+      return v2 === value ? undefined : v2;
76
+    };
77
+  } else {
78
+    wrappedHandler = Blockly.FieldAngle.angleValidator;
79
+  }
80
+  Blockly.FieldAngle.superClass_.setChangeHandler.call(this, wrappedHandler);
81
+};
82
+
83
+/**
84
+ * Round angles to the nearest 15 degrees when using mouse.
85
+ * Set to 0 to disable rounding.
86
+ */
87
+Blockly.FieldAngle.ROUND = 15;
88
+
89
+/**
90
+ * Half the width of protractor image.
91
+ */
92
+Blockly.FieldAngle.HALF = 100 / 2;
93
+
94
+/**
95
+ * Radius of protractor circle.  Slightly smaller than protractor size since
96
+ * otherwise SVG crops off half the border at the edges.
97
+ */
98
+Blockly.FieldAngle.RADIUS = Blockly.FieldAngle.HALF - 1;
99
+
100
+/**
101
+ * Clean up this FieldAngle, as well as the inherited FieldTextInput.
102
+ * @return {!Function} Closure to call on destruction of the WidgetDiv.
103
+ * @private
104
+ */
105
+Blockly.FieldAngle.prototype.dispose_ = function() {
106
+  var thisField = this;
107
+  return function() {
108
+    Blockly.FieldAngle.superClass_.dispose_.call(thisField)();
109
+    thisField.gauge_ = null;
110
+    if (thisField.clickWrapper_) {
111
+      Blockly.unbindEvent_(thisField.clickWrapper_);
112
+    }
113
+    if (thisField.moveWrapper1_) {
114
+      Blockly.unbindEvent_(thisField.moveWrapper1_);
115
+    }
116
+    if (thisField.moveWrapper2_) {
117
+      Blockly.unbindEvent_(thisField.moveWrapper2_);
118
+    }
119
+  };
120
+};
121
+
122
+/**
123
+ * Show the inline free-text editor on top of the text.
124
+ * @private
125
+ */
126
+Blockly.FieldAngle.prototype.showEditor_ = function() {
127
+  var noFocus =
128
+      goog.userAgent.MOBILE || goog.userAgent.ANDROID || goog.userAgent.IPAD;
129
+  // Mobile browsers have issues with in-line textareas (focus & keyboards).
130
+  Blockly.FieldAngle.superClass_.showEditor_.call(this, noFocus);
131
+  var div = Blockly.WidgetDiv.DIV;
132
+  if (!div.firstChild) {
133
+    // Mobile interface uses window.prompt.
134
+    return;
135
+  }
136
+  // Build the SVG DOM.
137
+  var svg = Blockly.createSvgElement('svg', {
138
+    'xmlns': 'http://www.w3.org/2000/svg',
139
+    'xmlns:html': 'http://www.w3.org/1999/xhtml',
140
+    'xmlns:xlink': 'http://www.w3.org/1999/xlink',
141
+    'version': '1.1',
142
+    'height': (Blockly.FieldAngle.HALF * 2) + 'px',
143
+    'width': (Blockly.FieldAngle.HALF * 2) + 'px'
144
+  }, div);
145
+  var circle = Blockly.createSvgElement('circle', {
146
+    'cx': Blockly.FieldAngle.HALF, 'cy': Blockly.FieldAngle.HALF,
147
+    'r': Blockly.FieldAngle.RADIUS,
148
+    'class': 'blocklyAngleCircle'
149
+  }, svg);
150
+  this.gauge_ = Blockly.createSvgElement('path',
151
+      {'class': 'blocklyAngleGauge'}, svg);
152
+  this.line_ = Blockly.createSvgElement('line',
153
+      {'x1': Blockly.FieldAngle.HALF,
154
+      'y1': Blockly.FieldAngle.HALF,
155
+      'class': 'blocklyAngleLine'}, svg);
156
+  // Draw markers around the edge.
157
+  for (var a = 0; a < 360; a += 15) {
158
+    Blockly.createSvgElement('line', {
159
+      'x1': Blockly.FieldAngle.HALF + Blockly.FieldAngle.RADIUS,
160
+      'y1': Blockly.FieldAngle.HALF,
161
+      'x2': Blockly.FieldAngle.HALF + Blockly.FieldAngle.RADIUS -
162
+          (a % 45 == 0 ? 10 : 5),
163
+      'y2': Blockly.FieldAngle.HALF,
164
+      'class': 'blocklyAngleMarks',
165
+      'transform': 'rotate(' + a + ',' +
166
+          Blockly.FieldAngle.HALF + ',' + Blockly.FieldAngle.HALF + ')'
167
+    }, svg);
168
+  }
169
+  svg.style.marginLeft = (15 - Blockly.FieldAngle.RADIUS) + 'px';
170
+  this.clickWrapper_ =
171
+      Blockly.bindEvent_(svg, 'click', this, Blockly.WidgetDiv.hide);
172
+  this.moveWrapper1_ =
173
+      Blockly.bindEvent_(circle, 'mousemove', this, this.onMouseMove);
174
+  this.moveWrapper2_ =
175
+      Blockly.bindEvent_(this.gauge_, 'mousemove', this, this.onMouseMove);
176
+  this.updateGraph_();
177
+};
178
+
179
+/**
180
+ * Set the angle to match the mouse's position.
181
+ * @param {!Event} e Mouse move event.
182
+ */
183
+Blockly.FieldAngle.prototype.onMouseMove = function(e) {
184
+  var bBox = this.gauge_.ownerSVGElement.getBoundingClientRect();
185
+  var dx = e.clientX - bBox.left - Blockly.FieldAngle.HALF;
186
+  var dy = e.clientY - bBox.top - Blockly.FieldAngle.HALF;
187
+  var angle = Math.atan(-dy / dx);
188
+  if (isNaN(angle)) {
189
+    // This shouldn't happen, but let's not let this error propogate further.
190
+    return;
191
+  }
192
+  angle = goog.math.toDegrees(angle);
193
+  // 0: East, 90: North, 180: West, 270: South.
194
+  if (dx < 0) {
195
+    angle += 180;
196
+  } else if (dy > 0) {
197
+    angle += 360;
198
+  }
199
+  if (Blockly.FieldAngle.ROUND) {
200
+    angle = Math.round(angle / Blockly.FieldAngle.ROUND) *
201
+        Blockly.FieldAngle.ROUND;
202
+  }
203
+  if (angle >= 360) {
204
+    // Rounding may have rounded up to 360.
205
+    angle -= 360;
206
+  }
207
+  angle = String(angle);
208
+  Blockly.FieldTextInput.htmlInput_.value = angle;
209
+  this.setText(angle);
210
+  this.validate_();
211
+};
212
+
213
+/**
214
+ * Insert a degree symbol.
215
+ * @param {?string} text New text.
216
+ */
217
+Blockly.FieldAngle.prototype.setText = function(text) {
218
+  Blockly.FieldAngle.superClass_.setText.call(this, text);
219
+  if (!this.textElement_) {
220
+    // Not rendered yet.
221
+    return;
222
+  }
223
+  this.updateGraph_();
224
+  // Insert degree symbol.
225
+  if (this.sourceBlock_.RTL) {
226
+    this.textElement_.insertBefore(this.symbol_, this.textElement_.firstChild);
227
+  } else {
228
+    this.textElement_.appendChild(this.symbol_);
229
+  }
230
+  // Cached width is obsolete.  Clear it.
231
+  this.size_.width = 0;
232
+};
233
+
234
+/**
235
+ * Redraw the graph with the current angle.
236
+ * @private
237
+ */
238
+Blockly.FieldAngle.prototype.updateGraph_ = function() {
239
+  if (!this.gauge_) {
240
+    return;
241
+  }
242
+  var angleRadians = goog.math.toRadians(Number(this.getText()));
243
+  if (isNaN(angleRadians)) {
244
+    this.gauge_.setAttribute('d',
245
+        'M ' + Blockly.FieldAngle.HALF + ',' + Blockly.FieldAngle.HALF);
246
+    this.line_.setAttribute('x2', Blockly.FieldAngle.HALF);
247
+    this.line_.setAttribute('y2', Blockly.FieldAngle.HALF);
248
+  } else {
249
+    var x = Blockly.FieldAngle.HALF + Math.cos(angleRadians) *
250
+        Blockly.FieldAngle.RADIUS;
251
+    var y = Blockly.FieldAngle.HALF + Math.sin(angleRadians) *
252
+        -Blockly.FieldAngle.RADIUS;
253
+    var largeFlag = (angleRadians > Math.PI) ? 1 : 0;
254
+    this.gauge_.setAttribute('d',
255
+        'M ' + Blockly.FieldAngle.HALF + ',' + Blockly.FieldAngle.HALF +
256
+        ' h ' + Blockly.FieldAngle.RADIUS +
257
+        ' A ' + Blockly.FieldAngle.RADIUS + ',' + Blockly.FieldAngle.RADIUS +
258
+        ' 0 ' + largeFlag + ' 0 ' + x + ',' + y + ' z');
259
+    this.line_.setAttribute('x2', x);
260
+    this.line_.setAttribute('y2', y);
261
+  }
262
+};
263
+
264
+/**
265
+ * Ensure that only an angle may be entered.
266
+ * @param {string} text The user's text.
267
+ * @return {?string} A string representing a valid angle, or null if invalid.
268
+ */
269
+Blockly.FieldAngle.angleValidator = function(text) {
270
+  var n = Blockly.FieldTextInput.numberValidator(text);
271
+  if (n !== null) {
272
+    n = n % 360;
273
+    if (n < 0) {
274
+      n += 360;
275
+    }
276
+    n = String(n);
277
+   }
278
+  return n;
279
+};

+ 117 - 0
src/blockly/core/field_checkbox.js

@@ -0,0 +1,117 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Checkbox field.  Checked or not checked.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.FieldCheckbox');
28
+
29
+goog.require('Blockly.Field');
30
+
31
+
32
+/**
33
+ * Class for a checkbox field.
34
+ * @param {string} state The initial state of the field ('TRUE' or 'FALSE').
35
+ * @param {Function=} opt_changeHandler A function that is executed when a new
36
+ *     option is selected.  Its sole argument is the new checkbox state.  If
37
+ *     it returns a value, this becomes the new checkbox state, unless the
38
+ *     value is null, in which case the change is aborted.
39
+ * @extends {Blockly.Field}
40
+ * @constructor
41
+ */
42
+Blockly.FieldCheckbox = function(state, opt_changeHandler) {
43
+  Blockly.FieldCheckbox.superClass_.constructor.call(this, '');
44
+
45
+  this.setChangeHandler(opt_changeHandler);
46
+  // Set the initial state.
47
+  this.setValue(state);
48
+};
49
+goog.inherits(Blockly.FieldCheckbox, Blockly.Field);
50
+
51
+/**
52
+ * Mouse cursor style when over the hotspot that initiates editability.
53
+ */
54
+Blockly.FieldCheckbox.prototype.CURSOR = 'default';
55
+
56
+/**
57
+ * Install this checkbox on a block.
58
+ * @param {!Blockly.Block} block The block containing this text.
59
+ */
60
+Blockly.FieldCheckbox.prototype.init = function(block) {
61
+  if (this.sourceBlock_) {
62
+    // Checkbox has already been initialized once.
63
+    return;
64
+  }
65
+  Blockly.FieldCheckbox.superClass_.init.call(this, block);
66
+  // The checkbox doesn't use the inherited text element.
67
+  // Instead it uses a custom checkmark element that is either visible or not.
68
+  this.checkElement_ = Blockly.createSvgElement('text',
69
+      {'class': 'blocklyText', 'x': -3, 'y': 14}, this.fieldGroup_);
70
+  var textNode = document.createTextNode('\u2713');
71
+  this.checkElement_.appendChild(textNode);
72
+  this.checkElement_.style.display = this.state_ ? 'block' : 'none';
73
+};
74
+
75
+/**
76
+ * Return 'TRUE' if the checkbox is checked, 'FALSE' otherwise.
77
+ * @return {string} Current state.
78
+ */
79
+Blockly.FieldCheckbox.prototype.getValue = function() {
80
+  return String(this.state_).toUpperCase();
81
+};
82
+
83
+/**
84
+ * Set the checkbox to be checked if strBool is 'TRUE', unchecks otherwise.
85
+ * @param {string} strBool New state.
86
+ */
87
+Blockly.FieldCheckbox.prototype.setValue = function(strBool) {
88
+  var newState = (strBool == 'TRUE');
89
+  if (this.state_ !== newState) {
90
+    this.state_ = newState;
91
+    if (this.checkElement_) {
92
+      this.checkElement_.style.display = newState ? 'block' : 'none';
93
+    }
94
+    if (this.sourceBlock_ && this.sourceBlock_.rendered) {
95
+      this.sourceBlock_.workspace.fireChangeEvent();
96
+    }
97
+  }
98
+};
99
+
100
+/**
101
+ * Toggle the state of the checkbox.
102
+ * @private
103
+ */
104
+Blockly.FieldCheckbox.prototype.showEditor_ = function() {
105
+  var newState = !this.state_;
106
+  if (this.sourceBlock_ && this.changeHandler_) {
107
+    // Call any change handler, and allow it to override.
108
+    var override = this.changeHandler_(newState);
109
+    if (override !== undefined) {
110
+      newState = override;
111
+    }
112
+  }
113
+  if (newState !== null) {
114
+    this.sourceBlock_.setShadow(false);
115
+    this.setValue(String(newState).toUpperCase());
116
+  }
117
+};

+ 242 - 0
src/blockly/core/field_colour.js

@@ -0,0 +1,242 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Colour input field.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.FieldColour');
28
+
29
+goog.require('Blockly.Field');
30
+goog.require('goog.dom');
31
+goog.require('goog.events');
32
+goog.require('goog.style');
33
+goog.require('goog.ui.ColorPicker');
34
+
35
+
36
+/**
37
+ * Class for a colour input field.
38
+ * @param {string} colour The initial colour in '#rrggbb' format.
39
+ * @param {Function=} opt_changeHandler A function that is executed when a new
40
+ *     colour is selected.  Its sole argument is the new colour value.  Its
41
+ *     return value becomes the selected colour, unless it is undefined, in
42
+ *     which case the new colour stands, or it is null, in which case the change
43
+ *     is aborted.
44
+ * @extends {Blockly.Field}
45
+ * @constructor
46
+ */
47
+Blockly.FieldColour = function(colour, opt_changeHandler) {
48
+  Blockly.FieldColour.superClass_.constructor.call(this, '\u00A0\u00A0\u00A0');
49
+
50
+  this.setChangeHandler(opt_changeHandler);
51
+  // Set the initial state.
52
+  this.setValue(colour);
53
+};
54
+goog.inherits(Blockly.FieldColour, Blockly.Field);
55
+
56
+/**
57
+ * By default use the global constants for colours.
58
+ * @type {Array.<string>}
59
+ * @private
60
+ */
61
+Blockly.FieldColour.prototype.colours_ = null;
62
+
63
+/**
64
+ * By default use the global constants for columns.
65
+ * @type {number}
66
+ * @private
67
+ */
68
+Blockly.FieldColour.prototype.columns_ = 0;
69
+
70
+/**
71
+ * Install this field on a block.
72
+ * @param {!Blockly.Block} block The block containing this field.
73
+ */
74
+Blockly.FieldColour.prototype.init = function(block) {
75
+  Blockly.FieldColour.superClass_.init.call(this, block);
76
+  this.borderRect_.style['fillOpacity'] = 1;
77
+  this.setValue(this.getValue());
78
+};
79
+
80
+/**
81
+ * Mouse cursor style when over the hotspot that initiates the editor.
82
+ */
83
+Blockly.FieldColour.prototype.CURSOR = 'default';
84
+
85
+/**
86
+ * Close the colour picker if this input is being deleted.
87
+ */
88
+Blockly.FieldColour.prototype.dispose = function() {
89
+  Blockly.WidgetDiv.hideIfOwner(this);
90
+  Blockly.FieldColour.superClass_.dispose.call(this);
91
+};
92
+
93
+/**
94
+ * Return the current colour.
95
+ * @return {string} Current colour in '#rrggbb' format.
96
+ */
97
+Blockly.FieldColour.prototype.getValue = function() {
98
+  return this.colour_;
99
+};
100
+
101
+/**
102
+ * Set the colour.
103
+ * @param {string} colour The new colour in '#rrggbb' format.
104
+ */
105
+Blockly.FieldColour.prototype.setValue = function(colour) {
106
+  this.colour_ = colour;
107
+  if (this.borderRect_) {
108
+    this.borderRect_.style.fill = colour;
109
+  }
110
+  if (this.sourceBlock_ && this.sourceBlock_.rendered) {
111
+    // Since we're not re-rendering we need to explicitly call
112
+    // Blockly.Realtime.blockChanged()
113
+    Blockly.Realtime.blockChanged(this.sourceBlock_);
114
+    this.sourceBlock_.workspace.fireChangeEvent();
115
+  }
116
+};
117
+
118
+/**
119
+ * Get the text from this field.  Used when the block is collapsed.
120
+ * @return {string} Current text.
121
+ */
122
+Blockly.FieldColour.prototype.getText = function() {
123
+  var colour = this.colour_;
124
+  var m = colour.match(/^#(.)\1(.)\2(.)\3$/);
125
+  if (m) {
126
+    colour = '#' + m[1] + m[2] + m[3];
127
+  }
128
+  return colour;
129
+};
130
+
131
+/**
132
+ * An array of colour strings for the palette.
133
+ * See bottom of this page for the default:
134
+ * http://docs.closure-library.googlecode.com/git/closure_goog_ui_colorpicker.js.source.html
135
+ * @type {!Array.<string>}
136
+ */
137
+Blockly.FieldColour.COLOURS = goog.ui.ColorPicker.SIMPLE_GRID_COLORS;
138
+
139
+/**
140
+ * Number of columns in the palette.
141
+ */
142
+Blockly.FieldColour.COLUMNS = 7;
143
+
144
+/**
145
+ * Set a custom colour grid for this field.
146
+ * @param {Array.<string>} colours Array of colours for this block,
147
+ *     or null to use default (Blockly.FieldColour.COLOURS).
148
+ * @return {!Blockly.FieldColour} Returns itself (for method chaining).
149
+ */
150
+Blockly.FieldColour.prototype.setColours = function(colours) {
151
+  this.colours_ = colours;
152
+  return this;
153
+};
154
+
155
+/**
156
+ * Set a custom grid size for this field.
157
+ * @param {number} columns Number of columns for this block,
158
+ *     or 0 to use default (Blockly.FieldColour.COLUMNS).
159
+ * @return {!Blockly.FieldColour} Returns itself (for method chaining).
160
+ */
161
+Blockly.FieldColour.prototype.setColumns = function(columns) {
162
+  this.columns_ = columns;
163
+  return this;
164
+};
165
+
166
+/**
167
+ * Create a palette under the colour field.
168
+ * @private
169
+ */
170
+Blockly.FieldColour.prototype.showEditor_ = function() {
171
+  Blockly.WidgetDiv.show(this, this.sourceBlock_.RTL,
172
+      Blockly.FieldColour.widgetDispose_);
173
+  // Create the palette using Closure.
174
+  var picker = new goog.ui.ColorPicker();
175
+  picker.setSize(this.columns_ || Blockly.FieldColour.COLUMNS);
176
+  picker.setColors(this.colours_ || Blockly.FieldColour.COLOURS);
177
+
178
+  // Position the palette to line up with the field.
179
+  // Record windowSize and scrollOffset before adding the palette.
180
+  var windowSize = goog.dom.getViewportSize();
181
+  var scrollOffset = goog.style.getViewportPageOffset(document);
182
+  var xy = this.getAbsoluteXY_();
183
+  var borderBBox = this.getScaledBBox_();
184
+  var div = Blockly.WidgetDiv.DIV;
185
+  picker.render(div);
186
+  picker.setSelectedColor(this.getValue());
187
+  // Record paletteSize after adding the palette.
188
+  var paletteSize = goog.style.getSize(picker.getElement());
189
+
190
+  // Flip the palette vertically if off the bottom.
191
+  if (xy.y + paletteSize.height + borderBBox.height >=
192
+      windowSize.height + scrollOffset.y) {
193
+    xy.y -= paletteSize.height - 1;
194
+  } else {
195
+    xy.y += borderBBox.height - 1;
196
+  }
197
+  if (this.sourceBlock_.RTL) {
198
+    xy.x += borderBBox.width;
199
+    xy.x -= paletteSize.width;
200
+    // Don't go offscreen left.
201
+    if (xy.x < scrollOffset.x) {
202
+      xy.x = scrollOffset.x;
203
+    }
204
+  } else {
205
+    // Don't go offscreen right.
206
+    if (xy.x > windowSize.width + scrollOffset.x - paletteSize.width) {
207
+      xy.x = windowSize.width + scrollOffset.x - paletteSize.width;
208
+    }
209
+  }
210
+  Blockly.WidgetDiv.position(xy.x, xy.y, windowSize, scrollOffset,
211
+                             this.sourceBlock_.RTL);
212
+
213
+  // Configure event handler.
214
+  var thisField = this;
215
+  Blockly.FieldColour.changeEventKey_ = goog.events.listen(picker,
216
+      goog.ui.ColorPicker.EventType.CHANGE,
217
+      function(event) {
218
+        var colour = event.target.getSelectedColor() || '#000000';
219
+        Blockly.WidgetDiv.hide();
220
+        if (thisField.sourceBlock_ && thisField.changeHandler_) {
221
+          // Call any change handler, and allow it to override.
222
+          var override = thisField.changeHandler_(colour);
223
+          if (override !== undefined) {
224
+            colour = override;
225
+          }
226
+        }
227
+        if (colour !== null) {
228
+          thisField.sourceBlock_.setShadow(false);
229
+          thisField.setValue(colour);
230
+        }
231
+      });
232
+};
233
+
234
+/**
235
+ * Hide the colour palette.
236
+ * @private
237
+ */
238
+Blockly.FieldColour.widgetDispose_ = function() {
239
+  if (Blockly.FieldColour.changeEventKey_) {
240
+    goog.events.unlistenByKey(Blockly.FieldColour.changeEventKey_);
241
+  }
242
+};

+ 350 - 0
src/blockly/core/field_date.js

@@ -0,0 +1,350 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2015 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Date input field.
23
+ * @author pkendall64@gmail.com (Paul Kendall)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.FieldDate');
28
+
29
+goog.require('Blockly.Field');
30
+goog.require('goog.date');
31
+goog.require('goog.dom');
32
+goog.require('goog.events');
33
+goog.require('goog.i18n.DateTimeSymbols');
34
+goog.require('goog.i18n.DateTimeSymbols_he');
35
+goog.require('goog.style');
36
+goog.require('goog.ui.DatePicker');
37
+
38
+
39
+/**
40
+ * Class for a date input field.
41
+ * @param {string} date The initial date.
42
+ * @param {Function=} opt_changeHandler A function that is executed when a new
43
+ *     date is selected.  Its sole argument is the new date value.  Its
44
+ *     return value becomes the selected date, unless it is undefined, in
45
+ *     which case the new date stands, or it is null, in which case the change
46
+ *     is aborted.
47
+ * @extends {Blockly.Field}
48
+ * @constructor
49
+ */
50
+Blockly.FieldDate = function(date, opt_changeHandler) {
51
+  if (!date) {
52
+    date = new goog.date.Date().toIsoString(true);
53
+  }
54
+  Blockly.FieldDate.superClass_.constructor.call(this, date);
55
+  this.setValue(date);
56
+  this.setChangeHandler(opt_changeHandler);
57
+};
58
+goog.inherits(Blockly.FieldDate, Blockly.Field);
59
+
60
+/**
61
+ * Mouse cursor style when over the hotspot that initiates the editor.
62
+ */
63
+Blockly.FieldDate.prototype.CURSOR = 'text';
64
+
65
+/**
66
+ * Close the colour picker if this input is being deleted.
67
+ */
68
+Blockly.FieldDate.prototype.dispose = function() {
69
+  Blockly.WidgetDiv.hideIfOwner(this);
70
+  Blockly.FieldDate.superClass_.dispose.call(this);
71
+};
72
+
73
+/**
74
+ * Return the current date.
75
+ * @return {string} Current date.
76
+ */
77
+Blockly.FieldDate.prototype.getValue = function() {
78
+  return this.date_;
79
+};
80
+
81
+/**
82
+ * Set the date.
83
+ * @param {string} date The new date.
84
+ */
85
+Blockly.FieldDate.prototype.setValue = function(date) {
86
+  if (this.sourceBlock_ && this.changeHandler_) {
87
+    var validated = this.changeHandler_(date);
88
+    // If the new date is invalid, validation returns null.
89
+    // In this case we still want to display the illegal result.
90
+    if (validated !== null && validated !== undefined) {
91
+      date = validated;
92
+    }
93
+  }
94
+  this.date_ = date;
95
+  Blockly.Field.prototype.setText.call(this, date);
96
+};
97
+
98
+/**
99
+ * Create a date picker under the date field.
100
+ * @private
101
+ */
102
+Blockly.FieldDate.prototype.showEditor_ = function() {
103
+  Blockly.WidgetDiv.show(this, this.sourceBlock_.RTL,
104
+      Blockly.FieldDate.widgetDispose_);
105
+  // Create the date picker using Closure.
106
+  Blockly.FieldDate.loadLanguage_();
107
+  var picker = new goog.ui.DatePicker();
108
+  picker.setAllowNone(false);
109
+  picker.setShowWeekNum(false);
110
+
111
+  // Position the picker to line up with the field.
112
+  // Record windowSize and scrollOffset before adding the picker.
113
+  var windowSize = goog.dom.getViewportSize();
114
+  var scrollOffset = goog.style.getViewportPageOffset(document);
115
+  var xy = this.getAbsoluteXY_();
116
+  var borderBBox = this.getScaledBBox_();
117
+  var div = Blockly.WidgetDiv.DIV;
118
+  picker.render(div);
119
+  picker.setDate(goog.date.fromIsoString(this.getValue()));
120
+  // Record pickerSize after adding the date picker.
121
+  var pickerSize = goog.style.getSize(picker.getElement());
122
+
123
+  // Flip the picker vertically if off the bottom.
124
+  if (xy.y + pickerSize.height + borderBBox.height >=
125
+      windowSize.height + scrollOffset.y) {
126
+    xy.y -= pickerSize.height - 1;
127
+  } else {
128
+    xy.y += borderBBox.height - 1;
129
+  }
130
+  if (this.sourceBlock_.RTL) {
131
+    xy.x += borderBBox.width;
132
+    xy.x -= pickerSize.width;
133
+    // Don't go offscreen left.
134
+    if (xy.x < scrollOffset.x) {
135
+      xy.x = scrollOffset.x;
136
+    }
137
+  } else {
138
+    // Don't go offscreen right.
139
+    if (xy.x > windowSize.width + scrollOffset.x - pickerSize.width) {
140
+      xy.x = windowSize.width + scrollOffset.x - pickerSize.width;
141
+    }
142
+  }
143
+  Blockly.WidgetDiv.position(xy.x, xy.y, windowSize, scrollOffset,
144
+                             this.sourceBlock_.RTL);
145
+
146
+  // Configure event handler.
147
+  var thisField = this;
148
+  Blockly.FieldDate.changeEventKey_ = goog.events.listen(picker,
149
+      goog.ui.DatePicker.Events.CHANGE,
150
+      function(event) {
151
+        var date = event.date ? event.date.toIsoString(true) : '';
152
+        Blockly.WidgetDiv.hide();
153
+        if (thisField.sourceBlock_ && thisField.changeHandler_) {
154
+          // Call any change handler, and allow it to override.
155
+          var override = thisField.changeHandler_(date);
156
+          if (override !== undefined) {
157
+            date = override;
158
+          }
159
+        }
160
+        thisField.setValue(date);
161
+      });
162
+};
163
+
164
+/**
165
+ * Hide the date picker.
166
+ * @private
167
+ */
168
+Blockly.FieldDate.widgetDispose_ = function() {
169
+  if (Blockly.FieldDate.changeEventKey_) {
170
+    goog.events.unlistenByKey(Blockly.FieldDate.changeEventKey_);
171
+  }
172
+};
173
+
174
+/**
175
+ * Load the best language pack by scanning the Blockly.Msg object for a
176
+ * language that matches the available languages in Closure.
177
+ * @private
178
+ */
179
+Blockly.FieldDate.loadLanguage_ = function() {
180
+  var reg = /^DateTimeSymbols_(.+)$/;
181
+  for (var prop in goog.i18n) {
182
+    var m = prop.match(reg);
183
+    if (m) {
184
+      var lang = m[1].toLowerCase().replace('_', '.');  // E.g. 'pt.br'
185
+      if (goog.getObjectByName(lang, Blockly.Msg)) {
186
+        goog.i18n.DateTimeSymbols = goog.i18n[prop];
187
+      }
188
+    }
189
+  }
190
+};
191
+
192
+/**
193
+ * CSS for date picker.  See css.js for use.
194
+ */
195
+Blockly.FieldDate.CSS = [
196
+  /* Copied from: goog/css/datepicker.css */
197
+  /*
198
+   * Copyright 2009 The Closure Library Authors. All Rights Reserved.
199
+   *
200
+   * Use of this source code is governed by the Apache License, Version 2.0.
201
+   * See the COPYING file for details.
202
+   */
203
+
204
+  /*
205
+   * Standard styling for a goog.ui.DatePicker.
206
+   *
207
+   * @author arv@google.com (Erik Arvidsson)
208
+   */
209
+
210
+  '.blocklyWidgetDiv .goog-date-picker,',
211
+  '.blocklyWidgetDiv .goog-date-picker th,',
212
+  '.blocklyWidgetDiv .goog-date-picker td {',
213
+  '  font: 13px Arial, sans-serif;',
214
+  '}',
215
+
216
+  '.blocklyWidgetDiv .goog-date-picker {',
217
+  '  -moz-user-focus: normal;',
218
+  '  -moz-user-select: none;',
219
+  '  position: relative;',
220
+  '  border: 1px solid #000;',
221
+  '  float: left;',
222
+  '  padding: 2px;',
223
+  '  color: #000;',
224
+  '  background: #c3d9ff;',
225
+  '  cursor: default;',
226
+  '}',
227
+
228
+  '.blocklyWidgetDiv .goog-date-picker th {',
229
+  '  text-align: center;',
230
+  '}',
231
+
232
+  '.blocklyWidgetDiv .goog-date-picker td {',
233
+  '  text-align: center;',
234
+  '  vertical-align: middle;',
235
+  '  padding: 1px 3px;',
236
+  '}',
237
+
238
+  '.blocklyWidgetDiv .goog-date-picker-menu {',
239
+  '  position: absolute;',
240
+  '  background: threedface;',
241
+  '  border: 1px solid gray;',
242
+  '  -moz-user-focus: normal;',
243
+  '  z-index: 1;',
244
+  '  outline: none;',
245
+  '}',
246
+
247
+  '.blocklyWidgetDiv .goog-date-picker-menu ul {',
248
+  '  list-style: none;',
249
+  '  margin: 0px;',
250
+  '  padding: 0px;',
251
+  '}',
252
+
253
+  '.blocklyWidgetDiv .goog-date-picker-menu ul li {',
254
+  '  cursor: default;',
255
+  '}',
256
+
257
+  '.blocklyWidgetDiv .goog-date-picker-menu-selected {',
258
+  '  background: #ccf;',
259
+  '}',
260
+
261
+  '.blocklyWidgetDiv .goog-date-picker th {',
262
+  '  font-size: .9em;',
263
+  '}',
264
+
265
+  '.blocklyWidgetDiv .goog-date-picker td div {',
266
+  '  float: left;',
267
+  '}',
268
+
269
+  '.blocklyWidgetDiv .goog-date-picker button {',
270
+  '  padding: 0px;',
271
+  '  margin: 1px 0;',
272
+  '  border: 0;',
273
+  '  color: #20c;',
274
+  '  font-weight: bold;',
275
+  '  background: transparent;',
276
+  '}',
277
+
278
+  '.blocklyWidgetDiv .goog-date-picker-date {',
279
+  '  background: #fff;',
280
+  '}',
281
+
282
+  '.blocklyWidgetDiv .goog-date-picker-week,',
283
+  '.blocklyWidgetDiv .goog-date-picker-wday {',
284
+  '  padding: 1px 3px;',
285
+  '  border: 0;',
286
+  '  border-color: #a2bbdd;',
287
+  '  border-style: solid;',
288
+  '}',
289
+
290
+  '.blocklyWidgetDiv .goog-date-picker-week {',
291
+  '  border-right-width: 1px;',
292
+  '}',
293
+
294
+  '.blocklyWidgetDiv .goog-date-picker-wday {',
295
+  '  border-bottom-width: 1px;',
296
+  '}',
297
+
298
+  '.blocklyWidgetDiv .goog-date-picker-head td {',
299
+  '  text-align: center;',
300
+  '}',
301
+
302
+  /** Use td.className instead of !important */
303
+  '.blocklyWidgetDiv td.goog-date-picker-today-cont {',
304
+  '  text-align: center;',
305
+  '}',
306
+
307
+  /** Use td.className instead of !important */
308
+  '.blocklyWidgetDiv td.goog-date-picker-none-cont {',
309
+  '  text-align: center;',
310
+  '}',
311
+
312
+  '.blocklyWidgetDiv .goog-date-picker-month {',
313
+  '  min-width: 11ex;',
314
+  '  white-space: nowrap;',
315
+  '}',
316
+
317
+  '.blocklyWidgetDiv .goog-date-picker-year {',
318
+  '  min-width: 6ex;',
319
+  '  white-space: nowrap;',
320
+  '}',
321
+
322
+  '.blocklyWidgetDiv .goog-date-picker-monthyear {',
323
+  '  white-space: nowrap;',
324
+  '}',
325
+
326
+  '.blocklyWidgetDiv .goog-date-picker table {',
327
+  '  border-collapse: collapse;',
328
+  '}',
329
+
330
+  '.blocklyWidgetDiv .goog-date-picker-other-month {',
331
+  '  color: #888;',
332
+  '}',
333
+
334
+  '.blocklyWidgetDiv .goog-date-picker-wkend-start,',
335
+  '.blocklyWidgetDiv .goog-date-picker-wkend-end {',
336
+  '  background: #eee;',
337
+  '}',
338
+
339
+  /** Use td.className instead of !important */
340
+  '.blocklyWidgetDiv td.goog-date-picker-selected {',
341
+  '  background: #c3d9ff;',
342
+  '}',
343
+
344
+  '.blocklyWidgetDiv .goog-date-picker-today {',
345
+  '  background: #9ab;',
346
+  '  font-weight: bold !important;',
347
+  '  border-color: #246 #9bd #9bd #246;',
348
+  '  color: #fff;',
349
+  '}'
350
+];

+ 321 - 0
src/blockly/core/field_dropdown.js

@@ -0,0 +1,321 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Dropdown input field.  Used for editable titles and variables.
23
+ * In the interests of a consistent UI, the toolbox shares some functions and
24
+ * properties with the context menu.
25
+ * @author fraser@google.com (Neil Fraser)
26
+ */
27
+'use strict';
28
+
29
+goog.provide('Blockly.FieldDropdown');
30
+
31
+goog.require('Blockly.Field');
32
+goog.require('goog.dom');
33
+goog.require('goog.events');
34
+goog.require('goog.style');
35
+goog.require('goog.ui.Menu');
36
+goog.require('goog.ui.MenuItem');
37
+goog.require('goog.userAgent');
38
+
39
+
40
+/**
41
+ * Class for an editable dropdown field.
42
+ * @param {(!Array.<!Array.<string>>|!Function)} menuGenerator An array of options
43
+ *     for a dropdown list, or a function which generates these options.
44
+ * @param {Function=} opt_changeHandler A function that is executed when a new
45
+ *     option is selected, with the newly selected value as its sole argument.
46
+ *     If it returns a value, that value (which must be one of the options) will
47
+ *     become selected in place of the newly selected option, unless the return
48
+ *     value is null, in which case the change is aborted.
49
+ * @extends {Blockly.Field}
50
+ * @constructor
51
+ */
52
+Blockly.FieldDropdown = function(menuGenerator, opt_changeHandler) {
53
+  this.menuGenerator_ = menuGenerator;
54
+  this.setChangeHandler(opt_changeHandler);
55
+  this.trimOptions_();
56
+  var firstTuple = this.getOptions_()[0];
57
+  this.value_ = firstTuple[1];
58
+
59
+  // Call parent's constructor.
60
+  Blockly.FieldDropdown.superClass_.constructor.call(this, firstTuple[0]);
61
+};
62
+goog.inherits(Blockly.FieldDropdown, Blockly.Field);
63
+
64
+/**
65
+ * Horizontal distance that a checkmark ovehangs the dropdown.
66
+ */
67
+Blockly.FieldDropdown.CHECKMARK_OVERHANG = 25;
68
+
69
+/**
70
+ * Android can't (in 2014) display "▾", so use "▼" instead.
71
+ */
72
+Blockly.FieldDropdown.ARROW_CHAR = goog.userAgent.ANDROID ? '\u25BC' : '\u25BE';
73
+
74
+/**
75
+ * Mouse cursor style when over the hotspot that initiates the editor.
76
+ */
77
+Blockly.FieldDropdown.prototype.CURSOR = 'default';
78
+
79
+/**
80
+ * Install this dropdown on a block.
81
+ * @param {!Blockly.Block} block The block containing this text.
82
+ */
83
+Blockly.FieldDropdown.prototype.init = function(block) {
84
+  if (this.sourceBlock_) {
85
+    // Dropdown has already been initialized once.
86
+    return;
87
+  }
88
+
89
+  // Add dropdown arrow: "option ▾" (LTR) or "▾ אופציה" (RTL)
90
+  this.arrow_ = Blockly.createSvgElement('tspan', {}, null);
91
+  this.arrow_.appendChild(document.createTextNode(
92
+      block.RTL ? Blockly.FieldDropdown.ARROW_CHAR + ' ' :
93
+          ' ' + Blockly.FieldDropdown.ARROW_CHAR));
94
+
95
+  Blockly.FieldDropdown.superClass_.init.call(this, block);
96
+  // Force a reset of the text to add the arrow.
97
+  var text = this.text_;
98
+  this.text_ = null;
99
+  this.setText(text);
100
+};
101
+
102
+/**
103
+ * Create a dropdown menu under the text.
104
+ * @private
105
+ */
106
+Blockly.FieldDropdown.prototype.showEditor_ = function() {
107
+  Blockly.WidgetDiv.show(this, this.sourceBlock_.RTL, null);
108
+  var thisField = this;
109
+
110
+  function callback(e) {
111
+    var menuItem = e.target;
112
+    if (menuItem) {
113
+      var value = menuItem.getValue();
114
+      if (thisField.sourceBlock_ && thisField.changeHandler_) {
115
+        // Call any change handler, and allow it to override.
116
+        var override = thisField.changeHandler_(value);
117
+        if (override !== undefined) {
118
+          value = override;
119
+        }
120
+      }
121
+      if (value !== null) {
122
+        thisField.sourceBlock_.setShadow(false);
123
+        thisField.setValue(value);
124
+      }
125
+    }
126
+    Blockly.WidgetDiv.hideIfOwner(thisField);
127
+  }
128
+
129
+  var menu = new goog.ui.Menu();
130
+  menu.setRightToLeft(this.sourceBlock_.RTL);
131
+  var options = this.getOptions_();
132
+  for (var x = 0; x < options.length; x++) {
133
+    var text = options[x][0];  // Human-readable text.
134
+    var value = options[x][1]; // Language-neutral value.
135
+    var menuItem = new goog.ui.MenuItem(text);
136
+    menuItem.setRightToLeft(this.sourceBlock_.RTL);
137
+    menuItem.setValue(value);
138
+    menuItem.setCheckable(true);
139
+    menu.addChild(menuItem, true);
140
+    menuItem.setChecked(value == this.value_);
141
+  }
142
+  // Listen for mouse/keyboard events.
143
+  goog.events.listen(menu, goog.ui.Component.EventType.ACTION, callback);
144
+  // Listen for touch events (why doesn't Closure handle this already?).
145
+  function callbackTouchStart(e) {
146
+    var control = this.getOwnerControl(/** @type {Node} */ (e.target));
147
+    // Highlight the menu item.
148
+    control.handleMouseDown(e);
149
+  }
150
+  function callbackTouchEnd(e) {
151
+    var control = this.getOwnerControl(/** @type {Node} */ (e.target));
152
+    // Activate the menu item.
153
+    control.performActionInternal(e);
154
+  }
155
+  menu.getHandler().listen(menu.getElement(), goog.events.EventType.TOUCHSTART,
156
+                           callbackTouchStart);
157
+  menu.getHandler().listen(menu.getElement(), goog.events.EventType.TOUCHEND,
158
+                           callbackTouchEnd);
159
+
160
+  // Record windowSize and scrollOffset before adding menu.
161
+  var windowSize = goog.dom.getViewportSize();
162
+  var scrollOffset = goog.style.getViewportPageOffset(document);
163
+  var xy = this.getAbsoluteXY_();
164
+  var borderBBox = this.getScaledBBox_();
165
+  var div = Blockly.WidgetDiv.DIV;
166
+  menu.render(div);
167
+  var menuDom = menu.getElement();
168
+  Blockly.addClass_(menuDom, 'blocklyDropdownMenu');
169
+  // Record menuSize after adding menu.
170
+  var menuSize = goog.style.getSize(menuDom);
171
+  // Recalculate height for the total content, not only box height.
172
+  menuSize.height = menuDom.scrollHeight;
173
+
174
+  // Position the menu.
175
+  // Flip menu vertically if off the bottom.
176
+  if (xy.y + menuSize.height + borderBBox.height >=
177
+      windowSize.height + scrollOffset.y) {
178
+    xy.y -= menuSize.height + 2;
179
+  } else {
180
+    xy.y += borderBBox.height;
181
+  }
182
+  if (this.sourceBlock_.RTL) {
183
+    xy.x += borderBBox.width;
184
+    xy.x += Blockly.FieldDropdown.CHECKMARK_OVERHANG;
185
+    // Don't go offscreen left.
186
+    if (xy.x < scrollOffset.x + menuSize.width) {
187
+      xy.x = scrollOffset.x + menuSize.width;
188
+    }
189
+  } else {
190
+    xy.x -= Blockly.FieldDropdown.CHECKMARK_OVERHANG;
191
+    // Don't go offscreen right.
192
+    if (xy.x > windowSize.width + scrollOffset.x - menuSize.width) {
193
+      xy.x = windowSize.width + scrollOffset.x - menuSize.width;
194
+    }
195
+  }
196
+  Blockly.WidgetDiv.position(xy.x, xy.y, windowSize, scrollOffset,
197
+                             this.sourceBlock_.RTL);
198
+  menu.setAllowAutoFocus(true);
199
+  menuDom.focus();
200
+};
201
+
202
+/**
203
+ * Factor out common words in statically defined options.
204
+ * Create prefix and/or suffix labels.
205
+ * @private
206
+ */
207
+Blockly.FieldDropdown.prototype.trimOptions_ = function() {
208
+  this.prefixField = null;
209
+  this.suffixField = null;
210
+  var options = this.menuGenerator_;
211
+  if (!goog.isArray(options) || options.length < 2) {
212
+    return;
213
+  }
214
+  var strings = options.map(function(t) {return t[0];});
215
+  var shortest = Blockly.shortestStringLength(strings);
216
+  var prefixLength = Blockly.commonWordPrefix(strings, shortest);
217
+  var suffixLength = Blockly.commonWordSuffix(strings, shortest);
218
+  if (!prefixLength && !suffixLength) {
219
+    return;
220
+  }
221
+  if (shortest <= prefixLength + suffixLength) {
222
+    // One or more strings will entirely vanish if we proceed.  Abort.
223
+    return;
224
+  }
225
+  if (prefixLength) {
226
+    this.prefixField = strings[0].substring(0, prefixLength - 1);
227
+  }
228
+  if (suffixLength) {
229
+    this.suffixField = strings[0].substr(1 - suffixLength);
230
+  }
231
+  // Remove the prefix and suffix from the options.
232
+  var newOptions = [];
233
+  for (var x = 0; x < options.length; x++) {
234
+    var text = options[x][0];
235
+    var value = options[x][1];
236
+    text = text.substring(prefixLength, text.length - suffixLength);
237
+    newOptions[x] = [text, value];
238
+  }
239
+  this.menuGenerator_ = newOptions;
240
+};
241
+
242
+/**
243
+ * Return a list of the options for this dropdown.
244
+ * @return {!Array.<!Array.<string>>} Array of option tuples:
245
+ *     (human-readable text, language-neutral name).
246
+ * @private
247
+ */
248
+Blockly.FieldDropdown.prototype.getOptions_ = function() {
249
+  if (goog.isFunction(this.menuGenerator_)) {
250
+    return this.menuGenerator_.call(this);
251
+  }
252
+  return /** @type {!Array.<!Array.<string>>} */ (this.menuGenerator_);
253
+};
254
+
255
+/**
256
+ * Get the language-neutral value from this dropdown menu.
257
+ * @return {string} Current text.
258
+ */
259
+Blockly.FieldDropdown.prototype.getValue = function() {
260
+  return this.value_;
261
+};
262
+
263
+/**
264
+ * Set the language-neutral value for this dropdown menu.
265
+ * @param {string} newValue New value to set.
266
+ */
267
+Blockly.FieldDropdown.prototype.setValue = function(newValue) {
268
+  this.value_ = newValue;
269
+  // Look up and display the human-readable text.
270
+  var options = this.getOptions_();
271
+  for (var x = 0; x < options.length; x++) {
272
+    // Options are tuples of human-readable text and language-neutral values.
273
+    if (options[x][1] == newValue) {
274
+      this.setText(options[x][0]);
275
+      return;
276
+    }
277
+  }
278
+  // Value not found.  Add it, maybe it will become valid once set
279
+  // (like variable names).
280
+  this.setText(newValue);
281
+};
282
+
283
+/**
284
+ * Set the text in this field.  Trigger a rerender of the source block.
285
+ * @param {?string} text New text.
286
+ */
287
+Blockly.FieldDropdown.prototype.setText = function(text) {
288
+  if (this.sourceBlock_ && this.arrow_) {
289
+    // Update arrow's colour.
290
+    this.arrow_.style.fill = Blockly.makeColour(this.sourceBlock_.getColour());
291
+  }
292
+  if (text === null || text === this.text_) {
293
+    // No change if null.
294
+    return;
295
+  }
296
+  this.text_ = text;
297
+  this.updateTextNode_();
298
+
299
+  if (this.textElement_) {
300
+    // Insert dropdown arrow.
301
+    if (this.sourceBlock_.RTL) {
302
+      this.textElement_.insertBefore(this.arrow_, this.textElement_.firstChild);
303
+    } else {
304
+      this.textElement_.appendChild(this.arrow_);
305
+    }
306
+  }
307
+
308
+  if (this.sourceBlock_ && this.sourceBlock_.rendered) {
309
+    this.sourceBlock_.render();
310
+    this.sourceBlock_.bumpNeighbours_();
311
+    this.sourceBlock_.workspace.fireChangeEvent();
312
+  }
313
+};
314
+
315
+/**
316
+ * Close the dropdown menu if this input is being deleted.
317
+ */
318
+Blockly.FieldDropdown.prototype.dispose = function() {
319
+  Blockly.WidgetDiv.hideIfOwner(this);
320
+  Blockly.FieldDropdown.superClass_.dispose.call(this);
321
+};

+ 168 - 0
src/blockly/core/field_image.js

@@ -0,0 +1,168 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Image field.  Used for titles, labels, etc.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.FieldImage');
28
+
29
+goog.require('Blockly.Field');
30
+goog.require('goog.dom');
31
+goog.require('goog.math.Size');
32
+goog.require('goog.userAgent');
33
+
34
+
35
+/**
36
+ * Class for an image.
37
+ * @param {string} src The URL of the image.
38
+ * @param {number} width Width of the image.
39
+ * @param {number} height Height of the image.
40
+ * @param {string=} opt_alt Optional alt text for when block is collapsed.
41
+ * @extends {Blockly.Field}
42
+ * @constructor
43
+ */
44
+Blockly.FieldImage = function(src, width, height, opt_alt) {
45
+  this.sourceBlock_ = null;
46
+  // Ensure height and width are numbers.  Strings are bad at math.
47
+  this.height_ = Number(height);
48
+  this.width_ = Number(width);
49
+  this.size_ = new goog.math.Size(this.width_,
50
+      this.height_ + 2 * Blockly.BlockSvg.INLINE_PADDING_Y);
51
+  this.text_ = opt_alt || '';
52
+  this.setValue(src);
53
+};
54
+goog.inherits(Blockly.FieldImage, Blockly.Field);
55
+
56
+/**
57
+ * Rectangular mask used by Firefox.
58
+ * @type {Element}
59
+ * @private
60
+ */
61
+Blockly.FieldImage.prototype.rectElement_ = null;
62
+
63
+/**
64
+ * Editable fields are saved by the XML renderer, non-editable fields are not.
65
+ */
66
+Blockly.FieldImage.prototype.EDITABLE = false;
67
+
68
+/**
69
+ * Install this image on a block.
70
+ * @param {!Blockly.Block} block The block containing this text.
71
+ */
72
+Blockly.FieldImage.prototype.init = function(block) {
73
+  if (this.sourceBlock_) {
74
+    // Image has already been initialized once.
75
+    return;
76
+  }
77
+  this.sourceBlock_ = block;
78
+  // Build the DOM.
79
+  this.fieldGroup_ = Blockly.createSvgElement('g', {}, null);
80
+  if (!this.visible_) {
81
+    this.fieldGroup_.style.display = 'none';
82
+  }
83
+  this.imageElement_ = Blockly.createSvgElement('image',
84
+      {'height': this.height_ + 'px',
85
+       'width': this.width_ + 'px'}, this.fieldGroup_);
86
+  this.setValue(this.src_);
87
+  if (goog.userAgent.GECKO) {
88
+    // Due to a Firefox bug which eats mouse events on image elements,
89
+    // a transparent rectangle needs to be placed on top of the image.
90
+    this.rectElement_ = Blockly.createSvgElement('rect',
91
+        {'height': this.height_ + 'px',
92
+         'width': this.width_ + 'px',
93
+         'fill-opacity': 0}, this.fieldGroup_);
94
+  }
95
+  block.getSvgRoot().appendChild(this.fieldGroup_);
96
+
97
+  // Configure the field to be transparent with respect to tooltips.
98
+  var topElement = this.rectElement_ || this.imageElement_;
99
+  topElement.tooltip = this.sourceBlock_;
100
+  Blockly.Tooltip.bindMouseEvents(topElement);
101
+};
102
+
103
+/**
104
+ * Dispose of all DOM objects belonging to this text.
105
+ */
106
+Blockly.FieldImage.prototype.dispose = function() {
107
+  goog.dom.removeNode(this.fieldGroup_);
108
+  this.fieldGroup_ = null;
109
+  this.imageElement_ = null;
110
+  this.rectElement_ = null;
111
+};
112
+
113
+/**
114
+ * Change the tooltip text for this field.
115
+ * @param {string|!Element} newTip Text for tooltip or a parent element to
116
+ *     link to for its tooltip.
117
+ */
118
+Blockly.FieldImage.prototype.setTooltip = function(newTip) {
119
+  var topElement = this.rectElement_ || this.imageElement_;
120
+  topElement.tooltip = newTip;
121
+};
122
+
123
+/**
124
+ * Get the source URL of this image.
125
+ * @return {string} Current text.
126
+ * @override
127
+ */
128
+Blockly.FieldImage.prototype.getValue = function() {
129
+  return this.src_;
130
+};
131
+
132
+/**
133
+ * Set the source URL of this image.
134
+ * @param {?string} src New source.
135
+ * @override
136
+ */
137
+Blockly.FieldImage.prototype.setValue = function(src) {
138
+  if (src === null) {
139
+    // No change if null.
140
+    return;
141
+  }
142
+  this.src_ = src;
143
+  if (this.imageElement_) {
144
+    this.imageElement_.setAttributeNS('http://www.w3.org/1999/xlink',
145
+        'xlink:href', goog.isString(src) ? src : '');
146
+  }
147
+};
148
+
149
+/**
150
+ * Set the alt text of this image.
151
+ * @param {?string} alt New alt text.
152
+ * @override
153
+ */
154
+Blockly.FieldImage.prototype.setText = function(alt) {
155
+  if (alt === null) {
156
+    // No change if null.
157
+    return;
158
+  }
159
+  this.text_ = alt;
160
+};
161
+
162
+/**
163
+ * Images are fixed width, no need to render.
164
+ * @private
165
+ */
166
+Blockly.FieldImage.prototype.render_ = function() {
167
+  // NOP
168
+};

+ 107 - 0
src/blockly/core/field_label.js

@@ -0,0 +1,107 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Non-editable text field.  Used for titles, labels, etc.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.FieldLabel');
28
+
29
+goog.require('Blockly.Field');
30
+goog.require('Blockly.Tooltip');
31
+goog.require('goog.dom');
32
+goog.require('goog.math.Size');
33
+
34
+
35
+/**
36
+ * Class for a non-editable field.
37
+ * @param {string} text The initial content of the field.
38
+ * @param {string=} opt_class Optional CSS class for the field's text.
39
+ * @extends {Blockly.Field}
40
+ * @constructor
41
+ */
42
+Blockly.FieldLabel = function(text, opt_class) {
43
+  this.size_ = new goog.math.Size(0, 17.5);
44
+  this.class_ = opt_class;
45
+  this.setText(text);
46
+};
47
+goog.inherits(Blockly.FieldLabel, Blockly.Field);
48
+
49
+/**
50
+ * Editable fields are saved by the XML renderer, non-editable fields are not.
51
+ */
52
+Blockly.FieldLabel.prototype.EDITABLE = false;
53
+
54
+/**
55
+ * Install this text on a block.
56
+ * @param {!Blockly.Block} block The block containing this text.
57
+ */
58
+Blockly.FieldLabel.prototype.init = function(block) {
59
+  if (this.sourceBlock_) {
60
+    // Text has already been initialized once.
61
+    return;
62
+  }
63
+  this.sourceBlock_ = block;
64
+
65
+  // Build the DOM.
66
+  this.textElement_ = Blockly.createSvgElement('text',
67
+      {'class': 'blocklyText', 'y': this.size_.height - 5}, null);
68
+  if (this.class_) {
69
+    Blockly.addClass_(this.textElement_, this.class_);
70
+  }
71
+  if (!this.visible_) {
72
+    this.textElement_.style.display = 'none';
73
+  }
74
+  block.getSvgRoot().appendChild(this.textElement_);
75
+
76
+  // Configure the field to be transparent with respect to tooltips.
77
+  this.textElement_.tooltip = this.sourceBlock_;
78
+  Blockly.Tooltip.bindMouseEvents(this.textElement_);
79
+  // Force a render.
80
+  this.updateTextNode_();
81
+};
82
+
83
+/**
84
+ * Dispose of all DOM objects belonging to this text.
85
+ */
86
+Blockly.FieldLabel.prototype.dispose = function() {
87
+  goog.dom.removeNode(this.textElement_);
88
+  this.textElement_ = null;
89
+};
90
+
91
+/**
92
+ * Gets the group element for this field.
93
+ * Used for measuring the size and for positioning.
94
+ * @return {!Element} The group element.
95
+ */
96
+Blockly.FieldLabel.prototype.getSvgRoot = function() {
97
+  return /** @type {!Element} */ (this.textElement_);
98
+};
99
+
100
+/**
101
+ * Change the tooltip text for this field.
102
+ * @param {string|!Element} newTip Text for tooltip or a parent element to
103
+ *     link to for its tooltip.
104
+ */
105
+Blockly.FieldLabel.prototype.setTooltip = function(newTip) {
106
+  this.textElement_.tooltip = newTip;
107
+};

+ 328 - 0
src/blockly/core/field_textinput.js

@@ -0,0 +1,328 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Text input field.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.FieldTextInput');
28
+
29
+goog.require('Blockly.Field');
30
+goog.require('Blockly.Msg');
31
+goog.require('goog.asserts');
32
+goog.require('goog.dom');
33
+goog.require('goog.userAgent');
34
+
35
+
36
+/**
37
+ * Class for an editable text field.
38
+ * @param {string} text The initial content of the field.
39
+ * @param {Function=} opt_changeHandler An optional function that is called
40
+ *     to validate any constraints on what the user entered.  Takes the new
41
+ *     text as an argument and returns either the accepted text, a replacement
42
+ *     text, or null to abort the change.
43
+ * @extends {Blockly.Field}
44
+ * @constructor
45
+ */
46
+Blockly.FieldTextInput = function(text, opt_changeHandler) {
47
+  Blockly.FieldTextInput.superClass_.constructor.call(this, text);
48
+  this.setChangeHandler(opt_changeHandler);
49
+};
50
+goog.inherits(Blockly.FieldTextInput, Blockly.Field);
51
+
52
+/**
53
+ * Point size of text.  Should match blocklyText's font-size in CSS.
54
+ */
55
+Blockly.FieldTextInput.FONTSIZE = 11;
56
+
57
+/**
58
+ * Mouse cursor style when over the hotspot that initiates the editor.
59
+ */
60
+Blockly.FieldTextInput.prototype.CURSOR = 'text';
61
+
62
+/**
63
+ * Allow browser to spellcheck this field.
64
+ * @private
65
+ */
66
+Blockly.FieldTextInput.prototype.spellcheck_ = true;
67
+
68
+/**
69
+ * Close the input widget if this input is being deleted.
70
+ */
71
+Blockly.FieldTextInput.prototype.dispose = function() {
72
+  Blockly.WidgetDiv.hideIfOwner(this);
73
+  Blockly.FieldTextInput.superClass_.dispose.call(this);
74
+};
75
+
76
+/**
77
+ * Set the text in this field.
78
+ * @param {?string} text New text.
79
+ * @override
80
+ */
81
+Blockly.FieldTextInput.prototype.setText = function(text) {
82
+  if (text === null) {
83
+    // No change if null.
84
+    return;
85
+  }
86
+  if (this.sourceBlock_ && this.changeHandler_) {
87
+    var validated = this.changeHandler_(text);
88
+    // If the new text is invalid, validation returns null.
89
+    // In this case we still want to display the illegal result.
90
+    if (validated !== null && validated !== undefined) {
91
+      text = validated;
92
+    }
93
+  }
94
+  Blockly.Field.prototype.setText.call(this, text);
95
+};
96
+
97
+/**
98
+ * Set whether this field is spellchecked by the browser.
99
+ * @param {boolean} check True if checked.
100
+ */
101
+Blockly.FieldTextInput.prototype.setSpellcheck = function(check) {
102
+  this.spellcheck_ = check;
103
+};
104
+
105
+/**
106
+ * Show the inline free-text editor on top of the text.
107
+ * @param {boolean=} opt_quietInput True if editor should be created without
108
+ *     focus.  Defaults to false.
109
+ * @private
110
+ */
111
+Blockly.FieldTextInput.prototype.showEditor_ = function(opt_quietInput) {
112
+  var quietInput = opt_quietInput || false;
113
+  if (!quietInput && (goog.userAgent.MOBILE || goog.userAgent.ANDROID ||
114
+                      goog.userAgent.IPAD)) {
115
+    // Mobile browsers have issues with in-line textareas (focus & keyboards).
116
+    var newValue = window.prompt(Blockly.Msg.CHANGE_VALUE_TITLE, this.text_);
117
+    if (this.sourceBlock_ && this.changeHandler_) {
118
+      var override = this.changeHandler_(newValue);
119
+      if (override !== undefined) {
120
+        newValue = override;
121
+      }
122
+    }
123
+    if (newValue !== null) {
124
+      this.setText(newValue);
125
+    }
126
+    return;
127
+  }
128
+
129
+  Blockly.WidgetDiv.show(this, this.sourceBlock_.RTL, this.widgetDispose_());
130
+  var div = Blockly.WidgetDiv.DIV;
131
+  // Create the input.
132
+  var htmlInput = goog.dom.createDom('input', 'blocklyHtmlInput');
133
+  htmlInput.setAttribute('spellcheck', this.spellcheck_);
134
+  var fontSize = (Blockly.FieldTextInput.FONTSIZE *
135
+                  this.sourceBlock_.workspace.scale) + 'pt';
136
+  div.style.fontSize = fontSize;
137
+  htmlInput.style.fontSize = fontSize;
138
+  /** @type {!HTMLInputElement} */
139
+  Blockly.FieldTextInput.htmlInput_ = htmlInput;
140
+  div.appendChild(htmlInput);
141
+
142
+  htmlInput.value = htmlInput.defaultValue = this.text_;
143
+  htmlInput.oldValue_ = null;
144
+  this.validate_();
145
+  this.resizeEditor_();
146
+  if (!quietInput) {
147
+    htmlInput.focus();
148
+    htmlInput.select();
149
+  }
150
+
151
+  // Bind to keydown -- trap Enter without IME and Esc to hide.
152
+  htmlInput.onKeyDownWrapper_ =
153
+      Blockly.bindEvent_(htmlInput, 'keydown', this, this.onHtmlInputKeyDown_);
154
+  // Bind to keyup -- trap Enter; resize after every keystroke.
155
+  htmlInput.onKeyUpWrapper_ =
156
+      Blockly.bindEvent_(htmlInput, 'keyup', this, this.onHtmlInputChange_);
157
+  // Bind to keyPress -- repeatedly resize when holding down a key.
158
+  htmlInput.onKeyPressWrapper_ =
159
+      Blockly.bindEvent_(htmlInput, 'keypress', this, this.onHtmlInputChange_);
160
+  var workspaceSvg = this.sourceBlock_.workspace.getCanvas();
161
+  htmlInput.onWorkspaceChangeWrapper_ =
162
+      Blockly.bindEvent_(workspaceSvg, 'blocklyWorkspaceChange', this,
163
+      this.resizeEditor_);
164
+};
165
+
166
+/**
167
+ * Handle key down to the editor.
168
+ * @param {!Event} e Keyboard event.
169
+ * @private
170
+ */
171
+Blockly.FieldTextInput.prototype.onHtmlInputKeyDown_ = function(e) {
172
+  var htmlInput = Blockly.FieldTextInput.htmlInput_;
173
+  var tabKey = 9, enterKey = 13, escKey = 27;
174
+  if (e.keyCode == enterKey) {
175
+    Blockly.WidgetDiv.hide();
176
+  } else if (e.keyCode == escKey) {
177
+    this.setText(htmlInput.defaultValue);
178
+    Blockly.WidgetDiv.hide();
179
+  } else if (e.keyCode == tabKey) {
180
+    Blockly.WidgetDiv.hide();
181
+    this.sourceBlock_.tab(this, !e.shiftKey);
182
+    e.preventDefault();
183
+  }
184
+};
185
+
186
+/**
187
+ * Handle a change to the editor.
188
+ * @param {!Event} e Keyboard event.
189
+ * @private
190
+ */
191
+Blockly.FieldTextInput.prototype.onHtmlInputChange_ = function(e) {
192
+  var htmlInput = Blockly.FieldTextInput.htmlInput_;
193
+  var escKey = 27;
194
+  if (e.keyCode != escKey) {
195
+    // Update source block.
196
+    var text = htmlInput.value;
197
+    if (text !== htmlInput.oldValue_) {
198
+      this.sourceBlock_.setShadow(false);
199
+      htmlInput.oldValue_ = text;
200
+      this.setText(text);
201
+      this.validate_();
202
+    } else if (goog.userAgent.WEBKIT) {
203
+      // Cursor key.  Render the source block to show the caret moving.
204
+      // Chrome only (version 26, OS X).
205
+      this.sourceBlock_.render();
206
+    }
207
+  }
208
+};
209
+
210
+/**
211
+ * Check to see if the contents of the editor validates.
212
+ * Style the editor accordingly.
213
+ * @private
214
+ */
215
+Blockly.FieldTextInput.prototype.validate_ = function() {
216
+  var valid = true;
217
+  goog.asserts.assertObject(Blockly.FieldTextInput.htmlInput_);
218
+  var htmlInput = Blockly.FieldTextInput.htmlInput_;
219
+  if (this.sourceBlock_ && this.changeHandler_) {
220
+    valid = this.changeHandler_(htmlInput.value);
221
+  }
222
+  if (valid === null) {
223
+    Blockly.addClass_(htmlInput, 'blocklyInvalidInput');
224
+  } else {
225
+    Blockly.removeClass_(htmlInput, 'blocklyInvalidInput');
226
+  }
227
+};
228
+
229
+/**
230
+ * Resize the editor and the underlying block to fit the text.
231
+ * @private
232
+ */
233
+Blockly.FieldTextInput.prototype.resizeEditor_ = function() {
234
+  var div = Blockly.WidgetDiv.DIV;
235
+  var bBox = this.fieldGroup_.getBBox();
236
+  div.style.width = bBox.width * this.sourceBlock_.workspace.scale + 'px';
237
+  div.style.height = bBox.height * this.sourceBlock_.workspace.scale + 'px';
238
+  var xy = this.getAbsoluteXY_();
239
+  // In RTL mode block fields and LTR input fields the left edge moves,
240
+  // whereas the right edge is fixed.  Reposition the editor.
241
+  if (this.sourceBlock_.RTL) {
242
+    var borderBBox = this.getScaledBBox_();
243
+    xy.x += borderBBox.width;
244
+    xy.x -= div.offsetWidth;
245
+  }
246
+  // Shift by a few pixels to line up exactly.
247
+  xy.y += 1;
248
+  if (goog.userAgent.GECKO && Blockly.WidgetDiv.DIV.style.top) {
249
+    // Firefox mis-reports the location of the border by a pixel
250
+    // once the WidgetDiv is moved into position.
251
+    xy.x -= 1;
252
+    xy.y -= 1;
253
+  }
254
+  if (goog.userAgent.WEBKIT) {
255
+    xy.y -= 3;
256
+  }
257
+  div.style.left = xy.x + 'px';
258
+  div.style.top = xy.y + 'px';
259
+};
260
+
261
+/**
262
+ * Close the editor, save the results, and dispose of the editable
263
+ * text field's elements.
264
+ * @return {!Function} Closure to call on destruction of the WidgetDiv.
265
+ * @private
266
+ */
267
+Blockly.FieldTextInput.prototype.widgetDispose_ = function() {
268
+  var thisField = this;
269
+  return function() {
270
+    var htmlInput = Blockly.FieldTextInput.htmlInput_;
271
+    // Save the edit (if it validates).
272
+    var text = htmlInput.value;
273
+    if (thisField.sourceBlock_ && thisField.changeHandler_) {
274
+      var text1 = thisField.changeHandler_(text);
275
+      if (text1 === null) {
276
+        // Invalid edit.
277
+        text = htmlInput.defaultValue;
278
+      } else if (text1 !== undefined) {
279
+        // Change handler has changed the text.
280
+        text = text1;
281
+      }
282
+    }
283
+    thisField.setText(text);
284
+    thisField.sourceBlock_.rendered && thisField.sourceBlock_.render();
285
+    Blockly.unbindEvent_(htmlInput.onKeyDownWrapper_);
286
+    Blockly.unbindEvent_(htmlInput.onKeyUpWrapper_);
287
+    Blockly.unbindEvent_(htmlInput.onKeyPressWrapper_);
288
+    Blockly.unbindEvent_(htmlInput.onWorkspaceChangeWrapper_);
289
+    Blockly.FieldTextInput.htmlInput_ = null;
290
+    // Delete style properties.
291
+    var style = Blockly.WidgetDiv.DIV.style;
292
+    style.width = 'auto';
293
+    style.height = 'auto';
294
+    style.fontSize = '';
295
+  };
296
+};
297
+
298
+/**
299
+ * Ensure that only a number may be entered.
300
+ * @param {string} text The user's text.
301
+ * @return {?string} A string representing a valid number, or null if invalid.
302
+ */
303
+Blockly.FieldTextInput.numberValidator = function(text) {
304
+  if (text === null) {
305
+    return null;
306
+  }
307
+  text = String(text);
308
+  // TODO: Handle cases like 'ten', '1.203,14', etc.
309
+  // 'O' is sometimes mistaken for '0' by inexperienced users.
310
+  text = text.replace(/O/ig, '0');
311
+  // Strip out thousands separators.
312
+  text = text.replace(/,/g, '');
313
+  var n = parseFloat(text || 0);
314
+  return isNaN(n) ? null : String(n);
315
+};
316
+
317
+/**
318
+ * Ensure that only a nonnegative integer may be entered.
319
+ * @param {string} text The user's text.
320
+ * @return {?string} A string representing a valid int, or null if invalid.
321
+ */
322
+Blockly.FieldTextInput.nonnegativeIntegerValidator = function(text) {
323
+  var n = Blockly.FieldTextInput.numberValidator(text);
324
+  if (n) {
325
+    n = String(Math.max(0, Math.floor(n)));
326
+  }
327
+  return n;
328
+};

+ 197 - 0
src/blockly/core/field_variable.js

@@ -0,0 +1,197 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Variable input field.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.FieldVariable');
28
+
29
+goog.require('Blockly.FieldDropdown');
30
+goog.require('Blockly.Msg');
31
+goog.require('Blockly.Variables');
32
+goog.require('goog.string');
33
+
34
+
35
+/**
36
+ * Class for a variable's dropdown field.
37
+ * @param {?string} varname The default name for the variable.  If null,
38
+ *     a unique variable name will be generated.
39
+ * @param {Function=} opt_changeHandler A function that is executed when a new
40
+ *     option is selected.  Its sole argument is the new option value.
41
+ * @extends {Blockly.FieldDropdown}
42
+ * @constructor
43
+ */
44
+Blockly.FieldVariable = function(varname, opt_changeHandler) {
45
+  Blockly.FieldVariable.superClass_.constructor.call(this,
46
+      Blockly.FieldVariable.dropdownCreate, null);
47
+  this.setChangeHandler(opt_changeHandler);
48
+  this.setValue(varname || '');
49
+};
50
+goog.inherits(Blockly.FieldVariable, Blockly.FieldDropdown);
51
+
52
+/**
53
+ * Sets a new change handler for angle field.
54
+ * @param {Function} handler New change handler, or null.
55
+ */
56
+Blockly.FieldVariable.prototype.setChangeHandler = function(handler) {
57
+  var wrappedHandler;
58
+  if (handler) {
59
+    // Wrap the user's change handler together with the variable rename handler.
60
+    wrappedHandler = function(value) {
61
+      var v1 = handler.call(this, value);
62
+      if (v1 === null) {
63
+        var v2 = v1;
64
+      } else {
65
+        if (v1 === undefined) {
66
+          v1 = value;
67
+        }
68
+        var v2 = Blockly.FieldVariable.dropdownChange.call(this, v1);
69
+        if (v2 !== undefined) {
70
+          v2 = v1;
71
+        }
72
+      }
73
+      return v2 === value ? undefined : v2;
74
+    };
75
+  } else {
76
+    wrappedHandler = Blockly.FieldVariable.dropdownChange;
77
+  }
78
+  Blockly.FieldVariable.superClass_.setChangeHandler.call(this, wrappedHandler);
79
+};
80
+
81
+/**
82
+ * Install this dropdown on a block.
83
+ * @param {!Blockly.Block} block The block containing this text.
84
+ */
85
+Blockly.FieldVariable.prototype.init = function(block) {
86
+  if (this.sourceBlock_) {
87
+    // Dropdown has already been initialized once.
88
+    return;
89
+  }
90
+
91
+  if (!this.getValue()) {
92
+    // Variables without names get uniquely named for this workspace.
93
+    if (block.isInFlyout) {
94
+      var workspace = block.workspace.targetWorkspace;
95
+    } else {
96
+      var workspace = block.workspace;
97
+    }
98
+    this.setValue(Blockly.Variables.generateUniqueName(workspace));
99
+  }
100
+  Blockly.FieldVariable.superClass_.init.call(this, block);
101
+};
102
+
103
+/**
104
+ * Get the variable's name (use a variableDB to convert into a real name).
105
+ * Unline a regular dropdown, variables are literal and have no neutral value.
106
+ * @return {string} Current text.
107
+ */
108
+Blockly.FieldVariable.prototype.getValue = function() {
109
+  return this.getText();
110
+};
111
+
112
+/**
113
+ * Set the variable name.
114
+ * @param {string} text New text.
115
+ */
116
+Blockly.FieldVariable.prototype.setValue = function(text) {
117
+  this.value_ = text;
118
+  this.setText(text);
119
+};
120
+
121
+/**
122
+ * Return a sorted list of variable names for variable dropdown menus.
123
+ * Include a special option at the end for creating a new variable name.
124
+ * @return {!Array.<string>} Array of variable names.
125
+ * @this {!Blockly.FieldVariable}
126
+ */
127
+Blockly.FieldVariable.dropdownCreate = function() {
128
+  if (this.sourceBlock_ && this.sourceBlock_.workspace) {
129
+    var variableList =
130
+        Blockly.Variables.allVariables(this.sourceBlock_.workspace);
131
+  } else {
132
+    var variableList = [];
133
+  }
134
+  // Ensure that the currently selected variable is an option.
135
+  var name = this.getText();
136
+  if (name && variableList.indexOf(name) == -1) {
137
+    variableList.push(name);
138
+  }
139
+  variableList.sort(goog.string.caseInsensitiveCompare);
140
+  variableList.push(Blockly.Msg.RENAME_VARIABLE);
141
+  variableList.push(Blockly.Msg.NEW_VARIABLE);
142
+  // Variables are not language-specific, use the name as both the user-facing
143
+  // text and the internal representation.
144
+  var options = [];
145
+  for (var x = 0; x < variableList.length; x++) {
146
+    options[x] = [variableList[x], variableList[x]];
147
+  }
148
+  return options;
149
+};
150
+
151
+/**
152
+ * Event handler for a change in variable name.
153
+ * Special case the 'New variable...' and 'Rename variable...' options.
154
+ * In both of these special cases, prompt the user for a new name.
155
+ * @param {string} text The selected dropdown menu option.
156
+ * @return {null|undefined|string} An acceptable new variable name, or null if
157
+ *     change is to be either aborted (cancel button) or has been already
158
+ *     handled (rename), or undefined if an existing variable was chosen.
159
+ * @this {!Blockly.FieldVariable}
160
+ */
161
+Blockly.FieldVariable.dropdownChange = function(text) {
162
+  function promptName(promptText, defaultText) {
163
+    Blockly.hideChaff();
164
+    var newVar = window.prompt(promptText, defaultText);
165
+    // Merge runs of whitespace.  Strip leading and trailing whitespace.
166
+    // Beyond this, all names are legal.
167
+    if (newVar) {
168
+      newVar = newVar.replace(/[\s\xa0]+/g, ' ').replace(/^ | $/g, '');
169
+      if (newVar == Blockly.Msg.RENAME_VARIABLE ||
170
+          newVar == Blockly.Msg.NEW_VARIABLE) {
171
+        // Ok, not ALL names are legal...
172
+        newVar = null;
173
+      }
174
+    }
175
+    return newVar;
176
+  }
177
+  var workspace = this.sourceBlock_.workspace;
178
+  if (text == Blockly.Msg.RENAME_VARIABLE) {
179
+    var oldVar = this.getText();
180
+    text = promptName(Blockly.Msg.RENAME_VARIABLE_TITLE.replace('%1', oldVar),
181
+                      oldVar);
182
+    if (text) {
183
+      Blockly.Variables.renameVariable(oldVar, text, workspace);
184
+    }
185
+    return null;
186
+  } else if (text == Blockly.Msg.NEW_VARIABLE) {
187
+    text = promptName(Blockly.Msg.NEW_VARIABLE_TITLE, '');
188
+    // Since variables are case-insensitive, ensure that if the new variable
189
+    // matches with an existing variable, the new case prevails throughout.
190
+    if (text) {
191
+      Blockly.Variables.renameVariable(text, text, workspace);
192
+      return text;
193
+    }
194
+    return null;
195
+  }
196
+  return undefined;
197
+};

+ 731 - 0
src/blockly/core/flyout.js

@@ -0,0 +1,731 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2011 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Flyout tray containing blocks which may be created.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Flyout');
28
+
29
+goog.require('Blockly.Block');
30
+goog.require('Blockly.Comment');
31
+goog.require('Blockly.WorkspaceSvg');
32
+goog.require('goog.dom');
33
+goog.require('goog.events');
34
+goog.require('goog.math.Rect');
35
+goog.require('goog.userAgent');
36
+
37
+
38
+/**
39
+ * Class for a flyout.
40
+ * @param {!Object} workspaceOptions Dictionary of options for the workspace.
41
+ * @constructor
42
+ */
43
+Blockly.Flyout = function(workspaceOptions) {
44
+  var flyout = this;
45
+  workspaceOptions.getMetrics = function() {return flyout.getMetrics_();};
46
+  workspaceOptions.setMetrics =
47
+      function(ratio) {return flyout.setMetrics_(ratio);};
48
+  /**
49
+   * @type {!Blockly.Workspace}
50
+   * @private
51
+   */
52
+  this.workspace_ = new Blockly.WorkspaceSvg(workspaceOptions);
53
+  this.workspace_.isFlyout = true;
54
+
55
+  /**
56
+   * Is RTL vs LTR.
57
+   * @type {boolean}
58
+   */
59
+  this.RTL = !!workspaceOptions.RTL;
60
+
61
+  /**
62
+   * Opaque data that can be passed to Blockly.unbindEvent_.
63
+   * @type {!Array.<!Array>}
64
+   * @private
65
+   */
66
+  this.eventWrappers_ = [];
67
+
68
+  /**
69
+   * List of background buttons that lurk behind each block to catch clicks
70
+   * landing in the blocks' lakes and bays.
71
+   * @type {!Array.<!Element>}
72
+   * @private
73
+   */
74
+  this.buttons_ = [];
75
+
76
+  /**
77
+   * List of event listeners.
78
+   * @type {!Array.<!Array>}
79
+   * @private
80
+   */
81
+  this.listeners_ = [];
82
+};
83
+
84
+/**
85
+ * Does the flyout automatically close when a block is created?
86
+ * @type {boolean}
87
+ */
88
+Blockly.Flyout.prototype.autoClose = true;
89
+
90
+/**
91
+ * Corner radius of the flyout background.
92
+ * @type {number}
93
+ * @const
94
+ */
95
+Blockly.Flyout.prototype.CORNER_RADIUS = 8;
96
+
97
+/**
98
+ * Top/bottom padding between scrollbar and edge of flyout background.
99
+ * @type {number}
100
+ * @const
101
+ */
102
+Blockly.Flyout.prototype.SCROLLBAR_PADDING = 2;
103
+
104
+/**
105
+ * Width of flyout.
106
+ * @type {number}
107
+ * @private
108
+ */
109
+Blockly.Flyout.prototype.width_ = 0;
110
+
111
+/**
112
+ * Height of flyout.
113
+ * @type {number}
114
+ * @private
115
+ */
116
+Blockly.Flyout.prototype.height_ = 0;
117
+
118
+/**
119
+ * Creates the flyout's DOM.  Only needs to be called once.
120
+ * @return {!Element} The flyout's SVG group.
121
+ */
122
+Blockly.Flyout.prototype.createDom = function() {
123
+  /*
124
+  <g>
125
+    <path class="blocklyFlyoutBackground"/>
126
+    <g class="blocklyFlyout"></g>
127
+  </g>
128
+  */
129
+  this.svgGroup_ = Blockly.createSvgElement('g',
130
+      {'class': 'blocklyFlyout'}, null);
131
+  this.svgBackground_ = Blockly.createSvgElement('path',
132
+      {'class': 'blocklyFlyoutBackground'}, this.svgGroup_);
133
+  this.svgGroup_.appendChild(this.workspace_.createDom());
134
+  return this.svgGroup_;
135
+};
136
+
137
+/**
138
+ * Initializes the flyout.
139
+ * @param {!Blockly.Workspace} targetWorkspace The workspace in which to create
140
+ *     new blocks.
141
+ */
142
+Blockly.Flyout.prototype.init = function(targetWorkspace) {
143
+  this.targetWorkspace_ = targetWorkspace;
144
+  this.workspace_.targetWorkspace = targetWorkspace;
145
+  // Add scrollbar.
146
+  this.scrollbar_ = new Blockly.Scrollbar(this.workspace_, false, false);
147
+
148
+  this.hide();
149
+
150
+  Array.prototype.push.apply(this.eventWrappers_,
151
+      Blockly.bindEvent_(this.svgGroup_, 'wheel', this, this.wheel_));
152
+  if (!this.autoClose) {
153
+    Array.prototype.push.apply(this.eventWrappers_,
154
+        Blockly.bindEvent_(this.targetWorkspace_.getCanvas(),
155
+        'blocklyWorkspaceChange', this, this.filterForCapacity_));
156
+  }
157
+  // Dragging the flyout up and down.
158
+  Array.prototype.push.apply(this.eventWrappers_,
159
+      Blockly.bindEvent_(this.svgGroup_, 'mousedown', this, this.onMouseDown_));
160
+};
161
+
162
+/**
163
+ * Dispose of this flyout.
164
+ * Unlink from all DOM elements to prevent memory leaks.
165
+ */
166
+Blockly.Flyout.prototype.dispose = function() {
167
+  this.hide();
168
+  Blockly.unbindEvent_(this.eventWrappers_);
169
+  if (this.scrollbar_) {
170
+    this.scrollbar_.dispose();
171
+    this.scrollbar_ = null;
172
+  }
173
+  if (this.workspace_) {
174
+    this.workspace_.targetWorkspace = null;
175
+    this.workspace_.dispose();
176
+    this.workspace_ = null;
177
+  }
178
+  if (this.svgGroup_) {
179
+    goog.dom.removeNode(this.svgGroup_);
180
+    this.svgGroup_ = null;
181
+  }
182
+  this.svgBackground_ = null;
183
+  this.targetWorkspace_ = null;
184
+};
185
+
186
+/**
187
+ * Return an object with all the metrics required to size scrollbars for the
188
+ * flyout.  The following properties are computed:
189
+ * .viewHeight: Height of the visible rectangle,
190
+ * .viewWidth: Width of the visible rectangle,
191
+ * .contentHeight: Height of the contents,
192
+ * .viewTop: Offset of top edge of visible rectangle from parent,
193
+ * .contentTop: Offset of the top-most content from the y=0 coordinate,
194
+ * .absoluteTop: Top-edge of view.
195
+ * .absoluteLeft: Left-edge of view.
196
+ * @return {Object} Contains size and position metrics of the flyout.
197
+ * @private
198
+ */
199
+Blockly.Flyout.prototype.getMetrics_ = function() {
200
+  if (!this.isVisible()) {
201
+    // Flyout is hidden.
202
+    return null;
203
+  }
204
+  var viewHeight = this.height_ - 2 * this.SCROLLBAR_PADDING;
205
+  var viewWidth = this.width_;
206
+  try {
207
+    var optionBox = this.workspace_.getCanvas().getBBox();
208
+  } catch (e) {
209
+    // Firefox has trouble with hidden elements (Bug 528969).
210
+    var optionBox = {height: 0, y: 0};
211
+  }
212
+  return {
213
+    viewHeight: viewHeight,
214
+    viewWidth: viewWidth,
215
+    contentHeight: (optionBox.height + optionBox.y) * this.workspace_.scale,
216
+    viewTop: -this.workspace_.scrollY,
217
+    contentTop: 0,
218
+    absoluteTop: this.SCROLLBAR_PADDING,
219
+    absoluteLeft: 0
220
+  };
221
+};
222
+
223
+/**
224
+ * Sets the Y translation of the flyout to match the scrollbars.
225
+ * @param {!Object} yRatio Contains a y property which is a float
226
+ *     between 0 and 1 specifying the degree of scrolling.
227
+ * @private
228
+ */
229
+Blockly.Flyout.prototype.setMetrics_ = function(yRatio) {
230
+  var metrics = this.getMetrics_();
231
+  // This is a fix to an apparent race condition.
232
+  if (!metrics) {
233
+    return;
234
+  }
235
+  if (goog.isNumber(yRatio.y)) {
236
+    this.workspace_.scrollY =
237
+        -metrics.contentHeight * yRatio.y - metrics.contentTop;
238
+  }
239
+  this.workspace_.translate(0, this.workspace_.scrollY + metrics.absoluteTop);
240
+};
241
+
242
+/**
243
+ * Move the toolbox to the edge of the workspace.
244
+ */
245
+Blockly.Flyout.prototype.position = function() {
246
+  if (!this.isVisible()) {
247
+    return;
248
+  }
249
+  var metrics = this.targetWorkspace_.getMetrics();
250
+  if (!metrics) {
251
+    // Hidden components will return null.
252
+    return;
253
+  }
254
+  var edgeWidth = this.width_ - this.CORNER_RADIUS;
255
+  if (this.RTL) {
256
+    edgeWidth *= -1;
257
+  }
258
+  var path = ['M ' + (this.RTL ? this.width_ : 0) + ',0'];
259
+  path.push('h', edgeWidth);
260
+  path.push('a', this.CORNER_RADIUS, this.CORNER_RADIUS, 0, 0,
261
+      this.RTL ? 0 : 1,
262
+      this.RTL ? -this.CORNER_RADIUS : this.CORNER_RADIUS,
263
+      this.CORNER_RADIUS);
264
+  path.push('v', Math.max(0, metrics.viewHeight - this.CORNER_RADIUS * 2));
265
+  path.push('a', this.CORNER_RADIUS, this.CORNER_RADIUS, 0, 0,
266
+      this.RTL ? 0 : 1,
267
+      this.RTL ? this.CORNER_RADIUS : -this.CORNER_RADIUS,
268
+      this.CORNER_RADIUS);
269
+  path.push('h', -edgeWidth);
270
+  path.push('z');
271
+  this.svgBackground_.setAttribute('d', path.join(' '));
272
+
273
+  var x = metrics.absoluteLeft;
274
+  if (this.RTL) {
275
+    x += metrics.viewWidth;
276
+    x -= this.width_;
277
+  }
278
+  this.svgGroup_.setAttribute('transform',
279
+      'translate(' + x + ',' + metrics.absoluteTop + ')');
280
+
281
+  // Record the height for Blockly.Flyout.getMetrics_.
282
+  this.height_ = metrics.viewHeight;
283
+
284
+  // Update the scrollbar (if one exists).
285
+  if (this.scrollbar_) {
286
+    this.scrollbar_.resize();
287
+  }
288
+};
289
+
290
+/**
291
+ * Scroll the flyout to the top.
292
+ */
293
+Blockly.Flyout.prototype.scrollToTop = function() {
294
+  this.scrollbar_.set(0);
295
+};
296
+
297
+/**
298
+ * Scroll the flyout up or down.
299
+ * @param {!Event} e Mouse wheel scroll event.
300
+ * @private
301
+ */
302
+Blockly.Flyout.prototype.wheel_ = function(e) {
303
+  var delta = e.deltaY;
304
+  if (delta) {
305
+    if (goog.userAgent.GECKO) {
306
+      // Firefox's deltas are a tenth that of Chrome/Safari.
307
+      delta *= 10;
308
+    }
309
+    var metrics = this.getMetrics_();
310
+    var y = metrics.viewTop + delta;
311
+    y = Math.min(y, metrics.contentHeight - metrics.viewHeight);
312
+    y = Math.max(y, 0);
313
+    this.scrollbar_.set(y);
314
+    // Don't scroll the page.
315
+    e.preventDefault();
316
+    // Don't propagate mousewheel event (zooming).
317
+    e.stopPropagation();
318
+  }
319
+};
320
+
321
+/**
322
+ * Is the flyout visible?
323
+ * @return {boolean} True if visible.
324
+ */
325
+Blockly.Flyout.prototype.isVisible = function() {
326
+  return this.svgGroup_ && this.svgGroup_.style.display == 'block';
327
+};
328
+
329
+/**
330
+ * Hide and empty the flyout.
331
+ */
332
+Blockly.Flyout.prototype.hide = function() {
333
+  if (!this.isVisible()) {
334
+    return;
335
+  }
336
+  this.svgGroup_.style.display = 'none';
337
+  // Delete all the event listeners.
338
+  for (var x = 0, listen; listen = this.listeners_[x]; x++) {
339
+    Blockly.unbindEvent_(listen);
340
+  }
341
+  this.listeners_.length = 0;
342
+  if (this.reflowWrapper_) {
343
+    Blockly.unbindEvent_(this.reflowWrapper_);
344
+    this.reflowWrapper_ = null;
345
+  }
346
+  // Do NOT delete the blocks here.  Wait until Flyout.show.
347
+  // https://neil.fraser.name/news/2014/08/09/
348
+};
349
+
350
+/**
351
+ * Show and populate the flyout.
352
+ * @param {!Array|string} xmlList List of blocks to show.
353
+ *     Variables and procedures have a custom set of blocks.
354
+ */
355
+Blockly.Flyout.prototype.show = function(xmlList) {
356
+  this.hide();
357
+  // Delete any blocks from a previous showing.
358
+  var blocks = this.workspace_.getTopBlocks(false);
359
+  for (var i = 0, block; block = blocks[i]; i++) {
360
+    if (block.workspace == this.workspace_) {
361
+      block.dispose(false, false);
362
+    }
363
+  }
364
+  // Delete any background buttons from a previous showing.
365
+  for (var i = 0, rect; rect = this.buttons_[i]; i++) {
366
+    goog.dom.removeNode(rect);
367
+  }
368
+  this.buttons_.length = 0;
369
+
370
+  if (xmlList == Blockly.Variables.NAME_TYPE) {
371
+    // Special category for variables.
372
+    xmlList =
373
+        Blockly.Variables.flyoutCategory(this.workspace_.targetWorkspace);
374
+  } else if (xmlList == Blockly.Procedures.NAME_TYPE) {
375
+    // Special category for procedures.
376
+    xmlList =
377
+        Blockly.Procedures.flyoutCategory(this.workspace_.targetWorkspace);
378
+  }
379
+
380
+  var margin = this.CORNER_RADIUS;
381
+  this.svgGroup_.style.display = 'block';
382
+  // Create the blocks to be shown in this flyout.
383
+  var blocks = [];
384
+  var gaps = [];
385
+  for (var i = 0, xml; xml = xmlList[i]; i++) {
386
+    if (xml.tagName && xml.tagName.toUpperCase() == 'BLOCK') {
387
+      var block = Blockly.Xml.domToBlock(
388
+          /** @type {!Blockly.Workspace} */ (this.workspace_), xml);
389
+      blocks.push(block);
390
+      var gap = parseInt(xml.getAttribute('gap'), 10);
391
+      gaps.push(gap || margin * 3);
392
+    }
393
+  }
394
+
395
+  // Lay out the blocks vertically.
396
+  var cursorY = margin;
397
+  for (var i = 0, block; block = blocks[i]; i++) {
398
+    var allBlocks = block.getDescendants();
399
+    for (var j = 0, child; child = allBlocks[j]; j++) {
400
+      // Mark blocks as being inside a flyout.  This is used to detect and
401
+      // prevent the closure of the flyout if the user right-clicks on such a
402
+      // block.
403
+      child.isInFlyout = true;
404
+      // There is no good way to handle comment bubbles inside the flyout.
405
+      // Blocks shouldn't come with predefined comments, but someone will
406
+      // try this, I'm sure.  Kill the comment.
407
+      child.setCommentText(null);
408
+    }
409
+    block.render();
410
+    var root = block.getSvgRoot();
411
+    var blockHW = block.getHeightWidth();
412
+    var x = this.RTL ? 0 : margin / this.workspace_.scale +
413
+        Blockly.BlockSvg.TAB_WIDTH;
414
+    block.moveBy(x, cursorY);
415
+    cursorY += blockHW.height + gaps[i];
416
+
417
+    // Create an invisible rectangle under the block to act as a button.  Just
418
+    // using the block as a button is poor, since blocks have holes in them.
419
+    var rect = Blockly.createSvgElement('rect', {'fill-opacity': 0}, null);
420
+    // Add the rectangles under the blocks, so that the blocks' tooltips work.
421
+    this.workspace_.getCanvas().insertBefore(rect, block.getSvgRoot());
422
+    block.flyoutRect_ = rect;
423
+    this.buttons_[i] = rect;
424
+
425
+    if (this.autoClose) {
426
+      this.listeners_.push(Blockly.bindEvent_(root, 'mousedown', null,
427
+          this.createBlockFunc_(block)));
428
+    } else {
429
+      this.listeners_.push(Blockly.bindEvent_(root, 'mousedown', null,
430
+          this.blockMouseDown_(block)));
431
+    }
432
+    this.listeners_.push(Blockly.bindEvent_(root, 'mouseover', block,
433
+        block.addSelect));
434
+    this.listeners_.push(Blockly.bindEvent_(root, 'mouseout', block,
435
+        block.removeSelect));
436
+    this.listeners_.push(Blockly.bindEvent_(rect, 'mousedown', null,
437
+        this.createBlockFunc_(block)));
438
+    this.listeners_.push(Blockly.bindEvent_(rect, 'mouseover', block,
439
+        block.addSelect));
440
+    this.listeners_.push(Blockly.bindEvent_(rect, 'mouseout', block,
441
+        block.removeSelect));
442
+  }
443
+
444
+  // IE 11 is an incompetant browser that fails to fire mouseout events.
445
+  // When the mouse is over the background, deselect all blocks.
446
+  var deselectAll = function(e) {
447
+    var blocks = this.workspace_.getTopBlocks(false);
448
+    for (var i = 0, block; block = blocks[i]; i++) {
449
+      block.removeSelect();
450
+    }
451
+  };
452
+  this.listeners_.push(Blockly.bindEvent_(this.svgBackground_, 'mouseover',
453
+      this, deselectAll));
454
+
455
+  this.width_ = 0;
456
+  this.reflow();
457
+
458
+  this.filterForCapacity_();
459
+
460
+  // Fire a resize event to update the flyout's scrollbar.
461
+  Blockly.fireUiEventNow(window, 'resize');
462
+  this.reflowWrapper_ = Blockly.bindEvent_(this.workspace_.getCanvas(),
463
+      'blocklyWorkspaceChange', this, this.reflow);
464
+  this.workspace_.fireChangeEvent();
465
+};
466
+
467
+/**
468
+ * Compute width of flyout.  Position button under each block.
469
+ * For RTL: Lay out the blocks right-aligned.
470
+ */
471
+Blockly.Flyout.prototype.reflow = function() {
472
+  this.workspace_.scale = this.targetWorkspace_.scale;
473
+  var flyoutWidth = 0;
474
+  var margin = this.CORNER_RADIUS;
475
+  var blocks = this.workspace_.getTopBlocks(false);
476
+  for (var x = 0, block; block = blocks[x]; x++) {
477
+    var width = block.getHeightWidth().width;
478
+    if (block.outputConnection) {
479
+      width -= Blockly.BlockSvg.TAB_WIDTH;
480
+    }
481
+    flyoutWidth = Math.max(flyoutWidth, width);
482
+  }
483
+  flyoutWidth += Blockly.BlockSvg.TAB_WIDTH;
484
+  flyoutWidth *= this.workspace_.scale;
485
+  flyoutWidth += margin * 1.5 + Blockly.Scrollbar.scrollbarThickness;
486
+  if (this.width_ != flyoutWidth) {
487
+    for (var x = 0, block; block = blocks[x]; x++) {
488
+      var blockHW = block.getHeightWidth();
489
+      if (this.RTL) {
490
+        // With the flyoutWidth known, right-align the blocks.
491
+        var oldX = block.getRelativeToSurfaceXY().x;
492
+        var dx = flyoutWidth - margin;
493
+        dx /= this.workspace_.scale;
494
+        dx -= Blockly.BlockSvg.TAB_WIDTH;
495
+        block.moveBy(dx - oldX, 0);
496
+      }
497
+      if (block.flyoutRect_) {
498
+        block.flyoutRect_.setAttribute('width', blockHW.width);
499
+        block.flyoutRect_.setAttribute('height', blockHW.height);
500
+        // Blocks with output tabs are shifted a bit.
501
+        var tab = block.outputConnection ? Blockly.BlockSvg.TAB_WIDTH : 0;
502
+        var blockXY = block.getRelativeToSurfaceXY();
503
+        block.flyoutRect_.setAttribute('x',
504
+            this.RTL ? blockXY.x - blockHW.width + tab : blockXY.x - tab);
505
+        block.flyoutRect_.setAttribute('y', blockXY.y);
506
+      }
507
+    }
508
+    // Record the width for .getMetrics_ and .position.
509
+    this.width_ = flyoutWidth;
510
+    // Fire a resize event to update the flyout's scrollbar.
511
+    Blockly.fireUiEvent(window, 'resize');
512
+  }
513
+};
514
+
515
+/**
516
+ * Handle a mouse-down on an SVG block in a non-closing flyout.
517
+ * @param {!Blockly.Block} block The flyout block to copy.
518
+ * @return {!Function} Function to call when block is clicked.
519
+ * @private
520
+ */
521
+Blockly.Flyout.prototype.blockMouseDown_ = function(block) {
522
+  var flyout = this;
523
+  return function(e) {
524
+    Blockly.terminateDrag_();
525
+    Blockly.hideChaff();
526
+    if (Blockly.isRightButton(e)) {
527
+      // Right-click.
528
+      block.showContextMenu_(e);
529
+    } else {
530
+      // Left-click (or middle click)
531
+      Blockly.removeAllRanges();
532
+      Blockly.Css.setCursor(Blockly.Css.Cursor.CLOSED);
533
+      // Record the current mouse position.
534
+      Blockly.Flyout.startDownEvent_ = e;
535
+      Blockly.Flyout.startBlock_ = block;
536
+      Blockly.Flyout.startFlyout_ = flyout;
537
+      Blockly.Flyout.onMouseUpWrapper_ = Blockly.bindEvent_(document,
538
+          'mouseup', this, Blockly.terminateDrag_);
539
+      Blockly.Flyout.onMouseMoveBlockWrapper_ = Blockly.bindEvent_(document,
540
+          'mousemove', this, flyout.onMouseMoveBlock_);
541
+    }
542
+    // This event has been handled.  No need to bubble up to the document.
543
+    e.stopPropagation();
544
+  };
545
+};
546
+
547
+/**
548
+ * Mouse down on the flyout background.  Start a vertical scroll drag.
549
+ * @param {!Event} e Mouse down event.
550
+ * @private
551
+ */
552
+Blockly.Flyout.prototype.onMouseDown_ = function(e) {
553
+  if (Blockly.isRightButton(e)) {
554
+    return;
555
+  }
556
+  Blockly.hideChaff(true);
557
+  Blockly.Flyout.terminateDrag_();
558
+  this.startDragMouseY_ = e.clientY;
559
+  Blockly.Flyout.onMouseMoveWrapper_ = Blockly.bindEvent_(document, 'mousemove',
560
+      this, this.onMouseMove_);
561
+  Blockly.Flyout.onMouseUpWrapper_ = Blockly.bindEvent_(document, 'mouseup',
562
+      this, Blockly.Flyout.terminateDrag_);
563
+  // This event has been handled.  No need to bubble up to the document.
564
+  e.preventDefault();
565
+  e.stopPropagation();
566
+};
567
+
568
+/**
569
+ * Handle a mouse-move to vertically drag the flyout.
570
+ * @param {!Event} e Mouse move event.
571
+ * @private
572
+ */
573
+Blockly.Flyout.prototype.onMouseMove_ = function(e) {
574
+  var dy = e.clientY - this.startDragMouseY_;
575
+  this.startDragMouseY_ = e.clientY;
576
+  var metrics = this.getMetrics_();
577
+  var y = metrics.viewTop - dy;
578
+  y = Math.min(y, metrics.contentHeight - metrics.viewHeight);
579
+  y = Math.max(y, 0);
580
+  this.scrollbar_.set(y);
581
+};
582
+
583
+/**
584
+ * Mouse button is down on a block in a non-closing flyout.  Create the block
585
+ * if the mouse moves beyond a small radius.  This allows one to play with
586
+ * fields without instantiating blocks that instantly self-destruct.
587
+ * @param {!Event} e Mouse move event.
588
+ * @private
589
+ */
590
+Blockly.Flyout.prototype.onMouseMoveBlock_ = function(e) {
591
+  if (e.type == 'mousemove' && e.clientX <= 1 && e.clientY == 0 &&
592
+      e.button == 0) {
593
+    /* HACK:
594
+     Safari Mobile 6.0 and Chrome for Android 18.0 fire rogue mousemove events
595
+     on certain touch actions. Ignore events with these signatures.
596
+     This may result in a one-pixel blind spot in other browsers,
597
+     but this shouldn't be noticable. */
598
+    e.stopPropagation();
599
+    return;
600
+  }
601
+  Blockly.removeAllRanges();
602
+  var dx = e.clientX - Blockly.Flyout.startDownEvent_.clientX;
603
+  var dy = e.clientY - Blockly.Flyout.startDownEvent_.clientY;
604
+  // Still dragging within the sticky DRAG_RADIUS.
605
+  if (Math.sqrt(dx * dx + dy * dy) > Blockly.DRAG_RADIUS) {
606
+    // Create the block.
607
+    Blockly.Flyout.startFlyout_.createBlockFunc_(Blockly.Flyout.startBlock_)(
608
+        Blockly.Flyout.startDownEvent_);
609
+  }
610
+};
611
+
612
+/**
613
+ * Create a copy of this block on the workspace.
614
+ * @param {!Blockly.Block} originBlock The flyout block to copy.
615
+ * @return {!Function} Function to call when block is clicked.
616
+ * @private
617
+ */
618
+Blockly.Flyout.prototype.createBlockFunc_ = function(originBlock) {
619
+  var flyout = this;
620
+  var workspace = this.targetWorkspace_;
621
+  return function(e) {
622
+    if (Blockly.isRightButton(e)) {
623
+      // Right-click.  Don't create a block, let the context menu show.
624
+      return;
625
+    }
626
+    if (originBlock.disabled) {
627
+      // Beyond capacity.
628
+      return;
629
+    }
630
+    // Create the new block by cloning the block in the flyout (via XML).
631
+    var xml = Blockly.Xml.blockToDom_(originBlock);
632
+    var block = Blockly.Xml.domToBlock(workspace, xml);
633
+    // Place it in the same spot as the flyout copy.
634
+    var svgRootOld = originBlock.getSvgRoot();
635
+    if (!svgRootOld) {
636
+      throw 'originBlock is not rendered.';
637
+    }
638
+    var xyOld = Blockly.getSvgXY_(svgRootOld, workspace);
639
+    // Scale the scroll (getSvgXY_ did not do this).
640
+    if (flyout.RTL) {
641
+      var width = workspace.getMetrics().viewWidth - flyout.width_;
642
+      xyOld.x += width / workspace.scale - width;
643
+    } else {
644
+      xyOld.x += flyout.workspace_.scrollX / flyout.workspace_.scale -
645
+          flyout.workspace_.scrollX;
646
+    }
647
+    xyOld.y += flyout.workspace_.scrollY / flyout.workspace_.scale -
648
+        flyout.workspace_.scrollY;
649
+    var svgRootNew = block.getSvgRoot();
650
+    if (!svgRootNew) {
651
+      throw 'block is not rendered.';
652
+    }
653
+    var xyNew = Blockly.getSvgXY_(svgRootNew, workspace);
654
+    // Scale the scroll (getSvgXY_ did not do this).
655
+    xyNew.x += workspace.scrollX / workspace.scale - workspace.scrollX;
656
+    xyNew.y += workspace.scrollY / workspace.scale - workspace.scrollY;
657
+    if (workspace.toolbox_ && !workspace.scrollbar) {
658
+      xyNew.x += workspace.toolbox_.width / workspace.scale;
659
+    }
660
+    block.moveBy(xyOld.x - xyNew.x, xyOld.y - xyNew.y);
661
+    if (flyout.autoClose) {
662
+      flyout.hide();
663
+    } else {
664
+      flyout.filterForCapacity_();
665
+    }
666
+    // Start a dragging operation on the new block.
667
+    block.onMouseDown_(e);
668
+  };
669
+};
670
+
671
+/**
672
+ * Filter the blocks on the flyout to disable the ones that are above the
673
+ * capacity limit.
674
+ * @private
675
+ */
676
+Blockly.Flyout.prototype.filterForCapacity_ = function() {
677
+  var remainingCapacity = this.targetWorkspace_.remainingCapacity();
678
+  var blocks = this.workspace_.getTopBlocks(false);
679
+  for (var i = 0, block; block = blocks[i]; i++) {
680
+    var allBlocks = block.getDescendants();
681
+    if (allBlocks.length > remainingCapacity) {
682
+      block.setDisabled(true);
683
+    }
684
+  }
685
+};
686
+
687
+/**
688
+ * Return the deletion rectangle for this flyout.
689
+ * @return {goog.math.Rect} Rectangle in which to delete.
690
+ */
691
+Blockly.Flyout.prototype.getRect = function() {
692
+  // BIG_NUM is offscreen padding so that blocks dragged beyond the shown flyout
693
+  // area are still deleted.  Must be larger than the largest screen size,
694
+  // but be smaller than half Number.MAX_SAFE_INTEGER (not available on IE).
695
+  var BIG_NUM = 1000000000;
696
+  var mainWorkspace = Blockly.mainWorkspace;
697
+  var x = Blockly.getSvgXY_(this.svgGroup_, mainWorkspace).x;
698
+  if (!this.RTL) {
699
+    x -= BIG_NUM;
700
+  }
701
+  // Fix scale if nested in zoomed workspace.
702
+  var scale = this.targetWorkspace_ == mainWorkspace ? 1 : mainWorkspace.scale;
703
+    return new goog.math.Rect(x, -BIG_NUM,
704
+        BIG_NUM + this.width_ * scale, BIG_NUM * 2);
705
+};
706
+
707
+/**
708
+ * Stop binding to the global mouseup and mousemove events.
709
+ * @private
710
+ */
711
+Blockly.Flyout.terminateDrag_ = function() {
712
+  if (Blockly.Flyout.onMouseUpWrapper_) {
713
+    Blockly.unbindEvent_(Blockly.Flyout.onMouseUpWrapper_);
714
+    Blockly.Flyout.onMouseUpWrapper_ = null;
715
+  }
716
+  if (Blockly.Flyout.onMouseMoveBlockWrapper_) {
717
+    Blockly.unbindEvent_(Blockly.Flyout.onMouseMoveBlockWrapper_);
718
+    Blockly.Flyout.onMouseMoveBlockWrapper_ = null;
719
+  }
720
+  if (Blockly.Flyout.onMouseMoveWrapper_) {
721
+    Blockly.unbindEvent_(Blockly.Flyout.onMouseMoveWrapper_);
722
+    Blockly.Flyout.onMouseMoveWrapper_ = null;
723
+  }
724
+  if (Blockly.Flyout.onMouseUpWrapper_) {
725
+    Blockly.unbindEvent_(Blockly.Flyout.onMouseUpWrapper_);
726
+    Blockly.Flyout.onMouseUpWrapper_ = null;
727
+  }
728
+  Blockly.Flyout.startDownEvent_ = null;
729
+  Blockly.Flyout.startBlock_ = null;
730
+  Blockly.Flyout.startFlyout_ = null;
731
+};

+ 328 - 0
src/blockly/core/generator.js

@@ -0,0 +1,328 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Utility functions for generating executable code from
23
+ * Blockly code.
24
+ * @author fraser@google.com (Neil Fraser)
25
+ */
26
+'use strict';
27
+
28
+goog.provide('Blockly.Generator');
29
+
30
+goog.require('Blockly.Block');
31
+goog.require('goog.asserts');
32
+
33
+
34
+/**
35
+ * Class for a code generator that translates the blocks into a language.
36
+ * @param {string} name Language name of this generator.
37
+ * @constructor
38
+ */
39
+Blockly.Generator = function(name) {
40
+  this.name_ = name;
41
+  this.FUNCTION_NAME_PLACEHOLDER_REGEXP_ =
42
+      new RegExp(this.FUNCTION_NAME_PLACEHOLDER_, 'g');
43
+};
44
+
45
+/**
46
+ * Category to separate generated function names from variables and procedures.
47
+ */
48
+Blockly.Generator.NAME_TYPE = 'generated_function';
49
+
50
+/**
51
+ * Arbitrary code to inject into locations that risk causing infinite loops.
52
+ * Any instances of '%1' will be replaced by the block ID that failed.
53
+ * E.g. '  checkTimeout(%1);\n'
54
+ * @type {?string}
55
+ */
56
+Blockly.Generator.prototype.INFINITE_LOOP_TRAP = null;
57
+
58
+/**
59
+ * Arbitrary code to inject before every statement.
60
+ * Any instances of '%1' will be replaced by the block ID of the statement.
61
+ * E.g. 'highlight(%1);\n'
62
+ * @type {?string}
63
+ */
64
+Blockly.Generator.prototype.STATEMENT_PREFIX = null;
65
+
66
+/**
67
+ * Generate code for all blocks in the workspace to the specified language.
68
+ * @param {Blockly.Workspace} workspace Workspace to generate code from.
69
+ * @return {string} Generated code.
70
+ */
71
+Blockly.Generator.prototype.workspaceToCode = function(workspace) {
72
+  if (!workspace) {
73
+    // Backwards compatability from before there could be multiple workspaces.
74
+    console.warn('No workspace specified in workspaceToCode call.  Guessing.');
75
+    workspace = Blockly.getMainWorkspace();
76
+  }
77
+  var code = [];
78
+  this.init(workspace);
79
+  var blocks = workspace.getTopBlocks(true);
80
+  for (var x = 0, block; block = blocks[x]; x++) {
81
+    var line = this.blockToCode(block);
82
+    if (goog.isArray(line)) {
83
+      // Value blocks return tuples of code and operator order.
84
+      // Top-level blocks don't care about operator order.
85
+      line = line[0];
86
+    }
87
+    if (line) {
88
+      if (block.outputConnection && this.scrubNakedValue) {
89
+        // This block is a naked value.  Ask the language's code generator if
90
+        // it wants to append a semicolon, or something.
91
+        line = this.scrubNakedValue(line);
92
+      }
93
+      code.push(line);
94
+    }
95
+  }
96
+  code = code.join('\n');  // Blank line between each section.
97
+  code = this.finish(code);
98
+  // Final scrubbing of whitespace.
99
+  code = code.replace(/^\s+\n/, '');
100
+  code = code.replace(/\n\s+$/, '\n');
101
+  code = code.replace(/[ \t]+\n/g, '\n');
102
+  return code;
103
+};
104
+
105
+// The following are some helpful functions which can be used by multiple
106
+// languages.
107
+
108
+/**
109
+ * Prepend a common prefix onto each line of code.
110
+ * @param {string} text The lines of code.
111
+ * @param {string} prefix The common prefix.
112
+ * @return {string} The prefixed lines of code.
113
+ */
114
+Blockly.Generator.prototype.prefixLines = function(text, prefix) {
115
+  return prefix + text.replace(/\n(.)/g, '\n' + prefix + '$1');
116
+};
117
+
118
+/**
119
+ * Recursively spider a tree of blocks, returning all their comments.
120
+ * @param {!Blockly.Block} block The block from which to start spidering.
121
+ * @return {string} Concatenated list of comments.
122
+ */
123
+Blockly.Generator.prototype.allNestedComments = function(block) {
124
+  var comments = [];
125
+  var blocks = block.getDescendants();
126
+  for (var x = 0; x < blocks.length; x++) {
127
+    var comment = blocks[x].getCommentText();
128
+    if (comment) {
129
+      comments.push(comment);
130
+    }
131
+  }
132
+  // Append an empty string to create a trailing line break when joined.
133
+  if (comments.length) {
134
+    comments.push('');
135
+  }
136
+  return comments.join('\n');
137
+};
138
+
139
+/**
140
+ * Generate code for the specified block (and attached blocks).
141
+ * @param {Blockly.Block} block The block to generate code for.
142
+ * @return {string|!Array} For statement blocks, the generated code.
143
+ *     For value blocks, an array containing the generated code and an
144
+ *     operator order value.  Returns '' if block is null.
145
+ */
146
+Blockly.Generator.prototype.blockToCode = function(block) {
147
+  if (!block) {
148
+    return '';
149
+  }
150
+  if (block.disabled) {
151
+    // Skip past this block if it is disabled.
152
+    return this.blockToCode(block.getNextBlock());
153
+  }
154
+
155
+  var func = this[block.type];
156
+  goog.asserts.assertFunction(func,
157
+      'Language "%s" does not know how to generate code for block type "%s".',
158
+      this.name_, block.type);
159
+  // First argument to func.call is the value of 'this' in the generator.
160
+  // Prior to 24 September 2013 'this' was the only way to access the block.
161
+  // The current prefered method of accessing the block is through the second
162
+  // argument to func.call, which becomes the first parameter to the generator.
163
+  var code = func.call(block, block);
164
+  if (goog.isArray(code)) {
165
+    // Value blocks return tuples of code and operator order.
166
+    return [this.scrub_(block, code[0]), code[1]];
167
+  } else if (goog.isString(code)) {
168
+    if (this.STATEMENT_PREFIX) {
169
+      code = this.STATEMENT_PREFIX.replace(/%1/g, '\'' + block.id + '\'') +
170
+          code;
171
+    }
172
+    return this.scrub_(block, code);
173
+  } else if (code === null) {
174
+    // Block has handled code generation itself.
175
+    return '';
176
+  } else {
177
+    goog.asserts.fail('Invalid code generated: %s', code);
178
+  }
179
+};
180
+
181
+/**
182
+ * Generate code representing the specified value input.
183
+ * @param {!Blockly.Block} block The block containing the input.
184
+ * @param {string} name The name of the input.
185
+ * @param {number} order The maximum binding strength (minimum order value)
186
+ *     of any operators adjacent to "block".
187
+ * @return {string} Generated code or '' if no blocks are connected or the
188
+ *     specified input does not exist.
189
+ */
190
+Blockly.Generator.prototype.valueToCode = function(block, name, order) {
191
+  if (isNaN(order)) {
192
+    goog.asserts.fail('Expecting valid order from block "%s".', block.type);
193
+  }
194
+  var targetBlock = block.getInputTargetBlock(name);
195
+  if (!targetBlock) {
196
+    return '';
197
+  }
198
+  var tuple = this.blockToCode(targetBlock);
199
+  if (tuple === '') {
200
+    // Disabled block.
201
+    return '';
202
+  }
203
+  // Value blocks must return code and order of operations info.
204
+  // Statement blocks must only return code.
205
+  goog.asserts.assertArray(tuple,
206
+      'Expecting tuple from value block "%s".', targetBlock.type);
207
+  var code = tuple[0];
208
+  var innerOrder = tuple[1];
209
+  if (isNaN(innerOrder)) {
210
+    goog.asserts.fail('Expecting valid order from value block "%s".',
211
+        targetBlock.type);
212
+  }
213
+  if (code && order <= innerOrder) {
214
+    if (order == innerOrder && (order == 0 || order == 99)) {
215
+      // Don't generate parens around NONE-NONE and ATOMIC-ATOMIC pairs.
216
+      // 0 is the atomic order, 99 is the none order.  No parentheses needed.
217
+      // In all known languages multiple such code blocks are not order
218
+      // sensitive.  In fact in Python ('a' 'b') 'c' would fail.
219
+    } else {
220
+      // The operators outside this code are stonger than the operators
221
+      // inside this code.  To prevent the code from being pulled apart,
222
+      // wrap the code in parentheses.
223
+      // Technically, this should be handled on a language-by-language basis.
224
+      // However all known (sane) languages use parentheses for grouping.
225
+      code = '(' + code + ')';
226
+    }
227
+  }
228
+  return code;
229
+};
230
+
231
+/**
232
+ * Generate code representing the statement.  Indent the code.
233
+ * @param {!Blockly.Block} block The block containing the input.
234
+ * @param {string} name The name of the input.
235
+ * @return {string} Generated code or '' if no blocks are connected.
236
+ */
237
+Blockly.Generator.prototype.statementToCode = function(block, name) {
238
+  var targetBlock = block.getInputTargetBlock(name);
239
+  var code = this.blockToCode(targetBlock);
240
+  // Value blocks must return code and order of operations info.
241
+  // Statement blocks must only return code.
242
+  goog.asserts.assertString(code,
243
+      'Expecting code from statement block "%s".',
244
+      targetBlock && targetBlock.type);
245
+  if (code) {
246
+    code = this.prefixLines(/** @type {string} */ (code), this.INDENT);
247
+  }
248
+  return code;
249
+};
250
+
251
+/**
252
+ * Add an infinite loop trap to the contents of a loop.
253
+ * If loop is empty, add a statment prefix for the loop block.
254
+ * @param {string} branch Code for loop contents.
255
+ * @param {string} id ID of enclosing block.
256
+ * @return {string} Loop contents, with infinite loop trap added.
257
+ */
258
+Blockly.Generator.prototype.addLoopTrap = function(branch, id) {
259
+  if (this.INFINITE_LOOP_TRAP) {
260
+    branch = this.INFINITE_LOOP_TRAP.replace(/%1/g, '\'' + id + '\'') + branch;
261
+  }
262
+  if (this.STATEMENT_PREFIX) {
263
+    branch += this.prefixLines(this.STATEMENT_PREFIX.replace(/%1/g,
264
+        '\'' + id + '\''), this.INDENT);
265
+  }
266
+  return branch;
267
+};
268
+
269
+/**
270
+ * The method of indenting.  Defaults to two spaces, but language generators
271
+ * may override this to increase indent or change to tabs.
272
+ * @type {string}
273
+ */
274
+Blockly.Generator.prototype.INDENT = '  ';
275
+
276
+/**
277
+ * Comma-separated list of reserved words.
278
+ * @type {string}
279
+ * @private
280
+ */
281
+Blockly.Generator.prototype.RESERVED_WORDS_ = '';
282
+
283
+/**
284
+ * Add one or more words to the list of reserved words for this language.
285
+ * @param {string} words Comma-separated list of words to add to the list.
286
+ *     No spaces.  Duplicates are ok.
287
+ */
288
+Blockly.Generator.prototype.addReservedWords = function(words) {
289
+  this.RESERVED_WORDS_ += words + ',';
290
+};
291
+
292
+/**
293
+ * This is used as a placeholder in functions defined using
294
+ * Blockly.Generator.provideFunction_.  It must not be legal code that could
295
+ * legitimately appear in a function definition (or comment), and it must
296
+ * not confuse the regular expression parser.
297
+ * @type {string}
298
+ * @private
299
+ */
300
+Blockly.Generator.prototype.FUNCTION_NAME_PLACEHOLDER_ = '{leCUI8hutHZI4480Dc}';
301
+
302
+/**
303
+ * Define a function to be included in the generated code.
304
+ * The first time this is called with a given desiredName, the code is
305
+ * saved and an actual name is generated.  Subsequent calls with the
306
+ * same desiredName have no effect but have the same return value.
307
+ *
308
+ * It is up to the caller to make sure the same desiredName is not
309
+ * used for different code values.
310
+ *
311
+ * The code gets output when Blockly.Generator.finish() is called.
312
+ *
313
+ * @param {string} desiredName The desired name of the function (e.g., isPrime).
314
+ * @param {!Array.<string>} code A list of Python statements.
315
+ * @return {string} The actual name of the new function.  This may differ
316
+ *     from desiredName if the former has already been taken by the user.
317
+ * @private
318
+ */
319
+Blockly.Generator.prototype.provideFunction_ = function(desiredName, code) {
320
+  if (!this.definitions_[desiredName]) {
321
+    var functionName =
322
+        this.variableDB_.getDistinctName(desiredName, this.NAME_TYPE);
323
+    this.functionNames_[desiredName] = functionName;
324
+    this.definitions_[desiredName] = code.join('\n').replace(
325
+        this.FUNCTION_NAME_PLACEHOLDER_REGEXP_, functionName);
326
+  }
327
+  return this.functionNames_[desiredName];
328
+};

+ 222 - 0
src/blockly/core/icon.js

@@ -0,0 +1,222 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2013 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Object representing an icon on a block.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Icon');
28
+
29
+goog.require('goog.dom');
30
+
31
+
32
+/**
33
+ * Class for an icon.
34
+ * @param {Blockly.Block} block The block associated with this icon.
35
+ * @constructor
36
+ */
37
+Blockly.Icon = function(block) {
38
+  this.block_ = block;
39
+};
40
+
41
+/**
42
+ * Does this icon get hidden when the block is collapsed.
43
+ */
44
+Blockly.Icon.prototype.collapseHidden = true;
45
+
46
+/**
47
+ * Height and width of icons.
48
+ */
49
+Blockly.Icon.prototype.SIZE = 17;
50
+
51
+/**
52
+ * Icon in base64 format.
53
+ * @private
54
+ */
55
+Blockly.Icon.prototype.png_ = '';
56
+
57
+/**
58
+ * Bubble UI (if visible).
59
+ * @type {Blockly.Bubble}
60
+ * @private
61
+ */
62
+Blockly.Icon.prototype.bubble_ = null;
63
+
64
+/**
65
+ * Absolute X coordinate of icon's center.
66
+ * @private
67
+ */
68
+Blockly.Icon.prototype.iconX_ = 0;
69
+
70
+/**
71
+ * Absolute Y coordinate of icon's centre.
72
+ * @private
73
+ */
74
+Blockly.Icon.prototype.iconY_ = 0;
75
+
76
+/**
77
+ * Create the icon on the block.
78
+ */
79
+Blockly.Icon.prototype.createIcon = function() {
80
+  if (this.iconGroup_) {
81
+    // Icon already exists.
82
+    return;
83
+  }
84
+  /* Here's the markup that will be generated:
85
+  <g class="blocklyIconGroup">
86
+    <image width="17" height="17"
87
+     xlink:href="..."></image>
88
+  </g>
89
+  */
90
+  this.iconGroup_ = Blockly.createSvgElement('g',
91
+      {'class': 'blocklyIconGroup'}, null);
92
+  var img = Blockly.createSvgElement('image',
93
+      {'width': this.SIZE, 'height': this.SIZE},
94
+      this.iconGroup_);
95
+  img.setAttributeNS('http://www.w3.org/1999/xlink', 'xlink:href', this.png_);
96
+
97
+  this.block_.getSvgRoot().appendChild(this.iconGroup_);
98
+  Blockly.bindEvent_(this.iconGroup_, 'mouseup', this, this.iconClick_);
99
+  this.updateEditable();
100
+};
101
+
102
+/**
103
+ * Dispose of this icon.
104
+ */
105
+Blockly.Icon.prototype.dispose = function() {
106
+  // Dispose of and unlink the icon.
107
+  goog.dom.removeNode(this.iconGroup_);
108
+  this.iconGroup_ = null;
109
+  // Dispose of and unlink the bubble.
110
+  this.setVisible(false);
111
+  this.block_ = null;
112
+};
113
+
114
+/**
115
+ * Add or remove the UI indicating if this icon may be clicked or not.
116
+ */
117
+Blockly.Icon.prototype.updateEditable = function() {
118
+  if (this.block_.isInFlyout || !this.block_.isEditable()) {
119
+    Blockly.addClass_(/** @type {!Element} */ (this.iconGroup_),
120
+                      'blocklyIconGroupReadonly');
121
+  } else {
122
+    Blockly.removeClass_(/** @type {!Element} */ (this.iconGroup_),
123
+                         'blocklyIconGroupReadonly');
124
+  }
125
+};
126
+
127
+/**
128
+ * Is the associated bubble visible?
129
+ * @return {boolean} True if the bubble is visible.
130
+ */
131
+Blockly.Icon.prototype.isVisible = function() {
132
+  return !!this.bubble_;
133
+};
134
+
135
+/**
136
+ * Clicking on the icon toggles if the bubble is visible.
137
+ * @param {!Event} e Mouse click event.
138
+ * @private
139
+ */
140
+Blockly.Icon.prototype.iconClick_ = function(e) {
141
+  if (Blockly.dragMode_ == 2) {
142
+    // Drag operation is concluding.  Don't open the editor.
143
+    return;
144
+  }
145
+  if (!this.block_.isInFlyout && !Blockly.isRightButton(e)) {
146
+    this.setVisible(!this.isVisible());
147
+  }
148
+};
149
+
150
+/**
151
+ * Change the colour of the associated bubble to match its block.
152
+ */
153
+Blockly.Icon.prototype.updateColour = function() {
154
+  if (this.isVisible()) {
155
+    var hexColour = Blockly.makeColour(this.block_.getColour());
156
+    this.bubble_.setColour(hexColour);
157
+  }
158
+};
159
+
160
+/**
161
+ * Render the icon.
162
+ * @param {number} cursorX Horizontal offset at which to position the icon.
163
+ * @return {number} Horizontal offset for next item to draw.
164
+ */
165
+Blockly.Icon.prototype.renderIcon = function(cursorX) {
166
+  if (this.collapseHidden && this.block_.isCollapsed()) {
167
+    this.iconGroup_.setAttribute('display', 'none');
168
+    return cursorX;
169
+  }
170
+  this.iconGroup_.setAttribute('display', 'block');
171
+
172
+  var TOP_MARGIN = 5;
173
+  var width = this.SIZE;
174
+  if (this.block_.RTL) {
175
+    cursorX -= width;
176
+  }
177
+  this.iconGroup_.setAttribute('transform',
178
+      'translate(' + cursorX + ',' + TOP_MARGIN + ')');
179
+  this.computeIconLocation();
180
+  if (this.block_.RTL) {
181
+    cursorX -= Blockly.BlockSvg.SEP_SPACE_X;
182
+  } else {
183
+    cursorX += width + Blockly.BlockSvg.SEP_SPACE_X;
184
+  }
185
+  return cursorX;
186
+};
187
+
188
+/**
189
+ * Notification that the icon has moved.  Update the arrow accordingly.
190
+ * @param {number} x Absolute horizontal location.
191
+ * @param {number} y Absolute vertical location.
192
+ */
193
+Blockly.Icon.prototype.setIconLocation = function(x, y) {
194
+  this.iconX_ = x;
195
+  this.iconY_ = y;
196
+  if (this.isVisible()) {
197
+    this.bubble_.setAnchorLocation(x, y);
198
+  }
199
+};
200
+
201
+/**
202
+ * Notification that the icon has moved, but we don't really know where.
203
+ * Recompute the icon's location from scratch.
204
+ */
205
+Blockly.Icon.prototype.computeIconLocation = function() {
206
+  // Find coordinates for the centre of the icon and update the arrow.
207
+  var blockXY = this.block_.getRelativeToSurfaceXY();
208
+  var iconXY = Blockly.getRelativeXY_(this.iconGroup_);
209
+  var newX = blockXY.x + iconXY.x + this.SIZE / 2;
210
+  var newY = blockXY.y + iconXY.y + this.SIZE / 2;
211
+  if (newX !== this.iconX_ || newY !== this.iconY_) {
212
+    this.setIconLocation(newX, newY);
213
+  }
214
+};
215
+
216
+/**
217
+ * Returns the center of the block's icon relative to the surface.
218
+ * @return {!Object} Object with x and y properties.
219
+ */
220
+Blockly.Icon.prototype.getIconLocation = function() {
221
+  return {x: this.iconX_, y: this.iconY_};
222
+};

+ 542 - 0
src/blockly/core/inject.js

@@ -0,0 +1,542 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2011 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Functions for injecting Blockly into a web page.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.inject');
28
+
29
+goog.require('Blockly.Css');
30
+goog.require('Blockly.WorkspaceSvg');
31
+goog.require('goog.dom');
32
+goog.require('goog.ui.Component');
33
+goog.require('goog.userAgent');
34
+
35
+
36
+/**
37
+ * Inject a Blockly editor into the specified container element (usually a div).
38
+ * @param {!Element|string} container Containing element or its ID.
39
+ * @param {Object=} opt_options Optional dictionary of options.
40
+ * @return {!Blockly.Workspace} Newly created main workspace.
41
+ */
42
+Blockly.inject = function(container, opt_options) {
43
+  if (goog.isString(container)) {
44
+    container = document.getElementById(container);
45
+  }
46
+  // Verify that the container is in document.
47
+  if (!goog.dom.contains(document, container)) {
48
+    throw 'Error: container is not in current document.';
49
+  }
50
+  var options = Blockly.parseOptions_(opt_options || {});
51
+  var workspace;
52
+  var startUi = function() {
53
+    var svg = Blockly.createDom_(container, options);
54
+    workspace = Blockly.createMainWorkspace_(svg, options);
55
+    Blockly.init_(workspace);
56
+    workspace.markFocused();
57
+    Blockly.bindEvent_(svg, 'focus', workspace, workspace.markFocused);
58
+  };
59
+  if (options.enableRealtime) {
60
+    var realtimeElement = document.getElementById('realtime');
61
+    if (realtimeElement) {
62
+      realtimeElement.style.display = 'block';
63
+    }
64
+    Blockly.Realtime.startRealtime(startUi, container, options.realtimeOptions);
65
+  } else {
66
+    startUi();
67
+  }
68
+  return workspace;
69
+};
70
+
71
+/**
72
+ * Parse the provided toolbox tree into a consistent DOM format.
73
+ * @param {Node|string} tree DOM tree of blocks, or text representation of same.
74
+ * @return {Node} DOM tree of blocks, or null.
75
+ * @private
76
+ */
77
+Blockly.parseToolboxTree_ = function(tree) {
78
+  if (tree) {
79
+    if (typeof tree != 'string' && typeof XSLTProcessor == 'undefined') {
80
+      // In this case the tree will not have been properly built by the
81
+      // browser. The HTML will be contained in the element, but it will
82
+      // not have the proper DOM structure since the browser doesn't support
83
+      // XSLTProcessor (XML -> HTML). This is the case in IE 9+.
84
+      tree = tree.outerHTML;
85
+    }
86
+    if (typeof tree == 'string') {
87
+      tree = Blockly.Xml.textToDom(tree);
88
+    }
89
+  } else {
90
+    tree = null;
91
+  }
92
+  return tree;
93
+};
94
+
95
+/**
96
+ * Configure Blockly to behave according to a set of options.
97
+ * @param {!Object} options Dictionary of options.  Specification:
98
+ *   https://developers.google.com/blockly/installation/overview#configuration
99
+ * @return {!Object} Dictionary of normalized options.
100
+ * @private
101
+ */
102
+Blockly.parseOptions_ = function(options) {
103
+  var readOnly = !!options['readOnly'];
104
+  if (readOnly) {
105
+    var languageTree = null;
106
+    var hasCategories = false;
107
+    var hasTrashcan = false;
108
+    var hasCollapse = false;
109
+    var hasComments = false;
110
+    var hasDisable = false;
111
+    var hasSounds = false;
112
+  } else {
113
+    var languageTree = Blockly.parseToolboxTree_(options['toolbox']);
114
+    var hasCategories = Boolean(languageTree &&
115
+        languageTree.getElementsByTagName('category').length);
116
+    var hasTrashcan = options['trashcan'];
117
+    if (hasTrashcan === undefined) {
118
+      hasTrashcan = hasCategories;
119
+    }
120
+    var hasCollapse = options['collapse'];
121
+    if (hasCollapse === undefined) {
122
+      hasCollapse = hasCategories;
123
+    }
124
+    var hasComments = options['comments'];
125
+    if (hasComments === undefined) {
126
+      hasComments = hasCategories;
127
+    }
128
+    var hasDisable = options['disable'];
129
+    if (hasDisable === undefined) {
130
+      hasDisable = hasCategories;
131
+    }
132
+    var hasSounds = options['sounds'];
133
+    if (hasSounds === undefined) {
134
+      hasSounds = true;
135
+    }
136
+  }
137
+  var hasScrollbars = options['scrollbars'];
138
+  if (hasScrollbars === undefined) {
139
+    hasScrollbars = hasCategories;
140
+  }
141
+  var hasCss = options['css'];
142
+  if (hasCss === undefined) {
143
+    hasCss = true;
144
+  }
145
+  // See grid documentation at:
146
+  // https://developers.google.com/blockly/installation/grid
147
+  var grid = options['grid'] || {};
148
+  var gridOptions = {};
149
+  gridOptions.spacing = parseFloat(grid['spacing']) || 0;
150
+  gridOptions.colour = grid['colour'] || '#888';
151
+  gridOptions.length = parseFloat(grid['length']) || 1;
152
+  gridOptions.snap = gridOptions.spacing > 0 && !!grid['snap'];
153
+  var pathToMedia = 'https://blockly-demo.appspot.com/static/media/';
154
+  if (options['media']) {
155
+    pathToMedia = options['media'];
156
+  } else if (options['path']) {
157
+    // 'path' is a deprecated option which has been replaced by 'media'.
158
+    pathToMedia = options['path'] + 'media/';
159
+  }
160
+
161
+/* TODO (fraser): Add documentation page:
162
+ * https://developers.google.com/blockly/installation/zoom
163
+ *
164
+ * controls
165
+ *
166
+ * Set to `true` to show zoom-in and zoom-out buttons.  Defaults to `false`.
167
+ *
168
+ * wheel
169
+ *
170
+ * Set to `true` to allow the mouse wheel to zoom.  Defaults to `false`.
171
+ *
172
+ * startScale
173
+ *
174
+ * Initial magnification factor.  Defaults to `1.0`.
175
+ *
176
+ * maxScale
177
+ *
178
+ * Maximum multiplication factor for how far one can zoom in.  Defaults to `3`.
179
+ *
180
+ * minScale
181
+ *
182
+ * Minimum multiplication factor for how far one can zoom out.  Defaults to `0.3`.
183
+ *
184
+ * scaleSpeed
185
+ *
186
+ * For each zooming in-out step the scale is multiplied
187
+ * or divided respectively by the scale speed, this means that:
188
+ * `scale = scaleSpeed ^ steps`, note that in this formula
189
+ * steps of zoom-out are subtracted and zoom-in steps are added.
190
+ */
191
+  // See zoom documentation at:
192
+  // https://developers.google.com/blockly/installation/zoom
193
+  var zoom = options['zoom'] || {};
194
+  var zoomOptions = {};
195
+  if (zoom['controls'] === undefined) {
196
+    zoomOptions.controls = false;
197
+  } else {
198
+    zoomOptions.controls = !!zoom['controls'];
199
+  }
200
+  if (zoom['wheel'] === undefined) {
201
+    zoomOptions.wheel = false;
202
+  } else {
203
+    zoomOptions.wheel = !!zoom['wheel'];
204
+  }
205
+  if (zoom['startScale'] === undefined) {
206
+    zoomOptions.startScale = 1;
207
+  } else {
208
+    zoomOptions.startScale = parseFloat(zoom['startScale']);
209
+  }
210
+  if (zoom['maxScale'] === undefined) {
211
+    zoomOptions.maxScale = 3;
212
+  } else {
213
+    zoomOptions.maxScale = parseFloat(zoom['maxScale']);
214
+  }
215
+  if (zoom['minScale'] === undefined) {
216
+    zoomOptions.minScale = 0.3;
217
+  } else {
218
+    zoomOptions.minScale = parseFloat(zoom['minScale']);
219
+  }
220
+  if (zoom['scaleSpeed'] === undefined) {
221
+    zoomOptions.scaleSpeed = 1.2;
222
+  } else {
223
+    zoomOptions.scaleSpeed = parseFloat(zoom['scaleSpeed']);
224
+  }
225
+
226
+  var enableRealtime = !!options['realtime'];
227
+  var realtimeOptions = enableRealtime ? options['realtimeOptions'] : undefined;
228
+
229
+  return {
230
+    RTL: !!options['rtl'],
231
+    collapse: hasCollapse,
232
+    comments: hasComments,
233
+    disable: hasDisable,
234
+    readOnly: readOnly,
235
+    maxBlocks: options['maxBlocks'] || Infinity,
236
+    pathToMedia: pathToMedia,
237
+    hasCategories: hasCategories,
238
+    hasScrollbars: hasScrollbars,
239
+    hasTrashcan: hasTrashcan,
240
+    hasSounds: hasSounds,
241
+    hasCss: hasCss,
242
+    languageTree: languageTree,
243
+    gridOptions: gridOptions,
244
+    zoomOptions: zoomOptions,
245
+    enableRealtime: enableRealtime,
246
+    realtimeOptions: realtimeOptions
247
+  };
248
+};
249
+
250
+/**
251
+ * Create the SVG image.
252
+ * @param {!Element} container Containing element.
253
+ * @param {Object} options Dictionary of options.
254
+ * @return {!Element} Newly created SVG image.
255
+ * @private
256
+ */
257
+Blockly.createDom_ = function(container, options) {
258
+  // Sadly browsers (Chrome vs Firefox) are currently inconsistent in laying
259
+  // out content in RTL mode.  Therefore Blockly forces the use of LTR,
260
+  // then manually positions content in RTL as needed.
261
+  container.setAttribute('dir', 'LTR');
262
+  // Closure can be trusted to create HTML widgets with the proper direction.
263
+  goog.ui.Component.setDefaultRightToLeft(options.RTL);
264
+
265
+  // Load CSS.
266
+  Blockly.Css.inject(options.hasCss, options.pathToMedia);
267
+
268
+  // Build the SVG DOM.
269
+  /*
270
+  <svg
271
+    xmlns="http://www.w3.org/2000/svg"
272
+    xmlns:html="http://www.w3.org/1999/xhtml"
273
+    xmlns:xlink="http://www.w3.org/1999/xlink"
274
+    version="1.1"
275
+    class="blocklySvg">
276
+    ...
277
+  </svg>
278
+  */
279
+  var svg = Blockly.createSvgElement('svg', {
280
+    'xmlns': 'http://www.w3.org/2000/svg',
281
+    'xmlns:html': 'http://www.w3.org/1999/xhtml',
282
+    'xmlns:xlink': 'http://www.w3.org/1999/xlink',
283
+    'version': '1.1',
284
+    'class': 'blocklySvg'
285
+  }, container);
286
+  /*
287
+  <defs>
288
+    ... filters go here ...
289
+  </defs>
290
+  */
291
+  var defs = Blockly.createSvgElement('defs', {}, svg);
292
+  var rnd = String(Math.random()).substring(2);
293
+  /*
294
+    <filter id="blocklyEmbossFilter837493">
295
+      <feGaussianBlur in="SourceAlpha" stdDeviation="1" result="blur"/>
296
+      <feSpecularLighting in="blur" surfaceScale="1" specularConstant="0.5"
297
+                          specularExponent="10" lighting-color="white"
298
+                          result="specOut">
299
+        <fePointLight x="-5000" y="-10000" z="20000"/>
300
+      </feSpecularLighting>
301
+      <feComposite in="specOut" in2="SourceAlpha" operator="in"
302
+                   result="specOut"/>
303
+      <feComposite in="SourceGraphic" in2="specOut" operator="arithmetic"
304
+                   k1="0" k2="1" k3="1" k4="0"/>
305
+    </filter>
306
+  */
307
+  var embossFilter = Blockly.createSvgElement('filter',
308
+      {'id': 'blocklyEmbossFilter' + rnd}, defs);
309
+  Blockly.createSvgElement('feGaussianBlur',
310
+      {'in': 'SourceAlpha', 'stdDeviation': 1, 'result': 'blur'}, embossFilter);
311
+  var feSpecularLighting = Blockly.createSvgElement('feSpecularLighting',
312
+      {'in': 'blur', 'surfaceScale': 1, 'specularConstant': 0.5,
313
+       'specularExponent': 10, 'lighting-color': 'white', 'result': 'specOut'},
314
+      embossFilter);
315
+  Blockly.createSvgElement('fePointLight',
316
+      {'x': -5000, 'y': -10000, 'z': 20000}, feSpecularLighting);
317
+  Blockly.createSvgElement('feComposite',
318
+      {'in': 'specOut', 'in2': 'SourceAlpha', 'operator': 'in',
319
+       'result': 'specOut'}, embossFilter);
320
+  Blockly.createSvgElement('feComposite',
321
+      {'in': 'SourceGraphic', 'in2': 'specOut', 'operator': 'arithmetic',
322
+       'k1': 0, 'k2': 1, 'k3': 1, 'k4': 0}, embossFilter);
323
+  options.embossFilterId = embossFilter.id;
324
+  /*
325
+    <pattern id="blocklyDisabledPattern837493" patternUnits="userSpaceOnUse"
326
+             width="10" height="10">
327
+      <rect width="10" height="10" fill="#aaa" />
328
+      <path d="M 0 0 L 10 10 M 10 0 L 0 10" stroke="#cc0" />
329
+    </pattern>
330
+  */
331
+  var disabledPattern = Blockly.createSvgElement('pattern',
332
+      {'id': 'blocklyDisabledPattern' + rnd,
333
+       'patternUnits': 'userSpaceOnUse',
334
+       'width': 10, 'height': 10}, defs);
335
+  Blockly.createSvgElement('rect',
336
+      {'width': 10, 'height': 10, 'fill': '#aaa'}, disabledPattern);
337
+  Blockly.createSvgElement('path',
338
+      {'d': 'M 0 0 L 10 10 M 10 0 L 0 10', 'stroke': '#cc0'}, disabledPattern);
339
+  options.disabledPatternId = disabledPattern.id;
340
+  /*
341
+    <pattern id="blocklyGridPattern837493" patternUnits="userSpaceOnUse">
342
+      <rect stroke="#888" />
343
+      <rect stroke="#888" />
344
+    </pattern>
345
+  */
346
+  var gridPattern = Blockly.createSvgElement('pattern',
347
+      {'id': 'blocklyGridPattern' + rnd,
348
+       'patternUnits': 'userSpaceOnUse'}, defs);
349
+  if (options.gridOptions['length'] > 0 && options.gridOptions['spacing'] > 0) {
350
+    Blockly.createSvgElement('line',
351
+        {'stroke': options.gridOptions['colour']},
352
+        gridPattern);
353
+    if (options.gridOptions['length'] > 1) {
354
+      Blockly.createSvgElement('line',
355
+          {'stroke': options.gridOptions['colour']},
356
+          gridPattern);
357
+    }
358
+    // x1, y1, x1, x2 properties will be set later in updateGridPattern_.
359
+  }
360
+  options.gridPattern = gridPattern;
361
+  options.svg = svg;
362
+  return svg;
363
+};
364
+
365
+/**
366
+ * Create a main workspace and add it to the SVG.
367
+ * @param {!Element} svg SVG element with pattern defined.
368
+ * @param {Object} options Dictionary of options.
369
+ * @return {!Blockly.Workspace} Newly created main workspace.
370
+ * @private
371
+ */
372
+Blockly.createMainWorkspace_ = function(svg, options) {
373
+  options.parentWorkspace = null;
374
+  options.getMetrics = Blockly.getMainWorkspaceMetrics_;
375
+  options.setMetrics = Blockly.setMainWorkspaceMetrics_;
376
+  var mainWorkspace = new Blockly.WorkspaceSvg(options);
377
+  mainWorkspace.scale = options.zoomOptions.startScale;
378
+  svg.appendChild(mainWorkspace.createDom('blocklyMainBackground'));
379
+  // A null translation will also apply the correct initial scale.
380
+  mainWorkspace.translate(0, 0);
381
+  mainWorkspace.markFocused();
382
+
383
+  if (!options.readOnly && !options.hasScrollbars) {
384
+    var workspaceChanged = function() {
385
+      if (Blockly.dragMode_ == 0) {
386
+        var metrics = mainWorkspace.getMetrics();
387
+        var edgeLeft = metrics.viewLeft + metrics.absoluteLeft;
388
+        var edgeTop = metrics.viewTop + metrics.absoluteTop;
389
+        if (metrics.contentTop < edgeTop ||
390
+            metrics.contentTop + metrics.contentHeight >
391
+            metrics.viewHeight + edgeTop ||
392
+            metrics.contentLeft <
393
+                (options.RTL ? metrics.viewLeft : edgeLeft) ||
394
+            metrics.contentLeft + metrics.contentWidth > (options.RTL ?
395
+                metrics.viewWidth : metrics.viewWidth + edgeLeft)) {
396
+          // One or more blocks may be out of bounds.  Bump them back in.
397
+          var MARGIN = 25;
398
+          var blocks = mainWorkspace.getTopBlocks(false);
399
+          for (var b = 0, block; block = blocks[b]; b++) {
400
+            var blockXY = block.getRelativeToSurfaceXY();
401
+            var blockHW = block.getHeightWidth();
402
+            // Bump any block that's above the top back inside.
403
+            var overflow = edgeTop + MARGIN - blockHW.height - blockXY.y;
404
+            if (overflow > 0) {
405
+              block.moveBy(0, overflow);
406
+            }
407
+            // Bump any block that's below the bottom back inside.
408
+            var overflow = edgeTop + metrics.viewHeight - MARGIN - blockXY.y;
409
+            if (overflow < 0) {
410
+              block.moveBy(0, overflow);
411
+            }
412
+            // Bump any block that's off the left back inside.
413
+            var overflow = MARGIN + edgeLeft -
414
+                blockXY.x - (options.RTL ? 0 : blockHW.width);
415
+            if (overflow > 0) {
416
+              block.moveBy(overflow, 0);
417
+            }
418
+            // Bump any block that's off the right back inside.
419
+            var overflow = edgeLeft + metrics.viewWidth - MARGIN -
420
+                blockXY.x + (options.RTL ? blockHW.width : 0);
421
+            if (overflow < 0) {
422
+              block.moveBy(overflow, 0);
423
+            }
424
+          }
425
+        }
426
+      }
427
+    };
428
+    mainWorkspace.addChangeListener(workspaceChanged);
429
+  }
430
+  // The SVG is now fully assembled.
431
+  Blockly.svgResize(mainWorkspace);
432
+  Blockly.WidgetDiv.createDom();
433
+  Blockly.Tooltip.createDom();
434
+  return mainWorkspace;
435
+};
436
+
437
+/**
438
+ * Initialize Blockly with various handlers.
439
+ * @param {!Blockly.Workspace} mainWorkspace Newly created main workspace.
440
+ * @private
441
+ */
442
+Blockly.init_ = function(mainWorkspace) {
443
+  var options = mainWorkspace.options;
444
+  var svg = mainWorkspace.options.svg;
445
+  // Supress the browser's context menu.
446
+  Blockly.bindEvent_(svg, 'contextmenu', null,
447
+      function(e) {
448
+        if (!Blockly.isTargetInput_(e)) {
449
+          e.preventDefault();
450
+        }
451
+      });
452
+  // Bind events for scrolling the workspace.
453
+  // Most of these events should be bound to the SVG's surface.
454
+  // However, 'mouseup' has to be on the whole document so that a block dragged
455
+  // out of bounds and released will know that it has been released.
456
+  // Also, 'keydown' has to be on the whole document since the browser doesn't
457
+  // understand a concept of focus on the SVG image.
458
+
459
+  Blockly.bindEvent_(window, 'resize', null,
460
+                     function() {Blockly.svgResize(mainWorkspace);});
461
+
462
+  if (!Blockly.documentEventsBound_) {
463
+    // Only bind the window/document events once.
464
+    // Destroying and reinjecting Blockly should not bind again.
465
+    Blockly.bindEvent_(document, 'keydown', null, Blockly.onKeyDown_);
466
+    Blockly.bindEvent_(document, 'touchend', null, Blockly.longStop_);
467
+    Blockly.bindEvent_(document, 'touchcancel', null, Blockly.longStop_);
468
+    // Don't use bindEvent_ for document's mouseup since that would create a
469
+    // corresponding touch handler that would squeltch the ability to interact
470
+    // with non-Blockly elements.
471
+    document.addEventListener('mouseup', Blockly.onMouseUp_, false);
472
+    // Some iPad versions don't fire resize after portrait to landscape change.
473
+    if (goog.userAgent.IPAD) {
474
+      Blockly.bindEvent_(window, 'orientationchange', document, function() {
475
+        Blockly.fireUiEvent(window, 'resize');
476
+      });
477
+    }
478
+    Blockly.documentEventsBound_ = true;
479
+  }
480
+
481
+  if (options.languageTree) {
482
+    if (mainWorkspace.toolbox_) {
483
+      mainWorkspace.toolbox_.init(mainWorkspace);
484
+    } else if (mainWorkspace.flyout_) {
485
+      // Build a fixed flyout with the root blocks.
486
+      mainWorkspace.flyout_.init(mainWorkspace);
487
+      mainWorkspace.flyout_.show(options.languageTree.childNodes);
488
+      // Translate the workspace sideways to avoid the fixed flyout.
489
+      mainWorkspace.scrollX = mainWorkspace.flyout_.width_;
490
+      if (options.RTL) {
491
+        mainWorkspace.scrollX *= -1;
492
+      }
493
+      var translation = 'translate(' + mainWorkspace.scrollX + ',0)';
494
+      mainWorkspace.getCanvas().setAttribute('transform', translation);
495
+      mainWorkspace.getBubbleCanvas().setAttribute('transform', translation);
496
+    }
497
+  }
498
+  if (options.hasScrollbars) {
499
+    mainWorkspace.scrollbar = new Blockly.ScrollbarPair(mainWorkspace);
500
+    mainWorkspace.scrollbar.resize();
501
+  }
502
+
503
+  // Load the sounds.
504
+  if (options.hasSounds) {
505
+    mainWorkspace.loadAudio_(
506
+        [options.pathToMedia + 'click.mp3',
507
+         options.pathToMedia + 'click.wav',
508
+         options.pathToMedia + 'click.ogg'], 'click');
509
+    mainWorkspace.loadAudio_(
510
+        [options.pathToMedia + 'disconnect.wav',
511
+         options.pathToMedia + 'disconnect.mp3',
512
+         options.pathToMedia + 'disconnect.ogg'], 'disconnect');
513
+    mainWorkspace.loadAudio_(
514
+        [options.pathToMedia + 'delete.mp3',
515
+         options.pathToMedia + 'delete.ogg',
516
+         options.pathToMedia + 'delete.wav'], 'delete');
517
+
518
+    // Bind temporary hooks that preload the sounds.
519
+    var soundBinds = [];
520
+    var unbindSounds = function() {
521
+      while (soundBinds.length) {
522
+        Blockly.unbindEvent_(soundBinds.pop());
523
+      }
524
+      mainWorkspace.preloadAudio_();
525
+    };
526
+    // Android ignores any sound not loaded as a result of a user action.
527
+    soundBinds.push(
528
+        Blockly.bindEvent_(document, 'mousemove', null, unbindSounds));
529
+    soundBinds.push(
530
+        Blockly.bindEvent_(document, 'touchstart', null, unbindSounds));
531
+  }
532
+};
533
+
534
+/**
535
+ * Modify the block tree on the existing toolbox.
536
+ * @param {Node|string} tree DOM tree of blocks, or text representation of same.
537
+ */
538
+Blockly.updateToolbox = function(tree) {
539
+  console.warn('Deprecated call to Blockly.updateToolbox, ' +
540
+               'use workspace.updateToolbox instead.');
541
+  Blockly.getMainWorkspace().updateToolbox(tree);
542
+};

+ 239 - 0
src/blockly/core/input.js

@@ -0,0 +1,239 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Object representing an input (value, statement, or dummy).
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Input');
28
+
29
+// TODO(scr): Fix circular dependencies
30
+// goog.require('Blockly.Block');
31
+goog.require('Blockly.Connection');
32
+goog.require('Blockly.FieldLabel');
33
+goog.require('goog.asserts');
34
+
35
+
36
+/**
37
+ * Class for an input with an optional field.
38
+ * @param {number} type The type of the input.
39
+ * @param {string} name Language-neutral identifier which may used to find this
40
+ *     input again.
41
+ * @param {!Blockly.Block} block The block containing this input.
42
+ * @param {Blockly.Connection} connection Optional connection for this input.
43
+ * @constructor
44
+ */
45
+Blockly.Input = function(type, name, block, connection) {
46
+  /** @type {number} */
47
+  this.type = type;
48
+  /** @type {string} */
49
+  this.name = name;
50
+  /** @type {!Blockly.Block} */
51
+  this.sourceBlock_ = block;
52
+  /** @type {Blockly.Connection} */
53
+  this.connection = connection;
54
+  /** @type {!Array.<!Blockly.Field>} */
55
+  this.fieldRow = [];
56
+};
57
+
58
+/**
59
+ * Alignment of input's fields (left, right or centre).
60
+ * @type {number}
61
+ */
62
+Blockly.Input.prototype.align = Blockly.ALIGN_LEFT;
63
+
64
+/**
65
+ * Is the input visible?
66
+ * @type {boolean}
67
+ * @private
68
+ */
69
+Blockly.Input.prototype.visible_ = true;
70
+
71
+/**
72
+ * Add an item to the end of the input's field row.
73
+ * @param {string|!Blockly.Field} field Something to add as a field.
74
+ * @param {string=} opt_name Language-neutral identifier which may used to find
75
+ *     this field again.  Should be unique to the host block.
76
+ * @return {!Blockly.Input} The input being append to (to allow chaining).
77
+ */
78
+Blockly.Input.prototype.appendField = function(field, opt_name) {
79
+  // Empty string, Null or undefined generates no field, unless field is named.
80
+  if (!field && !opt_name) {
81
+    return this;
82
+  }
83
+  // Generate a FieldLabel when given a plain text field.
84
+  if (goog.isString(field)) {
85
+    field = new Blockly.FieldLabel(/** @type {string} */ (field));
86
+  }
87
+  if (this.sourceBlock_.rendered) {
88
+    field.init(this.sourceBlock_);
89
+  }
90
+  field.name = opt_name;
91
+
92
+  if (field.prefixField) {
93
+    // Add any prefix.
94
+    this.appendField(field.prefixField);
95
+  }
96
+  // Add the field to the field row.
97
+  this.fieldRow.push(field);
98
+  if (field.suffixField) {
99
+    // Add any suffix.
100
+    this.appendField(field.suffixField);
101
+  }
102
+
103
+  if (this.sourceBlock_.rendered) {
104
+    this.sourceBlock_.render();
105
+    // Adding a field will cause the block to change shape.
106
+    this.sourceBlock_.bumpNeighbours_();
107
+  }
108
+  return this;
109
+};
110
+
111
+/**
112
+ * Add an item to the end of the input's field row.
113
+ * @param {*} field Something to add as a field.
114
+ * @param {string=} opt_name Language-neutral identifier which may used to find
115
+ *     this field again.  Should be unique to the host block.
116
+ * @return {!Blockly.Input} The input being append to (to allow chaining).
117
+ * @deprecated December 2013
118
+ */
119
+Blockly.Input.prototype.appendTitle = function(field, opt_name) {
120
+  console.warn('Deprecated call to appendTitle, use appendField instead.');
121
+  return this.appendField(field, opt_name);
122
+};
123
+
124
+/**
125
+ * Remove a field from this input.
126
+ * @param {string} name The name of the field.
127
+ * @throws {goog.asserts.AssertionError} if the field is not present.
128
+ */
129
+Blockly.Input.prototype.removeField = function(name) {
130
+  for (var i = 0, field; field = this.fieldRow[i]; i++) {
131
+    if (field.name === name) {
132
+      field.dispose();
133
+      this.fieldRow.splice(i, 1);
134
+      if (this.sourceBlock_.rendered) {
135
+        this.sourceBlock_.render();
136
+        // Removing a field will cause the block to change shape.
137
+        this.sourceBlock_.bumpNeighbours_();
138
+      }
139
+      return;
140
+    }
141
+  }
142
+  goog.asserts.fail('Field "%s" not found.', name);
143
+};
144
+
145
+/**
146
+ * Gets whether this input is visible or not.
147
+ * @return {boolean} True if visible.
148
+ */
149
+Blockly.Input.prototype.isVisible = function() {
150
+  return this.visible_;
151
+};
152
+
153
+/**
154
+ * Sets whether this input is visible or not.
155
+ * Used to collapse/uncollapse a block.
156
+ * @param {boolean} visible True if visible.
157
+ * @return {!Array.<!Blockly.Block>} List of blocks to render.
158
+ */
159
+Blockly.Input.prototype.setVisible = function(visible) {
160
+  var renderList = [];
161
+  if (this.visible_ == visible) {
162
+    return renderList;
163
+  }
164
+  this.visible_ = visible;
165
+
166
+  var display = visible ? 'block' : 'none';
167
+  for (var y = 0, field; field = this.fieldRow[y]; y++) {
168
+    field.setVisible(visible);
169
+  }
170
+  if (this.connection) {
171
+    // Has a connection.
172
+    if (visible) {
173
+      renderList = this.connection.unhideAll();
174
+    } else {
175
+      this.connection.hideAll();
176
+    }
177
+    var child = this.connection.targetBlock();
178
+    if (child) {
179
+      child.getSvgRoot().style.display = display;
180
+      if (!visible) {
181
+        child.rendered = false;
182
+      }
183
+    }
184
+  }
185
+  return renderList;
186
+};
187
+
188
+/**
189
+ * Change a connection's compatibility.
190
+ * @param {string|Array.<string>|null} check Compatible value type or
191
+ *     list of value types.  Null if all types are compatible.
192
+ * @return {!Blockly.Input} The input being modified (to allow chaining).
193
+ */
194
+Blockly.Input.prototype.setCheck = function(check) {
195
+  if (!this.connection) {
196
+    throw 'This input does not have a connection.';
197
+  }
198
+  this.connection.setCheck(check);
199
+  return this;
200
+};
201
+
202
+/**
203
+ * Change the alignment of the connection's field(s).
204
+ * @param {number} align One of Blockly.ALIGN_LEFT, ALIGN_CENTRE, ALIGN_RIGHT.
205
+ *   In RTL mode directions are reversed, and ALIGN_RIGHT aligns to the left.
206
+ * @return {!Blockly.Input} The input being modified (to allow chaining).
207
+ */
208
+Blockly.Input.prototype.setAlign = function(align) {
209
+  this.align = align;
210
+  if (this.sourceBlock_.rendered) {
211
+    this.sourceBlock_.render();
212
+  }
213
+  return this;
214
+};
215
+
216
+/**
217
+ * Initialize the fields on this input.
218
+ */
219
+Blockly.Input.prototype.init = function() {
220
+  if (!this.sourceBlock_.workspace.rendered) {
221
+    return;  // Headless blocks don't need fields initialized.
222
+  }
223
+  for (var x = 0; x < this.fieldRow.length; x++) {
224
+    this.fieldRow[x].init(this.sourceBlock_);
225
+  }
226
+};
227
+
228
+/**
229
+ * Sever all links to this input.
230
+ */
231
+Blockly.Input.prototype.dispose = function() {
232
+  for (var i = 0, field; field = this.fieldRow[i]; i++) {
233
+    field.dispose();
234
+  }
235
+  if (this.connection) {
236
+    this.connection.dispose();
237
+  }
238
+  this.sourceBlock_ = null;
239
+};

+ 62 - 0
src/blockly/core/msg.js

@@ -0,0 +1,62 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2013 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Core JavaScript library for Blockly.
23
+ * @author scr@google.com (Sheridan Rawlins)
24
+ */
25
+'use strict';
26
+
27
+/**
28
+ * Name space for the Msg singleton.
29
+ * Msg gets populated in the message files.
30
+ */
31
+goog.provide('Blockly.Msg');
32
+
33
+
34
+/**
35
+ * Back up original getMsg function.
36
+ * @type {!Function}
37
+ */
38
+goog.getMsgOrig = goog.getMsg;
39
+
40
+/**
41
+ * Gets a localized message.
42
+ * Overrides the default Closure function to check for a Blockly.Msg first.
43
+ * Used infrequently, only known case is TODAY button in date picker.
44
+ * @param {string} str Translatable string, places holders in the form {$foo}.
45
+ * @param {Object<string, string>=} opt_values Maps place holder name to value.
46
+ * @return {string} message with placeholders filled.
47
+ * @suppress {duplicate}
48
+ */
49
+goog.getMsg = function(str, opt_values) {
50
+  var key = goog.getMsg.blocklyMsgMap[str];
51
+  if (key) {
52
+    str = Blockly.Msg[key];
53
+  }
54
+  return goog.getMsgOrig(str, opt_values);
55
+};
56
+
57
+/**
58
+ * Mapping of Closure messages to Blockly.Msg names.
59
+ */
60
+goog.getMsg.blocklyMsgMap = {
61
+  'Today': 'TODAY'
62
+};

File diff suppressed because it is too large
+ 303 - 0
src/blockly/core/mutator.js


+ 143 - 0
src/blockly/core/names.js

@@ -0,0 +1,143 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Utility functions for handling variables and procedure names.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Names');
28
+
29
+
30
+/**
31
+ * Class for a database of entity names (variables, functions, etc).
32
+ * @param {string} reservedWords A comma-separated string of words that are
33
+ *     illegal for use as names in a language (e.g. 'new,if,this,...').
34
+ * @param {string=} opt_variablePrefix Some languages need a '$' or a namespace
35
+ *     before all variable names.
36
+ * @constructor
37
+ */
38
+Blockly.Names = function(reservedWords, opt_variablePrefix) {
39
+  this.variablePrefix_ = opt_variablePrefix || '';
40
+  this.reservedDict_ = Object.create(null);
41
+  if (reservedWords) {
42
+    var splitWords = reservedWords.split(',');
43
+    for (var i = 0; i < splitWords.length; i++) {
44
+      this.reservedDict_[splitWords[i]] = true;
45
+    }
46
+  }
47
+  this.reset();
48
+};
49
+
50
+/**
51
+ * When JavaScript (or most other languages) is generated, variable 'foo' and
52
+ * procedure 'foo' would collide.  However, Blockly has no such problems since
53
+ * variable get 'foo' and procedure call 'foo' are unambiguous.
54
+ * Therefore, Blockly keeps a separate type name to disambiguate.
55
+ * getName('foo', 'variable') -> 'foo'
56
+ * getName('foo', 'procedure') -> 'foo2'
57
+ */
58
+
59
+/**
60
+ * Empty the database and start from scratch.  The reserved words are kept.
61
+ */
62
+Blockly.Names.prototype.reset = function() {
63
+  this.db_ = Object.create(null);
64
+  this.dbReverse_ = Object.create(null);
65
+};
66
+
67
+/**
68
+ * Convert a Blockly entity name to a legal exportable entity name.
69
+ * @param {string} name The Blockly entity name (no constraints).
70
+ * @param {string} type The type of entity in Blockly
71
+ *     ('VARIABLE', 'PROCEDURE', 'BUILTIN', etc...).
72
+ * @return {string} An entity name legal for the exported language.
73
+ */
74
+Blockly.Names.prototype.getName = function(name, type) {
75
+  var normalized = name.toLowerCase() + '_' + type;
76
+  var prefix = (type == Blockly.Variables.NAME_TYPE) ?
77
+      this.variablePrefix_ : '';
78
+  if (normalized in this.db_) {
79
+    return prefix + this.db_[normalized];
80
+  }
81
+  var safeName = this.getDistinctName(name, type);
82
+  this.db_[normalized] = safeName.substr(prefix.length);
83
+  return safeName;
84
+};
85
+
86
+/**
87
+ * Convert a Blockly entity name to a legal exportable entity name.
88
+ * Ensure that this is a new name not overlapping any previously defined name.
89
+ * Also check against list of reserved words for the current language and
90
+ * ensure name doesn't collide.
91
+ * @param {string} name The Blockly entity name (no constraints).
92
+ * @param {string} type The type of entity in Blockly
93
+ *     ('VARIABLE', 'PROCEDURE', 'BUILTIN', etc...).
94
+ * @return {string} An entity name legal for the exported language.
95
+ */
96
+Blockly.Names.prototype.getDistinctName = function(name, type) {
97
+  var safeName = this.safeName_(name);
98
+  var i = '';
99
+  while (this.dbReverse_[safeName + i] ||
100
+         (safeName + i) in this.reservedDict_) {
101
+    // Collision with existing name.  Create a unique name.
102
+    i = i ? i + 1 : 2;
103
+  }
104
+  safeName += i;
105
+  this.dbReverse_[safeName] = true;
106
+  var prefix = (type == Blockly.Variables.NAME_TYPE) ?
107
+      this.variablePrefix_ : '';
108
+  return prefix + safeName;
109
+};
110
+
111
+/**
112
+ * Given a proposed entity name, generate a name that conforms to the
113
+ * [_A-Za-z][_A-Za-z0-9]* format that most languages consider legal for
114
+ * variables.
115
+ * @param {string} name Potentially illegal entity name.
116
+ * @return {string} Safe entity name.
117
+ * @private
118
+ */
119
+Blockly.Names.prototype.safeName_ = function(name) {
120
+  if (!name) {
121
+    name = 'unnamed';
122
+  } else {
123
+    // Unfortunately names in non-latin characters will look like
124
+    // _E9_9F_B3_E4_B9_90 which is pretty meaningless.
125
+    name = encodeURI(name.replace(/ /g, '_')).replace(/[^\w]/g, '_');
126
+    // Most languages don't allow names with leading numbers.
127
+    if ('0123456789'.indexOf(name[0]) != -1) {
128
+      name = 'my_' + name;
129
+    }
130
+  }
131
+  return name;
132
+};
133
+
134
+/**
135
+ * Do the given two entity names refer to the same entity?
136
+ * Blockly names are case-insensitive.
137
+ * @param {string} name1 First name.
138
+ * @param {string} name2 Second name.
139
+ * @return {boolean} True if names are the same.
140
+ */
141
+Blockly.Names.equals = function(name1, name2) {
142
+  return name1.toLowerCase() == name2.toLowerCase();
143
+};

+ 284 - 0
src/blockly/core/procedures.js

@@ -0,0 +1,284 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Utility functions for handling procedures.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Procedures');
28
+
29
+// TODO(scr): Fix circular dependencies
30
+// goog.require('Blockly.Block');
31
+goog.require('Blockly.Field');
32
+goog.require('Blockly.Names');
33
+goog.require('Blockly.Workspace');
34
+
35
+
36
+/**
37
+ * Category to separate procedure names from variables and generated functions.
38
+ */
39
+Blockly.Procedures.NAME_TYPE = 'PROCEDURE';
40
+
41
+/**
42
+ * Find all user-created procedure definitions in a workspace.
43
+ * @param {!Blockly.Workspace} root Root workspace.
44
+ * @return {!Array.<!Array.<!Array>>} Pair of arrays, the
45
+ *     first contains procedures without return variables, the second with.
46
+ *     Each procedure is defined by a three-element list of name, parameter
47
+ *     list, and return value boolean.
48
+ */
49
+Blockly.Procedures.allProcedures = function(root) {
50
+  var blocks = root.getAllBlocks();
51
+  var proceduresReturn = [];
52
+  var proceduresNoReturn = [];
53
+  for (var i = 0; i < blocks.length; i++) {
54
+    if (blocks[i].getProcedureDef) {
55
+      var tuple = blocks[i].getProcedureDef();
56
+      if (tuple) {
57
+        if (tuple[2]) {
58
+          proceduresReturn.push(tuple);
59
+        } else {
60
+          proceduresNoReturn.push(tuple);
61
+        }
62
+      }
63
+    }
64
+  }
65
+  proceduresNoReturn.sort(Blockly.Procedures.procTupleComparator_);
66
+  proceduresReturn.sort(Blockly.Procedures.procTupleComparator_);
67
+  return [proceduresNoReturn, proceduresReturn];
68
+};
69
+
70
+/**
71
+ * Comparison function for case-insensitive sorting of the first element of
72
+ * a tuple.
73
+ * @param {!Array} ta First tuple.
74
+ * @param {!Array} tb Second tuple.
75
+ * @return {number} -1, 0, or 1 to signify greater than, equality, or less than.
76
+ * @private
77
+ */
78
+Blockly.Procedures.procTupleComparator_ = function(ta, tb) {
79
+  return ta[0].toLowerCase().localeCompare(tb[0].toLowerCase());
80
+};
81
+
82
+/**
83
+ * Ensure two identically-named procedures don't exist.
84
+ * @param {string} name Proposed procedure name.
85
+ * @param {!Blockly.Block} block Block to disambiguate.
86
+ * @return {string} Non-colliding name.
87
+ */
88
+Blockly.Procedures.findLegalName = function(name, block) {
89
+  if (block.isInFlyout) {
90
+    // Flyouts can have multiple procedures called 'do something'.
91
+    return name;
92
+  }
93
+  while (!Blockly.Procedures.isLegalName(name, block.workspace, block)) {
94
+    // Collision with another procedure.
95
+    var r = name.match(/^(.*?)(\d+)$/);
96
+    if (!r) {
97
+      name += '2';
98
+    } else {
99
+      name = r[1] + (parseInt(r[2], 10) + 1);
100
+    }
101
+  }
102
+  return name;
103
+};
104
+
105
+/**
106
+ * Does this procedure have a legal name?  Illegal names include names of
107
+ * procedures already defined.
108
+ * @param {string} name The questionable name.
109
+ * @param {!Blockly.Workspace} workspace The workspace to scan for collisions.
110
+ * @param {Blockly.Block=} opt_exclude Optional block to exclude from
111
+ *     comparisons (one doesn't want to collide with oneself).
112
+ * @return {boolean} True if the name is legal.
113
+ */
114
+Blockly.Procedures.isLegalName = function(name, workspace, opt_exclude) {
115
+  var blocks = workspace.getAllBlocks();
116
+  // Iterate through every block and check the name.
117
+  for (var i = 0; i < blocks.length; i++) {
118
+    if (blocks[i] == opt_exclude) {
119
+      continue;
120
+    }
121
+    if (blocks[i].getProcedureDef) {
122
+      var procName = blocks[i].getProcedureDef();
123
+      if (Blockly.Names.equals(procName[0], name)) {
124
+        return false;
125
+      }
126
+    }
127
+  }
128
+  return true;
129
+};
130
+
131
+/**
132
+ * Rename a procedure.  Called by the editable field.
133
+ * @param {string} text The proposed new name.
134
+ * @return {string} The accepted name.
135
+ * @this {!Blockly.Field}
136
+ */
137
+Blockly.Procedures.rename = function(text) {
138
+  // Strip leading and trailing whitespace.  Beyond this, all names are legal.
139
+  text = text.replace(/^[\s\xa0]+|[\s\xa0]+$/g, '');
140
+
141
+  // Ensure two identically-named procedures don't exist.
142
+  text = Blockly.Procedures.findLegalName(text, this.sourceBlock_);
143
+  // Rename any callers.
144
+  var blocks = this.sourceBlock_.workspace.getAllBlocks();
145
+  for (var i = 0; i < blocks.length; i++) {
146
+    if (blocks[i].renameProcedure) {
147
+      blocks[i].renameProcedure(this.text_, text);
148
+    }
149
+  }
150
+  return text;
151
+};
152
+
153
+/**
154
+ * Construct the blocks required by the flyout for the procedure category.
155
+ * @param {!Blockly.Workspace} workspace The workspace contianing procedures.
156
+ * @return {!Array.<!Element>} Array of XML block elements.
157
+ */
158
+Blockly.Procedures.flyoutCategory = function(workspace) {
159
+  var xmlList = [];
160
+  if (Blockly.Blocks['procedures_defnoreturn']) {
161
+    // <block type="procedures_defnoreturn" gap="16"></block>
162
+    var block = goog.dom.createDom('block');
163
+    block.setAttribute('type', 'procedures_defnoreturn');
164
+    block.setAttribute('gap', 16);
165
+    xmlList.push(block);
166
+  }
167
+  if (Blockly.Blocks['procedures_defreturn']) {
168
+    // <block type="procedures_defreturn" gap="16"></block>
169
+    var block = goog.dom.createDom('block');
170
+    block.setAttribute('type', 'procedures_defreturn');
171
+    block.setAttribute('gap', 16);
172
+    xmlList.push(block);
173
+  }
174
+  if (Blockly.Blocks['procedures_ifreturn']) {
175
+    // <block type="procedures_ifreturn" gap="16"></block>
176
+    var block = goog.dom.createDom('block');
177
+    block.setAttribute('type', 'procedures_ifreturn');
178
+    block.setAttribute('gap', 16);
179
+    xmlList.push(block);
180
+  }
181
+  if (xmlList.length) {
182
+    // Add slightly larger gap between system blocks and user calls.
183
+    xmlList[xmlList.length - 1].setAttribute('gap', 24);
184
+  }
185
+
186
+  function populateProcedures(procedureList, templateName) {
187
+    for (var i = 0; i < procedureList.length; i++) {
188
+      var name = procedureList[i][0];
189
+      var args = procedureList[i][1];
190
+      // <block type="procedures_callnoreturn" gap="16">
191
+      //   <mutation name="do something">
192
+      //     <arg name="x"></arg>
193
+      //   </mutation>
194
+      // </block>
195
+      var block = goog.dom.createDom('block');
196
+      block.setAttribute('type', templateName);
197
+      block.setAttribute('gap', 16);
198
+      var mutation = goog.dom.createDom('mutation');
199
+      mutation.setAttribute('name', name);
200
+      block.appendChild(mutation);
201
+      for (var t = 0; t < args.length; t++) {
202
+        var arg = goog.dom.createDom('arg');
203
+        arg.setAttribute('name', args[t]);
204
+        mutation.appendChild(arg);
205
+      }
206
+      xmlList.push(block);
207
+    }
208
+  }
209
+
210
+  var tuple = Blockly.Procedures.allProcedures(workspace);
211
+  populateProcedures(tuple[0], 'procedures_callnoreturn');
212
+  populateProcedures(tuple[1], 'procedures_callreturn');
213
+  return xmlList;
214
+};
215
+
216
+/**
217
+ * Find all the callers of a named procedure.
218
+ * @param {string} name Name of procedure.
219
+ * @param {!Blockly.Workspace} workspace The workspace to find callers in.
220
+ * @return {!Array.<!Blockly.Block>} Array of caller blocks.
221
+ */
222
+Blockly.Procedures.getCallers = function(name, workspace) {
223
+  var callers = [];
224
+  var blocks = workspace.getAllBlocks();
225
+  // Iterate through every block and check the name.
226
+  for (var i = 0; i < blocks.length; i++) {
227
+    if (blocks[i].getProcedureCall) {
228
+      var procName = blocks[i].getProcedureCall();
229
+      // Procedure name may be null if the block is only half-built.
230
+      if (procName && Blockly.Names.equals(procName, name)) {
231
+        callers.push(blocks[i]);
232
+      }
233
+    }
234
+  }
235
+  return callers;
236
+};
237
+
238
+/**
239
+ * When a procedure definition is disposed of, find and dispose of all its
240
+ *     callers.
241
+ * @param {string} name Name of deleted procedure definition.
242
+ * @param {!Blockly.Workspace} workspace The workspace to delete callers from.
243
+ */
244
+Blockly.Procedures.disposeCallers = function(name, workspace) {
245
+  var callers = Blockly.Procedures.getCallers(name, workspace);
246
+  for (var i = 0; i < callers.length; i++) {
247
+    callers[i].dispose(true, false);
248
+  }
249
+};
250
+
251
+/**
252
+ * When a procedure definition changes its parameters, find and edit all its
253
+ * callers.
254
+ * @param {string} name Name of edited procedure definition.
255
+ * @param {!Blockly.Workspace} workspace The workspace to delete callers from.
256
+ * @param {!Array.<string>} paramNames Array of new parameter names.
257
+ * @param {!Array.<string>} paramIds Array of unique parameter IDs.
258
+ */
259
+Blockly.Procedures.mutateCallers = function(name, workspace,
260
+                                            paramNames, paramIds) {
261
+  var callers = Blockly.Procedures.getCallers(name, workspace);
262
+  for (var i = 0; i < callers.length; i++) {
263
+    callers[i].setProcedureParameters(paramNames, paramIds);
264
+  }
265
+};
266
+
267
+/**
268
+ * Find the definition block for the named procedure.
269
+ * @param {string} name Name of procedure.
270
+ * @param {!Blockly.Workspace} workspace The workspace to search.
271
+ * @return {Blockly.Block} The procedure definition block, or null not found.
272
+ */
273
+Blockly.Procedures.getDefinition = function(name, workspace) {
274
+  var blocks = workspace.getAllBlocks();
275
+  for (var i = 0; i < blocks.length; i++) {
276
+    if (blocks[i].getProcedureDef) {
277
+      var tuple = blocks[i].getProcedureDef();
278
+      if (tuple && Blockly.Names.equals(tuple[0], name)) {
279
+        return blocks[i];
280
+      }
281
+    }
282
+  }
283
+  return null;
284
+};

+ 500 - 0
src/blockly/core/realtime-client-utils.js

@@ -0,0 +1,500 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2013 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Common utility functionality for Google Drive Realtime API,
23
+ * including authorization and file loading. This functionality should serve
24
+ * mostly as a well-documented example, though is usable in its own right.
25
+ *
26
+ * You can find this code as part of the Google Drive Realtime API Quickstart at
27
+ * https://developers.google.com/drive/realtime/realtime-quickstart and also as
28
+ * part of the Google Drive Realtime Playground code at
29
+ * https://github.com/googledrive/realtime-playground/blob/master/js/realtime-client-utils.js
30
+ */
31
+'use strict';
32
+
33
+/**
34
+ * Realtime client utilities namespace.
35
+ */
36
+goog.provide('rtclient');
37
+
38
+
39
+/**
40
+ * OAuth 2.0 scope for installing Drive Apps.
41
+ * @const
42
+ */
43
+rtclient.INSTALL_SCOPE = 'https://www.googleapis.com/auth/drive.install';
44
+
45
+/**
46
+ * OAuth 2.0 scope for opening and creating files.
47
+ * @const
48
+ */
49
+rtclient.FILE_SCOPE = 'https://www.googleapis.com/auth/drive.file';
50
+
51
+/**
52
+ * OAuth 2.0 scope for accessing the appdata folder, a hidden folder private
53
+ * to this app.
54
+ * @const
55
+ */
56
+rtclient.APPDATA_SCOPE = 'https://www.googleapis.com/auth/drive.appdata';
57
+
58
+/**
59
+ * OAuth 2.0 scope for accessing the user's ID.
60
+ * @const
61
+ */
62
+rtclient.OPENID_SCOPE = 'openid';
63
+
64
+/**
65
+ * MIME type for newly created Realtime files.
66
+ * @const
67
+ */
68
+rtclient.REALTIME_MIMETYPE = 'application/vnd.google-apps.drive-sdk';
69
+
70
+/**
71
+ * Key used to store the folder id of the Drive folder in which we will store
72
+ * Realtime files.
73
+ * @type {string}
74
+ */
75
+rtclient.FOLDER_KEY = 'folderId';
76
+
77
+/**
78
+ * Parses the hash parameters to this page and returns them as an object.
79
+ * @return {!Object} Parameter object.
80
+ */
81
+rtclient.getParams = function() {
82
+  // Be careful with regards to node.js which has no window or location.
83
+  var location = goog.global['location'] || {};
84
+  var params = {};
85
+  function parseParams(fragment) {
86
+    // Split up the query string and store in an object.
87
+    var paramStrs = fragment.slice(1).split('&');
88
+    for (var i = 0; i < paramStrs.length; i++) {
89
+      var paramStr = paramStrs[i].split('=');
90
+      params[decodeURIComponent(paramStr[0])] = decodeURIComponent(paramStr[1]);
91
+    }
92
+  }
93
+  var hashFragment = location.hash;
94
+  if (hashFragment) {
95
+    parseParams(hashFragment);
96
+  }
97
+  // Opening from Drive will encode the state in a query search parameter.
98
+  var searchFragment = location.search;
99
+  if (searchFragment) {
100
+    parseParams(searchFragment);
101
+  }
102
+  return params;
103
+};
104
+
105
+/**
106
+ * Instance of the query parameters.
107
+ */
108
+rtclient.params = rtclient.getParams();
109
+
110
+/**
111
+ * Fetches an option from options or a default value, logging an error if
112
+ *     neither is available.
113
+ * @param {!Object} options Containing options.
114
+ * @param {string} key Option key.
115
+ * @param {*=} opt_defaultValue Default option value (optional).
116
+ * @return {*} Option value.
117
+ */
118
+rtclient.getOption = function(options, key, opt_defaultValue) {
119
+  if (options.hasOwnProperty(key)) {
120
+    return options[key];
121
+  }
122
+  if (opt_defaultValue === undefined) {
123
+    console.error(key + ' should be present in the options.');
124
+  }
125
+  return opt_defaultValue;
126
+};
127
+
128
+/**
129
+ * Creates a new Authorizer from the options.
130
+ * @constructor
131
+ * @param {!Object} options For authorizer. Two keys are required as mandatory,
132
+ *     these are:
133
+ *
134
+ *    1. "clientId", the Client ID from the console
135
+ *    2. "authButtonElementId", the is of the dom element to use for
136
+ *       authorizing.
137
+ */
138
+rtclient.Authorizer = function(options) {
139
+  this.clientId = rtclient.getOption(options, 'clientId');
140
+  // Get the user ID if it's available in the state query parameter.
141
+  this.userId = rtclient.params['userId'];
142
+  this.authButton = document.getElementById(rtclient.getOption(options,
143
+      'authButtonElementId'));
144
+  this.authDiv = document.getElementById(rtclient.getOption(options,
145
+      'authDivElementId'));
146
+};
147
+
148
+/**
149
+ * Start the authorization process.
150
+ * @param {Function} onAuthComplete To call once authorization has completed.
151
+ */
152
+rtclient.Authorizer.prototype.start = function(onAuthComplete) {
153
+  var _this = this;
154
+  gapi.load('auth:client,drive-realtime,drive-share', function() {
155
+    _this.authorize(onAuthComplete);
156
+  });
157
+};
158
+
159
+/**
160
+ * Reauthorize the client with no callback (used for authorization failure).
161
+ * @param {Function} onAuthComplete To call once authorization has completed.
162
+ */
163
+rtclient.Authorizer.prototype.authorize = function(onAuthComplete) {
164
+  var clientId = this.clientId;
165
+  var userId = this.userId;
166
+  var _this = this;
167
+  var handleAuthResult = function(authResult) {
168
+    if (authResult && !authResult.error) {
169
+      _this.authButton.disabled = true;
170
+      _this.fetchUserId(onAuthComplete);
171
+      _this.authDiv.style.display = 'none';
172
+    } else {
173
+      _this.authButton.disabled = false;
174
+      _this.authButton.onclick = authorizeWithPopup;
175
+      _this.authDiv.style.display = 'block';
176
+    }
177
+  };
178
+  var authorizeWithPopup = function() {
179
+    gapi.auth.authorize({
180
+      'client_id': clientId,
181
+      'scope': [
182
+        rtclient.INSTALL_SCOPE,
183
+        rtclient.FILE_SCOPE,
184
+        rtclient.OPENID_SCOPE,
185
+        rtclient.APPDATA_SCOPE
186
+      ],
187
+      'user_id': userId,
188
+      'immediate': false
189
+    }, handleAuthResult);
190
+  };
191
+  // Try with no popups first.
192
+  gapi.auth.authorize({
193
+    'client_id': clientId,
194
+    'scope': [
195
+      rtclient.INSTALL_SCOPE,
196
+      rtclient.FILE_SCOPE,
197
+      rtclient.OPENID_SCOPE,
198
+      rtclient.APPDATA_SCOPE
199
+    ],
200
+    'user_id': userId,
201
+    'immediate': true
202
+  }, handleAuthResult);
203
+};
204
+
205
+/**
206
+ * Fetch the user ID using the UserInfo API and save it locally.
207
+ * @param {Function} callback The callback to call after user ID has been
208
+ *     fetched.
209
+ */
210
+rtclient.Authorizer.prototype.fetchUserId = function(callback) {
211
+  var _this = this;
212
+  gapi.client.load('oauth2', 'v2', function() {
213
+    gapi.client.oauth2.userinfo.get().execute(function(resp) {
214
+      if (resp.id) {
215
+        _this.userId = resp.id;
216
+      }
217
+      if (callback) {
218
+        callback();
219
+      }
220
+    });
221
+  });
222
+};
223
+
224
+/**
225
+ * Creates a new Realtime file.
226
+ * @param {string} title Title of the newly created file.
227
+ * @param {string} mimeType The MIME type of the new file.
228
+ * @param {string} folderTitle Title of the folder to place the file in.
229
+ * @param {Function} callback The callback to call after creation.
230
+ */
231
+rtclient.createRealtimeFile = function(title, mimeType, folderTitle, callback) {
232
+
233
+  function insertFile(folderId) {
234
+    gapi.client.drive.files.insert({
235
+      'resource': {
236
+        'mimeType': mimeType,
237
+        'title': title,
238
+        'parents': [{'id': folderId}]
239
+      }
240
+    }).execute(callback);
241
+  }
242
+
243
+  function getOrCreateFolder() {
244
+
245
+    function storeInAppdataProperty(folderId) {
246
+      // Store folder id in a custom property of the appdata folder.  The
247
+      // 'appdata' folder is a special Google Drive folder that is only
248
+      // accessible by a specific app (i.e. identified by the client id).
249
+      gapi.client.drive.properties.insert({
250
+        'fileId': 'appdata',
251
+        'resource': { 'key': rtclient.FOLDER_KEY, 'value': folderId }
252
+      }).execute(function(resp) {
253
+        insertFile(folderId);
254
+      });
255
+    };
256
+
257
+    function createFolder() {
258
+      gapi.client.drive.files.insert({
259
+        'resource': {
260
+          'mimeType': 'application/vnd.google-apps.folder',
261
+          'title': folderTitle
262
+        }
263
+      }).execute(function(folder) {
264
+        storeInAppdataProperty(folder.id);
265
+      });
266
+    }
267
+
268
+    // Get the folder id from the appdata properties.
269
+    gapi.client.drive.properties.get({
270
+      'fileId': 'appdata',
271
+      'propertyKey': rtclient.FOLDER_KEY
272
+    }).execute(function(resp) {
273
+       if (resp.error) {
274
+        // There's no folder id stored yet so we create a new folder if a
275
+        // folderTitle has been supplied.
276
+        if (folderTitle) {
277
+          createFolder();
278
+        } else {
279
+          // There's no folder specified, so we just store the file in the
280
+          // user's root folder.
281
+          storeInAppdataProperty('root');
282
+        }
283
+      } else {
284
+         var folderId = resp.result.value;
285
+         gapi.client.drive.files.get({
286
+           'fileId': folderId
287
+         }).execute(function(resp) {
288
+           if (resp.error || resp.labels.trashed) {
289
+             // Folder doesn't exist or was deleted, so create a new one.
290
+             createFolder();
291
+           } else {
292
+             insertFile(folderId);
293
+           }
294
+         });
295
+      }
296
+    });
297
+  }
298
+
299
+  gapi.client.load('drive', 'v2', function() {
300
+    getOrCreateFolder();
301
+  });
302
+};
303
+
304
+/**
305
+ * Fetches the metadata for a Realtime file.
306
+ * @param {string} fileId The file to load metadata for.
307
+ * @param {Function} callback The callback to be called on completion,
308
+ *     with signature:
309
+ *
310
+ *    function onGetFileMetadata(file) {}
311
+ *
312
+ * where the file parameter is a Google Drive API file resource instance.
313
+ */
314
+rtclient.getFileMetadata = function(fileId, callback) {
315
+  gapi.client.load('drive', 'v2', function() {
316
+    gapi.client.drive.files.get({
317
+      'fileId': fileId
318
+    }).execute(callback);
319
+  });
320
+};
321
+
322
+/**
323
+ * Parses the state parameter passed from the Drive user interface after
324
+ *     Open With operations.
325
+ * @param {string} stateParam The state query parameter as a JSON string.
326
+ * @return {Object} The state query parameter as an object or null if
327
+ *     parsing failed.
328
+ */
329
+rtclient.parseState = function(stateParam) {
330
+  try {
331
+    var stateObj = JSON.parse(stateParam);
332
+    return stateObj;
333
+  } catch (e) {
334
+    return null;
335
+  }
336
+};
337
+
338
+/**
339
+ * Handles authorizing, parsing query parameters, loading and creating Realtime
340
+ *     documents.
341
+ * @constructor
342
+ * @param {!Object} options Options for loader. Four keys are required as
343
+ *     mandatory, these are:
344
+ *
345
+ *    1. "clientId", the Client ID from the console
346
+ *    2. "initializeModel", the callback to call when the file is loaded.
347
+ *    3. "onFileLoaded", the callback to call when the model is first created.
348
+ *
349
+ * and two keys are optional:
350
+ *
351
+ *    1. "defaultTitle", the title of newly created Realtime files.
352
+ *    2. "defaultFolderTitle", the folder to place in which to place newly
353
+ *       created Realtime files.
354
+ */
355
+rtclient.RealtimeLoader = function(options) {
356
+  // Initialize configuration variables.
357
+  this.onFileLoaded = rtclient.getOption(options, 'onFileLoaded');
358
+  this.newFileMimeType = rtclient.getOption(options, 'newFileMimeType',
359
+      rtclient.REALTIME_MIMETYPE);
360
+  this.initializeModel = rtclient.getOption(options, 'initializeModel');
361
+  this.registerTypes = rtclient.getOption(options, 'registerTypes',
362
+      function() {});
363
+  this.afterAuth = rtclient.getOption(options, 'afterAuth', function() {});
364
+  // This tells us if need to we automatically create a file after auth.
365
+  this.autoCreate = rtclient.getOption(options, 'autoCreate', false);
366
+  this.defaultTitle = rtclient.getOption(options, 'defaultTitle',
367
+      'New Realtime File');
368
+  this.defaultFolderTitle = rtclient.getOption(options, 'defaultFolderTitle',
369
+      '');
370
+  this.afterCreate = rtclient.getOption(options, 'afterCreate', function() {});
371
+  this.authorizer = new rtclient.Authorizer(options);
372
+};
373
+
374
+/**
375
+ * Redirects the browser back to the current page with an appropriate file ID.
376
+ * @param {Array.<string>} fileIds The IDs of the files to open.
377
+ * @param {string} userId The ID of the user.
378
+ */
379
+rtclient.RealtimeLoader.prototype.redirectTo = function(fileIds, userId) {
380
+  var params = [];
381
+  if (fileIds) {
382
+    params.push('fileIds=' + fileIds.join(','));
383
+  }
384
+  if (userId) {
385
+    params.push('userId=' + userId);
386
+  }
387
+  // Naive URL construction.
388
+  var newUrl = params.length == 0 ?
389
+      window.location.pathname :
390
+      (window.location.pathname + '#' + params.join('&'));
391
+  // Using HTML URL re-write if available.
392
+  if (window.history && window.history.replaceState) {
393
+    window.history.replaceState('Google Drive Realtime API Playground',
394
+        'Google Drive Realtime API Playground', newUrl);
395
+  } else {
396
+    window.location.href = newUrl;
397
+  }
398
+  // We are still here that means the page didn't reload.
399
+  rtclient.params = rtclient.getParams();
400
+  for (var index in fileIds) {
401
+    gapi.drive.realtime.load(fileIds[index], this.onFileLoaded,
402
+        this.initializeModel, this.handleErrors);
403
+  }
404
+};
405
+
406
+/**
407
+ * Starts the loader by authorizing.
408
+ */
409
+rtclient.RealtimeLoader.prototype.start = function() {
410
+  // Bind to local context to make them suitable for callbacks.
411
+  var _this = this;
412
+  this.authorizer.start(function() {
413
+    if (_this.registerTypes) {
414
+      _this.registerTypes();
415
+    }
416
+    if (_this.afterAuth) {
417
+      _this.afterAuth();
418
+    }
419
+    _this.load();
420
+  });
421
+};
422
+
423
+/**
424
+ * Handles errors thrown by the Realtime API.
425
+ * @param {!Error} e Error.
426
+ */
427
+rtclient.RealtimeLoader.prototype.handleErrors = function(e) {
428
+  if (e.type == gapi.drive.realtime.ErrorType.TOKEN_REFRESH_REQUIRED) {
429
+    this.authorizer.authorize();
430
+  } else if (e.type == gapi.drive.realtime.ErrorType.CLIENT_ERROR) {
431
+    alert('An Error happened: ' + e.message);
432
+    window.location.href = '/';
433
+  } else if (e.type == gapi.drive.realtime.ErrorType.NOT_FOUND) {
434
+    alert('The file was not found. It does not exist or you do not have ' +
435
+        'read access to the file.');
436
+    window.location.href = '/';
437
+  }
438
+};
439
+
440
+/**
441
+ * Loads or creates a Realtime file depending on the fileId and state query
442
+ * parameters.
443
+ */
444
+rtclient.RealtimeLoader.prototype.load = function() {
445
+  var fileIds = rtclient.params['fileIds'];
446
+  if (fileIds) {
447
+    fileIds = fileIds.split(',');
448
+  }
449
+  var userId = this.authorizer.userId;
450
+  var state = rtclient.params['state'];
451
+  // Creating the error callback.
452
+  var authorizer = this.authorizer;
453
+  // We have file IDs in the query parameters, so we will use them to load a
454
+  // file.
455
+  if (fileIds) {
456
+    for (var index in fileIds) {
457
+      gapi.drive.realtime.load(fileIds[index], this.onFileLoaded,
458
+          this.initializeModel, this.handleErrors);
459
+    }
460
+    return;
461
+  }
462
+  // We have a state parameter being redirected from the Drive UI.
463
+  // We will parse it and redirect to the fileId contained.
464
+  else if (state) {
465
+    var stateObj = rtclient.parseState(state);
466
+    // If opening a file from Drive.
467
+    if (stateObj.action == 'open') {
468
+      fileIds = stateObj.ids;
469
+      userId = stateObj.userId;
470
+      this.redirectTo(fileIds, userId);
471
+      return;
472
+    }
473
+  }
474
+  if (this.autoCreate) {
475
+    this.createNewFileAndRedirect();
476
+  }
477
+};
478
+
479
+/**
480
+ * Creates a new file and redirects to the URL to load it.
481
+ */
482
+rtclient.RealtimeLoader.prototype.createNewFileAndRedirect = function() {
483
+  // No fileId or state have been passed. We create a new Realtime file and
484
+  // redirect to it.
485
+  var _this = this;
486
+  rtclient.createRealtimeFile(this.defaultTitle, this.newFileMimeType,
487
+      this.defaultFolderTitle,
488
+      function(file) {
489
+        if (file.id) {
490
+          if (_this.afterCreate) {
491
+            _this.afterCreate(file.id);
492
+          }
493
+          _this.redirectTo([file.id], _this.authorizer.userId);
494
+        } else {
495
+          // File failed to be created, log why and do not attempt to redirect.
496
+          console.error('Error creating file.');
497
+          console.error(file);
498
+        }
499
+      });
500
+};

+ 869 - 0
src/blockly/core/realtime.js

@@ -0,0 +1,869 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2014 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * This file contains functions used by any Blockly app that wants to provide
23
+ * realtime collaboration functionality.
24
+ */
25
+
26
+/**
27
+ * @fileoverview Common support code for Blockly apps using realtime
28
+ * collaboration.
29
+ * Note that to use this you must set up a project via the Google Developers
30
+ * Console. Instructions on how to do that can be found at
31
+ * https://developers.google.com/blockly/realtime-collaboration
32
+ * Once you do that you can set the clientId in
33
+ * Blockly.Realtime.rtclientOptions_
34
+ * @author markf@google.com (Mark Friedman)
35
+ */
36
+'use strict';
37
+
38
+goog.provide('Blockly.Realtime');
39
+
40
+goog.require('goog.array');
41
+goog.require('goog.dom');
42
+goog.require('goog.style');
43
+goog.require('rtclient');
44
+
45
+/**
46
+ * Is realtime collaboration enabled?
47
+ * @type {boolean}
48
+ * @private
49
+ */
50
+Blockly.Realtime.enabled_ = false;
51
+
52
+/**
53
+ * The Realtime document being collaborated on.
54
+ * @type {gapi.drive.realtime.Document}
55
+ * @private
56
+ */
57
+Blockly.Realtime.document_ = null;
58
+
59
+/**
60
+ * The Realtime model of this doc.
61
+ * @type {gapi.drive.realtime.Model}
62
+ * @private
63
+ */
64
+Blockly.Realtime.model_ = null;
65
+
66
+/**
67
+ * The unique id associated with this editing session.
68
+ * @type {string}
69
+ * @private
70
+ */
71
+Blockly.Realtime.sessionId_ = null;
72
+
73
+/**
74
+ * The function used to initialize the UI after realtime is initialized.
75
+ * @type {function()}
76
+ * @private
77
+ */
78
+Blockly.Realtime.initUi_ = null;
79
+
80
+/**
81
+ * A map from block id to blocks.
82
+ * @type {gapi.drive.realtime.CollaborativeMap}
83
+ * @private
84
+ */
85
+Blockly.Realtime.blocksMap_ = null;
86
+
87
+/**
88
+ * Are currently syncing from another instance of this realtime doc.
89
+ * @type {boolean}
90
+ */
91
+Blockly.Realtime.withinSync = false;
92
+
93
+/**
94
+ * The current instance of the realtime loader client
95
+ * @type {rtclient.RealtimeLoader}
96
+ * @private
97
+ */
98
+Blockly.Realtime.realtimeLoader_ = null;
99
+
100
+/**
101
+ * The id of a text area to be used as a realtime chat box.
102
+ * @type {string}
103
+ * @private
104
+ */
105
+Blockly.Realtime.chatBoxElementId_ = null;
106
+
107
+/**
108
+ * The initial text to be placed in the realtime chat box.
109
+ * @type {string}
110
+ * @private
111
+ */
112
+Blockly.Realtime.chatBoxInitialText_ = null;
113
+
114
+/**
115
+ * Indicator of whether we are in the context of an undo or redo operation.
116
+ * @type {boolean}
117
+ * @private
118
+ */
119
+Blockly.Realtime.withinUndo_ = false;
120
+
121
+/**
122
+ * Returns whether realtime collaboration is enabled.
123
+ * @return {boolean}
124
+ */
125
+Blockly.Realtime.isEnabled = function() {
126
+  return Blockly.Realtime.enabled_;
127
+};
128
+
129
+/**
130
+ * The id of the button to use for undo.
131
+ * @type {string}
132
+ * @private
133
+ */
134
+Blockly.Realtime.undoElementId_ = null;
135
+
136
+/**
137
+ * The id of the button to use for redo.
138
+ * @type {string}
139
+ * @private
140
+ */
141
+Blockly.Realtime.redoElementId_ = null;
142
+
143
+/**
144
+ * URL of the animated progress indicator.
145
+ * @type {string}
146
+ * @private
147
+ */
148
+Blockly.Realtime.PROGRESS_URL_ = 'progress.gif';
149
+
150
+/**
151
+ * URL of the anonymous user image.
152
+ * @type {string}
153
+ * @private
154
+ */
155
+Blockly.Realtime.ANONYMOUS_URL_ = 'anon.jpeg';
156
+
157
+/**
158
+ * This function is called the first time that the Realtime model is created
159
+ * for a file. This function should be used to initialize any values of the
160
+ * model.
161
+ * @param {!gapi.drive.realtime.Model} model The Realtime root model object.
162
+ * @private
163
+ */
164
+Blockly.Realtime.initializeModel_ = function(model) {
165
+  Blockly.Realtime.model_ = model;
166
+  var blocksMap = model.createMap();
167
+  model.getRoot().set('blocks', blocksMap);
168
+  var topBlocks = model.createList();
169
+  model.getRoot().set('topBlocks', topBlocks);
170
+  if (Blockly.Realtime.chatBoxElementId_) {
171
+    model.getRoot().set(Blockly.Realtime.chatBoxElementId_,
172
+        model.createString(Blockly.Realtime.chatBoxInitialText_));
173
+  }
174
+};
175
+
176
+/**
177
+ * Delete a block from the realtime blocks map.
178
+ * @param {!Blockly.Block} block The block to remove.
179
+ */
180
+Blockly.Realtime.removeBlock = function(block) {
181
+  Blockly.Realtime.blocksMap_['delete'](block.id.toString());
182
+};
183
+
184
+/**
185
+ * Add to the list of top-level blocks.
186
+ * @param {!Blockly.Block} block The block to add.
187
+ */
188
+Blockly.Realtime.addTopBlock = function(block) {
189
+  if (Blockly.Realtime.topBlocks_.indexOf(block) == -1) {
190
+    Blockly.Realtime.topBlocks_.push(block);
191
+  }
192
+};
193
+
194
+/**
195
+ * Delete a block from the list of top-level blocks.
196
+ * @param {!Blockly.Block} block The block to remove.
197
+ */
198
+Blockly.Realtime.removeTopBlock = function(block) {
199
+  Blockly.Realtime.topBlocks_.removeValue(block);
200
+};
201
+
202
+/**
203
+ * Obtain a newly created block known by the Realtime API.
204
+ * @param {!Blockly.Workspace} workspace The workspace to put the block in.
205
+ * @param {string} prototypeName The name of the prototype for the block.
206
+ * @return {!Blockly.Block}
207
+ */
208
+Blockly.Realtime.obtainBlock = function(workspace, prototypeName) {
209
+  var newBlock =
210
+      Blockly.Realtime.model_.create(Blockly.Block, workspace, prototypeName);
211
+  return newBlock;
212
+};
213
+
214
+/**
215
+ * Get an existing block by id.
216
+ * @param {string} id The block's id.
217
+ * @return {Blockly.Block} The found block.
218
+ */
219
+Blockly.Realtime.getBlockById = function(id) {
220
+  return Blockly.Realtime.blocksMap_.get(id);
221
+};
222
+
223
+/**
224
+ * Log the event for debugging purposes.
225
+ * @param {gapi.drive.realtime.BaseModelEvent} evt The event that occurred.
226
+ * @private
227
+ */
228
+Blockly.Realtime.logEvent_ = function(evt) {
229
+  console.log('Object event:');
230
+  console.log('  id: ' + evt.target.id);
231
+  console.log('  type: ' + evt.type);
232
+  var events = evt.events;
233
+  if (events) {
234
+    var eventCount = events.length;
235
+    for (var i = 0; i < eventCount; i++) {
236
+      var event = events[i];
237
+      console.log('  child event:');
238
+      console.log('    id: ' + event.target.id);
239
+      console.log('    type: ' + event.type);
240
+    }
241
+  }
242
+};
243
+
244
+/**
245
+ * Event handler to call when a block is changed.
246
+ * @param {!gapi.drive.realtime.ObjectChangedEvent} evt The event that occurred.
247
+ * @private
248
+ */
249
+Blockly.Realtime.onObjectChange_ = function(evt) {
250
+  var events = evt.events;
251
+  var eventCount = evt.events.length;
252
+  for (var i = 0; i < eventCount; i++) {
253
+    var event = events[i];
254
+    if (!event.isLocal || Blockly.Realtime.withinUndo_) {
255
+      var block = event.target;
256
+      if (event.type == 'value_changed') {
257
+        if (event.property == 'xmlDom') {
258
+          Blockly.Realtime.doWithinSync_(function() {
259
+            Blockly.Realtime.placeBlockOnWorkspace_(block, false);
260
+            Blockly.Realtime.moveBlock_(block);
261
+          });
262
+        } else if (event.property == 'relativeX' ||
263
+            event.property == 'relativeY') {
264
+          Blockly.Realtime.doWithinSync_(function() {
265
+            if (!block.svg_) {
266
+              // If this is a move of a newly disconnected (i.e. newly top
267
+              // level) block it will not have any svg (because it has been
268
+              // disposed of by its parent), so we need to handle that here.
269
+              Blockly.Realtime.placeBlockOnWorkspace_(block, false);
270
+            }
271
+            Blockly.Realtime.moveBlock_(block);
272
+          });
273
+        }
274
+      }
275
+    }
276
+  }
277
+};
278
+
279
+/**
280
+ * Event handler to call when there is a change to the realtime blocks map.
281
+ * @param {!gapi.drive.realtime.ValueChangedEvent} evt The event that occurred.
282
+ * @private
283
+ */
284
+Blockly.Realtime.onBlocksMapChange_ = function(evt) {
285
+  if (!evt.isLocal || Blockly.Realtime.withinUndo_) {
286
+    var block = evt.newValue;
287
+    if (block) {
288
+      Blockly.Realtime.placeBlockOnWorkspace_(block, !(evt.oldValue));
289
+    } else {
290
+      block = evt.oldValue;
291
+      Blockly.Realtime.deleteBlock(block);
292
+    }
293
+  }
294
+};
295
+
296
+/**
297
+ * A convenient wrapper around code that synchronizes the local model being
298
+ * edited with changes from another non-local model.
299
+ * @param {!function()} thunk A thunk of code to call.
300
+ * @private
301
+ */
302
+Blockly.Realtime.doWithinSync_ = function(thunk) {
303
+  if (Blockly.Realtime.withinSync) {
304
+    thunk();
305
+  } else {
306
+    try {
307
+      Blockly.Realtime.withinSync = true;
308
+      thunk();
309
+    } finally {
310
+      Blockly.Realtime.withinSync = false;
311
+    }
312
+  }
313
+};
314
+
315
+/**
316
+ * Places a block to be synced on this docs main workspace.  The block might
317
+ * already exist on this doc, in which case it is updated and/or moved.
318
+ * @param {!Blockly.Block} block The block.
319
+ * @param {boolean} addToTop Whether to add the block to the workspace/s list of
320
+ *     top-level blocks.
321
+ * @private
322
+ */
323
+Blockly.Realtime.placeBlockOnWorkspace_ = function(block, addToTop) {
324
+  Blockly.Realtime.doWithinSync_(function() {
325
+//    if (!Blockly.Realtime.blocksMap_.has(block.id)) {
326
+//      Blockly.Realtime.blocksMap_.set(block.id, block);
327
+//    }
328
+    var blockDom = Blockly.Xml.textToDom(block.xmlDom).firstChild;
329
+    var newBlock =
330
+        Blockly.Xml.domToBlock(Blockly.mainWorkspace, blockDom, true);
331
+    // TODO: The following is for debugging.  It should never actually happen.
332
+    if (!newBlock) {
333
+      return;
334
+    }
335
+    // Since Blockly.Xml.blockDomToBlock() purposely won't add blocks to
336
+    // workspace.topBlocks_ we sometimes need to do it explicitly here.
337
+    if (addToTop) {
338
+      newBlock.workspace.addTopBlock(newBlock);
339
+    }
340
+    if (addToTop ||
341
+        goog.array.contains(Blockly.Realtime.topBlocks_, newBlock)) {
342
+      Blockly.Realtime.moveBlock_(newBlock);
343
+    }
344
+  });
345
+};
346
+
347
+/**
348
+ * Move a block.
349
+ * @param {!Blockly.Block} block The block to move.
350
+ * @private
351
+ */
352
+Blockly.Realtime.moveBlock_ = function(block) {
353
+  if (!isNaN(block.relativeX) && !isNaN(block.relativeY)) {
354
+    var width = Blockly.svgSize().width;
355
+    var curPos = block.getRelativeToSurfaceXY();
356
+    var dx = block.relativeX - curPos.x;
357
+    var dy = block.relativeY - curPos.y;
358
+    block.moveBy(Blockly.RTL ? width - dx : dx, dy);
359
+  }
360
+};
361
+
362
+/**
363
+ * Delete a block.
364
+ * @param {!Blockly.Block} block The block to delete.
365
+ */
366
+Blockly.Realtime.deleteBlock = function(block) {
367
+  Blockly.Realtime.doWithinSync_(function() {
368
+    block.dispose(true, true, true);
369
+  });
370
+};
371
+
372
+/**
373
+ * Load all the blocks from the realtime model's blocks map and place them
374
+ * appropriately on the main Blockly workspace.
375
+ * @private
376
+ */
377
+Blockly.Realtime.loadBlocks_ = function() {
378
+  var topBlocks = Blockly.Realtime.topBlocks_;
379
+  for (var j = 0; j < topBlocks.length; j++) {
380
+    var topBlock = topBlocks.get(j);
381
+    Blockly.Realtime.placeBlockOnWorkspace_(topBlock, true);
382
+  }
383
+};
384
+
385
+/**
386
+ * Cause a changed block to update the realtime model, and therefore to be
387
+ * synced with other apps editing this same doc.
388
+ * @param {!Blockly.Block} block The block that changed.
389
+ */
390
+Blockly.Realtime.blockChanged = function(block) {
391
+  if (block.workspace == Blockly.mainWorkspace &&
392
+      Blockly.Realtime.isEnabled() &&
393
+      !Blockly.Realtime.withinSync) {
394
+    var rootBlock = block.getRootBlock();
395
+    var xy = rootBlock.getRelativeToSurfaceXY();
396
+    var changed = false;
397
+    var xml = Blockly.Xml.blockToDom_(rootBlock);
398
+    xml.setAttribute('id', rootBlock.id);
399
+    var topXml = goog.dom.createDom('xml');
400
+    topXml.appendChild(xml);
401
+    var newXml = Blockly.Xml.domToText(topXml);
402
+    if (newXml != rootBlock.xmlDom) {
403
+      changed = true;
404
+      rootBlock.xmlDom = newXml;
405
+    }
406
+    if (rootBlock.relativeX != xy.x || rootBlock.relativeY != xy.y) {
407
+      rootBlock.relativeX = xy.x;
408
+      rootBlock.relativeY = xy.y;
409
+      changed = true;
410
+    }
411
+    if (changed) {
412
+      var blockId = rootBlock.id.toString();
413
+      Blockly.Realtime.blocksMap_.set(blockId, rootBlock);
414
+    }
415
+  }
416
+};
417
+
418
+/**
419
+ * This function is called when the Realtime file has been loaded. It should
420
+ * be used to initialize any user interface components and event handlers
421
+ * depending on the Realtime model. In this case, create a text control binder
422
+ * and bind it to our string model that we created in initializeModel.
423
+ * @param {!gapi.drive.realtime.Document} doc The Realtime document.
424
+ * @private
425
+ */
426
+Blockly.Realtime.onFileLoaded_ = function(doc) {
427
+  Blockly.Realtime.document_ = doc;
428
+  Blockly.Realtime.sessionId_ = Blockly.Realtime.getSessionId_(doc);
429
+  Blockly.Realtime.model_ = doc.getModel();
430
+  Blockly.Realtime.blocksMap_ =
431
+      Blockly.Realtime.model_.getRoot().get('blocks');
432
+  Blockly.Realtime.topBlocks_ =
433
+      Blockly.Realtime.model_.getRoot().get('topBlocks');
434
+
435
+  Blockly.Realtime.model_.getRoot().addEventListener(
436
+      gapi.drive.realtime.EventType.OBJECT_CHANGED,
437
+      Blockly.Realtime.onObjectChange_);
438
+  Blockly.Realtime.blocksMap_.addEventListener(
439
+      gapi.drive.realtime.EventType.VALUE_CHANGED,
440
+      Blockly.Realtime.onBlocksMapChange_);
441
+
442
+  Blockly.Realtime.initUi_();
443
+
444
+  //Adding Listeners for Collaborator events.
445
+  doc.addEventListener(gapi.drive.realtime.EventType.COLLABORATOR_JOINED,
446
+      Blockly.Realtime.onCollaboratorJoined_);
447
+  doc.addEventListener(gapi.drive.realtime.EventType.COLLABORATOR_LEFT,
448
+      Blockly.Realtime.onCollaboratorLeft_);
449
+  Blockly.Realtime.updateCollabUi_();
450
+
451
+  Blockly.Realtime.loadBlocks_();
452
+
453
+  // Add logic for undo button.
454
+  // TODO: Uncomment this when undo/redo are fixed.
455
+//
456
+//  var undoButton = document.getElementById(Blockly.Realtime.undoElementId_);
457
+//  var redoButton = document.getElementById(Blockly.Realtime.redoElementId_);
458
+//
459
+//  if (undoButton) {
460
+//    undoButton.onclick = function (e) {
461
+//      try {
462
+//        Blockly.Realtime.withinUndo_ = true;
463
+//        Blockly.Realtime.model_.undo();
464
+//      } finally {
465
+//        Blockly.Realtime.withinUndo_ = false;
466
+//      }
467
+//    };
468
+//  }
469
+//  if (redoButton) {
470
+//    redoButton.onclick = function (e) {
471
+//      try {
472
+//        Blockly.Realtime.withinUndo_ = true;
473
+//        Blockly.Realtime.model_.redo();
474
+//      } finally {
475
+//        Blockly.Realtime.withinUndo_ = false;
476
+//      }
477
+//    };
478
+//  }
479
+//
480
+//  // Add event handler for UndoRedoStateChanged events.
481
+//  var onUndoRedoStateChanged = function(e) {
482
+//    undoButton.disabled = !e.canUndo;
483
+//    redoButton.disabled = !e.canRedo;
484
+//  };
485
+//  Blockly.Realtime.model_.addEventListener(
486
+//      gapi.drive.realtime.EventType.UNDO_REDO_STATE_CHANGED,
487
+//      onUndoRedoStateChanged);
488
+
489
+};
490
+
491
+/**
492
+ * Get the sessionId associated with this editing session.  Note that it is
493
+ * unique to the current browser window/tab.
494
+ * @param {gapi.drive.realtime.Document} doc
495
+ * @return {*}
496
+ * @private
497
+ */
498
+Blockly.Realtime.getSessionId_ = function(doc) {
499
+  var collaborators = doc.getCollaborators();
500
+  for (var i = 0; i < collaborators.length; i++) {
501
+    var collaborator = collaborators[i];
502
+    if (collaborator.isMe) {
503
+      return collaborator.sessionId;
504
+    }
505
+  }
506
+  return undefined;
507
+};
508
+
509
+/**
510
+ * Register the Blockly types and attributes that are reflected in the realtime
511
+ * model.
512
+ * @private
513
+ */
514
+Blockly.Realtime.registerTypes_ = function() {
515
+  var custom = gapi.drive.realtime.custom;
516
+
517
+  custom.registerType(Blockly.Block, 'Block');
518
+  Blockly.Block.prototype.id = custom.collaborativeField('id');
519
+  Blockly.Block.prototype.xmlDom = custom.collaborativeField('xmlDom');
520
+  Blockly.Block.prototype.relativeX = custom.collaborativeField('relativeX');
521
+  Blockly.Block.prototype.relativeY = custom.collaborativeField('relativeY');
522
+
523
+  custom.setInitializer(Blockly.Block, Blockly.Block.prototype.initialize);
524
+};
525
+
526
+/**
527
+ * Time period for realtime re-authorization
528
+ * @type {number}
529
+ * @private
530
+ */
531
+Blockly.Realtime.REAUTH_INTERVAL_IN_MILLISECONDS_ = 30 * 60 * 1000;
532
+
533
+/**
534
+ * What to do after Realtime authorization.
535
+ * @private
536
+ */
537
+Blockly.Realtime.afterAuth_ = function() {
538
+  // This is a workaround for the fact that the code in realtime-client-utils.js
539
+  // doesn't deal with auth timeouts correctly.  So we explicitly reauthorize at
540
+  // regular intervals.
541
+  setTimeout(
542
+      function() {
543
+        Blockly.Realtime.realtimeLoader_.authorizer.authorize(
544
+            Blockly.Realtime.afterAuth_);
545
+      },
546
+      Blockly.Realtime.REAUTH_INTERVAL_IN_MILLISECONDS_);
547
+};
548
+
549
+/**
550
+ * Add "Anyone with the link" permissions to the file.
551
+ * @param {string} fileId the file id
552
+ * @private
553
+ */
554
+Blockly.Realtime.afterCreate_ = function(fileId) {
555
+  var resource = {
556
+    'type': 'anyone',
557
+    'role': 'writer',
558
+    'value': 'default',
559
+    'withLink': true
560
+  };
561
+  var request = gapi.client.drive.permissions.insert({
562
+    'fileId': fileId,
563
+    'resource': resource
564
+  });
565
+  request.execute(function(resp) {
566
+    // If we have an error try to just set the permission for all users
567
+    // of the domain.
568
+    if (resp.error) {
569
+      Blockly.Realtime.getUserDomain(fileId, function(domain) {
570
+        var resource = {
571
+          'type': 'domain',
572
+          'role': 'writer',
573
+          'value': domain,
574
+          'withLink': true
575
+        };
576
+        request = gapi.client.drive.permissions.insert({
577
+          'fileId': fileId,
578
+          'resource': resource
579
+        });
580
+        request.execute(function(resp) { });
581
+      });
582
+    }
583
+  });
584
+};
585
+
586
+/**
587
+ * Get the domain (if it exists) associated with a realtime file.  The callback
588
+ * will be called with the domain, if it exists.
589
+ * @param {string} fileId the id of the file
590
+ * @param {function(string)} callback a function to call back with the domain
591
+ */
592
+Blockly.Realtime.getUserDomain = function(fileId, callback) {
593
+  /**
594
+   * Note that there may be a more direct way to get the domain by, for example,
595
+   * using the Google profile API but this way we don't need any additional
596
+   * APIs or scopes.  But if it turns out that the permissions API stops
597
+   * providing the domain this might have to change.
598
+   */
599
+  var request = gapi.client.drive.permissions.list({
600
+    'fileId': fileId
601
+  });
602
+  request.execute(function(resp) {
603
+    for (var i = 0; i < resp.items.length; i++) {
604
+      var item = resp.items[i];
605
+      if (item.role == 'owner') {
606
+        callback(item.domain);
607
+        return;
608
+      }
609
+    }
610
+  });
611
+};
612
+
613
+/**
614
+ * Options for the Realtime loader.
615
+ * @private
616
+ */
617
+Blockly.Realtime.rtclientOptions_ = {
618
+  /**
619
+   * Client ID from the console.
620
+   * This will be set from the options passed into Blockly.Realtime.start()
621
+   */
622
+  'clientId': null,
623
+
624
+  /**
625
+   * The ID of the button to click to authorize. Must be a DOM element ID.
626
+   */
627
+  'authButtonElementId': 'authorizeButton',
628
+
629
+  /**
630
+   * The ID of the container of the authorize button.
631
+   */
632
+  'authDivElementId': 'authButtonDiv',
633
+
634
+  /**
635
+   * Function to be called when a Realtime model is first created.
636
+   */
637
+  'initializeModel': Blockly.Realtime.initializeModel_,
638
+
639
+  /**
640
+   * Autocreate files right after auth automatically.
641
+   */
642
+  'autoCreate': true,
643
+
644
+  /**
645
+   * The name of newly created Drive files.
646
+   */
647
+  'defaultTitle': 'Realtime Blockly File',
648
+
649
+  /**
650
+   * The name of the folder to place newly created Drive files in.
651
+   */
652
+  'defaultFolderTitle': 'Realtime Blockly Folder',
653
+
654
+  /**
655
+   * The MIME type of newly created Drive Files. By default the application
656
+   * specific MIME type will be used:
657
+   *     application/vnd.google-apps.drive-sdk.
658
+   */
659
+  'newFileMimeType': null, // Using default.
660
+
661
+  /**
662
+   * Function to be called every time a Realtime file is loaded.
663
+   */
664
+  'onFileLoaded': Blockly.Realtime.onFileLoaded_,
665
+
666
+  /**
667
+   * Function to be called to initialize custom Collaborative Objects types.
668
+   */
669
+  'registerTypes': Blockly.Realtime.registerTypes_,
670
+
671
+  /**
672
+   * Function to be called after authorization and before loading files.
673
+   */
674
+  'afterAuth': Blockly.Realtime.afterAuth_,
675
+
676
+  /**
677
+   * Function to be called after file creation, if autoCreate is true.
678
+   */
679
+  'afterCreate': Blockly.Realtime.afterCreate_
680
+};
681
+
682
+/**
683
+ * Parse options to startRealtime().
684
+ * @param {!Object} options Object containing the options.
685
+ * @private
686
+ */
687
+Blockly.Realtime.parseOptions_ = function(options) {
688
+  var chatBoxOptions = rtclient.getOption(options, 'chatbox');
689
+  if (chatBoxOptions) {
690
+    Blockly.Realtime.chatBoxElementId_ =
691
+        rtclient.getOption(chatBoxOptions, 'elementId');
692
+    Blockly.Realtime.chatBoxInitialText_ =
693
+        rtclient.getOption(chatBoxOptions, 'initText', Blockly.Msg.CHAT);
694
+  }
695
+  Blockly.Realtime.rtclientOptions_.clientId =
696
+      rtclient.getOption(options, 'clientId');
697
+  Blockly.Realtime.collabElementId =
698
+      rtclient.getOption(options, 'collabElementId');
699
+  // TODO: Uncomment this when undo/redo are fixed.
700
+//  Blockly.Realtime.undoElementId_ =
701
+//      rtclient.getOption(options, 'undoElementId', 'undoButton');
702
+//  Blockly.Realtime.redoElementId_ =
703
+//      rtclient.getOption(options, 'redoElementId', 'redoButton');
704
+};
705
+
706
+/**
707
+ * Setup the Blockly container for realtime authorization and start the
708
+ * Realtime loader.
709
+ * @param {function()} uiInitialize Function to initialize the Blockly UI.
710
+ * @param {!Element} uiContainer Container element for the Blockly UI.
711
+ * @param {!Object} options The realtime options.
712
+ */
713
+Blockly.Realtime.startRealtime = function(uiInitialize, uiContainer, options) {
714
+  Blockly.Realtime.parseOptions_(options);
715
+  Blockly.Realtime.enabled_ = true;
716
+  // Note that we need to setup the UI for realtime authorization before
717
+  // loading the realtime code (which, in turn, will handle initializing the
718
+  // rest of the Blockly UI).
719
+  var authDiv = Blockly.Realtime.addAuthUi_(uiContainer);
720
+  Blockly.Realtime.initUi_ = function() {
721
+    uiInitialize();
722
+    if (Blockly.Realtime.chatBoxElementId_) {
723
+      var chatText = Blockly.Realtime.model_.getRoot().get(
724
+          Blockly.Realtime.chatBoxElementId_);
725
+      var chatBox = document.getElementById(Blockly.Realtime.chatBoxElementId_);
726
+      gapi.drive.realtime.databinding.bindString(chatText, chatBox);
727
+      chatBox.disabled = false;
728
+    }
729
+  };
730
+  Blockly.Realtime.realtimeLoader_ =
731
+      new rtclient.RealtimeLoader(Blockly.Realtime.rtclientOptions_);
732
+  Blockly.Realtime.realtimeLoader_.start();
733
+};
734
+
735
+/**
736
+ * Setup the Blockly container for realtime authorization.
737
+ * @param {!Element} uiContainer A DOM container element for the Blockly UI.
738
+ * @return {!Element} The DOM element for the authorization UI.
739
+ * @private
740
+ */
741
+Blockly.Realtime.addAuthUi_ = function(uiContainer) {
742
+  // Add progress indicator to the UI container.
743
+  uiContainer.style.background = 'url(' + Blockly.pathToMedia +
744
+      Blockly.Realtime.PROGRESS_URL_ + ') no-repeat center center';
745
+  // Setup authorization button
746
+  var blocklyDivBounds = goog.style.getBounds(uiContainer);
747
+  var authButtonDiv = goog.dom.createDom('div');
748
+  authButtonDiv.id = Blockly.Realtime.rtclientOptions_['authDivElementId'];
749
+  var authText = goog.dom.createDom('p', null, Blockly.Msg.AUTH);
750
+  authButtonDiv.appendChild(authText);
751
+  var authButton = goog.dom.createDom('button', null, 'Authorize');
752
+  authButton.id = Blockly.Realtime.rtclientOptions_.authButtonElementId;
753
+  authButtonDiv.appendChild(authButton);
754
+  uiContainer.appendChild(authButtonDiv);
755
+
756
+  // TODO: I would have liked to set the style for the authButtonDiv in css.js
757
+  // but that CSS doesn't get injected until after this code gets run.
758
+  authButtonDiv.style.display = 'none';
759
+  authButtonDiv.style.position = 'relative';
760
+  authButtonDiv.style.textAlign = 'center';
761
+  authButtonDiv.style.border = '1px solid';
762
+  authButtonDiv.style.backgroundColor = '#f6f9ff';
763
+  authButtonDiv.style.borderRadius = '15px';
764
+  authButtonDiv.style.boxShadow = '10px 10px 5px #888';
765
+  authButtonDiv.style.width = (blocklyDivBounds.width / 3) + 'px';
766
+  var authButtonDivBounds = goog.style.getBounds(authButtonDiv);
767
+  authButtonDiv.style.left =
768
+      (blocklyDivBounds.width - authButtonDivBounds.width) / 3 + 'px';
769
+  authButtonDiv.style.top =
770
+      (blocklyDivBounds.height - authButtonDivBounds.height) / 4 + 'px';
771
+  return authButtonDiv;
772
+};
773
+
774
+/**
775
+ * Update the collaborators UI to include the latest set of users.
776
+ * @private
777
+ */
778
+Blockly.Realtime.updateCollabUi_ = function() {
779
+  if (!Blockly.Realtime.collabElementId) {
780
+    return;
781
+  }
782
+  var collabElement = goog.dom.getElement(Blockly.Realtime.collabElementId);
783
+  goog.dom.removeChildren(collabElement);
784
+  var collaboratorsList = Blockly.Realtime.document_.getCollaborators();
785
+  for (var i = 0; i < collaboratorsList.length; i++) {
786
+    var collaborator = collaboratorsList[i];
787
+    var imgSrc = collaborator.photoUrl ||
788
+        Blockly.pathToMedia + Blockly.Realtime.ANONYMOUS_URL_;
789
+    var img = goog.dom.createDom('img',
790
+        {
791
+          'src': imgSrc,
792
+          'alt': collaborator.displayName,
793
+          'title': collaborator.displayName +
794
+              (collaborator.isMe ? ' (' + Blockly.Msg.ME + ')' : '')});
795
+    img.style.backgroundColor = collaborator.color;
796
+    goog.dom.appendChild(collabElement, img);
797
+  }
798
+};
799
+
800
+/**
801
+ * Event handler for when collaborators join.
802
+ * @param {gapi.drive.realtime.CollaboratorJoinedEvent} event The event.
803
+ * @private
804
+ */
805
+Blockly.Realtime.onCollaboratorJoined_ = function(event) {
806
+  Blockly.Realtime.updateCollabUi_();
807
+};
808
+
809
+/**
810
+ * Event handler for when collaborators leave.
811
+ * @param {gapi.drive.realtime.CollaboratorLeftEvent} event The event.
812
+ * @private
813
+ */
814
+Blockly.Realtime.onCollaboratorLeft_ = function(event) {
815
+  Blockly.Realtime.updateCollabUi_();
816
+};
817
+
818
+/**
819
+ * Execute a command.  Generally, a command is the result of a user action
820
+ * e.g. a click, drag or context menu selection.
821
+ * @param {function()} cmdThunk A function representing the command execution.
822
+ */
823
+Blockly.Realtime.doCommand = function(cmdThunk) {
824
+  // TODO(): We'd like to use the realtime API compound operations as in the
825
+  // commented out code below.  However, it appears that the realtime API is
826
+  // re-ordering events when they're within compound operations in a way which
827
+  // breaks us.  We might need to implement our own compound operations as a
828
+  // workaround.  Doing so might give us some other advantages since we could
829
+  // then allow compound operations that span synchronous blocks of code (e.g.,
830
+  // span multiple Blockly events).  It would also allow us to deal with the
831
+  // fact that the current realtime API puts some operations into the undo stack
832
+  // that we would prefer weren't there; namely local changes that occur as a
833
+  // result of remote realtime events.
834
+//  try {
835
+//    Blockly.Realtime.model_.beginCompoundOperation();
836
+//    cmdThunk();
837
+//  } finally {
838
+//    Blockly.Realtime.model_.endCompoundOperation();
839
+//  }
840
+  cmdThunk();
841
+};
842
+
843
+/**
844
+ * Generate an id that is unique among the all the sessions that ever
845
+ * collaborated on this document.
846
+ * @param {string} extra A string id which is unique within this particular
847
+ * session.
848
+ * @return {string}
849
+ */
850
+Blockly.Realtime.genUid = function(extra) {
851
+  /* The idea here is that we use the extra string to ensure uniqueness within
852
+     this session and the current sessionId to ensure uniqueness across
853
+     all the current sessions.  There's still the (remote) chance that the
854
+     current sessionId is the same as some old (non-current) one, so we still
855
+     need to check that our uid hasn't been previously used.
856
+
857
+     Note that you could potentially use a random number to generate the id but
858
+     there remains the small chance of regenerating the same number that's been
859
+     used before and I'm paranoid.  It's not enough to just check that the
860
+     random uid hasn't been previously used because other concurrent sessions
861
+     might generate the same uid at the same time.  Like I said, I'm paranoid.
862
+   */
863
+  var potentialUid = Blockly.Realtime.sessionId_ + '-' + extra;
864
+  if (!Blockly.Realtime.blocksMap_.has(potentialUid)) {
865
+    return potentialUid;
866
+  } else {
867
+    return (Blockly.Realtime.genUid('-' + extra));
868
+  }
869
+};

+ 523 - 0
src/blockly/core/scrollbar.js

@@ -0,0 +1,523 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2011 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Library for creating scrollbars.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Scrollbar');
28
+goog.provide('Blockly.ScrollbarPair');
29
+
30
+goog.require('goog.dom');
31
+goog.require('goog.events');
32
+
33
+
34
+/**
35
+ * Class for a pair of scrollbars.  Horizontal and vertical.
36
+ * @param {!Blockly.Workspace} workspace Workspace to bind the scrollbars to.
37
+ * @constructor
38
+ */
39
+Blockly.ScrollbarPair = function(workspace) {
40
+  this.workspace_ = workspace;
41
+  this.hScroll = new Blockly.Scrollbar(workspace, true, true);
42
+  this.vScroll = new Blockly.Scrollbar(workspace, false, true);
43
+  this.corner_ = Blockly.createSvgElement('rect',
44
+      {'height': Blockly.Scrollbar.scrollbarThickness,
45
+      'width': Blockly.Scrollbar.scrollbarThickness,
46
+      'class': 'blocklyScrollbarBackground'}, null);
47
+  Blockly.Scrollbar.insertAfter_(this.corner_, workspace.getBubbleCanvas());
48
+};
49
+
50
+/**
51
+ * Previously recorded metrics from the workspace.
52
+ * @type {Object}
53
+ * @private
54
+ */
55
+Blockly.ScrollbarPair.prototype.oldHostMetrics_ = null;
56
+
57
+/**
58
+ * Dispose of this pair of scrollbars.
59
+ * Unlink from all DOM elements to prevent memory leaks.
60
+ */
61
+Blockly.ScrollbarPair.prototype.dispose = function() {
62
+  goog.dom.removeNode(this.corner_);
63
+  this.corner_ = null;
64
+  this.workspace_ = null;
65
+  this.oldHostMetrics_ = null;
66
+  this.hScroll.dispose();
67
+  this.hScroll = null;
68
+  this.vScroll.dispose();
69
+  this.vScroll = null;
70
+};
71
+
72
+/**
73
+ * Recalculate both of the scrollbars' locations and lengths.
74
+ * Also reposition the corner rectangle.
75
+ */
76
+Blockly.ScrollbarPair.prototype.resize = function() {
77
+  // Look up the host metrics once, and use for both scrollbars.
78
+  var hostMetrics = this.workspace_.getMetrics();
79
+  if (!hostMetrics) {
80
+    // Host element is likely not visible.
81
+    return;
82
+  }
83
+
84
+  // Only change the scrollbars if there has been a change in metrics.
85
+  var resizeH = false;
86
+  var resizeV = false;
87
+  if (!this.oldHostMetrics_ ||
88
+      this.oldHostMetrics_.viewWidth != hostMetrics.viewWidth ||
89
+      this.oldHostMetrics_.viewHeight != hostMetrics.viewHeight ||
90
+      this.oldHostMetrics_.absoluteTop != hostMetrics.absoluteTop ||
91
+      this.oldHostMetrics_.absoluteLeft != hostMetrics.absoluteLeft) {
92
+    // The window has been resized or repositioned.
93
+    resizeH = true;
94
+    resizeV = true;
95
+  } else {
96
+    // Has the content been resized or moved?
97
+    if (!this.oldHostMetrics_ ||
98
+        this.oldHostMetrics_.contentWidth != hostMetrics.contentWidth ||
99
+        this.oldHostMetrics_.viewLeft != hostMetrics.viewLeft ||
100
+        this.oldHostMetrics_.contentLeft != hostMetrics.contentLeft) {
101
+      resizeH = true;
102
+    }
103
+    if (!this.oldHostMetrics_ ||
104
+        this.oldHostMetrics_.contentHeight != hostMetrics.contentHeight ||
105
+        this.oldHostMetrics_.viewTop != hostMetrics.viewTop ||
106
+        this.oldHostMetrics_.contentTop != hostMetrics.contentTop) {
107
+      resizeV = true;
108
+    }
109
+  }
110
+  if (resizeH) {
111
+    this.hScroll.resize(hostMetrics);
112
+  }
113
+  if (resizeV) {
114
+    this.vScroll.resize(hostMetrics);
115
+  }
116
+
117
+  // Reposition the corner square.
118
+  if (!this.oldHostMetrics_ ||
119
+      this.oldHostMetrics_.viewWidth != hostMetrics.viewWidth ||
120
+      this.oldHostMetrics_.absoluteLeft != hostMetrics.absoluteLeft) {
121
+    this.corner_.setAttribute('x', this.vScroll.xCoordinate);
122
+  }
123
+  if (!this.oldHostMetrics_ ||
124
+      this.oldHostMetrics_.viewHeight != hostMetrics.viewHeight ||
125
+      this.oldHostMetrics_.absoluteTop != hostMetrics.absoluteTop) {
126
+    this.corner_.setAttribute('y', this.hScroll.yCoordinate);
127
+  }
128
+
129
+  // Cache the current metrics to potentially short-cut the next resize event.
130
+  this.oldHostMetrics_ = hostMetrics;
131
+};
132
+
133
+/**
134
+ * Set the sliders of both scrollbars to be at a certain position.
135
+ * @param {number} x Horizontal scroll value.
136
+ * @param {number} y Vertical scroll value.
137
+ */
138
+Blockly.ScrollbarPair.prototype.set = function(x, y) {
139
+  this.hScroll.set(x);
140
+  this.vScroll.set(y);
141
+};
142
+
143
+// --------------------------------------------------------------------
144
+
145
+/**
146
+ * Class for a pure SVG scrollbar.
147
+ * This technique offers a scrollbar that is guaranteed to work, but may not
148
+ * look or behave like the system's scrollbars.
149
+ * @param {!Blockly.Workspace} workspace Workspace to bind the scrollbar to.
150
+ * @param {boolean} horizontal True if horizontal, false if vertical.
151
+ * @param {boolean=} opt_pair True if the scrollbar is part of a horiz/vert pair.
152
+ * @constructor
153
+ */
154
+Blockly.Scrollbar = function(workspace, horizontal, opt_pair) {
155
+  this.workspace_ = workspace;
156
+  this.pair_ = opt_pair || false;
157
+  this.horizontal_ = horizontal;
158
+
159
+  this.createDom_();
160
+
161
+  if (horizontal) {
162
+    this.svgBackground_.setAttribute('height',
163
+        Blockly.Scrollbar.scrollbarThickness);
164
+    this.svgKnob_.setAttribute('height',
165
+        Blockly.Scrollbar.scrollbarThickness - 5);
166
+    this.svgKnob_.setAttribute('y', 2.5);
167
+  } else {
168
+    this.svgBackground_.setAttribute('width',
169
+        Blockly.Scrollbar.scrollbarThickness);
170
+    this.svgKnob_.setAttribute('width',
171
+        Blockly.Scrollbar.scrollbarThickness - 5);
172
+    this.svgKnob_.setAttribute('x', 2.5);
173
+  }
174
+  var scrollbar = this;
175
+  this.onMouseDownBarWrapper_ = Blockly.bindEvent_(this.svgBackground_,
176
+      'mousedown', scrollbar, scrollbar.onMouseDownBar_);
177
+  this.onMouseDownKnobWrapper_ = Blockly.bindEvent_(this.svgKnob_,
178
+      'mousedown', scrollbar, scrollbar.onMouseDownKnob_);
179
+};
180
+
181
+/**
182
+ * Width of vertical scrollbar or height of horizontal scrollbar.
183
+ * Increase the size of scrollbars on touch devices.
184
+ * Don't define if there is no document object (e.g. node.js).
185
+ */
186
+Blockly.Scrollbar.scrollbarThickness = 15;
187
+if (goog.events.BrowserFeature.TOUCH_ENABLED) {
188
+  Blockly.Scrollbar.scrollbarThickness = 25;
189
+}
190
+
191
+/**
192
+ * Dispose of this scrollbar.
193
+ * Unlink from all DOM elements to prevent memory leaks.
194
+ */
195
+Blockly.Scrollbar.prototype.dispose = function() {
196
+  this.onMouseUpKnob_();
197
+  Blockly.unbindEvent_(this.onMouseDownBarWrapper_);
198
+  this.onMouseDownBarWrapper_ = null;
199
+  Blockly.unbindEvent_(this.onMouseDownKnobWrapper_);
200
+  this.onMouseDownKnobWrapper_ = null;
201
+
202
+  goog.dom.removeNode(this.svgGroup_);
203
+  this.svgGroup_ = null;
204
+  this.svgBackground_ = null;
205
+  this.svgKnob_ = null;
206
+  this.workspace_ = null;
207
+};
208
+
209
+/**
210
+ * Recalculate the scrollbar's location and its length.
211
+ * @param {Object=} opt_metrics A data structure of from the describing all the
212
+ * required dimensions.  If not provided, it will be fetched from the host
213
+ * object.
214
+ */
215
+Blockly.Scrollbar.prototype.resize = function(opt_metrics) {
216
+  // Determine the location, height and width of the host element.
217
+  var hostMetrics = opt_metrics;
218
+  if (!hostMetrics) {
219
+    hostMetrics = this.workspace_.getMetrics();
220
+    if (!hostMetrics) {
221
+      // Host element is likely not visible.
222
+      return;
223
+    }
224
+  }
225
+  /* hostMetrics is an object with the following properties.
226
+   * .viewHeight: Height of the visible rectangle,
227
+   * .viewWidth: Width of the visible rectangle,
228
+   * .contentHeight: Height of the contents,
229
+   * .contentWidth: Width of the content,
230
+   * .viewTop: Offset of top edge of visible rectangle from parent,
231
+   * .viewLeft: Offset of left edge of visible rectangle from parent,
232
+   * .contentTop: Offset of the top-most content from the y=0 coordinate,
233
+   * .contentLeft: Offset of the left-most content from the x=0 coordinate,
234
+   * .absoluteTop: Top-edge of view.
235
+   * .absoluteLeft: Left-edge of view.
236
+   */
237
+  if (this.horizontal_) {
238
+    var outerLength = hostMetrics.viewWidth - 1;
239
+    if (this.pair_) {
240
+      // Shorten the scrollbar to make room for the corner square.
241
+      outerLength -= Blockly.Scrollbar.scrollbarThickness;
242
+    } else {
243
+      // Only show the scrollbar if needed.
244
+      // Ideally this would also apply to scrollbar pairs, but that's a bigger
245
+      // headache (due to interactions with the corner square).
246
+      this.setVisible(outerLength < hostMetrics.contentHeight);
247
+    }
248
+    this.ratio_ = outerLength / hostMetrics.contentWidth;
249
+    if (this.ratio_ === -Infinity || this.ratio_ === Infinity ||
250
+        isNaN(this.ratio_)) {
251
+      this.ratio_ = 0;
252
+    }
253
+    var innerLength = hostMetrics.viewWidth * this.ratio_;
254
+    var innerOffset = (hostMetrics.viewLeft - hostMetrics.contentLeft) *
255
+        this.ratio_;
256
+    this.svgKnob_.setAttribute('width', Math.max(0, innerLength));
257
+    this.xCoordinate = hostMetrics.absoluteLeft + 0.5;
258
+    if (this.pair_ && this.workspace_.RTL) {
259
+      this.xCoordinate += hostMetrics.absoluteLeft +
260
+          Blockly.Scrollbar.scrollbarThickness;
261
+    }
262
+    this.yCoordinate = hostMetrics.absoluteTop + hostMetrics.viewHeight -
263
+        Blockly.Scrollbar.scrollbarThickness - 0.5;
264
+    this.svgGroup_.setAttribute('transform',
265
+        'translate(' + this.xCoordinate + ',' + this.yCoordinate + ')');
266
+    this.svgBackground_.setAttribute('width', Math.max(0, outerLength));
267
+    this.svgKnob_.setAttribute('x', this.constrainKnob_(innerOffset));
268
+  } else {
269
+    var outerLength = hostMetrics.viewHeight - 1;
270
+    if (this.pair_) {
271
+      // Shorten the scrollbar to make room for the corner square.
272
+      outerLength -= Blockly.Scrollbar.scrollbarThickness;
273
+    } else {
274
+      // Only show the scrollbar if needed.
275
+      this.setVisible(outerLength < hostMetrics.contentHeight);
276
+    }
277
+    this.ratio_ = outerLength / hostMetrics.contentHeight;
278
+    if (this.ratio_ === -Infinity || this.ratio_ === Infinity ||
279
+        isNaN(this.ratio_)) {
280
+      this.ratio_ = 0;
281
+    }
282
+    var innerLength = hostMetrics.viewHeight * this.ratio_;
283
+    var innerOffset = (hostMetrics.viewTop - hostMetrics.contentTop) *
284
+        this.ratio_;
285
+    this.svgKnob_.setAttribute('height', Math.max(0, innerLength));
286
+    this.xCoordinate = hostMetrics.absoluteLeft + 0.5;
287
+    if (!this.workspace_.RTL) {
288
+      this.xCoordinate += hostMetrics.viewWidth -
289
+          Blockly.Scrollbar.scrollbarThickness - 1;
290
+    }
291
+    this.yCoordinate = hostMetrics.absoluteTop + 0.5;
292
+    this.svgGroup_.setAttribute('transform',
293
+        'translate(' + this.xCoordinate + ',' + this.yCoordinate + ')');
294
+    this.svgBackground_.setAttribute('height', Math.max(0, outerLength));
295
+    this.svgKnob_.setAttribute('y', this.constrainKnob_(innerOffset));
296
+  }
297
+  // Resizing may have caused some scrolling.
298
+  this.onScroll_();
299
+};
300
+
301
+/**
302
+ * Create all the DOM elements required for a scrollbar.
303
+ * The resulting widget is not sized.
304
+ * @private
305
+ */
306
+Blockly.Scrollbar.prototype.createDom_ = function() {
307
+  /* Create the following DOM:
308
+  <g class="blocklyScrollbarHorizontal">
309
+    <rect class="blocklyScrollbarBackground" />
310
+    <rect class="blocklyScrollbarKnob" rx="8" ry="8" />
311
+  </g>
312
+  */
313
+  var className = 'blocklyScrollbar' +
314
+      (this.horizontal_ ? 'Horizontal' : 'Vertical');
315
+  this.svgGroup_ = Blockly.createSvgElement('g', {'class': className}, null);
316
+  this.svgBackground_ = Blockly.createSvgElement('rect',
317
+      {'class': 'blocklyScrollbarBackground'}, this.svgGroup_);
318
+  var radius = Math.floor((Blockly.Scrollbar.scrollbarThickness - 5) / 2);
319
+  this.svgKnob_ = Blockly.createSvgElement('rect',
320
+      {'class': 'blocklyScrollbarKnob', 'rx': radius, 'ry': radius},
321
+      this.svgGroup_);
322
+  Blockly.Scrollbar.insertAfter_(this.svgGroup_,
323
+                                 this.workspace_.getBubbleCanvas());
324
+};
325
+
326
+/**
327
+ * Is the scrollbar visible.  Non-paired scrollbars disappear when they aren't
328
+ * needed.
329
+ * @return {boolean} True if visible.
330
+ */
331
+Blockly.Scrollbar.prototype.isVisible = function() {
332
+  return this.svgGroup_.getAttribute('display') != 'none';
333
+};
334
+
335
+/**
336
+ * Set whether the scrollbar is visible.
337
+ * Only applies to non-paired scrollbars.
338
+ * @param {boolean} visible True if visible.
339
+ */
340
+Blockly.Scrollbar.prototype.setVisible = function(visible) {
341
+  if (visible == this.isVisible()) {
342
+    return;
343
+  }
344
+  // Ideally this would also apply to scrollbar pairs, but that's a bigger
345
+  // headache (due to interactions with the corner square).
346
+  if (this.pair_) {
347
+    throw 'Unable to toggle visibility of paired scrollbars.';
348
+  }
349
+  if (visible) {
350
+    this.svgGroup_.setAttribute('display', 'block');
351
+  } else {
352
+    // Hide the scrollbar.
353
+    this.workspace_.setMetrics({x: 0, y: 0});
354
+    this.svgGroup_.setAttribute('display', 'none');
355
+  }
356
+};
357
+
358
+/**
359
+ * Scroll by one pageful.
360
+ * Called when scrollbar background is clicked.
361
+ * @param {!Event} e Mouse down event.
362
+ * @private
363
+ */
364
+Blockly.Scrollbar.prototype.onMouseDownBar_ = function(e) {
365
+  this.onMouseUpKnob_();
366
+  if (Blockly.isRightButton(e)) {
367
+    // Right-click.
368
+    // Scrollbars have no context menu.
369
+    e.stopPropagation();
370
+    return;
371
+  }
372
+  var mouseXY = Blockly.mouseToSvg(e, this.workspace_.options.svg);
373
+  var mouseLocation = this.horizontal_ ? mouseXY.x : mouseXY.y;
374
+
375
+  var knobXY = Blockly.getSvgXY_(this.svgKnob_, this.workspace_);
376
+  var knobStart = this.horizontal_ ? knobXY.x : knobXY.y;
377
+  var knobLength = parseFloat(
378
+      this.svgKnob_.getAttribute(this.horizontal_ ? 'width' : 'height'));
379
+  var knobValue = parseFloat(
380
+      this.svgKnob_.getAttribute(this.horizontal_ ? 'x' : 'y'));
381
+
382
+  var pageLength = knobLength * 0.95;
383
+  if (mouseLocation <= knobStart) {
384
+    // Decrease the scrollbar's value by a page.
385
+    knobValue -= pageLength;
386
+  } else if (mouseLocation >= knobStart + knobLength) {
387
+    // Increase the scrollbar's value by a page.
388
+    knobValue += pageLength;
389
+  }
390
+  this.svgKnob_.setAttribute(this.horizontal_ ? 'x' : 'y',
391
+                             this.constrainKnob_(knobValue));
392
+  this.onScroll_();
393
+  e.stopPropagation();
394
+};
395
+
396
+/**
397
+ * Start a dragging operation.
398
+ * Called when scrollbar knob is clicked.
399
+ * @param {!Event} e Mouse down event.
400
+ * @private
401
+ */
402
+Blockly.Scrollbar.prototype.onMouseDownKnob_ = function(e) {
403
+  this.onMouseUpKnob_();
404
+  if (Blockly.isRightButton(e)) {
405
+    // Right-click.
406
+    // Scrollbars have no context menu.
407
+    e.stopPropagation();
408
+    return;
409
+  }
410
+  // Look up the current translation and record it.
411
+  this.startDragKnob = parseFloat(
412
+      this.svgKnob_.getAttribute(this.horizontal_ ? 'x' : 'y'));
413
+  // Record the current mouse position.
414
+  this.startDragMouse = this.horizontal_ ? e.clientX : e.clientY;
415
+  Blockly.Scrollbar.onMouseUpWrapper_ = Blockly.bindEvent_(document,
416
+      'mouseup', this, this.onMouseUpKnob_);
417
+  Blockly.Scrollbar.onMouseMoveWrapper_ = Blockly.bindEvent_(document,
418
+      'mousemove', this, this.onMouseMoveKnob_);
419
+  e.stopPropagation();
420
+};
421
+
422
+/**
423
+ * Drag the scrollbar's knob.
424
+ * @param {!Event} e Mouse up event.
425
+ * @private
426
+ */
427
+Blockly.Scrollbar.prototype.onMouseMoveKnob_ = function(e) {
428
+  var currentMouse = this.horizontal_ ? e.clientX : e.clientY;
429
+  var mouseDelta = currentMouse - this.startDragMouse;
430
+  var knobValue = this.startDragKnob + mouseDelta;
431
+  // Position the bar.
432
+  this.svgKnob_.setAttribute(this.horizontal_ ? 'x' : 'y',
433
+                             this.constrainKnob_(knobValue));
434
+  this.onScroll_();
435
+};
436
+
437
+/**
438
+ * Stop binding to the global mouseup and mousemove events.
439
+ * @private
440
+ */
441
+Blockly.Scrollbar.prototype.onMouseUpKnob_ = function() {
442
+  Blockly.removeAllRanges();
443
+  Blockly.hideChaff(true);
444
+  if (Blockly.Scrollbar.onMouseUpWrapper_) {
445
+    Blockly.unbindEvent_(Blockly.Scrollbar.onMouseUpWrapper_);
446
+    Blockly.Scrollbar.onMouseUpWrapper_ = null;
447
+  }
448
+  if (Blockly.Scrollbar.onMouseMoveWrapper_) {
449
+    Blockly.unbindEvent_(Blockly.Scrollbar.onMouseMoveWrapper_);
450
+    Blockly.Scrollbar.onMouseMoveWrapper_ = null;
451
+  }
452
+};
453
+
454
+/**
455
+ * Constrain the knob's position within the minimum (0) and maximum
456
+ * (length of scrollbar) values allowed for the scrollbar.
457
+ * @param {number} value Value that is potentially out of bounds.
458
+ * @return {number} Constrained value.
459
+ * @private
460
+ */
461
+Blockly.Scrollbar.prototype.constrainKnob_ = function(value) {
462
+  if (value <= 0 || isNaN(value)) {
463
+    value = 0;
464
+  } else {
465
+    var axis = this.horizontal_ ? 'width' : 'height';
466
+    var barLength = parseFloat(this.svgBackground_.getAttribute(axis));
467
+    var knobLength = parseFloat(this.svgKnob_.getAttribute(axis));
468
+    value = Math.min(value, barLength - knobLength);
469
+  }
470
+  return value;
471
+};
472
+
473
+/**
474
+ * Called when scrollbar is moved.
475
+ * @private
476
+ */
477
+Blockly.Scrollbar.prototype.onScroll_ = function() {
478
+  var knobValue = parseFloat(
479
+      this.svgKnob_.getAttribute(this.horizontal_ ? 'x' : 'y'));
480
+  var barLength = parseFloat(
481
+      this.svgBackground_.getAttribute(this.horizontal_ ? 'width' : 'height'));
482
+  var ratio = knobValue / barLength;
483
+  if (isNaN(ratio)) {
484
+    ratio = 0;
485
+  }
486
+  var xyRatio = {};
487
+  if (this.horizontal_) {
488
+    xyRatio.x = ratio;
489
+  } else {
490
+    xyRatio.y = ratio;
491
+  }
492
+  this.workspace_.setMetrics(xyRatio);
493
+};
494
+
495
+/**
496
+ * Set the scrollbar slider's position.
497
+ * @param {number} value The distance from the top/left end of the bar.
498
+ */
499
+Blockly.Scrollbar.prototype.set = function(value) {
500
+  // Move the scrollbar slider.
501
+  this.svgKnob_.setAttribute(this.horizontal_ ? 'x' : 'y', value * this.ratio_);
502
+  this.onScroll_();
503
+};
504
+
505
+/**
506
+ * Insert a node after a reference node.
507
+ * Contrast with node.insertBefore function.
508
+ * @param {!Element} newNode New element to insert.
509
+ * @param {!Element} refNode Existing element to precede new node.
510
+ * @private
511
+ */
512
+Blockly.Scrollbar.insertAfter_ = function(newNode, refNode) {
513
+  var siblingNode = refNode.nextSibling;
514
+  var parentNode = refNode.parentNode;
515
+  if (!parentNode) {
516
+    throw 'Reference node has no parent.';
517
+  }
518
+  if (siblingNode) {
519
+    parentNode.insertBefore(newNode, siblingNode);
520
+  } else {
521
+    parentNode.appendChild(newNode);
522
+  }
523
+};

+ 475 - 0
src/blockly/core/toolbox.js

@@ -0,0 +1,475 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2011 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Toolbox from whence to create blocks.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Toolbox');
28
+
29
+goog.require('Blockly.Flyout');
30
+goog.require('goog.dom');
31
+goog.require('goog.events');
32
+goog.require('goog.events.BrowserFeature');
33
+goog.require('goog.html.SafeHtml');
34
+goog.require('goog.math.Rect');
35
+goog.require('goog.style');
36
+goog.require('goog.ui.tree.TreeControl');
37
+goog.require('goog.ui.tree.TreeNode');
38
+
39
+
40
+/**
41
+ * Class for a Toolbox.
42
+ * Creates the toolbox's DOM.
43
+ * @param {!Blockly.Workspace} workspace The workspace in which to create new
44
+ *     blocks.
45
+ * @constructor
46
+ */
47
+Blockly.Toolbox = function(workspace) {
48
+  /**
49
+   * @type {!Blockly.Workspace}
50
+   * @private
51
+   */
52
+  this.workspace_ = workspace;
53
+};
54
+
55
+/**
56
+ * Width of the toolbox.
57
+ * @type {number}
58
+ */
59
+Blockly.Toolbox.prototype.width = 0;
60
+
61
+/**
62
+ * The SVG group currently selected.
63
+ * @type {SVGGElement}
64
+ * @private
65
+ */
66
+Blockly.Toolbox.prototype.selectedOption_ = null;
67
+
68
+/**
69
+ * The tree node most recently selected.
70
+ * @type {goog.ui.tree.BaseNode}
71
+ * @private
72
+ */
73
+Blockly.Toolbox.prototype.lastCategory_ = null;
74
+
75
+/**
76
+ * Configuration constants for Closure's tree UI.
77
+ * @type {Object.<string,*>}
78
+ * @const
79
+ * @private
80
+ */
81
+Blockly.Toolbox.prototype.CONFIG_ = {
82
+  indentWidth: 19,
83
+  cssRoot: 'blocklyTreeRoot',
84
+  cssHideRoot: 'blocklyHidden',
85
+  cssItem: '',
86
+  cssTreeRow: 'blocklyTreeRow',
87
+  cssItemLabel: 'blocklyTreeLabel',
88
+  cssTreeIcon: 'blocklyTreeIcon',
89
+  cssExpandedFolderIcon: 'blocklyTreeIconOpen',
90
+  cssFileIcon: 'blocklyTreeIconNone',
91
+  cssSelectedRow: 'blocklyTreeSelected'
92
+};
93
+
94
+/**
95
+ * Initializes the toolbox.
96
+ */
97
+Blockly.Toolbox.prototype.init = function() {
98
+  var workspace = this.workspace_;
99
+
100
+  // Create an HTML container for the Toolbox menu.
101
+  this.HtmlDiv = goog.dom.createDom('div', 'blocklyToolboxDiv');
102
+  this.HtmlDiv.setAttribute('dir', workspace.RTL ? 'RTL' : 'LTR');
103
+  document.body.appendChild(this.HtmlDiv);
104
+
105
+  // Clicking on toolbar closes popups.
106
+  Blockly.bindEvent_(this.HtmlDiv, 'mousedown', this,
107
+      function(e) {
108
+        if (Blockly.isRightButton(e) || e.target == this.HtmlDiv) {
109
+          // Close flyout.
110
+          Blockly.hideChaff(false);
111
+        } else {
112
+          // Just close popups.
113
+          Blockly.hideChaff(true);
114
+        }
115
+      });
116
+  var workspaceOptions = {
117
+    disabledPatternId: workspace.options.disabledPatternId,
118
+    parentWorkspace: workspace,
119
+    RTL: workspace.RTL,
120
+    svg: workspace.options.svg
121
+  };
122
+  /**
123
+   * @type {!Blockly.Flyout}
124
+   * @private
125
+   */
126
+  this.flyout_ = new Blockly.Flyout(workspaceOptions);
127
+  goog.dom.insertSiblingAfter(this.flyout_.createDom(), workspace.svgGroup_);
128
+  this.flyout_.init(workspace);
129
+
130
+  this.CONFIG_['cleardotPath'] = workspace.options.pathToMedia + '1x1.gif';
131
+  this.CONFIG_['cssCollapsedFolderIcon'] =
132
+      'blocklyTreeIconClosed' + (workspace.RTL ? 'Rtl' : 'Ltr');
133
+  var tree = new Blockly.Toolbox.TreeControl(this, this.CONFIG_);
134
+  this.tree_ = tree;
135
+  tree.setShowRootNode(false);
136
+  tree.setShowLines(false);
137
+  tree.setShowExpandIcons(false);
138
+  tree.setSelectedItem(null);
139
+  this.hasColours_ = false;
140
+  this.populate_(workspace.options.languageTree);
141
+  tree.render(this.HtmlDiv);
142
+  if (this.hasColours_) {
143
+    this.addColour_(tree);
144
+  }
145
+  this.position();
146
+};
147
+
148
+/**
149
+ * Dispose of this toolbox.
150
+ */
151
+Blockly.Toolbox.prototype.dispose = function() {
152
+  this.flyout_.dispose();
153
+  this.tree_.dispose();
154
+  goog.dom.removeNode(this.HtmlDiv);
155
+  this.workspace_ = null;
156
+  this.lastCategory_ = null;
157
+};
158
+
159
+/**
160
+ * Move the toolbox to the edge.
161
+ */
162
+Blockly.Toolbox.prototype.position = function() {
163
+  var treeDiv = this.HtmlDiv;
164
+  if (!treeDiv) {
165
+    // Not initialized yet.
166
+    return;
167
+  }
168
+  var svg = this.workspace_.options.svg;
169
+  var svgPosition = goog.style.getPageOffset(svg);
170
+  var svgSize = Blockly.svgSize(svg);
171
+  if (this.workspace_.RTL) {
172
+    treeDiv.style.left =
173
+        (svgPosition.x + svgSize.width - treeDiv.offsetWidth) + 'px';
174
+  } else {
175
+    treeDiv.style.left = svgPosition.x + 'px';
176
+  }
177
+  treeDiv.style.height = svgSize.height + 'px';
178
+  treeDiv.style.top = svgPosition.y + 'px';
179
+  this.width = treeDiv.offsetWidth;
180
+  if (!this.workspace_.RTL) {
181
+    // For some reason the LTR toolbox now reports as 1px too wide.
182
+    this.width -= 1;
183
+  }
184
+  this.flyout_.position();
185
+};
186
+
187
+/**
188
+ * Fill the toolbox with categories and blocks.
189
+ * @param {Node} newTree DOM tree of blocks, or null.
190
+ * @private
191
+ */
192
+Blockly.Toolbox.prototype.populate_ = function(newTree) {
193
+  var rootOut = this.tree_;
194
+  rootOut.removeChildren();  // Delete any existing content.
195
+  rootOut.blocks = [];
196
+  var hasColours = false;
197
+  function syncTrees(treeIn, treeOut) {
198
+    for (var i = 0, childIn; childIn = treeIn.childNodes[i]; i++) {
199
+      if (!childIn.tagName) {
200
+        // Skip over text.
201
+        continue;
202
+      }
203
+      switch (childIn.tagName.toUpperCase()) {
204
+        case 'CATEGORY':
205
+          var childOut = rootOut.createNode(childIn.getAttribute('name'));
206
+          childOut.blocks = [];
207
+          treeOut.add(childOut);
208
+          var custom = childIn.getAttribute('custom');
209
+          if (custom) {
210
+            // Variables and procedures are special dynamic categories.
211
+            childOut.blocks = custom;
212
+          } else {
213
+            syncTrees(childIn, childOut);
214
+          }
215
+          var hue = childIn.getAttribute('colour');
216
+          if (goog.isString(hue)) {
217
+            childOut.hexColour = Blockly.makeColour(hue);
218
+            hasColours = true;
219
+          } else {
220
+            childOut.hexColour = '';
221
+          }
222
+          if (childIn.getAttribute('expanded') == 'true') {
223
+            if (childOut.blocks.length) {
224
+              rootOut.setSelectedItem(childOut);
225
+            }
226
+            childOut.setExpanded(true);
227
+          }
228
+          break;
229
+        case 'SEP':
230
+          treeOut.add(new Blockly.Toolbox.TreeSeparator());
231
+          break;
232
+        case 'BLOCK':
233
+          treeOut.blocks.push(childIn);
234
+          break;
235
+      }
236
+    }
237
+  }
238
+  syncTrees(newTree, this.tree_);
239
+  this.hasColours_ = hasColours;
240
+
241
+  if (rootOut.blocks.length) {
242
+    throw 'Toolbox cannot have both blocks and categories in the root level.';
243
+  }
244
+
245
+  // Fire a resize event since the toolbox may have changed width and height.
246
+  Blockly.fireUiEvent(window, 'resize');
247
+};
248
+
249
+/**
250
+ * Recursively add colours to this toolbox.
251
+ * @param {!Blockly.Toolbox.TreeNode}
252
+ * @private
253
+ */
254
+Blockly.Toolbox.prototype.addColour_ = function(tree) {
255
+  var children = tree.getChildren();
256
+  for (var i = 0, child; child = children[i]; i++) {
257
+    var element = child.getRowElement();
258
+    if (element) {
259
+      var border = '8px solid ' + (child.hexColour || '#ddd');
260
+      if (this.workspace_.RTL) {
261
+        element.style.borderRight = border;
262
+      } else {
263
+        element.style.borderLeft = border;
264
+      }
265
+    }
266
+    this.addColour_(child);
267
+  }
268
+};
269
+
270
+/**
271
+ * Unhighlight any previously specified option.
272
+ */
273
+Blockly.Toolbox.prototype.clearSelection = function() {
274
+  this.tree_.setSelectedItem(null);
275
+};
276
+
277
+/**
278
+ * Return the deletion rectangle for this toolbar.
279
+ * @return {goog.math.Rect} Rectangle in which to delete.
280
+ */
281
+Blockly.Toolbox.prototype.getRect = function() {
282
+  // BIG_NUM is offscreen padding so that blocks dragged beyond the toolbox
283
+  // area are still deleted.  Must be smaller than Infinity, but larger than
284
+  // the largest screen size.
285
+  var BIG_NUM = 10000000;
286
+  // Assumes that the toolbox is on the SVG edge.  If this changes
287
+  // (e.g. toolboxes in mutators) then this code will need to be more complex.
288
+  if (this.workspace_.RTL) {
289
+    var svgSize = Blockly.svgSize(this.workspace_.options.svg);
290
+    var x = svgSize.width - this.width;
291
+  } else {
292
+    var x = -BIG_NUM;
293
+  }
294
+  return new goog.math.Rect(x, -BIG_NUM, BIG_NUM + this.width, 2 * BIG_NUM);
295
+};
296
+
297
+// Extending Closure's Tree UI.
298
+
299
+/**
300
+ * Extention of a TreeControl object that uses a custom tree node.
301
+ * @param {Blockly.Toolbox} toolbox The parent toolbox for this tree.
302
+ * @param {Object} config The configuration for the tree. See
303
+ *    goog.ui.tree.TreeControl.DefaultConfig.
304
+ * @constructor
305
+ * @extends {goog.ui.tree.TreeControl}
306
+ */
307
+Blockly.Toolbox.TreeControl = function(toolbox, config) {
308
+  this.toolbox_ = toolbox;
309
+  goog.ui.tree.TreeControl.call(this, goog.html.SafeHtml.EMPTY, config);
310
+};
311
+goog.inherits(Blockly.Toolbox.TreeControl, goog.ui.tree.TreeControl);
312
+
313
+/**
314
+ * Adds touch handling to TreeControl.
315
+ * @override
316
+ */
317
+Blockly.Toolbox.TreeControl.prototype.enterDocument = function() {
318
+  Blockly.Toolbox.TreeControl.superClass_.enterDocument.call(this);
319
+
320
+  // Add touch handler.
321
+  if (goog.events.BrowserFeature.TOUCH_ENABLED) {
322
+    var el = this.getRowElement();
323
+    Blockly.bindEvent_(el, goog.events.EventType.TOUCHSTART, this,
324
+        this.handleTouchEvent_);
325
+  }
326
+};
327
+/**
328
+ * Handles touch events.
329
+ * @param {!goog.events.BrowserEvent} e The browser event.
330
+ * @private
331
+ */
332
+Blockly.Toolbox.TreeControl.prototype.handleTouchEvent_ = function(e) {
333
+  e.preventDefault();
334
+  var node = this.getNodeFromEvent_(e);
335
+  if (node && e.type === goog.events.EventType.TOUCHSTART) {
336
+    // Fire asynchronously since onMouseDown takes long enough that the browser
337
+    // would fire the default mouse event before this method returns.
338
+    setTimeout(function() {
339
+      node.onMouseDown(e);  // Same behaviour for click and touch.
340
+    }, 1);
341
+  }
342
+};
343
+
344
+/**
345
+ * Creates a new tree node using a custom tree node.
346
+ * @param {string=} opt_html The HTML content of the node label.
347
+ * @return {!goog.ui.tree.TreeNode} The new item.
348
+ * @override
349
+ */
350
+Blockly.Toolbox.TreeControl.prototype.createNode = function(opt_html) {
351
+  return new Blockly.Toolbox.TreeNode(this.toolbox_, opt_html ?
352
+      goog.html.SafeHtml.htmlEscape(opt_html) : goog.html.SafeHtml.EMPTY,
353
+      this.getConfig(), this.getDomHelper());
354
+};
355
+
356
+/**
357
+ * Display/hide the flyout when an item is selected.
358
+ * @param {goog.ui.tree.BaseNode} node The item to select.
359
+ * @override
360
+ */
361
+Blockly.Toolbox.TreeControl.prototype.setSelectedItem = function(node) {
362
+  Blockly.removeAllRanges();
363
+  var toolbox = this.toolbox_;
364
+  if (node == this.selectedItem_ || node == toolbox.tree_) {
365
+    return;
366
+  }
367
+  if (toolbox.lastCategory_) {
368
+    toolbox.lastCategory_.getRowElement().style.backgroundColor = '';
369
+  }
370
+  if (node) {
371
+    var hexColour = node.hexColour || '#57e';
372
+    node.getRowElement().style.backgroundColor = hexColour;
373
+    // Add colours to child nodes which may have been collapsed and thus
374
+    // not rendered.
375
+    toolbox.addColour_(node);
376
+  }
377
+  goog.ui.tree.TreeControl.prototype.setSelectedItem.call(this, node);
378
+  if (node && node.blocks && node.blocks.length) {
379
+    toolbox.flyout_.show(node.blocks);
380
+    // Scroll the flyout to the top if the category has changed.
381
+    if (toolbox.lastCategory_ != node) {
382
+      toolbox.flyout_.scrollToTop();
383
+    }
384
+  } else {
385
+    // Hide the flyout.
386
+    toolbox.flyout_.hide();
387
+  }
388
+  if (node) {
389
+    toolbox.lastCategory_ = node;
390
+  }
391
+};
392
+
393
+/**
394
+ * A single node in the tree, customized for Blockly's UI.
395
+ * @param {Blockly.Toolbox} toolbox The parent toolbox for this tree.
396
+ * @param {!goog.html.SafeHtml} html The HTML content of the node label.
397
+ * @param {Object=} opt_config The configuration for the tree. See
398
+ *    goog.ui.tree.TreeControl.DefaultConfig. If not specified, a default config
399
+ *    will be used.
400
+ * @param {goog.dom.DomHelper=} opt_domHelper Optional DOM helper.
401
+ * @constructor
402
+ * @extends {goog.ui.tree.TreeNode}
403
+ */
404
+Blockly.Toolbox.TreeNode = function(toolbox, html, opt_config, opt_domHelper) {
405
+  goog.ui.tree.TreeNode.call(this, html, opt_config, opt_domHelper);
406
+  if (toolbox) {
407
+    var resize = function() {
408
+      Blockly.fireUiEvent(window, 'resize');
409
+    };
410
+    // Fire a resize event since the toolbox may have changed width.
411
+    goog.events.listen(toolbox.tree_,
412
+        goog.ui.tree.BaseNode.EventType.EXPAND, resize);
413
+    goog.events.listen(toolbox.tree_,
414
+        goog.ui.tree.BaseNode.EventType.COLLAPSE, resize);
415
+  }
416
+};
417
+goog.inherits(Blockly.Toolbox.TreeNode, goog.ui.tree.TreeNode);
418
+
419
+/**
420
+ * Supress population of the +/- icon.
421
+ * @return {!goog.html.SafeHtml} The source for the icon.
422
+ * @override
423
+ */
424
+Blockly.Toolbox.TreeNode.prototype.getExpandIconSafeHtml = function() {
425
+  return goog.html.SafeHtml.create('span');
426
+};
427
+
428
+/**
429
+ * Expand or collapse the node on mouse click.
430
+ * @param {!goog.events.BrowserEvent} e The browser event.
431
+ * @override
432
+ */
433
+Blockly.Toolbox.TreeNode.prototype.onMouseDown = function(e) {
434
+  // Expand icon.
435
+  if (this.hasChildren() && this.isUserCollapsible_) {
436
+    this.toggle();
437
+    this.select();
438
+  } else if (this.isSelected()) {
439
+    this.getTree().setSelectedItem(null);
440
+  } else {
441
+    this.select();
442
+  }
443
+  this.updateRow();
444
+};
445
+
446
+/**
447
+ * Supress the inherited double-click behaviour.
448
+ * @param {!goog.events.BrowserEvent} e The browser event.
449
+ * @override
450
+ * @private
451
+ */
452
+Blockly.Toolbox.TreeNode.prototype.onDoubleClick_ = function(e) {
453
+  // NOP.
454
+};
455
+
456
+/**
457
+ * A blank separator node in the tree.
458
+ * @constructor
459
+ * @extends {Blockly.Toolbox.TreeNode}
460
+ */
461
+Blockly.Toolbox.TreeSeparator = function() {
462
+  Blockly.Toolbox.TreeNode.call(this, null, '',
463
+      Blockly.Toolbox.TreeSeparator.CONFIG_);
464
+};
465
+goog.inherits(Blockly.Toolbox.TreeSeparator, Blockly.Toolbox.TreeNode);
466
+
467
+/**
468
+ * Configuration constants for tree separator.
469
+ * @type {Object.<string,*>}
470
+ * @const
471
+ * @private
472
+ */
473
+Blockly.Toolbox.TreeSeparator.CONFIG_ = {
474
+  cssTreeRow: 'blocklyTreeSeparator'
475
+};

+ 438 - 0
src/blockly/core/tooltip.js

@@ -0,0 +1,438 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2011 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Library to create tooltips for Blockly.
23
+ * First, call Blockly.Tooltip.init() after onload.
24
+ * Second, set the 'tooltip' property on any SVG element that needs a tooltip.
25
+ * If the tooltip is a string, then that message will be displayed.
26
+ * If the tooltip is an SVG element, then that object's tooltip will be used.
27
+ * Third, call Blockly.Tooltip.bindMouseEvents(e) passing the SVG element.
28
+ * @author fraser@google.com (Neil Fraser)
29
+ */
30
+'use strict';
31
+
32
+goog.provide('Blockly.Tooltip');
33
+
34
+goog.require('goog.dom');
35
+
36
+
37
+/**
38
+ * Is a tooltip currently showing?
39
+ */
40
+Blockly.Tooltip.visible = false;
41
+
42
+/**
43
+ * Maximum width (in characters) of a tooltip.
44
+ */
45
+Blockly.Tooltip.LIMIT = 50;
46
+
47
+/**
48
+ * PID of suspended thread to clear tooltip on mouse out.
49
+ * @private
50
+ */
51
+Blockly.Tooltip.mouseOutPid_ = 0;
52
+
53
+/**
54
+ * PID of suspended thread to show the tooltip.
55
+ * @private
56
+ */
57
+Blockly.Tooltip.showPid_ = 0;
58
+
59
+/**
60
+ * Last observed X location of the mouse pointer (freezes when tooltip appears).
61
+ * @private
62
+ */
63
+Blockly.Tooltip.lastX_ = 0;
64
+
65
+/**
66
+ * Last observed Y location of the mouse pointer (freezes when tooltip appears).
67
+ * @private
68
+ */
69
+Blockly.Tooltip.lastY_ = 0;
70
+
71
+/**
72
+ * Current element being pointed at.
73
+ * @private
74
+ */
75
+Blockly.Tooltip.element_ = null;
76
+
77
+/**
78
+ * Once a tooltip has opened for an element, that element is 'poisoned' and
79
+ * cannot respawn a tooltip until the pointer moves over a different element.
80
+ * @private
81
+ */
82
+Blockly.Tooltip.poisonedElement_ = null;
83
+
84
+/**
85
+ * Horizontal offset between mouse cursor and tooltip.
86
+ */
87
+Blockly.Tooltip.OFFSET_X = 0;
88
+
89
+/**
90
+ * Vertical offset between mouse cursor and tooltip.
91
+ */
92
+Blockly.Tooltip.OFFSET_Y = 10;
93
+
94
+/**
95
+ * Radius mouse can move before killing tooltip.
96
+ */
97
+Blockly.Tooltip.RADIUS_OK = 10;
98
+
99
+/**
100
+ * Delay before tooltip appears.
101
+ */
102
+Blockly.Tooltip.HOVER_MS = 1000;
103
+
104
+/**
105
+ * Horizontal padding between tooltip and screen edge.
106
+ */
107
+Blockly.Tooltip.MARGINS = 5;
108
+
109
+/**
110
+ * The HTML container.  Set once by Blockly.Tooltip.createDom.
111
+ * @type {Element}
112
+ */
113
+Blockly.Tooltip.DIV = null;
114
+
115
+/**
116
+ * Create the tooltip div and inject it onto the page.
117
+ */
118
+Blockly.Tooltip.createDom = function() {
119
+  if (Blockly.Tooltip.DIV) {
120
+    return;  // Already created.
121
+  }
122
+  // Create an HTML container for popup overlays (e.g. editor widgets).
123
+  Blockly.Tooltip.DIV = goog.dom.createDom('div', 'blocklyTooltipDiv');
124
+  document.body.appendChild(Blockly.Tooltip.DIV);
125
+};
126
+
127
+/**
128
+ * Binds the required mouse events onto an SVG element.
129
+ * @param {!Element} element SVG element onto which tooltip is to be bound.
130
+ */
131
+Blockly.Tooltip.bindMouseEvents = function(element) {
132
+  Blockly.bindEvent_(element, 'mouseover', null, Blockly.Tooltip.onMouseOver_);
133
+  Blockly.bindEvent_(element, 'mouseout', null, Blockly.Tooltip.onMouseOut_);
134
+  Blockly.bindEvent_(element, 'mousemove', null, Blockly.Tooltip.onMouseMove_);
135
+};
136
+
137
+/**
138
+ * Hide the tooltip if the mouse is over a different object.
139
+ * Initialize the tooltip to potentially appear for this object.
140
+ * @param {!Event} e Mouse event.
141
+ * @private
142
+ */
143
+Blockly.Tooltip.onMouseOver_ = function(e) {
144
+  // If the tooltip is an object, treat it as a pointer to the next object in
145
+  // the chain to look at.  Terminate when a string or function is found.
146
+  var element = e.target;
147
+  while (!goog.isString(element.tooltip) && !goog.isFunction(element.tooltip)) {
148
+    element = element.tooltip;
149
+  }
150
+  if (Blockly.Tooltip.element_ != element) {
151
+    Blockly.Tooltip.hide();
152
+    Blockly.Tooltip.poisonedElement_ = null;
153
+    Blockly.Tooltip.element_ = element;
154
+  }
155
+  // Forget about any immediately preceeding mouseOut event.
156
+  clearTimeout(Blockly.Tooltip.mouseOutPid_);
157
+};
158
+
159
+/**
160
+ * Hide the tooltip if the mouse leaves the object and enters the workspace.
161
+ * @param {!Event} e Mouse event.
162
+ * @private
163
+ */
164
+Blockly.Tooltip.onMouseOut_ = function(e) {
165
+  // Moving from one element to another (overlapping or with no gap) generates
166
+  // a mouseOut followed instantly by a mouseOver.  Fork off the mouseOut
167
+  // event and kill it if a mouseOver is received immediately.
168
+  // This way the task only fully executes if mousing into the void.
169
+  Blockly.Tooltip.mouseOutPid_ = setTimeout(function() {
170
+        Blockly.Tooltip.element_ = null;
171
+        Blockly.Tooltip.poisonedElement_ = null;
172
+        Blockly.Tooltip.hide();
173
+      }, 1);
174
+  clearTimeout(Blockly.Tooltip.showPid_);
175
+};
176
+
177
+/**
178
+ * When hovering over an element, schedule a tooltip to be shown.  If a tooltip
179
+ * is already visible, hide it if the mouse strays out of a certain radius.
180
+ * @param {!Event} e Mouse event.
181
+ * @private
182
+ */
183
+Blockly.Tooltip.onMouseMove_ = function(e) {
184
+  if (!Blockly.Tooltip.element_ || !Blockly.Tooltip.element_.tooltip) {
185
+    // No tooltip here to show.
186
+    return;
187
+  } else if (Blockly.dragMode_ != 0) {
188
+    // Don't display a tooltip during a drag.
189
+    return;
190
+  } else if (Blockly.WidgetDiv.isVisible()) {
191
+    // Don't display a tooltip if a widget is open (tooltip would be under it).
192
+    return;
193
+  }
194
+  if (Blockly.Tooltip.visible) {
195
+    // Compute the distance between the mouse position when the tooltip was
196
+    // shown and the current mouse position.  Pythagorean theorem.
197
+    var dx = Blockly.Tooltip.lastX_ - e.pageX;
198
+    var dy = Blockly.Tooltip.lastY_ - e.pageY;
199
+    if (Math.sqrt(dx * dx + dy * dy) > Blockly.Tooltip.RADIUS_OK) {
200
+      Blockly.Tooltip.hide();
201
+    }
202
+  } else if (Blockly.Tooltip.poisonedElement_ != Blockly.Tooltip.element_) {
203
+    // The mouse moved, clear any previously scheduled tooltip.
204
+    clearTimeout(Blockly.Tooltip.showPid_);
205
+    // Maybe this time the mouse will stay put.  Schedule showing of tooltip.
206
+    Blockly.Tooltip.lastX_ = e.pageX;
207
+    Blockly.Tooltip.lastY_ = e.pageY;
208
+    Blockly.Tooltip.showPid_ =
209
+        setTimeout(Blockly.Tooltip.show_, Blockly.Tooltip.HOVER_MS);
210
+  }
211
+};
212
+
213
+/**
214
+ * Hide the tooltip.
215
+ */
216
+Blockly.Tooltip.hide = function() {
217
+  if (Blockly.Tooltip.visible) {
218
+    Blockly.Tooltip.visible = false;
219
+    if (Blockly.Tooltip.DIV) {
220
+      Blockly.Tooltip.DIV.style.display = 'none';
221
+    }
222
+  }
223
+  clearTimeout(Blockly.Tooltip.showPid_);
224
+};
225
+
226
+/**
227
+ * Create the tooltip and show it.
228
+ * @private
229
+ */
230
+Blockly.Tooltip.show_ = function() {
231
+  Blockly.Tooltip.poisonedElement_ = Blockly.Tooltip.element_;
232
+  if (!Blockly.Tooltip.DIV) {
233
+    return;
234
+  }
235
+  // Erase all existing text.
236
+  goog.dom.removeChildren(/** @type {!Element} */ (Blockly.Tooltip.DIV));
237
+  // Get the new text.
238
+  var tip = Blockly.Tooltip.element_.tooltip;
239
+  if (goog.isFunction(tip)) {
240
+    tip = tip();
241
+  }
242
+  tip = Blockly.Tooltip.wrap_(tip, Blockly.Tooltip.LIMIT);
243
+  // Create new text, line by line.
244
+  var lines = tip.split('\n');
245
+  for (var i = 0; i < lines.length; i++) {
246
+    var div = document.createElement('div');
247
+    div.appendChild(document.createTextNode(lines[i]));
248
+    Blockly.Tooltip.DIV.appendChild(div);
249
+  }
250
+  var rtl = Blockly.Tooltip.element_.RTL;
251
+  var windowSize = goog.dom.getViewportSize();
252
+  // Display the tooltip.
253
+  Blockly.Tooltip.DIV.style.direction = rtl ? 'rtl' : 'ltr';
254
+  Blockly.Tooltip.DIV.style.display = 'block';
255
+  Blockly.Tooltip.visible = true;
256
+  // Move the tooltip to just below the cursor.
257
+  var anchorX = Blockly.Tooltip.lastX_;
258
+  if (rtl) {
259
+    anchorX -= Blockly.Tooltip.OFFSET_X + Blockly.Tooltip.DIV.offsetWidth;
260
+  } else {
261
+    anchorX += Blockly.Tooltip.OFFSET_X;
262
+  }
263
+  var anchorY = Blockly.Tooltip.lastY_ + Blockly.Tooltip.OFFSET_Y;
264
+
265
+  if (anchorY + Blockly.Tooltip.DIV.offsetHeight >
266
+      windowSize.height + window.scrollY) {
267
+    // Falling off the bottom of the screen; shift the tooltip up.
268
+    anchorY -= Blockly.Tooltip.DIV.offsetHeight + 2 * Blockly.Tooltip.OFFSET_Y;
269
+  }
270
+  if (rtl) {
271
+    // Prevent falling off left edge in RTL mode.
272
+    anchorX = Math.max(Blockly.Tooltip.MARGINS - window.scrollX, anchorX);
273
+  } else {
274
+    if (anchorX + Blockly.Tooltip.DIV.offsetWidth >
275
+        windowSize.width + window.scrollX - 2 * Blockly.Tooltip.MARGINS) {
276
+      // Falling off the right edge of the screen;
277
+      // clamp the tooltip on the edge.
278
+      anchorX = windowSize.width - Blockly.Tooltip.DIV.offsetWidth -
279
+          2 * Blockly.Tooltip.MARGINS;
280
+    }
281
+  }
282
+  Blockly.Tooltip.DIV.style.top = anchorY + 'px';
283
+  Blockly.Tooltip.DIV.style.left = anchorX + 'px';
284
+};
285
+
286
+/**
287
+ * Wrap text to the specified width.
288
+ * @param {string} text Text to wrap.
289
+ * @param {number} limit Width to wrap each line.
290
+ * @return {string} Wrapped text.
291
+ * @private
292
+ */
293
+Blockly.Tooltip.wrap_ = function(text, limit) {
294
+  if (text.length <= limit) {
295
+    // Short text, no need to wrap.
296
+    return text;
297
+  }
298
+  // Split the text into words.
299
+  var words = text.trim().split(/\s+/);
300
+  // Set limit to be the length of the largest word.
301
+  for (var i = 0; i < words.length; i++) {
302
+    if (words[i].length > limit) {
303
+      limit = words[i].length;
304
+    }
305
+  }
306
+
307
+  var lastScore;
308
+  var score = -Infinity;
309
+  var lastText;
310
+  var lineCount = 1;
311
+  do {
312
+    lastScore = score;
313
+    lastText = text;
314
+    // Create a list of booleans representing if a space (false) or
315
+    // a break (true) appears after each word.
316
+    var wordBreaks = [];
317
+    // Seed the list with evenly spaced linebreaks.
318
+    var steps = words.length / lineCount;
319
+    var insertedBreaks = 1;
320
+    for (var i = 0; i < words.length - 1; i++) {
321
+      if (insertedBreaks < (i + 1.5) / steps) {
322
+        insertedBreaks++;
323
+        wordBreaks[i] = true;
324
+      } else {
325
+        wordBreaks[i] = false;
326
+      }
327
+    }
328
+    wordBreaks = Blockly.Tooltip.wrapMutate_(words, wordBreaks, limit);
329
+    score = Blockly.Tooltip.wrapScore_(words, wordBreaks, limit);
330
+    text = Blockly.Tooltip.wrapToText_(words, wordBreaks);
331
+    lineCount++;
332
+  } while (score > lastScore);
333
+  return lastText;
334
+};
335
+
336
+/**
337
+ * Compute a score for how good the wrapping is.
338
+ * @param {!Array.<string>} words Array of each word.
339
+ * @param {!Array.<boolean>} wordBreaks Array of line breaks.
340
+ * @param {number} limit Width to wrap each line.
341
+ * @return {number} Larger the better.
342
+ * @private
343
+ */
344
+Blockly.Tooltip.wrapScore_ = function(words, wordBreaks, limit) {
345
+  // If this function becomes a performance liability, add caching.
346
+  // Compute the length of each line.
347
+  var lineLengths = [0];
348
+  var linePunctuation = [];
349
+  for (var i = 0; i < words.length; i++) {
350
+    lineLengths[lineLengths.length - 1] += words[i].length;
351
+    if (wordBreaks[i] === true) {
352
+      lineLengths.push(0);
353
+      linePunctuation.push(words[i].charAt(words[i].length - 1));
354
+    } else if (wordBreaks[i] === false) {
355
+      lineLengths[lineLengths.length - 1]++;
356
+    }
357
+  }
358
+  var maxLength = Math.max.apply(Math, lineLengths);
359
+
360
+  var score = 0;
361
+  for (var i = 0; i < lineLengths.length; i++) {
362
+    // Optimize for width.
363
+    // -2 points per char over limit (scaled to the power of 1.5).
364
+    score -= Math.pow(Math.abs(limit - lineLengths[i]), 1.5) * 2;
365
+    // Optimize for even lines.
366
+    // -1 point per char smaller than max (scaled to the power of 1.5).
367
+    score -= Math.pow(maxLength - lineLengths[i], 1.5);
368
+    // Optimize for structure.
369
+    // Add score to line endings after punctuation.
370
+    if ('.?!'.indexOf(linePunctuation[i]) != -1) {
371
+      score += limit / 3;
372
+    } else if (',;)]}'.indexOf(linePunctuation[i]) != -1) {
373
+      score += limit / 4;
374
+    }
375
+  }
376
+  // All else being equal, the last line should not be longer than the
377
+  // previous line.  For example, this looks wrong:
378
+  // aaa bbb
379
+  // ccc ddd eee
380
+  if (lineLengths.length > 1 && lineLengths[lineLengths.length - 1] <=
381
+      lineLengths[lineLengths.length - 2]) {
382
+    score += 0.5;
383
+  }
384
+  return score;
385
+};
386
+
387
+/**
388
+ * Mutate the array of line break locations until an optimal solution is found.
389
+ * No line breaks are added or deleted, they are simply moved around.
390
+ * @param {!Array.<string>} words Array of each word.
391
+ * @param {!Array.<boolean>} wordBreaks Array of line breaks.
392
+ * @param {number} limit Width to wrap each line.
393
+ * @return {!Array.<boolean>} New array of optimal line breaks.
394
+ * @private
395
+ */
396
+Blockly.Tooltip.wrapMutate_ = function(words, wordBreaks, limit) {
397
+  var bestScore = Blockly.Tooltip.wrapScore_(words, wordBreaks, limit);
398
+  var bestBreaks;
399
+  // Try shifting every line break forward or backward.
400
+  for (var i = 0; i < wordBreaks.length - 1; i++) {
401
+    if (wordBreaks[i] == wordBreaks[i + 1]) {
402
+      continue;
403
+    }
404
+    var mutatedWordBreaks = [].concat(wordBreaks);
405
+    mutatedWordBreaks[i] = !mutatedWordBreaks[i];
406
+    mutatedWordBreaks[i + 1] = !mutatedWordBreaks[i + 1];
407
+    var mutatedScore =
408
+        Blockly.Tooltip.wrapScore_(words, mutatedWordBreaks, limit);
409
+    if (mutatedScore > bestScore) {
410
+      bestScore = mutatedScore;
411
+      bestBreaks = mutatedWordBreaks;
412
+    }
413
+  }
414
+  if (bestBreaks) {
415
+    // Found an improvement.  See if it may be improved further.
416
+    return Blockly.Tooltip.wrapMutate_(words, bestBreaks, limit);
417
+  }
418
+  // No improvements found.  Done.
419
+  return wordBreaks;
420
+};
421
+
422
+/**
423
+ * Reassemble the array of words into text, with the specified line breaks.
424
+ * @param {!Array.<string>} words Array of each word.
425
+ * @param {!Array.<boolean>} wordBreaks Array of line breaks.
426
+ * @return {string} Plain text.
427
+ * @private
428
+ */
429
+Blockly.Tooltip.wrapToText_ = function(words, wordBreaks) {
430
+  var text = [];
431
+  for (var i = 0; i < words.length; i++) {
432
+    text.push(words[i]);
433
+    if (wordBreaks[i] !== undefined) {
434
+      text.push(wordBreaks[i] ? '\n' : ' ');
435
+    }
436
+  }
437
+  return text.join('');
438
+};

+ 297 - 0
src/blockly/core/trashcan.js

@@ -0,0 +1,297 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2011 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Object representing a trash can icon.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Trashcan');
28
+
29
+goog.require('goog.Timer');
30
+goog.require('goog.dom');
31
+goog.require('goog.math');
32
+goog.require('goog.math.Rect');
33
+
34
+
35
+/**
36
+ * Class for a trash can.
37
+ * @param {!Blockly.Workspace} workspace The workspace to sit in.
38
+ * @constructor
39
+ */
40
+Blockly.Trashcan = function(workspace) {
41
+  this.workspace_ = workspace;
42
+};
43
+
44
+/**
45
+ * Width of both the trash can and lid images.
46
+ * @type {number}
47
+ * @private
48
+ */
49
+Blockly.Trashcan.prototype.WIDTH_ = 47;
50
+
51
+/**
52
+ * Height of the trashcan image (minus lid).
53
+ * @type {number}
54
+ * @private
55
+ */
56
+Blockly.Trashcan.prototype.BODY_HEIGHT_ = 44;
57
+
58
+/**
59
+ * Height of the lid image.
60
+ * @type {number}
61
+ * @private
62
+ */
63
+Blockly.Trashcan.prototype.LID_HEIGHT_ = 16;
64
+
65
+/**
66
+ * Distance between trashcan and bottom edge of workspace.
67
+ * @type {number}
68
+ * @private
69
+ */
70
+Blockly.Trashcan.prototype.MARGIN_BOTTOM_ = 20;
71
+
72
+/**
73
+ * Distance between trashcan and right edge of workspace.
74
+ * @type {number}
75
+ * @private
76
+ */
77
+Blockly.Trashcan.prototype.MARGIN_SIDE_ = 20;
78
+
79
+/**
80
+ * Extent of hotspot on all sides beyond the size of the image.
81
+ * @type {number}
82
+ * @private
83
+ */
84
+Blockly.Trashcan.prototype.MARGIN_HOTSPOT_ = 25;
85
+
86
+/**
87
+ * Current open/close state of the lid.
88
+ * @type {boolean}
89
+ */
90
+Blockly.Trashcan.prototype.isOpen = false;
91
+
92
+/**
93
+ * The SVG group containing the trash can.
94
+ * @type {Element}
95
+ * @private
96
+ */
97
+Blockly.Trashcan.prototype.svgGroup_ = null;
98
+
99
+/**
100
+ * The SVG image element of the trash can lid.
101
+ * @type {Element}
102
+ * @private
103
+ */
104
+Blockly.Trashcan.prototype.svgLid_ = null;
105
+
106
+/**
107
+ * Task ID of opening/closing animation.
108
+ * @type {number}
109
+ * @private
110
+ */
111
+Blockly.Trashcan.prototype.lidTask_ = 0;
112
+
113
+/**
114
+ * Current state of lid opening (0.0 = closed, 1.0 = open).
115
+ * @type {number}
116
+ * @private
117
+ */
118
+Blockly.Trashcan.prototype.lidOpen_ = 0;
119
+
120
+/**
121
+ * Left coordinate of the trash can.
122
+ * @type {number}
123
+ * @private
124
+ */
125
+Blockly.Trashcan.prototype.left_ = 0;
126
+
127
+/**
128
+ * Top coordinate of the trash can.
129
+ * @type {number}
130
+ * @private
131
+ */
132
+Blockly.Trashcan.prototype.top_ = 0;
133
+
134
+/**
135
+ * Create the trash can elements.
136
+ * @return {!Element} The trash can's SVG group.
137
+ */
138
+Blockly.Trashcan.prototype.createDom = function() {
139
+  /* Here's the markup that will be generated:
140
+  <g class="blocklyTrash">
141
+    <clippath id="blocklyTrashBodyClipPath837493">
142
+      <rect width="47" height="45" y="15"></rect>
143
+    </clippath>
144
+    <image width="64" height="92" y="-32" xlink:href="media/sprites.png"
145
+        clip-path="url(#blocklyTrashBodyClipPath837493)"></image>
146
+    <clippath id="blocklyTrashLidClipPath837493">
147
+      <rect width="47" height="15"></rect>
148
+    </clippath>
149
+    <image width="84" height="92" y="-32" xlink:href="media/sprites.png"
150
+        clip-path="url(#blocklyTrashLidClipPath837493)"></image>
151
+  </g>
152
+  */
153
+  this.svgGroup_ = Blockly.createSvgElement('g',
154
+      {'class': 'blocklyTrash'}, null);
155
+  var rnd = String(Math.random()).substring(2);
156
+  var clip = Blockly.createSvgElement('clipPath',
157
+      {'id': 'blocklyTrashBodyClipPath' + rnd},
158
+      this.svgGroup_);
159
+  Blockly.createSvgElement('rect',
160
+      {'width': this.WIDTH_, 'height': this.BODY_HEIGHT_,
161
+       'y': this.LID_HEIGHT_},
162
+      clip);
163
+  var body = Blockly.createSvgElement('image',
164
+      {'width': Blockly.SPRITE.width, 'height': Blockly.SPRITE.height, 'y': -32,
165
+       'clip-path': 'url(#blocklyTrashBodyClipPath' + rnd + ')'},
166
+      this.svgGroup_);
167
+  body.setAttributeNS('http://www.w3.org/1999/xlink', 'xlink:href',
168
+      this.workspace_.options.pathToMedia + Blockly.SPRITE.url);
169
+
170
+  var clip = Blockly.createSvgElement('clipPath',
171
+      {'id': 'blocklyTrashLidClipPath' + rnd},
172
+      this.svgGroup_);
173
+  Blockly.createSvgElement('rect',
174
+      {'width': this.WIDTH_, 'height': this.LID_HEIGHT_}, clip);
175
+  this.svgLid_ = Blockly.createSvgElement('image',
176
+      {'width': Blockly.SPRITE.width, 'height': Blockly.SPRITE.height, 'y': -32,
177
+       'clip-path': 'url(#blocklyTrashLidClipPath' + rnd + ')'},
178
+      this.svgGroup_);
179
+  this.svgLid_.setAttributeNS('http://www.w3.org/1999/xlink', 'xlink:href',
180
+      this.workspace_.options.pathToMedia + Blockly.SPRITE.url);
181
+
182
+  Blockly.bindEvent_(this.svgGroup_, 'mouseup', this, this.click);
183
+  this.animateLid_();
184
+  return this.svgGroup_;
185
+};
186
+
187
+/**
188
+ * Initialize the trash can.
189
+ * @param {number} bottom Distance from workspace bottom to bottom of trashcan.
190
+ * @return {number} Distance from workspace bottom to the top of trashcan.
191
+ */
192
+Blockly.Trashcan.prototype.init = function(bottom) {
193
+  this.bottom_ =  this.MARGIN_BOTTOM_ + bottom;
194
+  this.setOpen_(false);
195
+  return this.bottom_ + this.BODY_HEIGHT_ + this.LID_HEIGHT_;
196
+};
197
+
198
+/**
199
+ * Dispose of this trash can.
200
+ * Unlink from all DOM elements to prevent memory leaks.
201
+ */
202
+Blockly.Trashcan.prototype.dispose = function() {
203
+  if (this.svgGroup_) {
204
+    goog.dom.removeNode(this.svgGroup_);
205
+    this.svgGroup_ = null;
206
+  }
207
+  this.svgLid_ = null;
208
+  this.workspace_ = null;
209
+  goog.Timer.clear(this.lidTask_);
210
+};
211
+
212
+/**
213
+ * Move the trash can to the bottom-right corner.
214
+ */
215
+Blockly.Trashcan.prototype.position = function() {
216
+  var metrics = this.workspace_.getMetrics();
217
+  if (!metrics) {
218
+    // There are no metrics available (workspace is probably not visible).
219
+    return;
220
+  }
221
+  if (this.workspace_.RTL) {
222
+    this.left_ = this.MARGIN_SIDE_ + Blockly.Scrollbar.scrollbarThickness;
223
+  } else {
224
+    this.left_ = metrics.viewWidth + metrics.absoluteLeft -
225
+        this.WIDTH_ - this.MARGIN_SIDE_ - Blockly.Scrollbar.scrollbarThickness;
226
+  }
227
+  this.top_ = metrics.viewHeight + metrics.absoluteTop -
228
+      (this.BODY_HEIGHT_ + this.LID_HEIGHT_) - this.bottom_;
229
+  this.svgGroup_.setAttribute('transform',
230
+      'translate(' + this.left_ + ',' + this.top_ + ')');
231
+};
232
+
233
+/**
234
+ * Return the deletion rectangle for this trash can.
235
+ * @return {goog.math.Rect} Rectangle in which to delete.
236
+ */
237
+Blockly.Trashcan.prototype.getRect = function() {
238
+  var trashXY = Blockly.getSvgXY_(this.svgGroup_, this.workspace_);
239
+  return new goog.math.Rect(
240
+      trashXY.x - this.MARGIN_HOTSPOT_,
241
+      trashXY.y - this.MARGIN_HOTSPOT_,
242
+      this.WIDTH_ + 2 * this.MARGIN_HOTSPOT_,
243
+      this.BODY_HEIGHT_ + this.LID_HEIGHT_ + 2 * this.MARGIN_HOTSPOT_);
244
+};
245
+
246
+/**
247
+ * Flip the lid open or shut.
248
+ * @param {boolean} state True if open.
249
+ * @private
250
+ */
251
+Blockly.Trashcan.prototype.setOpen_ = function(state) {
252
+  if (this.isOpen == state) {
253
+    return;
254
+  }
255
+  goog.Timer.clear(this.lidTask_);
256
+  this.isOpen = state;
257
+  this.animateLid_();
258
+};
259
+
260
+/**
261
+ * Rotate the lid open or closed by one step.  Then wait and recurse.
262
+ * @private
263
+ */
264
+Blockly.Trashcan.prototype.animateLid_ = function() {
265
+  this.lidOpen_ += this.isOpen ? 0.2 : -0.2;
266
+  this.lidOpen_ = goog.math.clamp(this.lidOpen_, 0, 1);
267
+  var lidAngle = this.lidOpen_ * 45;
268
+  this.svgLid_.setAttribute('transform', 'rotate(' +
269
+      (this.workspace_.RTL ? -lidAngle : lidAngle) + ',' +
270
+      (this.workspace_.RTL ? 4 : this.WIDTH_ - 4) + ',' +
271
+      (this.LID_HEIGHT_ - 2) + ')');
272
+  var opacity = goog.math.lerp(0.4, 0.8, this.lidOpen_);
273
+  this.svgGroup_.style.opacity = opacity;
274
+  if (this.lidOpen_ > 0 && this.lidOpen_ < 1) {
275
+    this.lidTask_ = goog.Timer.callOnce(this.animateLid_, 20, this);
276
+  }
277
+};
278
+
279
+/**
280
+ * Flip the lid shut.
281
+ * Called externally after a drag.
282
+ */
283
+Blockly.Trashcan.prototype.close = function() {
284
+  this.setOpen_(false);
285
+};
286
+
287
+/**
288
+ * Inspect the contents of the trash.
289
+ */
290
+Blockly.Trashcan.prototype.click = function() {
291
+  var dx = this.workspace_.startScrollX - this.workspace_.scrollX;
292
+  var dy = this.workspace_.startScrollY - this.workspace_.scrollY;
293
+  if (Math.sqrt(dx * dx + dy * dy) > Blockly.DRAG_RADIUS) {
294
+    return;
295
+  }
296
+  console.log('TODO: Inspect trash.');
297
+};

+ 553 - 0
src/blockly/core/utils.js

@@ -0,0 +1,553 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Utility methods.
23
+ * These methods are not specific to Blockly, and could be factored out into
24
+ * a JavaScript framework such as Closure.
25
+ * @author fraser@google.com (Neil Fraser)
26
+ */
27
+'use strict';
28
+
29
+goog.provide('Blockly.utils');
30
+
31
+goog.require('goog.dom');
32
+goog.require('goog.events.BrowserFeature');
33
+goog.require('goog.math.Coordinate');
34
+goog.require('goog.userAgent');
35
+
36
+
37
+/**
38
+ * Add a CSS class to a element.
39
+ * Similar to Closure's goog.dom.classes.add, except it handles SVG elements.
40
+ * @param {!Element} element DOM element to add class to.
41
+ * @param {string} className Name of class to add.
42
+ * @private
43
+ */
44
+Blockly.addClass_ = function(element, className) {
45
+  var classes = element.getAttribute('class') || '';
46
+  if ((' ' + classes + ' ').indexOf(' ' + className + ' ') == -1) {
47
+    if (classes) {
48
+      classes += ' ';
49
+    }
50
+    element.setAttribute('class', classes + className);
51
+  }
52
+};
53
+
54
+/**
55
+ * Remove a CSS class from a element.
56
+ * Similar to Closure's goog.dom.classes.remove, except it handles SVG elements.
57
+ * @param {!Element} element DOM element to remove class from.
58
+ * @param {string} className Name of class to remove.
59
+ * @private
60
+ */
61
+Blockly.removeClass_ = function(element, className) {
62
+  var classes = element.getAttribute('class');
63
+  if ((' ' + classes + ' ').indexOf(' ' + className + ' ') != -1) {
64
+    var classList = classes.split(/\s+/);
65
+    for (var i = 0; i < classList.length; i++) {
66
+      if (!classList[i] || classList[i] == className) {
67
+        classList.splice(i, 1);
68
+        i--;
69
+      }
70
+    }
71
+    if (classList.length) {
72
+      element.setAttribute('class', classList.join(' '));
73
+    } else {
74
+      element.removeAttribute('class');
75
+    }
76
+  }
77
+};
78
+
79
+/**
80
+ * Checks if an element has the specified CSS class.
81
+ * Similar to Closure's goog.dom.classes.has, except it handles SVG elements.
82
+ * @param {!Element} element DOM element to check.
83
+ * @param {string} className Name of class to check.
84
+ * @return {boolean} True if class exists, false otherwise.
85
+ * @private
86
+ */
87
+Blockly.hasClass_ = function(element, className) {
88
+  var classes = element.getAttribute('class');
89
+  return (' ' + classes + ' ').indexOf(' ' + className + ' ') != -1;
90
+};
91
+
92
+/**
93
+ * Bind an event to a function call.
94
+ * @param {!Node} node Node upon which to listen.
95
+ * @param {string} name Event name to listen to (e.g. 'mousedown').
96
+ * @param {Object} thisObject The value of 'this' in the function.
97
+ * @param {!Function} func Function to call when event is triggered.
98
+ * @return {!Array.<!Array>} Opaque data that can be passed to unbindEvent_.
99
+ * @private
100
+ */
101
+Blockly.bindEvent_ = function(node, name, thisObject, func) {
102
+  if (thisObject) {
103
+    var wrapFunc = function(e) {
104
+      func.call(thisObject, e);
105
+    };
106
+  } else {
107
+    var wrapFunc = func;
108
+  }
109
+  node.addEventListener(name, wrapFunc, false);
110
+  var bindData = [[node, name, wrapFunc]];
111
+  // Add equivalent touch event.
112
+  if (name in Blockly.bindEvent_.TOUCH_MAP) {
113
+    wrapFunc = function(e) {
114
+      // Punt on multitouch events.
115
+      if (e.changedTouches.length == 1) {
116
+        // Map the touch event's properties to the event.
117
+        var touchPoint = e.changedTouches[0];
118
+        e.clientX = touchPoint.clientX;
119
+        e.clientY = touchPoint.clientY;
120
+      }
121
+      func.call(thisObject, e);
122
+      // Stop the browser from scrolling/zooming the page.
123
+      e.preventDefault();
124
+    };
125
+    for (var i = 0, eventName;
126
+         eventName = Blockly.bindEvent_.TOUCH_MAP[name][i]; i++) {
127
+      node.addEventListener(eventName, wrapFunc, false);
128
+      bindData.push([node, eventName, wrapFunc]);
129
+    }
130
+  }
131
+  return bindData;
132
+};
133
+
134
+/**
135
+ * The TOUCH_MAP lookup dictionary specifies additional touch events to fire,
136
+ * in conjunction with mouse events.
137
+ * @type {Object}
138
+ */
139
+Blockly.bindEvent_.TOUCH_MAP = {};
140
+if (goog.events.BrowserFeature.TOUCH_ENABLED) {
141
+  Blockly.bindEvent_.TOUCH_MAP = {
142
+    'mousedown': ['touchstart'],
143
+    'mousemove': ['touchmove'],
144
+    'mouseup': ['touchend', 'touchcancel']
145
+  };
146
+}
147
+
148
+/**
149
+ * Unbind one or more events event from a function call.
150
+ * @param {!Array.<!Array>} bindData Opaque data from bindEvent_.  This list is
151
+ *     emptied during the course of calling this function.
152
+ * @return {!Function} The function call.
153
+ * @private
154
+ */
155
+Blockly.unbindEvent_ = function(bindData) {
156
+  while (bindData.length) {
157
+    var bindDatum = bindData.pop();
158
+    var node = bindDatum[0];
159
+    var name = bindDatum[1];
160
+    var func = bindDatum[2];
161
+    node.removeEventListener(name, func, false);
162
+  }
163
+  return func;
164
+};
165
+
166
+/**
167
+ * Fire a synthetic event synchronously.
168
+ * @param {!EventTarget} node The event's target node.
169
+ * @param {string} eventName Name of event (e.g. 'click').
170
+ */
171
+Blockly.fireUiEventNow = function(node, eventName) {
172
+  // Remove the event from the anti-duplicate database.
173
+  var list = Blockly.fireUiEvent.DB_[eventName];
174
+  if (list) {
175
+    var i = list.indexOf(node);
176
+    if (i != -1) {
177
+      list.splice(i, 1);
178
+    }
179
+  }
180
+  // Fire the event in a browser-compatible way.
181
+  if (document.createEvent) {
182
+    // W3
183
+    var evt = document.createEvent('UIEvents');
184
+    evt.initEvent(eventName, true, true);  // event type, bubbling, cancelable
185
+    node.dispatchEvent(evt);
186
+  } else if (document.createEventObject) {
187
+    // MSIE
188
+    var evt = document.createEventObject();
189
+    node.fireEvent('on' + eventName, evt);
190
+  } else {
191
+    throw 'FireEvent: No event creation mechanism.';
192
+  }
193
+};
194
+
195
+/**
196
+ * Fire a synthetic event asynchronously.  Groups of simultaneous events (e.g.
197
+ * a tree of blocks being deleted) are merged into one event.
198
+ * @param {!EventTarget} node The event's target node.
199
+ * @param {string} eventName Name of event (e.g. 'click').
200
+ */
201
+Blockly.fireUiEvent = function(node, eventName) {
202
+  var list = Blockly.fireUiEvent.DB_[eventName];
203
+  if (list) {
204
+    if (list.indexOf(node) != -1) {
205
+      // This event is already scheduled to fire.
206
+      return;
207
+    }
208
+    list.push(node);
209
+  } else {
210
+    Blockly.fireUiEvent.DB_[eventName] = [node];
211
+  }
212
+  var fire = function() {
213
+    Blockly.fireUiEventNow(node, eventName);
214
+  };
215
+  setTimeout(fire, 0);
216
+};
217
+
218
+/**
219
+ * Database of upcoming firing event types.
220
+ * Used to fire only one event after multiple changes.
221
+ * @type {!Object}
222
+ * @private
223
+ */
224
+Blockly.fireUiEvent.DB_ = {};
225
+
226
+/**
227
+ * Don't do anything for this event, just halt propagation.
228
+ * @param {!Event} e An event.
229
+ */
230
+Blockly.noEvent = function(e) {
231
+  // This event has been handled.  No need to bubble up to the document.
232
+  e.preventDefault();
233
+  e.stopPropagation();
234
+};
235
+
236
+/**
237
+ * Is this event targeting a text input widget?
238
+ * @param {!Event} e An event.
239
+ * @return {boolean} True if text input.
240
+ * @private
241
+ */
242
+Blockly.isTargetInput_ = function(e) {
243
+  return e.target.type == 'textarea' || e.target.type == 'text' ||
244
+         e.target.type == 'number' || e.target.type == 'email' ||
245
+         e.target.type == 'password' || e.target.type == 'search' ||
246
+         e.target.type == 'tel' || e.target.type == 'url' ||
247
+         e.target.isContentEditable;
248
+};
249
+
250
+/**
251
+ * Return the coordinates of the top-left corner of this element relative to
252
+ * its parent.  Only for SVG elements and children (e.g. rect, g, path).
253
+ * @param {!Element} element SVG element to find the coordinates of.
254
+ * @return {!goog.math.Coordinate} Object with .x and .y properties.
255
+ * @private
256
+ */
257
+Blockly.getRelativeXY_ = function(element) {
258
+  var xy = new goog.math.Coordinate(0, 0);
259
+  // First, check for x and y attributes.
260
+  var x = element.getAttribute('x');
261
+  if (x) {
262
+    xy.x = parseInt(x, 10);
263
+  }
264
+  var y = element.getAttribute('y');
265
+  if (y) {
266
+    xy.y = parseInt(y, 10);
267
+  }
268
+  // Second, check for transform="translate(...)" attribute.
269
+  var transform = element.getAttribute('transform');
270
+  var r = transform && transform.match(Blockly.getRelativeXY_.XY_REGEXP_);
271
+  if (r) {
272
+    xy.x += parseFloat(r[1]);
273
+    if (r[3]) {
274
+      xy.y += parseFloat(r[3]);
275
+    }
276
+  }
277
+  return xy;
278
+};
279
+
280
+/**
281
+ * Static regex to pull the x,y values out of an SVG translate() directive.
282
+ * Note that Firefox and IE (9,10) return 'translate(12)' instead of
283
+ * 'translate(12, 0)'.
284
+ * Note that IE (9,10) returns 'translate(16 8)' instead of 'translate(16, 8)'.
285
+ * Note that IE has been reported to return scientific notation (0.123456e-42).
286
+ * @type {!RegExp}
287
+ * @private
288
+ */
289
+Blockly.getRelativeXY_.XY_REGEXP_ =
290
+    /translate\(\s*([-+\d.e]+)([ ,]\s*([-+\d.e]+)\s*\))?/;
291
+
292
+/**
293
+ * Return the absolute coordinates of the top-left corner of this element,
294
+ * scales that after canvas SVG element, if it's a descendant.
295
+ * The origin (0,0) is the top-left corner of the Blockly SVG.
296
+ * @param {!Element} element Element to find the coordinates of.
297
+ * @param {!Blockly.Workspace} workspace Element must be in this workspace.
298
+ * @return {!goog.math.Coordinate} Object with .x and .y properties.
299
+ * @private
300
+ */
301
+Blockly.getSvgXY_ = function(element, workspace) {
302
+  var x = 0;
303
+  var y = 0;
304
+  var scale = 1;
305
+  if (goog.dom.contains(workspace.getCanvas(), element) ||
306
+      goog.dom.contains(workspace.getBubbleCanvas(), element)) {
307
+    // Before the SVG canvas, scale the coordinates.
308
+    scale = workspace.scale;
309
+  }
310
+  do {
311
+    // Loop through this block and every parent.
312
+    var xy = Blockly.getRelativeXY_(element);
313
+    if (element == workspace.getCanvas() ||
314
+        element == workspace.getBubbleCanvas()) {
315
+      // After the SVG canvas, don't scale the coordinates.
316
+      scale = 1;
317
+    }
318
+    x += xy.x * scale;
319
+    y += xy.y * scale;
320
+    element = element.parentNode;
321
+  } while (element && element != workspace.options.svg);
322
+  return new goog.math.Coordinate(x, y);
323
+};
324
+
325
+/**
326
+ * Helper method for creating SVG elements.
327
+ * @param {string} name Element's tag name.
328
+ * @param {!Object} attrs Dictionary of attribute names and values.
329
+ * @param {Element} parent Optional parent on which to append the element.
330
+ * @param {Blockly.Workspace=} opt_workspace Optional workspace for access to
331
+ *     context (scale...).
332
+ * @return {!SVGElement} Newly created SVG element.
333
+ */
334
+Blockly.createSvgElement = function(name, attrs, parent, opt_workspace) {
335
+  var e = /** @type {!SVGElement} */ (
336
+      document.createElementNS(Blockly.SVG_NS, name));
337
+  for (var key in attrs) {
338
+    e.setAttribute(key, attrs[key]);
339
+  }
340
+  // IE defines a unique attribute "runtimeStyle", it is NOT applied to
341
+  // elements created with createElementNS. However, Closure checks for IE
342
+  // and assumes the presence of the attribute and crashes.
343
+  if (document.body.runtimeStyle) {  // Indicates presence of IE-only attr.
344
+    e.runtimeStyle = e.currentStyle = e.style;
345
+  }
346
+  if (parent) {
347
+    parent.appendChild(e);
348
+  }
349
+  return e;
350
+};
351
+
352
+/**
353
+ * Deselect any selections on the webpage.
354
+ * Chrome will select text outside the SVG when double-clicking.
355
+ * Deselect this text, so that it doesn't mess up any subsequent drag.
356
+ */
357
+Blockly.removeAllRanges = function() {
358
+  if (window.getSelection) {
359
+    setTimeout(function() {
360
+        try {
361
+          var selection = window.getSelection();
362
+          if (!selection.isCollapsed) {
363
+            selection.removeAllRanges();
364
+          }
365
+        } catch (e) {
366
+          // MSIE throws 'error 800a025e' here.
367
+        }
368
+      }, 0);
369
+  }
370
+};
371
+
372
+/**
373
+ * Is this event a right-click?
374
+ * @param {!Event} e Mouse event.
375
+ * @return {boolean} True if right-click.
376
+ */
377
+Blockly.isRightButton = function(e) {
378
+  if (e.ctrlKey && goog.userAgent.MAC) {
379
+    // Control-clicking on Mac OS X is treated as a right-click.
380
+    // WebKit on Mac OS X fails to change button to 2 (but Gecko does).
381
+    return true;
382
+  }
383
+  return e.button == 2;
384
+};
385
+
386
+/**
387
+ * Return the converted coordinates of the given mouse event.
388
+ * The origin (0,0) is the top-left corner of the Blockly svg.
389
+ * @param {!Event} e Mouse event.
390
+ * @param {!Element} svg SVG element.
391
+ * @return {!Object} Object with .x and .y properties.
392
+ */
393
+Blockly.mouseToSvg = function(e, svg) {
394
+  var svgPoint = svg.createSVGPoint();
395
+  svgPoint.x = e.clientX;
396
+  svgPoint.y = e.clientY;
397
+  var matrix = svg.getScreenCTM();
398
+  matrix = matrix.inverse();
399
+  return svgPoint.matrixTransform(matrix);
400
+};
401
+
402
+/**
403
+ * Given an array of strings, return the length of the shortest one.
404
+ * @param {!Array.<string>} array Array of strings.
405
+ * @return {number} Length of shortest string.
406
+ */
407
+Blockly.shortestStringLength = function(array) {
408
+  if (!array.length) {
409
+    return 0;
410
+  }
411
+  var len = array[0].length;
412
+  for (var i = 1; i < array.length; i++) {
413
+    len = Math.min(len, array[i].length);
414
+  }
415
+  return len;
416
+};
417
+
418
+/**
419
+ * Given an array of strings, return the length of the common prefix.
420
+ * Words may not be split.  Any space after a word is included in the length.
421
+ * @param {!Array.<string>} array Array of strings.
422
+ * @param {number=} opt_shortest Length of shortest string.
423
+ * @return {number} Length of common prefix.
424
+ */
425
+Blockly.commonWordPrefix = function(array, opt_shortest) {
426
+  if (!array.length) {
427
+    return 0;
428
+  } else if (array.length == 1) {
429
+    return array[0].length;
430
+  }
431
+  var wordPrefix = 0;
432
+  var max = opt_shortest || Blockly.shortestStringLength(array);
433
+  for (var len = 0; len < max; len++) {
434
+    var letter = array[0][len];
435
+    for (var i = 1; i < array.length; i++) {
436
+      if (letter != array[i][len]) {
437
+        return wordPrefix;
438
+      }
439
+    }
440
+    if (letter == ' ') {
441
+      wordPrefix = len + 1;
442
+    }
443
+  }
444
+  for (var i = 1; i < array.length; i++) {
445
+    var letter = array[i][len];
446
+    if (letter && letter != ' ') {
447
+      return wordPrefix;
448
+    }
449
+  }
450
+  return max;
451
+};
452
+
453
+/**
454
+ * Given an array of strings, return the length of the common suffix.
455
+ * Words may not be split.  Any space after a word is included in the length.
456
+ * @param {!Array.<string>} array Array of strings.
457
+ * @param {number=} opt_shortest Length of shortest string.
458
+ * @return {number} Length of common suffix.
459
+ */
460
+Blockly.commonWordSuffix = function(array, opt_shortest) {
461
+  if (!array.length) {
462
+    return 0;
463
+  } else if (array.length == 1) {
464
+    return array[0].length;
465
+  }
466
+  var wordPrefix = 0;
467
+  var max = opt_shortest || Blockly.shortestStringLength(array);
468
+  for (var len = 0; len < max; len++) {
469
+    var letter = array[0].substr(-len - 1, 1);
470
+    for (var i = 1; i < array.length; i++) {
471
+      if (letter != array[i].substr(-len - 1, 1)) {
472
+        return wordPrefix;
473
+      }
474
+    }
475
+    if (letter == ' ') {
476
+      wordPrefix = len + 1;
477
+    }
478
+  }
479
+  for (var i = 1; i < array.length; i++) {
480
+    var letter = array[i].charAt(array[i].length - len - 1);
481
+    if (letter && letter != ' ') {
482
+      return wordPrefix;
483
+    }
484
+  }
485
+  return max;
486
+};
487
+
488
+/**
489
+ * Is the given string a number (includes negative and decimals).
490
+ * @param {string} str Input string.
491
+ * @return {boolean} True if number, false otherwise.
492
+ */
493
+Blockly.isNumber = function(str) {
494
+  return !!str.match(/^\s*-?\d+(\.\d+)?\s*$/);
495
+};
496
+
497
+/**
498
+ * Parse a string with any number of interpolation tokens (%1, %2, ...).
499
+ * '%' characters may be self-escaped (%%).
500
+ * @param {string} message Text containing interpolation tokens.
501
+ * @return {!Array.<string|number>} Array of strings and numbers.
502
+ */
503
+Blockly.tokenizeInterpolation = function(message) {
504
+  var tokens = [];
505
+  var chars = message.split('');
506
+  chars.push('');  // End marker.
507
+  // Parse the message with a finite state machine.
508
+  // 0 - Base case.
509
+  // 1 - % found.
510
+  // 2 - Digit found.
511
+  var state = 0;
512
+  var buffer = [];
513
+  var number = null;
514
+  for (var i = 0; i < chars.length; i++) {
515
+    var c = chars[i];
516
+    if (state == 0) {
517
+      if (c == '%') {
518
+        state = 1;  // Start escape.
519
+      } else {
520
+        buffer.push(c);  // Regular char.
521
+      }
522
+    } else if (state == 1) {
523
+      if (c == '%') {
524
+        buffer.push(c);  // Escaped %: %%
525
+        state = 0;
526
+      } else if ('0' <= c && c <= '9') {
527
+        state = 2;
528
+        number = c;
529
+        var text = buffer.join('');
530
+        if (text) {
531
+          tokens.push(text);
532
+        }
533
+        buffer.length = 0;
534
+      } else {
535
+        buffer.push('%', c);  // Not an escape: %a
536
+        state = 0;
537
+      }
538
+    } else if (state == 2) {
539
+      if ('0' <= c && c <= '9') {
540
+        number += c;  // Multi-digit number.
541
+      } else {
542
+        tokens.push(parseInt(number, 10));
543
+        i--;  // Parse this char again.
544
+        state = 0;
545
+      }
546
+    }
547
+  }
548
+  var text = buffer.join('');
549
+  if (text) {
550
+    tokens.push(text);
551
+  }
552
+  return tokens;
553
+};

+ 189 - 0
src/blockly/core/variables.js

@@ -0,0 +1,189 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Utility functions for handling variables.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Variables');
28
+
29
+// TODO(scr): Fix circular dependencies
30
+// goog.require('Blockly.Block');
31
+goog.require('Blockly.Workspace');
32
+goog.require('goog.string');
33
+
34
+
35
+/**
36
+ * Category to separate variable names from procedures and generated functions.
37
+ */
38
+Blockly.Variables.NAME_TYPE = 'VARIABLE';
39
+
40
+/**
41
+ * Find all user-created variables.
42
+ * @param {!Blockly.Block|!Blockly.Workspace} root Root block or workspace.
43
+ * @return {!Array.<string>} Array of variable names.
44
+ */
45
+Blockly.Variables.allVariables = function(root) {
46
+  var blocks;
47
+  if (root.getDescendants) {
48
+    // Root is Block.
49
+    blocks = root.getDescendants();
50
+  } else if (root.getAllBlocks) {
51
+    // Root is Workspace.
52
+    blocks = root.getAllBlocks();
53
+  } else {
54
+    throw 'Not Block or Workspace: ' + root;
55
+  }
56
+  var variableHash = Object.create(null);
57
+  // Iterate through every block and add each variable to the hash.
58
+  for (var x = 0; x < blocks.length; x++) {
59
+    if (blocks[x].getVars) {
60
+      var blockVariables = blocks[x].getVars();
61
+      for (var y = 0; y < blockVariables.length; y++) {
62
+        var varName = blockVariables[y];
63
+        // Variable name may be null if the block is only half-built.
64
+        if (varName) {
65
+          variableHash[varName.toLowerCase()] = varName;
66
+        }
67
+      }
68
+    }
69
+  }
70
+  // Flatten the hash into a list.
71
+  var variableList = [];
72
+  for (var name in variableHash) {
73
+    variableList.push(variableHash[name]);
74
+  }
75
+  return variableList;
76
+};
77
+
78
+/**
79
+ * Find all instances of the specified variable and rename them.
80
+ * @param {string} oldName Variable to rename.
81
+ * @param {string} newName New variable name.
82
+ * @param {!Blockly.Workspace} workspace Workspace rename variables in.
83
+ */
84
+Blockly.Variables.renameVariable = function(oldName, newName, workspace) {
85
+  var blocks = workspace.getAllBlocks();
86
+  // Iterate through every block.
87
+  for (var i = 0; i < blocks.length; i++) {
88
+    if (blocks[i].renameVar) {
89
+      blocks[i].renameVar(oldName, newName);
90
+    }
91
+  }
92
+};
93
+
94
+/**
95
+ * Construct the blocks required by the flyout for the variable category.
96
+ * @param {!Blockly.Workspace} workspace The workspace contianing variables.
97
+ * @return {!Array.<!Element>} Array of XML block elements.
98
+ */
99
+Blockly.Variables.flyoutCategory = function(workspace) {
100
+  var variableList = Blockly.Variables.allVariables(workspace);
101
+  variableList.sort(goog.string.caseInsensitiveCompare);
102
+  // In addition to the user's variables, we also want to display the default
103
+  // variable name at the top.  We also don't want this duplicated if the
104
+  // user has created a variable of the same name.
105
+  goog.array.remove(variableList, Blockly.Msg.VARIABLES_DEFAULT_NAME);
106
+  variableList.unshift(Blockly.Msg.VARIABLES_DEFAULT_NAME);
107
+
108
+  var xmlList = [];
109
+  for (var i = 0; i < variableList.length; i++) {
110
+    if (Blockly.Blocks['variables_set']) {
111
+      // <block type="variables_set" gap="8">
112
+      //   <field name="VAR">item</field>
113
+      // </block>
114
+      var block = goog.dom.createDom('block');
115
+      block.setAttribute('type', 'variables_set');
116
+      if (Blockly.Blocks['variables_get']) {
117
+        block.setAttribute('gap', 8);
118
+      }
119
+      var field = goog.dom.createDom('field', null, variableList[i]);
120
+      field.setAttribute('name', 'VAR');
121
+      block.appendChild(field);
122
+      xmlList.push(block);
123
+    }
124
+    if (Blockly.Blocks['variables_get']) {
125
+      // <block type="variables_get" gap="24">
126
+      //   <field name="VAR">item</field>
127
+      // </block>
128
+      var block = goog.dom.createDom('block');
129
+      block.setAttribute('type', 'variables_get');
130
+      if (Blockly.Blocks['variables_set']) {
131
+        block.setAttribute('gap', 24);
132
+      }
133
+      var field = goog.dom.createDom('field', null, variableList[i]);
134
+      field.setAttribute('name', 'VAR');
135
+      block.appendChild(field);
136
+      xmlList.push(block);
137
+    }
138
+  }
139
+  return xmlList;
140
+};
141
+
142
+/**
143
+* Return a new variable name that is not yet being used. This will try to
144
+* generate single letter variable names in the range 'i' to 'z' to start with.
145
+* If no unique name is located it will try 'i' to 'z', 'a' to 'h',
146
+* then 'i2' to 'z2' etc.  Skip 'l'.
147
+ * @param {!Blockly.Workspace} workspace The workspace to be unique in.
148
+* @return {string} New variable name.
149
+*/
150
+Blockly.Variables.generateUniqueName = function(workspace) {
151
+  var variableList = Blockly.Variables.allVariables(workspace);
152
+  var newName = '';
153
+  if (variableList.length) {
154
+    var nameSuffix = 1;
155
+    var letters = 'ijkmnopqrstuvwxyzabcdefgh';  // No 'l'.
156
+    var letterIndex = 0;
157
+    var potName = letters.charAt(letterIndex);
158
+    while (!newName) {
159
+      var inUse = false;
160
+      for (var i = 0; i < variableList.length; i++) {
161
+        if (variableList[i].toLowerCase() == potName) {
162
+          // This potential name is already used.
163
+          inUse = true;
164
+          break;
165
+        }
166
+      }
167
+      if (inUse) {
168
+        // Try the next potential name.
169
+        letterIndex++;
170
+        if (letterIndex == letters.length) {
171
+          // Reached the end of the character sequence so back to 'i'.
172
+          // a new suffix.
173
+          letterIndex = 0;
174
+          nameSuffix++;
175
+        }
176
+        potName = letters.charAt(letterIndex);
177
+        if (nameSuffix > 1) {
178
+          potName += nameSuffix;
179
+        }
180
+      } else {
181
+        // We can use the current potential name.
182
+        newName = potName;
183
+      }
184
+    }
185
+  } else {
186
+    newName = 'i';
187
+  }
188
+  return newName;
189
+};

File diff suppressed because it is too large
+ 165 - 0
src/blockly/core/warning.js


+ 152 - 0
src/blockly/core/widgetdiv.js

@@ -0,0 +1,152 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2013 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview A div that floats on top of Blockly.  This singleton contains
23
+ *     temporary HTML UI widgets that the user is currently interacting with.
24
+ *     E.g. text input areas, colour pickers, context menus.
25
+ * @author fraser@google.com (Neil Fraser)
26
+ */
27
+'use strict';
28
+
29
+goog.provide('Blockly.WidgetDiv');
30
+
31
+goog.require('Blockly.Css');
32
+goog.require('goog.dom');
33
+goog.require('goog.style');
34
+
35
+
36
+/**
37
+ * The HTML container.  Set once by Blockly.WidgetDiv.createDom.
38
+ * @type {Element}
39
+ */
40
+Blockly.WidgetDiv.DIV = null;
41
+
42
+/**
43
+ * The object currently using this container.
44
+ * @type {Object}
45
+ * @private
46
+ */
47
+Blockly.WidgetDiv.owner_ = null;
48
+
49
+/**
50
+ * Optional cleanup function set by whichever object uses the widget.
51
+ * @type {Function}
52
+ * @private
53
+ */
54
+Blockly.WidgetDiv.dispose_ = null;
55
+
56
+/**
57
+ * Create the widget div and inject it onto the page.
58
+ */
59
+Blockly.WidgetDiv.createDom = function() {
60
+  if (Blockly.WidgetDiv.DIV) {
61
+    return;  // Already created.
62
+  }
63
+  // Create an HTML container for popup overlays (e.g. editor widgets).
64
+  Blockly.WidgetDiv.DIV = goog.dom.createDom('div', 'blocklyWidgetDiv');
65
+  document.body.appendChild(Blockly.WidgetDiv.DIV);
66
+};
67
+
68
+/**
69
+ * Initialize and display the widget div.  Close the old one if needed.
70
+ * @param {!Object} newOwner The object that will be using this container.
71
+ * @param {boolean} rtl Right-to-left (true) or left-to-right (false).
72
+ * @param {Function} dispose Optional cleanup function to be run when the widget
73
+ *   is closed.
74
+ */
75
+Blockly.WidgetDiv.show = function(newOwner, rtl, dispose) {
76
+  Blockly.WidgetDiv.hide();
77
+  Blockly.WidgetDiv.owner_ = newOwner;
78
+  Blockly.WidgetDiv.dispose_ = dispose;
79
+  // Temporarily move the widget to the top of the screen so that it does not
80
+  // cause a scrollbar jump in Firefox when displayed.
81
+  var xy = goog.style.getViewportPageOffset(document);
82
+  Blockly.WidgetDiv.DIV.style.top = xy.y + 'px';
83
+  Blockly.WidgetDiv.DIV.style.direction = rtl ? 'rtl' : 'ltr';
84
+  Blockly.WidgetDiv.DIV.style.display = 'block';
85
+};
86
+
87
+/**
88
+ * Destroy the widget and hide the div.
89
+ */
90
+Blockly.WidgetDiv.hide = function() {
91
+  if (Blockly.WidgetDiv.owner_) {
92
+    Blockly.WidgetDiv.DIV.style.display = 'none';
93
+    Blockly.WidgetDiv.DIV.style.left = '';
94
+    Blockly.WidgetDiv.DIV.style.top = '';
95
+    Blockly.WidgetDiv.DIV.style.height = '';
96
+    Blockly.WidgetDiv.dispose_ && Blockly.WidgetDiv.dispose_();
97
+    Blockly.WidgetDiv.owner_ = null;
98
+    Blockly.WidgetDiv.dispose_ = null;
99
+    goog.dom.removeChildren(Blockly.WidgetDiv.DIV);
100
+  }
101
+};
102
+
103
+/**
104
+ * Is the container visible?
105
+ * @return {boolean} True if visible.
106
+ */
107
+Blockly.WidgetDiv.isVisible = function() {
108
+  return !!Blockly.WidgetDiv.owner_;
109
+};
110
+
111
+/**
112
+ * Destroy the widget and hide the div if it is being used by the specified
113
+ *   object.
114
+ * @param {!Object} oldOwner The object that was using this container.
115
+ */
116
+Blockly.WidgetDiv.hideIfOwner = function(oldOwner) {
117
+  if (Blockly.WidgetDiv.owner_ == oldOwner) {
118
+    Blockly.WidgetDiv.hide();
119
+  }
120
+};
121
+
122
+/**
123
+ * Position the widget at a given location.  Prevent the widget from going
124
+ * offscreen top or left (right in RTL).
125
+ * @param {number} anchorX Horizontal location (window coorditates, not body).
126
+ * @param {number} anchorY Vertical location (window coorditates, not body).
127
+ * @param {!goog.math.Size} windowSize Height/width of window.
128
+ * @param {!goog.math.Coordinate} scrollOffset X/y of window scrollbars.
129
+ * @param {boolean} rtl True if RTL, false if LTR.
130
+ */
131
+Blockly.WidgetDiv.position = function(anchorX, anchorY, windowSize,
132
+                                      scrollOffset, rtl) {
133
+  // Don't let the widget go above the top edge of the window.
134
+  if (anchorY < scrollOffset.y) {
135
+    anchorY = scrollOffset.y;
136
+  }
137
+  if (rtl) {
138
+    // Don't let the widget go right of the right edge of the window.
139
+    if (anchorX > windowSize.width + scrollOffset.x) {
140
+      anchorX = windowSize.width + scrollOffset.x;
141
+    }
142
+  } else {
143
+    // Don't let the widget go left of the left edge of the window.
144
+    if (anchorX < scrollOffset.x) {
145
+      anchorX = scrollOffset.x;
146
+    }
147
+  }
148
+  Blockly.WidgetDiv.DIV.style.left = anchorX + 'px';
149
+  Blockly.WidgetDiv.DIV.style.top = anchorY + 'px';
150
+  Blockly.WidgetDiv.DIV.style.height =
151
+      (windowSize.height - anchorY + scrollOffset.y) + 'px';
152
+};

+ 187 - 0
src/blockly/core/workspace.js

@@ -0,0 +1,187 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Object representing a workspace.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Workspace');
28
+
29
+goog.require('goog.math');
30
+
31
+
32
+/**
33
+ * Class for a workspace.  This is a data structure that contains blocks.
34
+ * There is no UI, and can be created headlessly.
35
+ * @param {Object=} opt_options Dictionary of options.
36
+ * @constructor
37
+ */
38
+Blockly.Workspace = function(opt_options) {
39
+  /** @type {!Object} */
40
+  this.options = opt_options || {};
41
+  /** @type {boolean} */
42
+  this.RTL = !!this.options.RTL;
43
+  /** @type {!Array.<!Blockly.Block>} */
44
+  this.topBlocks_ = [];
45
+};
46
+
47
+/**
48
+ * Workspaces may be headless.
49
+ * @type {boolean} True if visible.  False if headless.
50
+ */
51
+Blockly.Workspace.prototype.rendered = false;
52
+
53
+/**
54
+ * Dispose of this workspace.
55
+ * Unlink from all DOM elements to prevent memory leaks.
56
+ */
57
+Blockly.Workspace.prototype.dispose = function() {
58
+  this.clear();
59
+};
60
+
61
+/**
62
+ * Angle away from the horizontal to sweep for blocks.  Order of execution is
63
+ * generally top to bottom, but a small angle changes the scan to give a bit of
64
+ * a left to right bias (reversed in RTL).  Units are in degrees.
65
+ * See: http://tvtropes.org/pmwiki/pmwiki.php/Main/DiagonalBilling.
66
+ */
67
+Blockly.Workspace.SCAN_ANGLE = 3;
68
+
69
+/**
70
+ * Add a block to the list of top blocks.
71
+ * @param {!Blockly.Block} block Block to remove.
72
+ */
73
+Blockly.Workspace.prototype.addTopBlock = function(block) {
74
+  this.topBlocks_.push(block);
75
+  this.fireChangeEvent();
76
+};
77
+
78
+/**
79
+ * Remove a block from the list of top blocks.
80
+ * @param {!Blockly.Block} block Block to remove.
81
+ */
82
+Blockly.Workspace.prototype.removeTopBlock = function(block) {
83
+  var found = false;
84
+  for (var child, i = 0; child = this.topBlocks_[i]; i++) {
85
+    if (child == block) {
86
+      this.topBlocks_.splice(i, 1);
87
+      found = true;
88
+      break;
89
+    }
90
+  }
91
+  if (!found) {
92
+    throw 'Block not present in workspace\'s list of top-most blocks.';
93
+  }
94
+  this.fireChangeEvent();
95
+};
96
+
97
+/**
98
+ * Finds the top-level blocks and returns them.  Blocks are optionally sorted
99
+ * by position; top to bottom (with slight LTR or RTL bias).
100
+ * @param {boolean} ordered Sort the list if true.
101
+ * @return {!Array.<!Blockly.Block>} The top-level block objects.
102
+ */
103
+Blockly.Workspace.prototype.getTopBlocks = function(ordered) {
104
+  // Copy the topBlocks_ list.
105
+  var blocks = [].concat(this.topBlocks_);
106
+  if (ordered && blocks.length > 1) {
107
+    var offset = Math.sin(goog.math.toRadians(Blockly.Workspace.SCAN_ANGLE));
108
+    if (this.RTL) {
109
+      offset *= -1;
110
+    }
111
+    blocks.sort(function(a, b) {
112
+      var aXY = a.getRelativeToSurfaceXY();
113
+      var bXY = b.getRelativeToSurfaceXY();
114
+      return (aXY.y + offset * aXY.x) - (bXY.y + offset * bXY.x);
115
+    });
116
+  }
117
+  return blocks;
118
+};
119
+
120
+/**
121
+ * Find all blocks in workspace.  No particular order.
122
+ * @return {!Array.<!Blockly.Block>} Array of blocks.
123
+ */
124
+Blockly.Workspace.prototype.getAllBlocks = function() {
125
+  var blocks = this.getTopBlocks(false);
126
+  for (var i = 0; i < blocks.length; i++) {
127
+    blocks.push.apply(blocks, blocks[i].getChildren());
128
+  }
129
+  return blocks;
130
+};
131
+
132
+/**
133
+ * Dispose of all blocks in workspace.
134
+ */
135
+Blockly.Workspace.prototype.clear = function() {
136
+  while (this.topBlocks_.length) {
137
+    this.topBlocks_[0].dispose();
138
+  }
139
+};
140
+
141
+/**
142
+ * Returns the horizontal offset of the workspace.
143
+ * Intended for LTR/RTL compatibility in XML.
144
+ * Not relevant for a headless workspace.
145
+ * @return {number} Width.
146
+ */
147
+Blockly.Workspace.prototype.getWidth = function() {
148
+  return 0;
149
+};
150
+
151
+/**
152
+ * Finds the block with the specified ID in this workspace.
153
+ * @param {string} id ID of block to find.
154
+ * @return {Blockly.Block} The matching block, or null if not found.
155
+ */
156
+Blockly.Workspace.prototype.getBlockById = function(id) {
157
+  // If this O(n) function fails to scale well, maintain a hash table of IDs.
158
+  var blocks = this.getAllBlocks();
159
+  for (var i = 0, block; block = blocks[i]; i++) {
160
+    if (block.id == id) {
161
+      return block;
162
+    }
163
+  }
164
+  return null;
165
+};
166
+
167
+/**
168
+ * The number of blocks that may be added to the workspace before reaching
169
+ *     the maxBlocks.
170
+ * @return {number} Number of blocks left.
171
+ */
172
+Blockly.Workspace.prototype.remainingCapacity = function() {
173
+  if (isNaN(this.options.maxBlocks)) {
174
+    return Infinity;
175
+  }
176
+  return this.options.maxBlocks - this.getAllBlocks().length;
177
+};
178
+
179
+/**
180
+ * Something on this workspace has changed.
181
+ */
182
+Blockly.Workspace.prototype.fireChangeEvent = function() {
183
+  // NOP.
184
+};
185
+
186
+// Export symbols that would otherwise be renamed by Closure compiler.
187
+Blockly.Workspace.prototype['clear'] = Blockly.Workspace.prototype.clear;

File diff suppressed because it is too large
+ 1031 - 0
src/blockly/core/workspace_svg.js


+ 552 - 0
src/blockly/core/xml.js

@@ -0,0 +1,552 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2012 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview XML reader and writer.
23
+ * @author fraser@google.com (Neil Fraser)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.Xml');
28
+
29
+// TODO(scr): Fix circular dependencies
30
+// goog.require('Blockly.Block');
31
+goog.require('goog.dom');
32
+
33
+
34
+/**
35
+ * Encode a block tree as XML.
36
+ * @param {!Blockly.Workspace} workspace The workspace containing blocks.
37
+ * @return {!Element} XML document.
38
+ */
39
+Blockly.Xml.workspaceToDom = function(workspace) {
40
+  var width;  // Not used in LTR.
41
+  if (workspace.RTL) {
42
+    width = workspace.getWidth();
43
+  }
44
+  var xml = goog.dom.createDom('xml');
45
+  var blocks = workspace.getTopBlocks(true);
46
+  for (var i = 0, block; block = blocks[i]; i++) {
47
+    var element = Blockly.Xml.blockToDom_(block);
48
+    var xy = block.getRelativeToSurfaceXY();
49
+    element.setAttribute('x', Math.round(workspace.RTL ? width - xy.x : xy.x));
50
+    element.setAttribute('y', Math.round(xy.y));
51
+    xml.appendChild(element);
52
+  }
53
+  return xml;
54
+};
55
+
56
+/**
57
+ * Encode a block subtree as XML.
58
+ * @param {!Blockly.Block} block The root block to encode.
59
+ * @return {!Element} Tree of XML elements.
60
+ * @private
61
+ */
62
+Blockly.Xml.blockToDom_ = function(block) {
63
+  var element = goog.dom.createDom(block.isShadow() ? 'shadow' : 'block');
64
+  element.setAttribute('type', block.type);
65
+  if (Blockly.Realtime.isEnabled()) {
66
+    // Only used by realtime.
67
+    element.setAttribute('id', block.id);
68
+  }
69
+  if (block.mutationToDom) {
70
+    // Custom data for an advanced block.
71
+    var mutation = block.mutationToDom();
72
+    if (mutation && (mutation.hasChildNodes() || mutation.hasAttributes())) {
73
+      element.appendChild(mutation);
74
+    }
75
+  }
76
+  function fieldToDom(field) {
77
+    if (field.name && field.EDITABLE) {
78
+      var container = goog.dom.createDom('field', null, field.getValue());
79
+      container.setAttribute('name', field.name);
80
+      element.appendChild(container);
81
+    }
82
+  }
83
+  for (var i = 0, input; input = block.inputList[i]; i++) {
84
+    for (var j = 0, field; field = input.fieldRow[j]; j++) {
85
+      fieldToDom(field);
86
+    }
87
+  }
88
+
89
+  var commentText = block.getCommentText();
90
+  if (commentText) {
91
+    var commentElement = goog.dom.createDom('comment', null, commentText);
92
+    if (typeof block.comment == 'object') {
93
+      commentElement.setAttribute('pinned', block.comment.isVisible());
94
+      var hw = block.comment.getBubbleSize();
95
+      commentElement.setAttribute('h', hw.height);
96
+      commentElement.setAttribute('w', hw.width);
97
+    }
98
+    element.appendChild(commentElement);
99
+  }
100
+
101
+  if (block.data) {
102
+    // Optional text data that round-trips beween blocks and XML.
103
+    // Has no effect.  May be used by 3rd parties for meta information.
104
+    var dataElement = goog.dom.createDom('data', null, block.data);
105
+    element.appendChild(dataElement);
106
+  }
107
+
108
+  for (var i = 0, input; input = block.inputList[i]; i++) {
109
+    var container;
110
+    var empty = true;
111
+    if (input.type == Blockly.DUMMY_INPUT) {
112
+      continue;
113
+    } else {
114
+      var childBlock = input.connection.targetBlock();
115
+      if (input.type == Blockly.INPUT_VALUE) {
116
+        container = goog.dom.createDom('value');
117
+      } else if (input.type == Blockly.NEXT_STATEMENT) {
118
+        container = goog.dom.createDom('statement');
119
+      }
120
+      var shadow = input.connection.getShadowDom();
121
+      if (shadow && (!childBlock || !childBlock.isShadow())) {
122
+        container.appendChild(Blockly.Xml.cloneShadow_(shadow));
123
+      }
124
+      if (childBlock) {
125
+        container.appendChild(Blockly.Xml.blockToDom_(childBlock));
126
+        empty = false;
127
+      }
128
+    }
129
+    container.setAttribute('name', input.name);
130
+    if (!empty) {
131
+      element.appendChild(container);
132
+    }
133
+  }
134
+  if (block.inputsInlineDefault != block.inputsInline) {
135
+    element.setAttribute('inline', block.inputsInline);
136
+  }
137
+  if (block.isCollapsed()) {
138
+    element.setAttribute('collapsed', true);
139
+  }
140
+  if (block.disabled) {
141
+    element.setAttribute('disabled', true);
142
+  }
143
+  if (!block.isDeletable() && !block.isShadow()) {
144
+    element.setAttribute('deletable', false);
145
+  }
146
+  if (!block.isMovable() && !block.isShadow()) {
147
+    element.setAttribute('movable', false);
148
+  }
149
+  if (!block.isEditable()) {
150
+    element.setAttribute('editable', false);
151
+  }
152
+
153
+  var nextBlock = block.getNextBlock();
154
+  if (nextBlock) {
155
+    var container = goog.dom.createDom('next', null,
156
+        Blockly.Xml.blockToDom_(nextBlock));
157
+    element.appendChild(container);
158
+  }
159
+  var shadow = block.nextConnection && block.nextConnection.getShadowDom();
160
+  if (shadow && (!nextBlock || !nextBlock.isShadow())) {
161
+    container.appendChild(Blockly.Xml.cloneShadow_(shadow));
162
+  }
163
+
164
+  return element;
165
+};
166
+
167
+/**
168
+ * Deeply clone the shadow's DOM so that changes don't back-wash to the block.
169
+ * @param {!Element} shadow A tree of XML elements.
170
+ * @return {!Element} A tree of XML elements.
171
+ * @private
172
+ */
173
+Blockly.Xml.cloneShadow_ = function(shadow) {
174
+  shadow = shadow.cloneNode(true);
175
+  // Walk the tree looking for whitespace.  Don't prune whitespace in a tag.
176
+  var node = shadow;
177
+  var textNode;
178
+  while (node) {
179
+    if (node.firstChild) {
180
+      node = node.firstChild;
181
+    } else {
182
+      while (node && !node.nextSibling) {
183
+        textNode = node;
184
+        node = node.parentNode;
185
+        if (textNode.nodeType == 3 && textNode.data.trim() == '' &&
186
+            node.firstChild != textNode) {
187
+          // Prune whitespace after a tag.
188
+          goog.dom.removeNode(textNode);
189
+        }
190
+      }
191
+      if (node) {
192
+        textNode = node;
193
+        node = node.nextSibling;
194
+        if (textNode.nodeType == 3 && textNode.data.trim() == '') {
195
+          // Prune whitespace before a tag.
196
+          goog.dom.removeNode(textNode);
197
+        }
198
+      }
199
+    }
200
+  }
201
+  return shadow;
202
+};
203
+
204
+/**
205
+ * Converts a DOM structure into plain text.
206
+ * Currently the text format is fairly ugly: all one line with no whitespace.
207
+ * @param {!Element} dom A tree of XML elements.
208
+ * @return {string} Text representation.
209
+ */
210
+Blockly.Xml.domToText = function(dom) {
211
+  var oSerializer = new XMLSerializer();
212
+  return oSerializer.serializeToString(dom);
213
+};
214
+
215
+/**
216
+ * Converts a DOM structure into properly indented text.
217
+ * @param {!Element} dom A tree of XML elements.
218
+ * @return {string} Text representation.
219
+ */
220
+Blockly.Xml.domToPrettyText = function(dom) {
221
+  // This function is not guaranteed to be correct for all XML.
222
+  // But it handles the XML that Blockly generates.
223
+  var blob = Blockly.Xml.domToText(dom);
224
+  // Place every open and close tag on its own line.
225
+  var lines = blob.split('<');
226
+  // Indent every line.
227
+  var indent = '';
228
+  for (var i = 1; i < lines.length; i++) {
229
+    var line = lines[i];
230
+    if (line[0] == '/') {
231
+      indent = indent.substring(2);
232
+    }
233
+    lines[i] = indent + '<' + line;
234
+    if (line[0] != '/' && line.slice(-2) != '/>') {
235
+      indent += '  ';
236
+    }
237
+  }
238
+  // Pull simple tags back together.
239
+  // E.g. <foo></foo>
240
+  var text = lines.join('\n');
241
+  text = text.replace(/(<(\w+)\b[^>]*>[^\n]*)\n *<\/\2>/g, '$1</$2>');
242
+  // Trim leading blank line.
243
+  return text.replace(/^\n/, '');
244
+};
245
+
246
+/**
247
+ * Converts plain text into a DOM structure.
248
+ * Throws an error if XML doesn't parse.
249
+ * @param {string} text Text representation.
250
+ * @return {!Element} A tree of XML elements.
251
+ */
252
+Blockly.Xml.textToDom = function(text) {
253
+  var oParser = new DOMParser();
254
+  var dom = oParser.parseFromString(text, 'text/xml');
255
+  // The DOM should have one and only one top-level node, an XML tag.
256
+  if (!dom || !dom.firstChild ||
257
+      dom.firstChild.nodeName.toLowerCase() != 'xml' ||
258
+      dom.firstChild !== dom.lastChild) {
259
+    // Whatever we got back from the parser is not XML.
260
+    throw 'Blockly.Xml.textToDom did not obtain a valid XML tree.';
261
+  }
262
+  return dom.firstChild;
263
+};
264
+
265
+/**
266
+ * Decode an XML DOM and create blocks on the workspace.
267
+ * @param {!Blockly.Workspace} workspace The workspace.
268
+ * @param {!Element} xml XML DOM.
269
+ */
270
+Blockly.Xml.domToWorkspace = function(workspace, xml) {
271
+  var width;  // Not used in LTR.
272
+  if (workspace.RTL) {
273
+    width = workspace.getWidth();
274
+  }
275
+  Blockly.Field.startCache();
276
+  // Safari 7.1.3 is known to provide node lists with extra references to
277
+  // children beyond the lists' length.  Trust the length, do not use the
278
+  // looping pattern of checking the index for an object.
279
+  var childCount = xml.childNodes.length;
280
+  for (var i = 0; i < childCount; i++) {
281
+    var xmlChild = xml.childNodes[i];
282
+    var name = xmlChild.nodeName.toLowerCase();
283
+    if (name == 'block' || name == 'shadow') {
284
+      var block = Blockly.Xml.domToBlock(workspace, xmlChild);
285
+      var blockX = parseInt(xmlChild.getAttribute('x'), 10);
286
+      var blockY = parseInt(xmlChild.getAttribute('y'), 10);
287
+      if (!isNaN(blockX) && !isNaN(blockY)) {
288
+        block.moveBy(workspace.RTL ? width - blockX : blockX, blockY);
289
+      }
290
+    }
291
+  }
292
+  Blockly.Field.stopCache();
293
+};
294
+
295
+/**
296
+ * Decode an XML block tag and create a block (and possibly sub blocks) on the
297
+ * workspace.
298
+ * @param {!Blockly.Workspace} workspace The workspace.
299
+ * @param {!Element} xmlBlock XML block element.
300
+ * @param {boolean=} opt_reuseBlock Optional arg indicating whether to
301
+ *     reinitialize an existing block.
302
+ * @return {!Blockly.Block} The root block created.
303
+ */
304
+Blockly.Xml.domToBlock = function(workspace, xmlBlock, opt_reuseBlock) {
305
+  // Create top-level block.
306
+  var topBlock = Blockly.Xml.domToBlockHeadless_(workspace, xmlBlock,
307
+                                                 opt_reuseBlock);
308
+  if (workspace.rendered) {
309
+    // Hide connections to speed up assembly.
310
+    topBlock.setConnectionsHidden(true);
311
+    // Generate list of all blocks.
312
+    var blocks = topBlock.getDescendants();
313
+    // Render each block.
314
+    for (var i = blocks.length - 1; i >= 0; i--) {
315
+      blocks[i].initSvg();
316
+    }
317
+    for (var i = blocks.length - 1; i >= 0; i--) {
318
+      blocks[i].render(false);
319
+    }
320
+    // Populating the connection database may be defered until after the blocks
321
+    // have renderend.
322
+    setTimeout(function() {
323
+      if (topBlock.workspace) {  // Check that the block hasn't been deleted.
324
+        topBlock.setConnectionsHidden(false);
325
+      }
326
+    }, 1);
327
+    topBlock.updateDisabled();
328
+    // Fire an event to allow scrollbars to resize.
329
+    Blockly.fireUiEvent(window, 'resize');
330
+  }
331
+  return topBlock;
332
+};
333
+
334
+/**
335
+ * Decode an XML block tag and create a block (and possibly sub blocks) on the
336
+ * workspace.
337
+ * @param {!Blockly.Workspace} workspace The workspace.
338
+ * @param {!Element} xmlBlock XML block element.
339
+ * @param {boolean=} opt_reuseBlock Optional arg indicating whether to
340
+ *     reinitialize an existing block.
341
+ * @return {!Blockly.Block} The root block created.
342
+ * @private
343
+ */
344
+Blockly.Xml.domToBlockHeadless_ =
345
+    function(workspace, xmlBlock, opt_reuseBlock) {
346
+  var block = null;
347
+  var prototypeName = xmlBlock.getAttribute('type');
348
+  if (!prototypeName) {
349
+    throw 'Block type unspecified: \n' + xmlBlock.outerHTML;
350
+  }
351
+  var id = xmlBlock.getAttribute('id');
352
+  if (opt_reuseBlock && id) {
353
+    // Only used by realtime.
354
+    block = Blockly.Block.getById(id, workspace);
355
+    // TODO: The following is for debugging.  It should never actually happen.
356
+    if (!block) {
357
+      throw 'Couldn\'t get Block with id: ' + id;
358
+    }
359
+    var parentBlock = block.getParent();
360
+    // If we've already filled this block then we will dispose of it and then
361
+    // re-fill it.
362
+    if (block.workspace) {
363
+      block.dispose(true, false, true);
364
+    }
365
+    block.fill(workspace, prototypeName);
366
+    block.parent_ = parentBlock;
367
+  } else {
368
+    block = Blockly.Block.obtain(workspace, prototypeName);
369
+  }
370
+
371
+  var blockChild = null;
372
+  for (var i = 0, xmlChild; xmlChild = xmlBlock.childNodes[i]; i++) {
373
+    if (xmlChild.nodeType == 3) {
374
+      // Ignore any text at the <block> level.  It's all whitespace anyway.
375
+      continue;
376
+    }
377
+    var input;
378
+
379
+    // Find any enclosed blocks or shadows in this tag.
380
+    var childBlockNode = null;
381
+    var childShadowNode = null;
382
+    var shadowActive = false;
383
+    for (var j = 0, grandchildNode; grandchildNode = xmlChild.childNodes[j];
384
+         j++) {
385
+      if (grandchildNode.nodeType == 1) {
386
+        if (grandchildNode.nodeName.toLowerCase() == 'block') {
387
+          childBlockNode = grandchildNode;
388
+        } else if (grandchildNode.nodeName.toLowerCase() == 'shadow') {
389
+          childShadowNode = grandchildNode;
390
+        }
391
+      }
392
+    }
393
+    // Use the shadow block if there is no child block.
394
+    if (!childBlockNode && childShadowNode) {
395
+      childBlockNode = childShadowNode;
396
+      shadowActive = true;
397
+    }
398
+
399
+    var name = xmlChild.getAttribute('name');
400
+    switch (xmlChild.nodeName.toLowerCase()) {
401
+      case 'mutation':
402
+        // Custom data for an advanced block.
403
+        if (block.domToMutation) {
404
+          block.domToMutation(xmlChild);
405
+          if (block.initSvg) {
406
+            // Mutation may have added some elements that need initalizing.
407
+            block.initSvg();
408
+          }
409
+        }
410
+        break;
411
+      case 'comment':
412
+        block.setCommentText(xmlChild.textContent);
413
+        var visible = xmlChild.getAttribute('pinned');
414
+        if (visible) {
415
+          // Give the renderer a millisecond to render and position the block
416
+          // before positioning the comment bubble.
417
+          setTimeout(function() {
418
+            if (block.comment && block.comment.setVisible) {
419
+              block.comment.setVisible(visible == 'true');
420
+            }
421
+          }, 1);
422
+        }
423
+        var bubbleW = parseInt(xmlChild.getAttribute('w'), 10);
424
+        var bubbleH = parseInt(xmlChild.getAttribute('h'), 10);
425
+        if (!isNaN(bubbleW) && !isNaN(bubbleH) &&
426
+            block.comment && block.comment.setVisible) {
427
+          block.comment.setBubbleSize(bubbleW, bubbleH);
428
+        }
429
+        break;
430
+      case 'data':
431
+        // Optional text data that round-trips beween blocks and XML.
432
+        // Has no effect.  May be used by 3rd parties for meta information.
433
+        block.data = xmlChild.textContent;
434
+        break;
435
+      case 'title':
436
+        // Titles were renamed to field in December 2013.
437
+        // Fall through.
438
+      case 'field':
439
+        var field = block.getField(name);
440
+        if (!field) {
441
+          console.warn('Ignoring non-existent field ' + name + ' in block ' +
442
+                       prototypeName);
443
+          break;
444
+        }
445
+        field.setValue(xmlChild.textContent);
446
+        break;
447
+      case 'value':
448
+      case 'statement':
449
+        input = block.getInput(name);
450
+        if (!input) {
451
+          console.warn('Ignoring non-existent input ' + name + ' in block ' +
452
+                       prototypeName);
453
+          break;
454
+        }
455
+        if (childShadowNode) {
456
+          input.connection.setShadowDom(childShadowNode);
457
+        }
458
+        if (childBlockNode) {
459
+          blockChild = Blockly.Xml.domToBlockHeadless_(workspace,
460
+              childBlockNode, opt_reuseBlock);
461
+          if (blockChild.outputConnection) {
462
+            input.connection.connect(blockChild.outputConnection);
463
+          } else if (blockChild.previousConnection) {
464
+            input.connection.connect(blockChild.previousConnection);
465
+          } else {
466
+            throw 'Child block does not have output or previous statement.';
467
+          }
468
+        }
469
+        break;
470
+      case 'next':
471
+        if (childShadowNode && block.nextConnection) {
472
+          block.nextConnection.setShadowDom(childShadowNode);
473
+        }
474
+        if (childBlockNode) {
475
+          if (!block.nextConnection) {
476
+            throw 'Next statement does not exist.';
477
+          } else if (block.nextConnection.targetConnection) {
478
+            // This could happen if there is more than one XML 'next' tag.
479
+            throw 'Next statement is already connected.';
480
+          }
481
+          blockChild = Blockly.Xml.domToBlockHeadless_(workspace,
482
+              childBlockNode, opt_reuseBlock);
483
+          if (!blockChild.previousConnection) {
484
+            throw 'Next block does not have previous statement.';
485
+          }
486
+          block.nextConnection.connect(blockChild.previousConnection);
487
+        }
488
+        break;
489
+      default:
490
+        // Unknown tag; ignore.  Same principle as HTML parsers.
491
+        console.warn('Ignoring unknown tag: ' + xmlChild.nodeName);
492
+    }
493
+  }
494
+
495
+  var inline = xmlBlock.getAttribute('inline');
496
+  if (inline) {
497
+    block.setInputsInline(inline == 'true');
498
+  }
499
+  var disabled = xmlBlock.getAttribute('disabled');
500
+  if (disabled) {
501
+    block.setDisabled(disabled == 'true');
502
+  }
503
+  var deletable = xmlBlock.getAttribute('deletable');
504
+  if (deletable) {
505
+    block.setDeletable(deletable == 'true');
506
+  }
507
+  var movable = xmlBlock.getAttribute('movable');
508
+  if (movable) {
509
+    block.setMovable(movable == 'true');
510
+  }
511
+  var editable = xmlBlock.getAttribute('editable');
512
+  if (editable) {
513
+    block.setEditable(editable == 'true');
514
+  }
515
+  var collapsed = xmlBlock.getAttribute('collapsed');
516
+  if (collapsed) {
517
+    block.setCollapsed(collapsed == 'true');
518
+  }
519
+  if (xmlBlock.nodeName.toLowerCase() == 'shadow') {
520
+    block.setShadow(true);
521
+  }
522
+  // Give the block a chance to clean up any initial inputs.
523
+  if (block.validate) {
524
+    block.validate();
525
+  }
526
+  return block;
527
+};
528
+
529
+/**
530
+ * Remove any 'next' block (statements in a stack).
531
+ * @param {!Element} xmlBlock XML block element.
532
+ */
533
+Blockly.Xml.deleteNext = function(xmlBlock) {
534
+  for (var i = 0, child; child = xmlBlock.childNodes[i]; i++) {
535
+    if (child.nodeName.toLowerCase() == 'next') {
536
+      xmlBlock.removeChild(child);
537
+      break;
538
+    }
539
+  }
540
+};
541
+
542
+// Export symbols that would otherwise be renamed by Closure compiler.
543
+if (!goog.global['Blockly']) {
544
+  goog.global['Blockly'] = {};
545
+}
546
+if (!goog.global['Blockly']['Xml']) {
547
+  goog.global['Blockly']['Xml'] = {};
548
+}
549
+goog.global['Blockly']['Xml']['domToText'] = Blockly.Xml.domToText;
550
+goog.global['Blockly']['Xml']['domToWorkspace'] = Blockly.Xml.domToWorkspace;
551
+goog.global['Blockly']['Xml']['textToDom'] = Blockly.Xml.textToDom;
552
+goog.global['Blockly']['Xml']['workspaceToDom'] = Blockly.Xml.workspaceToDom;

+ 219 - 0
src/blockly/core/zoom_controls.js

@@ -0,0 +1,219 @@
1
+/**
2
+ * @license
3
+ * Visual Blocks Editor
4
+ *
5
+ * Copyright 2015 Google Inc.
6
+ * https://developers.google.com/blockly/
7
+ *
8
+ * Licensed under the Apache License, Version 2.0 (the "License");
9
+ * you may not use this file except in compliance with the License.
10
+ * You may obtain a copy of the License at
11
+ *
12
+ *   http://www.apache.org/licenses/LICENSE-2.0
13
+ *
14
+ * Unless required by applicable law or agreed to in writing, software
15
+ * distributed under the License is distributed on an "AS IS" BASIS,
16
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
+ * See the License for the specific language governing permissions and
18
+ * limitations under the License.
19
+ */
20
+
21
+/**
22
+ * @fileoverview Object representing a zoom icons.
23
+ * @author carloslfu@gmail.com (Carlos Galarza)
24
+ */
25
+'use strict';
26
+
27
+goog.provide('Blockly.ZoomControls');
28
+
29
+goog.require('goog.dom');
30
+
31
+
32
+/**
33
+ * Class for a zoom controls.
34
+ * @param {!Blockly.Workspace} workspace The workspace to sit in.
35
+ * @constructor
36
+ */
37
+Blockly.ZoomControls = function(workspace) {
38
+  this.workspace_ = workspace;
39
+};
40
+
41
+/**
42
+ * Width of the zoom controls.
43
+ * @type {number}
44
+ * @private
45
+ */
46
+Blockly.ZoomControls.prototype.WIDTH_ = 32;
47
+
48
+/**
49
+ * Height of the zoom controls.
50
+ * @type {number}
51
+ * @private
52
+ */
53
+Blockly.ZoomControls.prototype.HEIGHT_ = 110;
54
+
55
+/**
56
+ * Distance between zoom controls and bottom edge of workspace.
57
+ * @type {number}
58
+ * @private
59
+ */
60
+Blockly.ZoomControls.prototype.MARGIN_BOTTOM_ = 20;
61
+
62
+/**
63
+ * Distance between zoom controls and right edge of workspace.
64
+ * @type {number}
65
+ * @private
66
+ */
67
+Blockly.ZoomControls.prototype.MARGIN_SIDE_ = 20;
68
+
69
+/**
70
+ * The SVG group containing the zoom controls.
71
+ * @type {Element}
72
+ * @private
73
+ */
74
+Blockly.ZoomControls.prototype.svgGroup_ = null;
75
+
76
+/**
77
+ * Left coordinate of the zoom controls.
78
+ * @type {number}
79
+ * @private
80
+ */
81
+Blockly.ZoomControls.prototype.left_ = 0;
82
+
83
+/**
84
+ * Top coordinate of the zoom controls.
85
+ * @type {number}
86
+ * @private
87
+ */
88
+Blockly.ZoomControls.prototype.top_ = 0;
89
+
90
+/**
91
+ * Create the zoom controls.
92
+ * @return {!Element} The zoom controls SVG group.
93
+ */
94
+Blockly.ZoomControls.prototype.createDom = function() {
95
+  var workspace = this.workspace_;
96
+  /* Here's the markup that will be generated:
97
+  <g class="blocklyZoom">
98
+    <clippath id="blocklyZoomoutClipPath837493">
99
+      <rect width="32" height="32" y="77"></rect>
100
+    </clippath>
101
+    <image width="96" height="124" x="-64" y="-15" xlink:href="media/sprites.png">
102
+        clip-path="url(#blocklyZoomoutClipPath837493)"></image>
103
+    <clippath id="blocklyZoominClipPath837493">
104
+      <rect width="32" height="32" y="43"></rect>
105
+    </clippath>
106
+    <image width="96" height="124" x="-32" y="-49" xlink:href="media/sprites.png">
107
+        clip-path="url(#blocklyZoominClipPath837493)"></image>
108
+    <clippath id="blocklyZoomresetClipPath837493">
109
+      <rect width="32" height="32"></rect>
110
+    </clippath>
111
+    <image width="96" height="124" y="-92" xlink:href="media/sprites.png">
112
+        clip-path="url(#blocklyZoomresetClipPath837493)"></image>
113
+  </g>
114
+  */
115
+  this.svgGroup_ = Blockly.createSvgElement('g',
116
+      {'class': 'blocklyZoom'}, null);
117
+  var rnd = String(Math.random()).substring(2);
118
+
119
+  var clip = Blockly.createSvgElement('clipPath',
120
+      {'id': 'blocklyZoomoutClipPath' + rnd},
121
+      this.svgGroup_);
122
+  Blockly.createSvgElement('rect',
123
+      {'width': 32, 'height': 32, 'y': 77},
124
+      clip);
125
+  var zoomoutSvg = Blockly.createSvgElement('image',
126
+      {'width': Blockly.SPRITE.width,
127
+       'height': Blockly.SPRITE.height, 'x': -64,
128
+       'y': -15,
129
+       'clip-path': 'url(#blocklyZoomoutClipPath' + rnd + ')'},
130
+      this.svgGroup_);
131
+  zoomoutSvg.setAttributeNS('http://www.w3.org/1999/xlink', 'xlink:href',
132
+      workspace.options.pathToMedia + Blockly.SPRITE.url);
133
+
134
+  var clip = Blockly.createSvgElement('clipPath',
135
+      {'id': 'blocklyZoominClipPath' + rnd},
136
+      this.svgGroup_);
137
+  Blockly.createSvgElement('rect',
138
+      {'width': 32, 'height': 32, 'y': 43},
139
+      clip);
140
+  var zoominSvg = Blockly.createSvgElement('image',
141
+      {'width': Blockly.SPRITE.width,
142
+       'height': Blockly.SPRITE.height,
143
+       'x': -32,
144
+       'y': -49,
145
+       'clip-path': 'url(#blocklyZoominClipPath' + rnd + ')'},
146
+      this.svgGroup_);
147
+  zoominSvg.setAttributeNS('http://www.w3.org/1999/xlink', 'xlink:href',
148
+      workspace.options.pathToMedia + Blockly.SPRITE.url);
149
+
150
+  var clip = Blockly.createSvgElement('clipPath',
151
+      {'id': 'blocklyZoomresetClipPath' + rnd},
152
+      this.svgGroup_);
153
+  Blockly.createSvgElement('rect',
154
+      {'width': 32, 'height': 32},
155
+      clip);
156
+  var zoomresetSvg = Blockly.createSvgElement('image',
157
+      {'width': Blockly.SPRITE.width,
158
+       'height': Blockly.SPRITE.height, 'y': -92,
159
+       'clip-path': 'url(#blocklyZoomresetClipPath' + rnd + ')'},
160
+      this.svgGroup_);
161
+  zoomresetSvg.setAttributeNS('http://www.w3.org/1999/xlink', 'xlink:href',
162
+      workspace.options.pathToMedia + Blockly.SPRITE.url);
163
+
164
+  // Attach event listeners.
165
+  Blockly.bindEvent_(zoomresetSvg, 'mousedown', workspace, workspace.zoomReset);
166
+  Blockly.bindEvent_(zoominSvg, 'mousedown', null, function(e) {
167
+    workspace.zoomCenter(1);
168
+    e.stopPropagation();  // Don't start a workspace scroll.
169
+  });
170
+  Blockly.bindEvent_(zoomoutSvg, 'mousedown', null, function(e) {
171
+    workspace.zoomCenter(-1);
172
+    e.stopPropagation();  // Don't start a workspace scroll.
173
+  });
174
+
175
+  return this.svgGroup_;
176
+};
177
+
178
+/**
179
+ * Initialize the zoom controls.
180
+ * @param {number} bottom Distance from workspace bottom to bottom of controls.
181
+ * @return {number} Distance from workspace bottom to the top of controls.
182
+ */
183
+Blockly.ZoomControls.prototype.init = function(bottom) {
184
+  this.bottom_ = this.MARGIN_BOTTOM_ + bottom;
185
+  return this.bottom_ + this.HEIGHT_;
186
+};
187
+
188
+/**
189
+ * Dispose of this zoom controls.
190
+ * Unlink from all DOM elements to prevent memory leaks.
191
+ */
192
+Blockly.ZoomControls.prototype.dispose = function() {
193
+  if (this.svgGroup_) {
194
+    goog.dom.removeNode(this.svgGroup_);
195
+    this.svgGroup_ = null;
196
+  }
197
+  this.workspace_ = null;
198
+};
199
+
200
+/**
201
+ * Move the zoom controls to the bottom-right corner.
202
+ */
203
+Blockly.ZoomControls.prototype.position = function() {
204
+  var metrics = this.workspace_.getMetrics();
205
+  if (!metrics) {
206
+    // There are no metrics available (workspace is probably not visible).
207
+    return;
208
+  }
209
+  if (this.workspace_.RTL) {
210
+    this.left_ = this.MARGIN_SIDE_ + Blockly.Scrollbar.scrollbarThickness;
211
+  } else {
212
+    this.left_ = metrics.viewWidth + metrics.absoluteLeft -
213
+        this.WIDTH_ - this.MARGIN_SIDE_ - Blockly.Scrollbar.scrollbarThickness;
214
+  }
215
+  this.top_ = metrics.viewHeight + metrics.absoluteTop -
216
+      this.HEIGHT_ - this.bottom_;
217
+  this.svgGroup_.setAttribute('transform',
218
+      'translate(' + this.left_ + ',' + this.top_ + ')');
219
+};

File diff suppressed because it is too large
+ 82 - 0
src/blockly/dart_compressed.js


+ 754 - 0
src/blockly/demos/blockfactory/blocks.js

@@ -0,0 +1,754 @@
1
+/**
2
+ * Blockly Demos: Block Factory Blocks
3
+ *
4
+ * Copyright 2012 Google Inc.
5
+ * https://developers.google.com/blockly/
6
+ *
7
+ * Licensed under the Apache License, Version 2.0 (the "License");
8
+ * you may not use this file except in compliance with the License.
9
+ * You may obtain a copy of the License at
10
+ *
11
+ *   http://www.apache.org/licenses/LICENSE-2.0
12
+ *
13
+ * Unless required by applicable law or agreed to in writing, software
14
+ * distributed under the License is distributed on an "AS IS" BASIS,
15
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ * See the License for the specific language governing permissions and
17
+ * limitations under the License.
18
+ */
19
+
20
+/**
21
+ * @fileoverview Blocks for Blockly's Block Factory application.
22
+ * @author fraser@google.com (Neil Fraser)
23
+ */
24
+'use strict';
25
+
26
+Blockly.Blocks['factory_base'] = {
27
+  // Base of new block.
28
+  init: function() {
29
+    this.setColour(120);
30
+    this.appendDummyInput()
31
+        .appendField('name')
32
+        .appendField(new Blockly.FieldTextInput('math_foo'), 'NAME');
33
+    this.appendStatementInput('INPUTS')
34
+        .setCheck('Input')
35
+        .appendField('inputs');
36
+    var dropdown = new Blockly.FieldDropdown([
37
+        ['automatic inputs', 'AUTO'],
38
+        ['external inputs', 'EXT'],
39
+        ['inline inputs', 'INT']]);
40
+    this.appendDummyInput()
41
+        .appendField(dropdown, 'INLINE');
42
+    dropdown = new Blockly.FieldDropdown([
43
+        ['no connections', 'NONE'],
44
+        ['← left output', 'LEFT'],
45
+        ['↕ top+bottom connections', 'BOTH'],
46
+        ['↑ top connection', 'TOP'],
47
+        ['↓ bottom connection', 'BOTTOM']],
48
+        function(option) {
49
+          this.sourceBlock_.updateShape_(option);
50
+        });
51
+    this.appendDummyInput()
52
+        .appendField(dropdown, 'CONNECTIONS');
53
+    this.appendValueInput('COLOUR')
54
+        .setCheck('Colour')
55
+        .appendField('colour');
56
+    /*
57
+    this.appendValueInput('TOOLTIP')
58
+        .setCheck('String')
59
+        .appendField('tooltip');
60
+    this.appendValueInput('HELP')
61
+        .setCheck('String')
62
+        .appendField('help url');
63
+    */
64
+    this.setTooltip('Build a custom block by plugging\n' +
65
+        'fields, inputs and other blocks here.');
66
+    this.setHelpUrl(
67
+        'https://developers.google.com/blockly/custom-blocks/block-factory');
68
+  },
69
+  mutationToDom: function() {
70
+    var container = document.createElement('mutation');
71
+    container.setAttribute('connections', this.getFieldValue('CONNECTIONS'));
72
+    return container;
73
+  },
74
+  domToMutation: function(xmlElement) {
75
+    var connections = xmlElement.getAttribute('connections');
76
+    this.updateShape_(connections);
77
+  },
78
+  updateShape_: function(option) {
79
+    var outputExists = this.getInput('OUTPUTTYPE');
80
+    var topExists = this.getInput('TOPTYPE');
81
+    var bottomExists = this.getInput('BOTTOMTYPE');
82
+    if (option == 'LEFT') {
83
+      if (!outputExists) {
84
+        this.appendValueInput('OUTPUTTYPE')
85
+            .setCheck('Type')
86
+            .appendField('output type');
87
+        this.moveInputBefore('OUTPUTTYPE', 'COLOUR');
88
+      }
89
+    } else if (outputExists) {
90
+      this.removeInput('OUTPUTTYPE');
91
+    }
92
+    if (option == 'TOP' || option == 'BOTH') {
93
+      if (!topExists) {
94
+        this.appendValueInput('TOPTYPE')
95
+            .setCheck('Type')
96
+            .appendField('top type');
97
+        this.moveInputBefore('TOPTYPE', 'COLOUR');
98
+      }
99
+    } else if (topExists) {
100
+      this.removeInput('TOPTYPE');
101
+    }
102
+    if (option == 'BOTTOM' || option == 'BOTH') {
103
+      if (!bottomExists) {
104
+        this.appendValueInput('BOTTOMTYPE')
105
+            .setCheck('Type')
106
+            .appendField('bottom type');
107
+        this.moveInputBefore('BOTTOMTYPE', 'COLOUR');
108
+      }
109
+    } else if (bottomExists) {
110
+      this.removeInput('BOTTOMTYPE');
111
+    }
112
+  }
113
+};
114
+
115
+var ALIGNMENT_OPTIONS =
116
+    [['left', 'LEFT'], ['right', 'RIGHT'], ['centre', 'CENTRE']];
117
+
118
+Blockly.Blocks['input_value'] = {
119
+  // Value input.
120
+  init: function() {
121
+    this.setColour(210);
122
+    this.appendDummyInput()
123
+        .appendField('value input')
124
+        .appendField(new Blockly.FieldTextInput('NAME'), 'INPUTNAME');
125
+    this.appendStatementInput('FIELDS')
126
+        .setCheck('Field')
127
+        .appendField('fields')
128
+        .appendField(new Blockly.FieldDropdown(ALIGNMENT_OPTIONS), 'ALIGN');
129
+    this.appendValueInput('TYPE')
130
+        .setCheck('Type')
131
+        .setAlign(Blockly.ALIGN_RIGHT)
132
+        .appendField('type');
133
+    this.setPreviousStatement(true, 'Input');
134
+    this.setNextStatement(true, 'Input');
135
+    this.setTooltip('A value socket for horizontal connections.');
136
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=71');
137
+  },
138
+  onchange: function() {
139
+    if (!this.workspace) {
140
+      // Block has been deleted.
141
+      return;
142
+    }
143
+    inputNameCheck(this);
144
+  }
145
+};
146
+
147
+Blockly.Blocks['input_statement'] = {
148
+  // Statement input.
149
+  init: function() {
150
+    this.setColour(210);
151
+    this.appendDummyInput()
152
+        .appendField('statement input')
153
+        .appendField(new Blockly.FieldTextInput('NAME'), 'INPUTNAME');
154
+    this.appendStatementInput('FIELDS')
155
+        .setCheck('Field')
156
+        .appendField('fields')
157
+        .appendField(new Blockly.FieldDropdown(ALIGNMENT_OPTIONS), 'ALIGN');
158
+    this.appendValueInput('TYPE')
159
+        .setCheck('Type')
160
+        .setAlign(Blockly.ALIGN_RIGHT)
161
+        .appendField('type');
162
+    this.setPreviousStatement(true, 'Input');
163
+    this.setNextStatement(true, 'Input');
164
+    this.setTooltip('A statement socket for enclosed vertical stacks.');
165
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=246');
166
+  },
167
+  onchange: function() {
168
+    if (!this.workspace) {
169
+      // Block has been deleted.
170
+      return;
171
+    }
172
+    inputNameCheck(this);
173
+  }
174
+};
175
+
176
+Blockly.Blocks['input_dummy'] = {
177
+  // Dummy input.
178
+  init: function() {
179
+    this.setColour(210);
180
+    this.appendDummyInput()
181
+        .appendField('dummy input');
182
+    this.appendStatementInput('FIELDS')
183
+        .setCheck('Field')
184
+        .appendField('fields')
185
+        .appendField(new Blockly.FieldDropdown(ALIGNMENT_OPTIONS), 'ALIGN');
186
+    this.setPreviousStatement(true, 'Input');
187
+    this.setNextStatement(true, 'Input');
188
+    this.setTooltip('For adding fields on a separate row with no ' +
189
+                    'connections. Alignment options (left, right, centre) ' +
190
+                    'apply only to multi-line fields.');
191
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=293');
192
+  }
193
+};
194
+
195
+Blockly.Blocks['field_static'] = {
196
+  // Text value.
197
+  init: function() {
198
+    this.setColour(160);
199
+    this.appendDummyInput()
200
+        .appendField('text')
201
+        .appendField(new Blockly.FieldTextInput(''), 'TEXT');
202
+    this.setPreviousStatement(true, 'Field');
203
+    this.setNextStatement(true, 'Field');
204
+    this.setTooltip('Static text that serves as a label.');
205
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=88');
206
+  }
207
+};
208
+
209
+Blockly.Blocks['field_input'] = {
210
+  // Text input.
211
+  init: function() {
212
+    this.setColour(160);
213
+    this.appendDummyInput()
214
+        .appendField('text input')
215
+        .appendField(new Blockly.FieldTextInput('default'), 'TEXT')
216
+        .appendField(',')
217
+        .appendField(new Blockly.FieldTextInput('NAME'), 'FIELDNAME');
218
+    this.setPreviousStatement(true, 'Field');
219
+    this.setNextStatement(true, 'Field');
220
+    this.setTooltip('An input field for the user to enter text.');
221
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=319');
222
+  },
223
+  onchange: function() {
224
+    if (!this.workspace) {
225
+      // Block has been deleted.
226
+      return;
227
+    }
228
+    fieldNameCheck(this);
229
+  }
230
+};
231
+
232
+Blockly.Blocks['field_angle'] = {
233
+  // Angle input.
234
+  init: function() {
235
+    this.setColour(160);
236
+    this.appendDummyInput()
237
+        .appendField('angle input')
238
+        .appendField(new Blockly.FieldAngle('90'), 'ANGLE')
239
+        .appendField(',')
240
+        .appendField(new Blockly.FieldTextInput('NAME'), 'FIELDNAME');
241
+    this.setPreviousStatement(true, 'Field');
242
+    this.setNextStatement(true, 'Field');
243
+    this.setTooltip('An input field for the user to enter an angle.');
244
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=372');
245
+  },
246
+  onchange: function() {
247
+    if (!this.workspace) {
248
+      // Block has been deleted.
249
+      return;
250
+    }
251
+    fieldNameCheck(this);
252
+  }
253
+};
254
+
255
+Blockly.Blocks['field_dropdown'] = {
256
+  // Dropdown menu.
257
+  init: function() {
258
+    this.setColour(160);
259
+    this.appendDummyInput()
260
+        .appendField('dropdown')
261
+        .appendField(new Blockly.FieldTextInput('NAME'), 'FIELDNAME');
262
+    this.appendDummyInput('OPTION0')
263
+        .appendField(new Blockly.FieldTextInput('option'), 'USER0')
264
+        .appendField(',')
265
+        .appendField(new Blockly.FieldTextInput('OPTIONNAME'), 'CPU0');
266
+    this.appendDummyInput('OPTION1')
267
+        .appendField(new Blockly.FieldTextInput('option'), 'USER1')
268
+        .appendField(',')
269
+        .appendField(new Blockly.FieldTextInput('OPTIONNAME'), 'CPU1');
270
+    this.appendDummyInput('OPTION2')
271
+        .appendField(new Blockly.FieldTextInput('option'), 'USER2')
272
+        .appendField(',')
273
+        .appendField(new Blockly.FieldTextInput('OPTIONNAME'), 'CPU2');
274
+    this.setPreviousStatement(true, 'Field');
275
+    this.setNextStatement(true, 'Field');
276
+    this.setMutator(new Blockly.Mutator(['field_dropdown_option']));
277
+    this.setTooltip('Dropdown menu with a list of options.');
278
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=386');
279
+    this.optionCount_ = 3;
280
+  },
281
+  mutationToDom: function(workspace) {
282
+    var container = document.createElement('mutation');
283
+    container.setAttribute('options', this.optionCount_);
284
+    return container;
285
+  },
286
+  domToMutation: function(container) {
287
+    for (var x = 0; x < this.optionCount_; x++) {
288
+      this.removeInput('OPTION' + x);
289
+    }
290
+    this.optionCount_ = parseInt(container.getAttribute('options'), 10);
291
+    for (var x = 0; x < this.optionCount_; x++) {
292
+      var input = this.appendDummyInput('OPTION' + x);
293
+      input.appendField(new Blockly.FieldTextInput('option'), 'USER' + x);
294
+      input.appendField(',');
295
+      input.appendField(new Blockly.FieldTextInput('OPTIONNAME'), 'CPU' + x);
296
+    }
297
+  },
298
+  decompose: function(workspace) {
299
+    var containerBlock =
300
+        Blockly.Block.obtain(workspace, 'field_dropdown_container');
301
+    containerBlock.initSvg();
302
+    var connection = containerBlock.getInput('STACK').connection;
303
+    for (var x = 0; x < this.optionCount_; x++) {
304
+      var optionBlock =
305
+          Blockly.Block.obtain(workspace, 'field_dropdown_option');
306
+      optionBlock.initSvg();
307
+      connection.connect(optionBlock.previousConnection);
308
+      connection = optionBlock.nextConnection;
309
+    }
310
+    return containerBlock;
311
+  },
312
+  compose: function(containerBlock) {
313
+    // Disconnect all input blocks and remove all inputs.
314
+    for (var x = this.optionCount_ - 1; x >= 0; x--) {
315
+      this.removeInput('OPTION' + x);
316
+    }
317
+    this.optionCount_ = 0;
318
+    // Rebuild the block's inputs.
319
+    var optionBlock = containerBlock.getInputTargetBlock('STACK');
320
+    while (optionBlock) {
321
+      this.appendDummyInput('OPTION' + this.optionCount_)
322
+          .appendField(new Blockly.FieldTextInput(
323
+              optionBlock.userData_ || 'option'), 'USER' + this.optionCount_)
324
+          .appendField(',')
325
+          .appendField(new Blockly.FieldTextInput(
326
+              optionBlock.cpuData_ || 'OPTIONNAME'), 'CPU' + this.optionCount_);
327
+      this.optionCount_++;
328
+      optionBlock = optionBlock.nextConnection &&
329
+          optionBlock.nextConnection.targetBlock();
330
+    }
331
+  },
332
+  saveConnections: function(containerBlock) {
333
+    // Store names and values for each option.
334
+    var optionBlock = containerBlock.getInputTargetBlock('STACK');
335
+    var x = 0;
336
+    while (optionBlock) {
337
+      optionBlock.userData_ = this.getFieldValue('USER' + x);
338
+      optionBlock.cpuData_ = this.getFieldValue('CPU' + x);
339
+      x++;
340
+      optionBlock = optionBlock.nextConnection &&
341
+          optionBlock.nextConnection.targetBlock();
342
+    }
343
+  },
344
+  onchange: function() {
345
+    if (!this.workspace) {
346
+      // Block has been deleted.
347
+      return;
348
+    }
349
+    if (this.optionCount_ < 1) {
350
+      this.setWarningText('Drop down menu must\nhave at least one option.');
351
+    } else {
352
+      fieldNameCheck(this);
353
+    }
354
+  }
355
+};
356
+
357
+Blockly.Blocks['field_dropdown_container'] = {
358
+  // Container.
359
+  init: function() {
360
+    this.setColour(160);
361
+    this.appendDummyInput()
362
+        .appendField('add options');
363
+    this.appendStatementInput('STACK');
364
+    this.setTooltip('Add, remove, or reorder options\n' +
365
+                    'to reconfigure this dropdown menu.');
366
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=386');
367
+    this.contextMenu = false;
368
+  }
369
+};
370
+
371
+Blockly.Blocks['field_dropdown_option'] = {
372
+  // Add option.
373
+  init: function() {
374
+    this.setColour(160);
375
+    this.appendDummyInput()
376
+        .appendField('option');
377
+    this.setPreviousStatement(true);
378
+    this.setNextStatement(true);
379
+    this.setTooltip('Add a new option to the dropdown menu.');
380
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=386');
381
+    this.contextMenu = false;
382
+  }
383
+};
384
+
385
+Blockly.Blocks['field_checkbox'] = {
386
+  // Checkbox.
387
+  init: function() {
388
+    this.setColour(160);
389
+    this.appendDummyInput()
390
+        .appendField('checkbox')
391
+        .appendField(new Blockly.FieldCheckbox('TRUE'), 'CHECKED')
392
+        .appendField(',')
393
+        .appendField(new Blockly.FieldTextInput('NAME'), 'FIELDNAME');
394
+    this.setPreviousStatement(true, 'Field');
395
+    this.setNextStatement(true, 'Field');
396
+    this.setTooltip('Checkbox field.');
397
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=485');
398
+  },
399
+  onchange: function() {
400
+    if (!this.workspace) {
401
+      // Block has been deleted.
402
+      return;
403
+    }
404
+    fieldNameCheck(this);
405
+  }
406
+};
407
+
408
+Blockly.Blocks['field_colour'] = {
409
+  // Colour input.
410
+  init: function() {
411
+    this.setColour(160);
412
+    this.appendDummyInput()
413
+        .appendField('colour')
414
+        .appendField(new Blockly.FieldColour('#ff0000'), 'COLOUR')
415
+        .appendField(',')
416
+        .appendField(new Blockly.FieldTextInput('NAME'), 'FIELDNAME');
417
+    this.setPreviousStatement(true, 'Field');
418
+    this.setNextStatement(true, 'Field');
419
+    this.setTooltip('Colour input field.');
420
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=495');
421
+  },
422
+  onchange: function() {
423
+    if (!this.workspace) {
424
+      // Block has been deleted.
425
+      return;
426
+    }
427
+    fieldNameCheck(this);
428
+  }
429
+};
430
+
431
+Blockly.Blocks['field_date'] = {
432
+  // Date input.
433
+  init: function() {
434
+    this.setColour(160);
435
+    this.appendDummyInput()
436
+        .appendField('date')
437
+        .appendField(new Blockly.FieldDate(), 'DATE')
438
+        .appendField(',')
439
+        .appendField(new Blockly.FieldTextInput('NAME'), 'FIELDNAME');
440
+    this.setPreviousStatement(true, 'Field');
441
+    this.setNextStatement(true, 'Field');
442
+    this.setTooltip('Date input field.');
443
+  },
444
+  onchange: function() {
445
+    if (!this.workspace) {
446
+      // Block has been deleted.
447
+      return;
448
+    }
449
+    fieldNameCheck(this);
450
+  }
451
+};
452
+
453
+Blockly.Blocks['field_variable'] = {
454
+  // Dropdown for variables.
455
+  init: function() {
456
+    this.setColour(160);
457
+    this.appendDummyInput()
458
+        .appendField('variable')
459
+        .appendField(new Blockly.FieldTextInput('item'), 'TEXT')
460
+        .appendField(',')
461
+        .appendField(new Blockly.FieldTextInput('NAME'), 'FIELDNAME');
462
+    this.setPreviousStatement(true, 'Field');
463
+    this.setNextStatement(true, 'Field');
464
+    this.setTooltip('Dropdown menu for variable names.');
465
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=510');
466
+  },
467
+  onchange: function() {
468
+    if (!this.workspace) {
469
+      // Block has been deleted.
470
+      return;
471
+    }
472
+    fieldNameCheck(this);
473
+  }
474
+};
475
+
476
+Blockly.Blocks['field_image'] = {
477
+  // Image.
478
+  init: function() {
479
+    this.setColour(160);
480
+    var src = 'https://www.gstatic.com/codesite/ph/images/star_on.gif';
481
+    this.appendDummyInput()
482
+        .appendField('image')
483
+        .appendField(new Blockly.FieldTextInput(src), 'SRC');
484
+    this.appendDummyInput()
485
+        .appendField('width')
486
+        .appendField(new Blockly.FieldTextInput('15',
487
+            Blockly.FieldTextInput.numberValidator), 'WIDTH')
488
+        .appendField('height')
489
+        .appendField(new Blockly.FieldTextInput('15',
490
+            Blockly.FieldTextInput.numberValidator), 'HEIGHT')
491
+        .appendField('alt text')
492
+        .appendField(new Blockly.FieldTextInput('*'), 'ALT');
493
+    this.setPreviousStatement(true, 'Field');
494
+    this.setNextStatement(true, 'Field');
495
+    this.setTooltip('Static image (JPEG, PNG, GIF, SVG, BMP).\n' +
496
+                    'Retains aspect ratio regardless of height and width.\n' +
497
+                    'Alt text is for when collapsed.');
498
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=567');
499
+  }
500
+};
501
+
502
+Blockly.Blocks['type_group'] = {
503
+  // Group of types.
504
+  init: function() {
505
+    this.setColour(230);
506
+    this.appendValueInput('TYPE0')
507
+        .setCheck('Type')
508
+        .appendField('any of');
509
+    this.appendValueInput('TYPE1')
510
+        .setCheck('Type');
511
+    this.setOutput(true, 'Type');
512
+    this.setMutator(new Blockly.Mutator(['type_group_item']));
513
+    this.setTooltip('Allows more than one type to be accepted.');
514
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=677');
515
+    this.typeCount_ = 2;
516
+  },
517
+  mutationToDom: function(workspace) {
518
+    var container = document.createElement('mutation');
519
+    container.setAttribute('types', this.typeCount_);
520
+    return container;
521
+  },
522
+  domToMutation: function(container) {
523
+    for (var x = 0; x < this.typeCount_; x++) {
524
+      this.removeInput('TYPE' + x);
525
+    }
526
+    this.typeCount_ = parseInt(container.getAttribute('types'), 10);
527
+    for (var x = 0; x < this.typeCount_; x++) {
528
+      var input = this.appendValueInput('TYPE' + x)
529
+                      .setCheck('Type');
530
+      if (x == 0) {
531
+        input.appendField('any of');
532
+      }
533
+    }
534
+  },
535
+  decompose: function(workspace) {
536
+    var containerBlock =
537
+        Blockly.Block.obtain(workspace, 'type_group_container');
538
+    containerBlock.initSvg();
539
+    var connection = containerBlock.getInput('STACK').connection;
540
+    for (var x = 0; x < this.typeCount_; x++) {
541
+      var typeBlock = Blockly.Block.obtain(workspace, 'type_group_item');
542
+      typeBlock.initSvg();
543
+      connection.connect(typeBlock.previousConnection);
544
+      connection = typeBlock.nextConnection;
545
+    }
546
+    return containerBlock;
547
+  },
548
+  compose: function(containerBlock) {
549
+    // Disconnect all input blocks and remove all inputs.
550
+    for (var x = this.typeCount_ - 1; x >= 0; x--) {
551
+      this.removeInput('TYPE' + x);
552
+    }
553
+    this.typeCount_ = 0;
554
+    // Rebuild the block's inputs.
555
+    var typeBlock = containerBlock.getInputTargetBlock('STACK');
556
+    while (typeBlock) {
557
+      var input = this.appendValueInput('TYPE' + this.typeCount_)
558
+                      .setCheck('Type');
559
+      if (this.typeCount_ == 0) {
560
+        input.appendField('any of');
561
+      }
562
+      // Reconnect any child blocks.
563
+      if (typeBlock.valueConnection_) {
564
+        input.connection.connect(typeBlock.valueConnection_);
565
+      }
566
+      this.typeCount_++;
567
+      typeBlock = typeBlock.nextConnection &&
568
+          typeBlock.nextConnection.targetBlock();
569
+    }
570
+  },
571
+  saveConnections: function(containerBlock) {
572
+    // Store a pointer to any connected child blocks.
573
+    var typeBlock = containerBlock.getInputTargetBlock('STACK');
574
+    var x = 0;
575
+    while (typeBlock) {
576
+      var input = this.getInput('TYPE' + x);
577
+      typeBlock.valueConnection_ = input && input.connection.targetConnection;
578
+      x++;
579
+      typeBlock = typeBlock.nextConnection &&
580
+          typeBlock.nextConnection.targetBlock();
581
+    }
582
+  }
583
+};
584
+
585
+Blockly.Blocks['type_group_container'] = {
586
+  // Container.
587
+  init: function() {
588
+    this.setColour(230);
589
+    this.appendDummyInput()
590
+        .appendField('add types');
591
+    this.appendStatementInput('STACK');
592
+    this.setTooltip('Add, or remove allowed type.');
593
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=677');
594
+    this.contextMenu = false;
595
+  }
596
+};
597
+
598
+Blockly.Blocks['type_group_item'] = {
599
+  // Add type.
600
+  init: function() {
601
+    this.setColour(230);
602
+    this.appendDummyInput()
603
+        .appendField('type');
604
+    this.setPreviousStatement(true);
605
+    this.setNextStatement(true);
606
+    this.setTooltip('Add a new allowed type.');
607
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=677');
608
+    this.contextMenu = false;
609
+  }
610
+};
611
+
612
+Blockly.Blocks['type_null'] = {
613
+  // Null type.
614
+  valueType: null,
615
+  init: function() {
616
+    this.setColour(230);
617
+    this.appendDummyInput()
618
+        .appendField('any');
619
+    this.setOutput(true, 'Type');
620
+    this.setTooltip('Any type is allowed.');
621
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=602');
622
+  }
623
+};
624
+
625
+Blockly.Blocks['type_boolean'] = {
626
+  // Boolean type.
627
+  valueType: 'Boolean',
628
+  init: function() {
629
+    this.setColour(230);
630
+    this.appendDummyInput()
631
+        .appendField('boolean');
632
+    this.setOutput(true, 'Type');
633
+    this.setTooltip('Booleans (true/false) are allowed.');
634
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=602');
635
+  }
636
+};
637
+
638
+Blockly.Blocks['type_number'] = {
639
+  // Number type.
640
+  valueType: 'Number',
641
+  init: function() {
642
+    this.setColour(230);
643
+    this.appendDummyInput()
644
+        .appendField('number');
645
+    this.setOutput(true, 'Type');
646
+    this.setTooltip('Numbers (int/float) are allowed.');
647
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=602');
648
+  }
649
+};
650
+
651
+Blockly.Blocks['type_string'] = {
652
+  // String type.
653
+  valueType: 'String',
654
+  init: function() {
655
+    this.setColour(230);
656
+    this.appendDummyInput()
657
+        .appendField('string');
658
+    this.setOutput(true, 'Type');
659
+    this.setTooltip('Strings (text) are allowed.');
660
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=602');
661
+  }
662
+};
663
+
664
+Blockly.Blocks['type_list'] = {
665
+  // List type.
666
+  valueType: 'Array',
667
+  init: function() {
668
+    this.setColour(230);
669
+    this.appendDummyInput()
670
+        .appendField('list');
671
+    this.setOutput(true, 'Type');
672
+    this.setTooltip('Arrays (lists) are allowed.');
673
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=602');
674
+  }
675
+};
676
+
677
+Blockly.Blocks['type_other'] = {
678
+  // Other type.
679
+  init: function() {
680
+    this.setColour(230);
681
+    this.appendDummyInput()
682
+        .appendField('other')
683
+        .appendField(new Blockly.FieldTextInput(''), 'TYPE');
684
+    this.setOutput(true, 'Type');
685
+    this.setTooltip('Custom type to allow.');
686
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=702');
687
+  }
688
+};
689
+
690
+Blockly.Blocks['colour_hue'] = {
691
+  // Set the colour of the block.
692
+  init: function() {
693
+    this.appendDummyInput()
694
+        .appendField('hue:')
695
+        .appendField(new Blockly.FieldAngle('0', this.validator), 'HUE');
696
+    this.setOutput(true, 'Colour');
697
+    this.setTooltip('Paint the block with this colour.');
698
+    this.setHelpUrl('https://www.youtube.com/watch?v=s2_xaEvcVI0#t=55');
699
+  },
700
+  validator: function(text) {
701
+    // Update the current block's colour to match.
702
+    this.sourceBlock_.setColour(text);
703
+  },
704
+  mutationToDom: function(workspace) {
705
+    var container = document.createElement('mutation');
706
+    container.setAttribute('colour', this.getColour());
707
+    return container;
708
+  },
709
+  domToMutation: function(container) {
710
+    this.setColour(container.getAttribute('colour'));
711
+  }
712
+};
713
+
714
+/**
715
+ * Check to see if more than one field has this name.
716
+ * Highly inefficient (On^2), but n is small.
717
+ * @param {!Blockly.Block} referenceBlock Block to check.
718
+ */
719
+function fieldNameCheck(referenceBlock) {
720
+  var name = referenceBlock.getFieldValue('FIELDNAME').toLowerCase();
721
+  var count = 0;
722
+  var blocks = referenceBlock.workspace.getAllBlocks();
723
+  for (var x = 0, block; block = blocks[x]; x++) {
724
+    var otherName = block.getFieldValue('FIELDNAME');
725
+    if (!block.disabled && !block.getInheritedDisabled() &&
726
+        otherName && otherName.toLowerCase() == name) {
727
+      count++;
728
+    }
729
+  }
730
+  var msg = (count > 1) ?
731
+      'There are ' + count + ' field blocks\n with this name.' : null;
732
+  referenceBlock.setWarningText(msg);
733
+}
734
+
735
+/**
736
+ * Check to see if more than one input has this name.
737
+ * Highly inefficient (On^2), but n is small.
738
+ * @param {!Blockly.Block} referenceBlock Block to check.
739
+ */
740
+function inputNameCheck(referenceBlock) {
741
+  var name = referenceBlock.getFieldValue('INPUTNAME').toLowerCase();
742
+  var count = 0;
743
+  var blocks = referenceBlock.workspace.getAllBlocks();
744
+  for (var x = 0, block; block = blocks[x]; x++) {
745
+    var otherName = block.getFieldValue('INPUTNAME');
746
+    if (!block.disabled && !block.getInheritedDisabled() &&
747
+        otherName && otherName.toLowerCase() == name) {
748
+      count++;
749
+    }
750
+  }
751
+  var msg = (count > 1) ?
752
+      'There are ' + count + ' input blocks\n with this name.' : null;
753
+  referenceBlock.setWarningText(msg);
754
+}

+ 793 - 0
src/blockly/demos/blockfactory/factory.js

@@ -0,0 +1,793 @@
1
+/**
2
+ * Blockly Demos: Block Factory
3
+ *
4
+ * Copyright 2012 Google Inc.
5
+ * https://developers.google.com/blockly/
6
+ *
7
+ * Licensed under the Apache License, Version 2.0 (the "License");
8
+ * you may not use this file except in compliance with the License.
9
+ * You may obtain a copy of the License at
10
+ *
11
+ *   http://www.apache.org/licenses/LICENSE-2.0
12
+ *
13
+ * Unless required by applicable law or agreed to in writing, software
14
+ * distributed under the License is distributed on an "AS IS" BASIS,
15
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ * See the License for the specific language governing permissions and
17
+ * limitations under the License.
18
+ */
19
+
20
+/**
21
+ * @fileoverview JavaScript for Blockly's Block Factory application.
22
+ * @author fraser@google.com (Neil Fraser)
23
+ */
24
+'use strict';
25
+
26
+/**
27
+ * Workspace for user to build block.
28
+ * @type {Blockly.Workspace}
29
+ */
30
+var mainWorkspace = null;
31
+
32
+/**
33
+ * Workspace for preview of block.
34
+ * @type {Blockly.Workspace}
35
+ */
36
+var previewWorkspace = null;
37
+
38
+/**
39
+ * Name of block if not named.
40
+ */
41
+var UNNAMED = 'unnamed';
42
+
43
+/**
44
+ * Change the language code format.
45
+ */
46
+function formatChange() {
47
+  var mask = document.getElementById('blocklyMask');
48
+  var languagePre = document.getElementById('languagePre');
49
+  var languageTA = document.getElementById('languageTA');
50
+  if (document.getElementById('format').value == 'Manual') {
51
+    Blockly.hideChaff();
52
+    mask.style.display = 'block';
53
+    languagePre.style.display = 'none';
54
+    languageTA.style.display = 'block';
55
+    var code = languagePre.textContent.trim();
56
+    languageTA.value = code;
57
+    languageTA.focus();
58
+    updatePreview();
59
+  } else {
60
+    mask.style.display = 'none';
61
+    languageTA.style.display = 'none';
62
+    languagePre.style.display = 'block';
63
+    updateLanguage();
64
+  }
65
+  disableEnableLink();
66
+}
67
+
68
+/**
69
+ * Update the language code based on constructs made in Blockly.
70
+ */
71
+function updateLanguage() {
72
+  var rootBlock = getRootBlock();
73
+  if (!rootBlock) {
74
+    return;
75
+  }
76
+  var blockType = rootBlock.getFieldValue('NAME').trim().toLowerCase();
77
+  if (!blockType) {
78
+    blockType = UNNAMED;
79
+  }
80
+  blockType = blockType.replace(/\W/g, '_').replace(/^(\d)/, '_\\1');
81
+  switch (document.getElementById('format').value) {
82
+    case 'JSON':
83
+      var code = formatJson_(blockType, rootBlock);
84
+      break;
85
+    case 'JavaScript':
86
+      var code = formatJavaScript_(blockType, rootBlock);
87
+      break;
88
+  }
89
+  injectCode(code, 'languagePre');
90
+  updatePreview();
91
+}
92
+
93
+/**
94
+ * Update the language code as JSON.
95
+ * @param {string} blockType Name of block.
96
+ * @param {!Blockly.Block} rootBlock Factory_base block.
97
+ * @return {string} Generanted language code.
98
+ * @private
99
+ */
100
+function formatJson_(blockType, rootBlock) {
101
+  var JS = {};
102
+  // ID is not used by Blockly, but may be used by a loader.
103
+  JS.id = blockType;
104
+  // Generate inputs.
105
+  var message = [];
106
+  var args = [];
107
+  var contentsBlock = rootBlock.getInputTargetBlock('INPUTS');
108
+  var lastInput = null;
109
+  while (contentsBlock) {
110
+    if (!contentsBlock.disabled && !contentsBlock.getInheritedDisabled()) {
111
+      var fields = getFieldsJson_(contentsBlock.getInputTargetBlock('FIELDS'));
112
+      for (var i = 0; i < fields.length; i++) {
113
+        if (typeof fields[i] == 'string') {
114
+          message.push(fields[i].replace(/%/g, '%%'));
115
+        } else {
116
+          args.push(fields[i]);
117
+          message.push('%' + args.length);
118
+        }
119
+      }
120
+
121
+      var input = {type: contentsBlock.type};
122
+      // Dummy inputs don't have names.  Other inputs do.
123
+      if (contentsBlock.type != 'input_dummy') {
124
+        input.name = contentsBlock.getFieldValue('INPUTNAME');
125
+      }
126
+      var check = JSON.parse(getOptTypesFrom(contentsBlock, 'TYPE') || 'null');
127
+      if (check) {
128
+        input.check = check;
129
+      }
130
+      var align = contentsBlock.getFieldValue('ALIGN');
131
+      if (align != 'LEFT') {
132
+        input.align = align;
133
+      }
134
+      args.push(input);
135
+      message.push('%' + args.length);
136
+      lastInput = contentsBlock;
137
+    }
138
+    contentsBlock = contentsBlock.nextConnection &&
139
+        contentsBlock.nextConnection.targetBlock();
140
+  }
141
+  // Remove last input if dummy and not empty.
142
+  if (lastInput && lastInput.type == 'input_dummy') {
143
+    var fields = lastInput.getInputTargetBlock('FIELDS');
144
+    if (fields && getFieldsJson_(fields).join('').trim() != '') {
145
+      var align = lastInput.getFieldValue('ALIGN');
146
+      if (align != 'LEFT') {
147
+        JS.lastDummyAlign0 = align;
148
+      }
149
+      args.pop();
150
+      message.pop();
151
+    }
152
+  }
153
+  JS.message0 = message.join(' ');
154
+  JS.args0 = args;
155
+  // Generate inline/external switch.
156
+  if (rootBlock.getFieldValue('INLINE') == 'EXT') {
157
+    JS.inputsInline = false;
158
+  } else if (rootBlock.getFieldValue('INLINE') == 'INT') {
159
+    JS.inputsInline = true;
160
+  }
161
+  // Generate output, or next/previous connections.
162
+  switch (rootBlock.getFieldValue('CONNECTIONS')) {
163
+    case 'LEFT':
164
+      JS.output =
165
+          JSON.parse(getOptTypesFrom(rootBlock, 'OUTPUTTYPE') || 'null');
166
+      break;
167
+    case 'BOTH':
168
+      JS.previousStatement =
169
+          JSON.parse(getOptTypesFrom(rootBlock, 'TOPTYPE') || 'null');
170
+      JS.nextStatement =
171
+          JSON.parse(getOptTypesFrom(rootBlock, 'BOTTOMTYPE') || 'null');
172
+      break;
173
+    case 'TOP':
174
+      JS.previousStatement =
175
+          JSON.parse(getOptTypesFrom(rootBlock, 'TOPTYPE') || 'null');
176
+      break;
177
+    case 'BOTTOM':
178
+      JS.nextStatement =
179
+          JSON.parse(getOptTypesFrom(rootBlock, 'BOTTOMTYPE') || 'null');
180
+      break;
181
+  }
182
+  // Generate colour.
183
+  var colourBlock = rootBlock.getInputTargetBlock('COLOUR');
184
+  if (colourBlock && !colourBlock.disabled) {
185
+    var hue = parseInt(colourBlock.getFieldValue('HUE'), 10);
186
+    JS.colour = hue;
187
+  }
188
+  JS.tooltip = '';
189
+  JS.helpUrl = 'http://www.example.com/';
190
+  return JSON.stringify(JS, null, '  ');
191
+}
192
+
193
+/**
194
+ * Update the language code as JavaScript.
195
+ * @param {string} blockType Name of block.
196
+ * @param {!Blockly.Block} rootBlock Factory_base block.
197
+ * @return {string} Generanted language code.
198
+ * @private
199
+ */
200
+function formatJavaScript_(blockType, rootBlock) {
201
+  var code = [];
202
+  code.push("Blockly.Blocks['" + blockType + "'] = {");
203
+  code.push("  init: function() {");
204
+  // Generate inputs.
205
+  var TYPES = {'input_value': 'appendValueInput',
206
+               'input_statement': 'appendStatementInput',
207
+               'input_dummy': 'appendDummyInput'};
208
+  var contentsBlock = rootBlock.getInputTargetBlock('INPUTS');
209
+  while (contentsBlock) {
210
+    if (!contentsBlock.disabled && !contentsBlock.getInheritedDisabled()) {
211
+      var name = '';
212
+      // Dummy inputs don't have names.  Other inputs do.
213
+      if (contentsBlock.type != 'input_dummy') {
214
+        name = escapeString(contentsBlock.getFieldValue('INPUTNAME'));
215
+      }
216
+      code.push('    this.' + TYPES[contentsBlock.type] + '(' + name + ')');
217
+      var check = getOptTypesFrom(contentsBlock, 'TYPE');
218
+      if (check) {
219
+        code.push('        .setCheck(' + check + ')');
220
+      }
221
+      var align = contentsBlock.getFieldValue('ALIGN');
222
+      if (align != 'LEFT') {
223
+        code.push('        .setAlign(Blockly.ALIGN_' + align + ')');
224
+      }
225
+      var fields = getFieldsJs_(contentsBlock.getInputTargetBlock('FIELDS'));
226
+      for (var i = 0; i < fields.length; i++) {
227
+        code.push('        .appendField(' + fields[i] + ')');
228
+      }
229
+      // Add semicolon to last line to finish the statement.
230
+      code[code.length - 1] += ';';
231
+    }
232
+    contentsBlock = contentsBlock.nextConnection &&
233
+        contentsBlock.nextConnection.targetBlock();
234
+  }
235
+  // Generate inline/external switch.
236
+  if (rootBlock.getFieldValue('INLINE') == 'EXT') {
237
+    code.push('    this.setInputsInline(false);');
238
+  } else if (rootBlock.getFieldValue('INLINE') == 'INT') {
239
+    code.push('    this.setInputsInline(true);');
240
+  }
241
+  // Generate output, or next/previous connections.
242
+  switch (rootBlock.getFieldValue('CONNECTIONS')) {
243
+    case 'LEFT':
244
+      code.push(connectionLineJs_('setOutput', 'OUTPUTTYPE'));
245
+      break;
246
+    case 'BOTH':
247
+      code.push(connectionLineJs_('setPreviousStatement', 'TOPTYPE'));
248
+      code.push(connectionLineJs_('setNextStatement', 'BOTTOMTYPE'));
249
+      break;
250
+    case 'TOP':
251
+      code.push(connectionLineJs_('setPreviousStatement', 'TOPTYPE'));
252
+      break;
253
+    case 'BOTTOM':
254
+      code.push(connectionLineJs_('setNextStatement', 'BOTTOMTYPE'));
255
+      break;
256
+  }
257
+  // Generate colour.
258
+  var colourBlock = rootBlock.getInputTargetBlock('COLOUR');
259
+  if (colourBlock && !colourBlock.disabled) {
260
+    var hue = parseInt(colourBlock.getFieldValue('HUE'), 10);
261
+    code.push('    this.setColour(' + hue + ');');
262
+  }
263
+  code.push("    this.setTooltip('');");
264
+  code.push("    this.setHelpUrl('http://www.example.com/');");
265
+  code.push('  }');
266
+  code.push('};');
267
+  return code.join('\n');
268
+}
269
+
270
+/**
271
+ * Create JS code required to create a top, bottom, or value connection.
272
+ * @param {string} functionName JavaScript function name.
273
+ * @param {string} typeName Name of type input.
274
+ * @return {string} Line of JavaScript code to create connection.
275
+ * @private
276
+ */
277
+function connectionLineJs_(functionName, typeName) {
278
+  var type = getOptTypesFrom(getRootBlock(), typeName);
279
+  if (type) {
280
+    type = ', ' + type;
281
+  } else {
282
+    type = '';
283
+  }
284
+  return '    this.' + functionName + '(true' + type + ');';
285
+}
286
+
287
+/**
288
+ * Returns field strings and any config.
289
+ * @param {!Blockly.Block} block Input block.
290
+ * @return {!Array.<string>} Field strings.
291
+ * @private
292
+ */
293
+function getFieldsJs_(block) {
294
+  var fields = [];
295
+  while (block) {
296
+    if (!block.disabled && !block.getInheritedDisabled()) {
297
+      switch (block.type) {
298
+        case 'field_static':
299
+          // Result: 'hello'
300
+          fields.push(escapeString(block.getFieldValue('TEXT')));
301
+          break;
302
+        case 'field_input':
303
+          // Result: new Blockly.FieldTextInput('Hello'), 'GREET'
304
+          fields.push('new Blockly.FieldTextInput(' +
305
+              escapeString(block.getFieldValue('TEXT')) + '), ' +
306
+              escapeString(block.getFieldValue('FIELDNAME')));
307
+          break;
308
+        case 'field_angle':
309
+          // Result: new Blockly.FieldAngle(90), 'ANGLE'
310
+          fields.push('new Blockly.FieldAngle(' +
311
+              escapeString(block.getFieldValue('ANGLE')) + '), ' +
312
+              escapeString(block.getFieldValue('FIELDNAME')));
313
+          break;
314
+        case 'field_checkbox':
315
+          // Result: new Blockly.FieldCheckbox('TRUE'), 'CHECK'
316
+          fields.push('new Blockly.FieldCheckbox(' +
317
+              escapeString(block.getFieldValue('CHECKED')) + '), ' +
318
+              escapeString(block.getFieldValue('FIELDNAME')));
319
+          break;
320
+        case 'field_colour':
321
+          // Result: new Blockly.FieldColour('#ff0000'), 'COLOUR'
322
+          fields.push('new Blockly.FieldColour(' +
323
+              escapeString(block.getFieldValue('COLOUR')) + '), ' +
324
+              escapeString(block.getFieldValue('FIELDNAME')));
325
+          break;
326
+        case 'field_date':
327
+          // Result: new Blockly.FieldDate('2015-02-04'), 'DATE'
328
+          fields.push('new Blockly.FieldDate(' +
329
+              escapeString(block.getFieldValue('DATE')) + '), ' +
330
+              escapeString(block.getFieldValue('FIELDNAME')));
331
+          break;
332
+        case 'field_variable':
333
+          // Result: new Blockly.FieldVariable('item'), 'VAR'
334
+          var varname = escapeString(block.getFieldValue('TEXT') || null);
335
+          fields.push('new Blockly.FieldVariable(' + varname + '), ' +
336
+              escapeString(block.getFieldValue('FIELDNAME')));
337
+          break;
338
+        case 'field_dropdown':
339
+          // Result:
340
+          // new Blockly.FieldDropdown([['yes', '1'], ['no', '0']]), 'TOGGLE'
341
+          var options = [];
342
+          for (var i = 0; i < block.optionCount_; i++) {
343
+            options[i] = '[' + escapeString(block.getFieldValue('USER' + i)) +
344
+                ', ' + escapeString(block.getFieldValue('CPU' + i)) + ']';
345
+          }
346
+          if (options.length) {
347
+            fields.push('new Blockly.FieldDropdown([' +
348
+                options.join(', ') + ']), ' +
349
+                escapeString(block.getFieldValue('FIELDNAME')));
350
+          }
351
+          break;
352
+        case 'field_image':
353
+          // Result: new Blockly.FieldImage('http://...', 80, 60)
354
+          var src = escapeString(block.getFieldValue('SRC'));
355
+          var width = Number(block.getFieldValue('WIDTH'));
356
+          var height = Number(block.getFieldValue('HEIGHT'));
357
+          var alt = escapeString(block.getFieldValue('ALT'));
358
+          fields.push('new Blockly.FieldImage(' +
359
+              src + ', ' + width + ', ' + height + ', ' + alt + ')');
360
+          break;
361
+      }
362
+    }
363
+    block = block.nextConnection && block.nextConnection.targetBlock();
364
+  }
365
+  return fields;
366
+}
367
+
368
+/**
369
+ * Returns field strings and any config.
370
+ * @param {!Blockly.Block} block Input block.
371
+ * @return {!Array.<string|!Object>} Array of static text and field configs.
372
+ * @private
373
+ */
374
+function getFieldsJson_(block) {
375
+  var fields = [];
376
+  while (block) {
377
+    if (!block.disabled && !block.getInheritedDisabled()) {
378
+      switch (block.type) {
379
+        case 'field_static':
380
+          // Result: 'hello'
381
+          fields.push(block.getFieldValue('TEXT'));
382
+          break;
383
+        case 'field_input':
384
+          fields.push({
385
+            type: block.type,
386
+            name: block.getFieldValue('FIELDNAME'),
387
+            text: block.getFieldValue('TEXT')
388
+          });
389
+          break;
390
+        case 'field_angle':
391
+          fields.push({
392
+            type: block.type,
393
+            name: block.getFieldValue('FIELDNAME'),
394
+            angle: Number(block.getFieldValue('ANGLE'))
395
+          });
396
+          break;
397
+        case 'field_checkbox':
398
+          fields.push({
399
+            type: block.type,
400
+            name: block.getFieldValue('FIELDNAME'),
401
+            checked: block.getFieldValue('CHECKED') == 'TRUE'
402
+          });
403
+          break;
404
+        case 'field_colour':
405
+          fields.push({
406
+            type: block.type,
407
+            name: block.getFieldValue('FIELDNAME'),
408
+            colour: block.getFieldValue('COLOUR')
409
+          });
410
+          break;
411
+        case 'field_date':
412
+          fields.push({
413
+            type: block.type,
414
+            name: block.getFieldValue('FIELDNAME'),
415
+            date: block.getFieldValue('DATE')
416
+          });
417
+          break;
418
+        case 'field_variable':
419
+          fields.push({
420
+            type: block.type,
421
+            name: block.getFieldValue('FIELDNAME'),
422
+            variable: block.getFieldValue('TEXT') || null
423
+          });
424
+          break;
425
+        case 'field_dropdown':
426
+          var options = [];
427
+          for (var i = 0; i < block.optionCount_; i++) {
428
+            options[i] = [block.getFieldValue('USER' + i),
429
+                block.getFieldValue('CPU' + i)];
430
+          }
431
+          if (options.length) {
432
+            fields.push({
433
+              type: block.type,
434
+              name: block.getFieldValue('FIELDNAME'),
435
+              options: options
436
+            });
437
+          }
438
+          break;
439
+        case 'field_image':
440
+          fields.push({
441
+            type: block.type,
442
+            src: block.getFieldValue('SRC'),
443
+            width: Number(block.getFieldValue('WIDTH')),
444
+            height: Number(block.getFieldValue('HEIGHT')),
445
+            alt: block.getFieldValue('ALT')
446
+          });
447
+          break;
448
+      }
449
+    }
450
+    block = block.nextConnection && block.nextConnection.targetBlock();
451
+  }
452
+  return fields;
453
+}
454
+
455
+/**
456
+ * Escape a string.
457
+ * @param {string} string String to escape.
458
+ * @return {string} Escaped string surrouned by quotes.
459
+ */
460
+function escapeString(string) {
461
+  return JSON.stringify(string);
462
+}
463
+
464
+/**
465
+ * Fetch the type(s) defined in the given input.
466
+ * Format as a string for appending to the generated code.
467
+ * @param {!Blockly.Block} block Block with input.
468
+ * @param {string} name Name of the input.
469
+ * @return {?string} String defining the types.
470
+ */
471
+function getOptTypesFrom(block, name) {
472
+  var types = getTypesFrom_(block, name);
473
+  if (types.length == 0) {
474
+    return undefined;
475
+  } else if (types.indexOf('null') != -1) {
476
+    return 'null';
477
+  } else if (types.length == 1) {
478
+    return types[0];
479
+  } else {
480
+    return '[' + types.join(', ') + ']';
481
+  }
482
+}
483
+
484
+/**
485
+ * Fetch the type(s) defined in the given input.
486
+ * @param {!Blockly.Block} block Block with input.
487
+ * @param {string} name Name of the input.
488
+ * @return {!Array.<string>} List of types.
489
+ * @private
490
+ */
491
+function getTypesFrom_(block, name) {
492
+  var typeBlock = block.getInputTargetBlock(name);
493
+  var types;
494
+  if (!typeBlock || typeBlock.disabled) {
495
+    types = [];
496
+  } else if (typeBlock.type == 'type_other') {
497
+    types = [escapeString(typeBlock.getFieldValue('TYPE'))];
498
+  } else if (typeBlock.type == 'type_group') {
499
+    types = [];
500
+    for (var n = 0; n < typeBlock.typeCount_; n++) {
501
+      types = types.concat(getTypesFrom_(typeBlock, 'TYPE' + n));
502
+    }
503
+    // Remove duplicates.
504
+    var hash = Object.create(null);
505
+    for (var n = types.length - 1; n >= 0; n--) {
506
+      if (hash[types[n]]) {
507
+        types.splice(n, 1);
508
+      }
509
+      hash[types[n]] = true;
510
+    }
511
+  } else {
512
+    types = [escapeString(typeBlock.valueType)];
513
+  }
514
+  return types;
515
+}
516
+
517
+/**
518
+ * Update the generator code.
519
+ * @param {!Blockly.Block} block Rendered block in preview workspace.
520
+ */
521
+function updateGenerator(block) {
522
+  function makeVar(root, name) {
523
+    name = name.toLowerCase().replace(/\W/g, '_');
524
+    return '  var ' + root + '_' + name;
525
+  }
526
+  var language = document.getElementById('language').value;
527
+  var code = [];
528
+  code.push("Blockly." + language + "['" + block.type +
529
+            "'] = function(block) {");
530
+
531
+  // Generate getters for any fields or inputs.
532
+  for (var i = 0, input; input = block.inputList[i]; i++) {
533
+    for (var j = 0, field; field = input.fieldRow[j]; j++) {
534
+      var name = field.name;
535
+      if (!name) {
536
+        continue;
537
+      }
538
+      if (field instanceof Blockly.FieldVariable) {
539
+        // Subclass of Blockly.FieldDropdown, must test first.
540
+        code.push(makeVar('variable', name) +
541
+                  " = Blockly." + language +
542
+                  ".variableDB_.getName(block.getFieldValue('" + name +
543
+                  "'), Blockly.Variables.NAME_TYPE);");
544
+      } else if (field instanceof Blockly.FieldAngle) {
545
+        // Subclass of Blockly.FieldTextInput, must test first.
546
+        code.push(makeVar('angle', name) +
547
+                  " = block.getFieldValue('" + name + "');");
548
+      } else if (Blockly.FieldDate && field instanceof Blockly.FieldDate) {
549
+        // Blockly.FieldDate may not be compiled into Blockly.
550
+        code.push(makeVar('date', name) +
551
+                  " = block.getFieldValue('" + name + "');");
552
+      } else if (field instanceof Blockly.FieldColour) {
553
+        code.push(makeVar('colour', name) +
554
+                  " = block.getFieldValue('" + name + "');");
555
+      } else if (field instanceof Blockly.FieldCheckbox) {
556
+        code.push(makeVar('checkbox', name) +
557
+                  " = block.getFieldValue('" + name + "') == 'TRUE';");
558
+      } else if (field instanceof Blockly.FieldDropdown) {
559
+        code.push(makeVar('dropdown', name) +
560
+                  " = block.getFieldValue('" + name + "');");
561
+      } else if (field instanceof Blockly.FieldTextInput) {
562
+        code.push(makeVar('text', name) +
563
+                  " = block.getFieldValue('" + name + "');");
564
+      }
565
+    }
566
+    var name = input.name;
567
+    if (name) {
568
+      if (input.type == Blockly.INPUT_VALUE) {
569
+        code.push(makeVar('value', name) +
570
+                  " = Blockly." + language + ".valueToCode(block, '" + name +
571
+                  "', Blockly." + language + ".ORDER_ATOMIC);");
572
+      } else if (input.type == Blockly.NEXT_STATEMENT) {
573
+        code.push(makeVar('statements', name) +
574
+                  " = Blockly." + language + ".statementToCode(block, '" +
575
+                  name + "');");
576
+      }
577
+    }
578
+  }
579
+  code.push("  // TODO: Assemble " + language + " into code variable.");
580
+  code.push("  var code = \'...\';");
581
+  if (block.outputConnection) {
582
+    code.push("  // TODO: Change ORDER_NONE to the correct strength.");
583
+    code.push("  return [code, Blockly." + language + ".ORDER_NONE];");
584
+  } else {
585
+    code.push("  return code;");
586
+  }
587
+  code.push("};");
588
+
589
+  injectCode(code.join('\n'), 'generatorPre');
590
+}
591
+
592
+/**
593
+ * Existing direction ('ltr' vs 'rtl') of preview.
594
+ */
595
+var oldDir = null;
596
+
597
+/**
598
+ * Update the preview display.
599
+ */
600
+function updatePreview() {
601
+  // Toggle between LTR/RTL if needed (also used in first display).
602
+  var newDir = document.getElementById('direction').value;
603
+  if (oldDir != newDir) {
604
+    if (previewWorkspace) {
605
+      previewWorkspace.dispose();
606
+    }
607
+    var rtl = newDir == 'rtl';
608
+    previewWorkspace = Blockly.inject('preview',
609
+        {rtl: rtl,
610
+         media: '../../media/',
611
+         scrollbars: true});
612
+    oldDir = newDir;
613
+  }
614
+  previewWorkspace.clear();
615
+
616
+  // Fetch the code and determine its format (JSON or JavaScript).
617
+  var format = document.getElementById('format').value;
618
+  if (format == 'Manual') {
619
+    var code = document.getElementById('languageTA').value;
620
+    // If the code is JSON, it will parse, otherwise treat as JS.
621
+    try {
622
+      JSON.parse(code);
623
+      format = 'JSON';
624
+    } catch (e) {
625
+      format = 'JavaScript';
626
+    }
627
+  } else {
628
+    var code = document.getElementById('languagePre').textContent;
629
+  }
630
+  if (!code.trim()) {
631
+    // Nothing to render.  Happens while cloud storage is loading.
632
+    return;
633
+  }
634
+
635
+  // Backup Blockly.Blocks object so that main workspace and preview don't
636
+  // collide if user creates a 'factory_base' block, for instance.
637
+  var backupBlocks = Blockly.Blocks;
638
+  try {
639
+    // Make a shallow copy.
640
+    Blockly.Blocks = {};
641
+    for (var prop in backupBlocks) {
642
+      Blockly.Blocks[prop] = backupBlocks[prop];
643
+    }
644
+
645
+    if (format == 'JSON') {
646
+      var json = JSON.parse(code);
647
+      Blockly.Blocks[json.id || UNNAMED] = {
648
+        init: function() {
649
+          this.jsonInit(json);
650
+        }
651
+      };
652
+    } else if (format == 'JavaScript') {
653
+      eval(code);
654
+    } else {
655
+      throw 'Unknown format: ' + format;
656
+    }
657
+
658
+    // Look for a block on Blockly.Blocks that does not match the backup.
659
+    var blockType = null;
660
+    for (var type in Blockly.Blocks) {
661
+      if (typeof Blockly.Blocks[type].init == 'function' &&
662
+          Blockly.Blocks[type] != backupBlocks[type]) {
663
+        blockType = type;
664
+        break;
665
+      }
666
+    }
667
+    if (!blockType) {
668
+      return;
669
+    }
670
+
671
+    // Create the preview block.
672
+    var previewBlock = Blockly.Block.obtain(previewWorkspace, blockType);
673
+    previewBlock.initSvg();
674
+    previewBlock.render();
675
+    previewBlock.setMovable(false);
676
+    previewBlock.setDeletable(false);
677
+    previewBlock.moveBy(15, 10);
678
+
679
+    updateGenerator(previewBlock);
680
+  } finally {
681
+    Blockly.Blocks = backupBlocks;
682
+  }
683
+}
684
+
685
+/**
686
+ * Inject code into a pre tag, with syntax highlighting.
687
+ * Safe from HTML/script injection.
688
+ * @param {string} code Lines of code.
689
+ * @param {string} id ID of <pre> element to inject into.
690
+ */
691
+function injectCode(code, id) {
692
+  Blockly.removeAllRanges();
693
+  var pre = document.getElementById(id);
694
+  pre.textContent = code;
695
+  code = pre.innerHTML;
696
+  code = prettyPrintOne(code, 'js');
697
+  pre.innerHTML = code;
698
+}
699
+
700
+/**
701
+ * Return the uneditable container block that everything else attaches to.
702
+ * @return {Blockly.Block}
703
+ */
704
+function getRootBlock() {
705
+  var blocks = mainWorkspace.getTopBlocks(false);
706
+  for (var i = 0, block; block = blocks[i]; i++) {
707
+    if (block.type == 'factory_base') {
708
+      return block;
709
+    }
710
+  }
711
+  return null;
712
+}
713
+
714
+/**
715
+ * Disable the link button if the format is 'Manual', enable otherwise.
716
+ */
717
+function disableEnableLink() {
718
+  var linkButton = document.getElementById('linkButton');
719
+  linkButton.disabled = document.getElementById('format').value == 'Manual';
720
+}
721
+
722
+/**
723
+ * Initialize Blockly and layout.  Called on page load.
724
+ */
725
+function init() {
726
+  if ('BlocklyStorage' in window) {
727
+    BlocklyStorage.HTTPREQUEST_ERROR =
728
+        'There was a problem with the request.\n';
729
+    BlocklyStorage.LINK_ALERT =
730
+        'Share your blocks with this link:\n\n%1';
731
+    BlocklyStorage.HASH_ERROR =
732
+        'Sorry, "%1" doesn\'t correspond with any saved Blockly file.';
733
+    BlocklyStorage.XML_ERROR = 'Could not load your saved file.\n'+
734
+        'Perhaps it was created with a different version of Blockly?';
735
+    var linkButton = document.getElementById('linkButton');
736
+    linkButton.style.display = 'inline-block';
737
+    linkButton.addEventListener('click',
738
+        function() {BlocklyStorage.link(mainWorkspace);});
739
+    disableEnableLink();
740
+  }
741
+
742
+  document.getElementById('helpButton').addEventListener('click',
743
+    function() {
744
+      open('https://developers.google.com/blockly/custom-blocks/block-factory',
745
+           'BlockFactoryHelp');
746
+    });
747
+
748
+  var expandList = [
749
+    document.getElementById('blockly'),
750
+    document.getElementById('blocklyMask'),
751
+    document.getElementById('preview'),
752
+    document.getElementById('languagePre'),
753
+    document.getElementById('languageTA'),
754
+    document.getElementById('generatorPre')
755
+  ];
756
+  var onresize = function(e) {
757
+    for (var i = 0, expand; expand = expandList[i]; i++) {
758
+      expand.style.width = (expand.parentNode.offsetWidth - 2) + 'px';
759
+      expand.style.height = (expand.parentNode.offsetHeight - 2) + 'px';
760
+    }
761
+  };
762
+  onresize();
763
+  window.addEventListener('resize', onresize);
764
+
765
+  var toolbox = document.getElementById('toolbox');
766
+  mainWorkspace = Blockly.inject('blockly',
767
+      {toolbox: toolbox, media: '../../media/'});
768
+
769
+  // Create the root block.
770
+  if ('BlocklyStorage' in window && window.location.hash.length > 1) {
771
+    BlocklyStorage.retrieveXml(window.location.hash.substring(1),
772
+                               mainWorkspace);
773
+  } else {
774
+    var rootBlock = Blockly.Block.obtain(mainWorkspace, 'factory_base');
775
+    rootBlock.initSvg();
776
+    rootBlock.render();
777
+    rootBlock.setMovable(false);
778
+    rootBlock.setDeletable(false);
779
+  }
780
+
781
+  mainWorkspace.addChangeListener(updateLanguage);
782
+  document.getElementById('direction')
783
+      .addEventListener('change', updatePreview);
784
+  document.getElementById('languageTA')
785
+      .addEventListener('change', updatePreview);
786
+  document.getElementById('languageTA')
787
+      .addEventListener('keyup', updatePreview);
788
+  document.getElementById('format')
789
+      .addEventListener('change', formatChange);
790
+  document.getElementById('language')
791
+      .addEventListener('change', updatePreview);
792
+}
793
+window.addEventListener('load', init);

BIN
src/blockly/demos/blockfactory/icon.png


+ 220 - 0
src/blockly/demos/blockfactory/index.html

@@ -0,0 +1,220 @@
1
+<!DOCTYPE html>
2
+<html>
3
+<head>
4
+  <meta charset="utf-8">
5
+  <meta name="viewport" content="target-densitydpi=device-dpi, height=660, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
6
+  <title>Blockly Demo: Block Factory</title>
7
+  <script src="/storage.js"></script>
8
+  <script src="factory.js"></script>
9
+  <script src="../../blockly_compressed.js"></script>
10
+  <script src="../../msg/messages.js"></script>
11
+  <script src="blocks.js"></script>
12
+  <style>
13
+    html, body {
14
+      height: 100%;
15
+    }
16
+    body {
17
+      background-color: #fff;
18
+      font-family: sans-serif;
19
+      margin: 0 5px;
20
+      overflow: hidden
21
+    }
22
+    h1 {
23
+      font-weight: normal;
24
+      font-size: 140%;
25
+    }
26
+    h3 {
27
+      margin-top: 5px;
28
+      margin-bottom: 0;
29
+    }
30
+    table {
31
+      height: 100%;
32
+      width: 100%;
33
+    }
34
+    td {
35
+      vertical-align: top;
36
+      padding: 0;
37
+    }
38
+    #blockly {
39
+      position: fixed;
40
+    }
41
+    #blocklyMask {
42
+      background-color: #000;
43
+      cursor: not-allowed;
44
+      display: none;
45
+      position: fixed;
46
+      opacity: 0.2;
47
+      z-index: 9;
48
+    }
49
+    #preview {
50
+      position: absolute;
51
+    }
52
+    pre,
53
+    #languageTA {
54
+      border: #ddd 1px solid;
55
+      margin-top: 0;
56
+      position: absolute;
57
+      overflow: scroll;
58
+    }
59
+    #languageTA {
60
+      display: none;
61
+      font-family: monospace;
62
+      font-size: 10pt;
63
+    }
64
+
65
+    button {
66
+      border-radius: 4px;
67
+      border: 1px solid #ddd;
68
+      background-color: #eee;
69
+      color: #000;
70
+      padding: 10px;
71
+      margin: 0 5px;
72
+      font-size: large;
73
+    }
74
+    button:hover:not(:disabled) {
75
+      box-shadow: 2px 2px 5px #888;
76
+    }
77
+    button:disabled {
78
+      opacity: 0.6;
79
+    }
80
+    button>* {
81
+      opacity: 0.6;
82
+      vertical-align: text-bottom;
83
+    }
84
+    button:hover:not(:disabled)>* {
85
+      opacity: 1;
86
+    }
87
+    #linkButton {
88
+      display: none;
89
+    }
90
+  </style>
91
+  <link rel="stylesheet" href="../prettify.css">
92
+  <script src="../prettify.js"></script>
93
+</head>
94
+<body>
95
+  <table>
96
+    <tr>
97
+      <td width="50%" height="5%">
98
+        <h1><a href="https://developers.google.com/blockly/">Blockly</a> &gt;
99
+          <a href="../index.html">Demos</a> &gt; Block Factory</h1>
100
+      </td>
101
+      <td width="50%" height="5%">
102
+        <table>
103
+          <tr>
104
+            <td style="vertical-align: bottom;">
105
+              <h3>Preview:
106
+                <select id="direction">
107
+                  <option value="ltr">LTR</option>
108
+                  <option value="rtl">RTL</option>
109
+                </select>
110
+              </h3>
111
+            </td>
112
+            <td style="vertical-align: middle; text-align: right;">
113
+              <button id="linkButton" title="Save and link to blocks.">
114
+                <img src="link.png" height="21" width="21">
115
+              </button>
116
+              <button id="linkButton" title="Save and link to blocks.">
117
+                <img src="link.png" height="21" width="21">
118
+              </button>
119
+              <button id="helpButton" title="View documentation in new window.">
120
+                <span>Help</span>
121
+              </button>
122
+            </td>
123
+          </tr>
124
+        </table>
125
+      </td>
126
+    </tr>
127
+    <tr>
128
+      <td width="50%" height="95%" style="padding: 2px;">
129
+        <div id="blockly"></div>
130
+        <div id="blocklyMask"></div>
131
+      </td>
132
+      <td width="50%" height="95%">
133
+        <table>
134
+          <tr>
135
+            <td height="30%">
136
+              <div id="preview"></div>
137
+            </td>
138
+          </tr>
139
+          <tr>
140
+            <td height="5%">
141
+              <h3>Language code:
142
+                <select id="format">
143
+                  <option value="JavaScript">JavaScript</option>
144
+                  <option value="JSON">JSON</option>
145
+                  <option value="Manual">Manual edit...</option>
146
+                </select>
147
+              </h3>
148
+            </td>
149
+          </tr>
150
+          <tr>
151
+            <td height="30%">
152
+              <pre id="languagePre"></pre>
153
+              <textarea id="languageTA"></textarea>
154
+            </td>
155
+          </tr>
156
+          <tr>
157
+            <td height="5%">
158
+              <h3>Generator stub:
159
+                <select id="language">
160
+                  <option value="JavaScript">JavaScript</option>
161
+                  <option value="Python">Python</option>
162
+                  <option value="PHP">PHP</option>
163
+                  <option value="Dart">Dart</option>
164
+                </select>
165
+              </h3>
166
+            </td>
167
+          </tr>
168
+          <tr>
169
+            <td height="30%">
170
+              <pre id="generatorPre"></pre>
171
+            </td>
172
+          </tr>
173
+        </table>
174
+      </td>
175
+    </tr>
176
+  </table>
177
+  <xml id="toolbox" style="display: none">
178
+    <category name="Input">
179
+      <block type="input_value"></block>
180
+      <block type="input_statement"></block>
181
+      <block type="input_dummy"></block>
182
+    </category>
183
+    <category name="Field">
184
+      <block type="field_static"></block>
185
+      <block type="field_input"></block>
186
+      <block type="field_angle"></block>
187
+      <block type="field_dropdown"></block>
188
+      <block type="field_checkbox"></block>
189
+      <block type="field_colour"></block>
190
+      <!--
191
+      Date picker commented out since it increases footprint by 60%.
192
+      Add it only if you need it.  See also goog.require in blockly.js.
193
+      <block type="field_date"></block>
194
+      -->
195
+      <block type="field_variable"></block>
196
+      <block type="field_image"></block>
197
+    </category>
198
+    <category name="Type">
199
+      <block type="type_group"></block>
200
+      <block type="type_null"></block>
201
+      <block type="type_boolean"></block>
202
+      <block type="type_number"></block>
203
+      <block type="type_string"></block>
204
+      <block type="type_list"></block>
205
+      <block type="type_other"></block>
206
+    </category>
207
+    <category name="Colour" id="colourCategory">
208
+      <block type="colour_hue"><mutation colour="20"></mutation><field name="HUE">20</field></block>
209
+      <block type="colour_hue"><mutation colour="65"></mutation><field name="HUE">65</field></block>
210
+      <block type="colour_hue"><mutation colour="120"></mutation><field name="HUE">120</field></block>
211
+      <block type="colour_hue"><mutation colour="160"></mutation><field name="HUE">160</field></block>
212
+      <block type="colour_hue"><mutation colour="210"></mutation><field name="HUE">210</field></block>
213
+      <block type="colour_hue"><mutation colour="230"></mutation><field name="HUE">230</field></block>
214
+      <block type="colour_hue"><mutation colour="260"></mutation><field name="HUE">260</field></block>
215
+      <block type="colour_hue"><mutation colour="290"></mutation><field name="HUE">290</field></block>
216
+      <block type="colour_hue"><mutation colour="330"></mutation><field name="HUE">330</field></block>
217
+    </category>
218
+  </xml>
219
+</body>
220
+</html>

BIN
src/blockly/demos/blockfactory/link.png


+ 533 - 0
src/blockly/demos/code/code.js

@@ -0,0 +1,533 @@
1
+/**
2
+ * Blockly Demos: Code
3
+ *
4
+ * Copyright 2012 Google Inc.
5
+ * https://developers.google.com/blockly/
6
+ *
7
+ * Licensed under the Apache License, Version 2.0 (the "License");
8
+ * you may not use this file except in compliance with the License.
9
+ * You may obtain a copy of the License at
10
+ *
11
+ *   http://www.apache.org/licenses/LICENSE-2.0
12
+ *
13
+ * Unless required by applicable law or agreed to in writing, software
14
+ * distributed under the License is distributed on an "AS IS" BASIS,
15
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ * See the License for the specific language governing permissions and
17
+ * limitations under the License.
18
+ */
19
+
20
+/**
21
+ * @fileoverview JavaScript for Blockly's Code demo.
22
+ * @author fraser@google.com (Neil Fraser)
23
+ */
24
+'use strict';
25
+
26
+/**
27
+ * Create a namespace for the application.
28
+ */
29
+var Code = {};
30
+
31
+/**
32
+ * Lookup for names of supported languages.  Keys should be in ISO 639 format.
33
+ */
34
+Code.LANGUAGE_NAME = {
35
+  'ar': 'العربية',
36
+  'be-tarask': 'Taraškievica',
37
+  'br': 'Brezhoneg',
38
+  'ca': 'Català',
39
+  'cs': 'Česky',
40
+  'da': 'Dansk',
41
+  'de': 'Deutsch',
42
+  'el': 'Ελληνικά',
43
+  'en': 'English',
44
+  'es': 'Español',
45
+  'fa': 'فارسی',
46
+  'fr': 'Français',
47
+  'he': 'עברית',
48
+  'hrx': 'Hunsrik',
49
+  'hu': 'Magyar',
50
+  'ia': 'Interlingua',
51
+  'is': 'Íslenska',
52
+  'it': 'Italiano',
53
+  'ja': '日本語',
54
+  'ko': '한국어',
55
+  'mk': 'Македонски',
56
+  'ms': 'Bahasa Melayu',
57
+  'nb': 'Norsk Bokmål',
58
+  'nl': 'Nederlands, Vlaams',
59
+  'oc': 'Lenga d\'òc',
60
+  'pl': 'Polski',
61
+  'pms': 'Piemontèis',
62
+  'pt-br': 'Português Brasileiro',
63
+  'ro': 'Română',
64
+  'ru': 'Русский',
65
+  'sc': 'Sardu',
66
+  'sk': 'Slovenčina',
67
+  'sr': 'Српски',
68
+  'sv': 'Svenska',
69
+  'th': 'ภาษาไทย',
70
+  'tlh': 'tlhIngan Hol',
71
+  'tr': 'Türkçe',
72
+  'uk': 'Українська',
73
+  'vi': 'Tiếng Việt',
74
+  'zh-hans': '簡體中文',
75
+  'zh-hant': '正體中文'
76
+};
77
+
78
+/**
79
+ * List of RTL languages.
80
+ */
81
+Code.LANGUAGE_RTL = ['ar', 'fa', 'he'];
82
+
83
+/**
84
+ * Blockly's main workspace.
85
+ * @type {Blockly.WorkspaceSvg}
86
+ */
87
+Code.workspace = null;
88
+
89
+/**
90
+ * Extracts a parameter from the URL.
91
+ * If the parameter is absent default_value is returned.
92
+ * @param {string} name The name of the parameter.
93
+ * @param {string} defaultValue Value to return if paramater not found.
94
+ * @return {string} The parameter value or the default value if not found.
95
+ */
96
+Code.getStringParamFromUrl = function(name, defaultValue) {
97
+  var val = location.search.match(new RegExp('[?&]' + name + '=([^&]+)'));
98
+  return val ? decodeURIComponent(val[1].replace(/\+/g, '%20')) : defaultValue;
99
+};
100
+
101
+/**
102
+ * Get the language of this user from the URL.
103
+ * @return {string} User's language.
104
+ */
105
+Code.getLang = function() {
106
+  var lang = Code.getStringParamFromUrl('lang', '');
107
+  if (Code.LANGUAGE_NAME[lang] === undefined) {
108
+    // Default to English.
109
+    lang = 'en';
110
+  }
111
+  return lang;
112
+};
113
+
114
+/**
115
+ * Is the current language (Code.LANG) an RTL language?
116
+ * @return {boolean} True if RTL, false if LTR.
117
+ */
118
+Code.isRtl = function() {
119
+  return Code.LANGUAGE_RTL.indexOf(Code.LANG) != -1;
120
+};
121
+
122
+/**
123
+ * Load blocks saved on App Engine Storage or in session/local storage.
124
+ * @param {string} defaultXml Text representation of default blocks.
125
+ */
126
+Code.loadBlocks = function(defaultXml) {
127
+  try {
128
+    var loadOnce = window.sessionStorage.loadOnceBlocks;
129
+  } catch(e) {
130
+    // Firefox sometimes throws a SecurityError when accessing sessionStorage.
131
+    // Restarting Firefox fixes this, so it looks like a bug.
132
+    var loadOnce = null;
133
+  }
134
+  if ('BlocklyStorage' in window && window.location.hash.length > 1) {
135
+    // An href with #key trigers an AJAX call to retrieve saved blocks.
136
+    BlocklyStorage.retrieveXml(window.location.hash.substring(1));
137
+  } else if (loadOnce) {
138
+    // Language switching stores the blocks during the reload.
139
+    delete window.sessionStorage.loadOnceBlocks;
140
+    var xml = Blockly.Xml.textToDom(loadOnce);
141
+    Blockly.Xml.domToWorkspace(Code.workspace, xml);
142
+  } else if (defaultXml) {
143
+    // Load the editor with default starting blocks.
144
+    var xml = Blockly.Xml.textToDom(defaultXml);
145
+    Blockly.Xml.domToWorkspace(Code.workspace, xml);
146
+  } else if ('BlocklyStorage' in window) {
147
+    // Restore saved blocks in a separate thread so that subsequent
148
+    // initialization is not affected from a failed load.
149
+    window.setTimeout(BlocklyStorage.restoreBlocks, 0);
150
+  }
151
+};
152
+
153
+/**
154
+ * Save the blocks and reload with a different language.
155
+ */
156
+Code.changeLanguage = function() {
157
+  // Store the blocks for the duration of the reload.
158
+  // This should be skipped for the index page, which has no blocks and does
159
+  // not load Blockly.
160
+  // MSIE 11 does not support sessionStorage on file:// URLs.
161
+  if (typeof Blockly != 'undefined' && window.sessionStorage) {
162
+    var xml = Blockly.Xml.workspaceToDom(Code.workspace);
163
+    var text = Blockly.Xml.domToText(xml);
164
+    window.sessionStorage.loadOnceBlocks = text;
165
+  }
166
+
167
+  var languageMenu = document.getElementById('languageMenu');
168
+  var newLang = encodeURIComponent(
169
+      languageMenu.options[languageMenu.selectedIndex].value);
170
+  var search = window.location.search;
171
+  if (search.length <= 1) {
172
+    search = '?lang=' + newLang;
173
+  } else if (search.match(/[?&]lang=[^&]*/)) {
174
+    search = search.replace(/([?&]lang=)[^&]*/, '$1' + newLang);
175
+  } else {
176
+    search = search.replace(/\?/, '?lang=' + newLang + '&');
177
+  }
178
+
179
+  window.location = window.location.protocol + '//' +
180
+      window.location.host + window.location.pathname + search;
181
+};
182
+
183
+/**
184
+ * Bind a function to a button's click event.
185
+ * On touch enabled browsers, ontouchend is treated as equivalent to onclick.
186
+ * @param {!Element|string} el Button element or ID thereof.
187
+ * @param {!Function} func Event handler to bind.
188
+ */
189
+Code.bindClick = function(el, func) {
190
+  if (typeof el == 'string') {
191
+    el = document.getElementById(el);
192
+  }
193
+  el.addEventListener('click', func, true);
194
+  el.addEventListener('touchend', func, true);
195
+};
196
+
197
+/**
198
+ * Load the Prettify CSS and JavaScript.
199
+ */
200
+Code.importPrettify = function() {
201
+  //<link rel="stylesheet" href="../prettify.css">
202
+  //<script src="../prettify.js"></script>
203
+  var link = document.createElement('link');
204
+  link.setAttribute('rel', 'stylesheet');
205
+  link.setAttribute('href', '../prettify.css');
206
+  document.head.appendChild(link);
207
+  var script = document.createElement('script');
208
+  script.setAttribute('src', '../prettify.js');
209
+  document.head.appendChild(script);
210
+};
211
+
212
+/**
213
+ * Compute the absolute coordinates and dimensions of an HTML element.
214
+ * @param {!Element} element Element to match.
215
+ * @return {!Object} Contains height, width, x, and y properties.
216
+ * @private
217
+ */
218
+Code.getBBox_ = function(element) {
219
+  var height = element.offsetHeight;
220
+  var width = element.offsetWidth;
221
+  var x = 0;
222
+  var y = 0;
223
+  do {
224
+    x += element.offsetLeft;
225
+    y += element.offsetTop;
226
+    element = element.offsetParent;
227
+  } while (element);
228
+  return {
229
+    height: height,
230
+    width: width,
231
+    x: x,
232
+    y: y
233
+  };
234
+};
235
+
236
+/**
237
+ * User's language (e.g. "en").
238
+ * @type {string}
239
+ */
240
+Code.LANG = Code.getLang();
241
+
242
+/**
243
+ * List of tab names.
244
+ * @private
245
+ */
246
+Code.TABS_ = ['blocks', 'javascript', 'php', 'python', 'dart', 'arduino', 'xml'];
247
+
248
+Code.selected = 'blocks';
249
+
250
+/**
251
+ * Switch the visible pane when a tab is clicked.
252
+ * @param {string} clickedName Name of tab clicked.
253
+ */
254
+Code.tabClick = function(clickedName) {
255
+  // If the XML tab was open, save and render the content.
256
+  if (document.getElementById('tab_xml').className == 'tabon') {
257
+    var xmlTextarea = document.getElementById('content_xml');
258
+    var xmlText = xmlTextarea.value;
259
+    var xmlDom = null;
260
+    try {
261
+      xmlDom = Blockly.Xml.textToDom(xmlText);
262
+    } catch (e) {
263
+      var q =
264
+          window.confirm(MSG['badXml'].replace('%1', e));
265
+      if (!q) {
266
+        // Leave the user on the XML tab.
267
+        return;
268
+      }
269
+    }
270
+    if (xmlDom) {
271
+      Code.workspace.clear();
272
+      Blockly.Xml.domToWorkspace(Code.workspace, xmlDom);
273
+    }
274
+  }
275
+
276
+  if (document.getElementById('tab_blocks').className == 'tabon') {
277
+    Code.workspace.setVisible(false);
278
+  }
279
+  // Deselect all tabs and hide all panes.
280
+  for (var i = 0; i < Code.TABS_.length; i++) {
281
+    var name = Code.TABS_[i];
282
+    document.getElementById('tab_' + name).className = 'taboff';
283
+    document.getElementById('content_' + name).style.visibility = 'hidden';
284
+  }
285
+
286
+  // Select the active tab.
287
+  Code.selected = clickedName;
288
+  document.getElementById('tab_' + clickedName).className = 'tabon';
289
+  // Show the selected pane.
290
+  document.getElementById('content_' + clickedName).style.visibility =
291
+      'visible';
292
+  Code.renderContent();
293
+  if (clickedName == 'blocks') {
294
+    Code.workspace.setVisible(true);
295
+  }
296
+  Blockly.fireUiEvent(window, 'resize');
297
+};
298
+
299
+/**
300
+ * Populate the currently selected pane with content generated from the blocks.
301
+ */
302
+Code.renderContent = function() {
303
+  var content = document.getElementById('content_' + Code.selected);
304
+  // Initialize the pane.
305
+  if (content.id == 'content_xml') {
306
+    var xmlTextarea = document.getElementById('content_xml');
307
+    var xmlDom = Blockly.Xml.workspaceToDom(Code.workspace);
308
+    var xmlText = Blockly.Xml.domToPrettyText(xmlDom);
309
+    xmlTextarea.value = xmlText;
310
+    xmlTextarea.focus();
311
+  } else if (content.id == 'content_javascript') {
312
+    var code = Blockly.JavaScript.workspaceToCode(Code.workspace);
313
+    content.textContent = code;
314
+    if (typeof prettyPrintOne == 'function') {
315
+      code = content.innerHTML;
316
+      code = prettyPrintOne(code, 'js');
317
+      content.innerHTML = code;
318
+    }
319
+  } else if (content.id == 'content_python') {
320
+    code = Blockly.Python.workspaceToCode(Code.workspace);
321
+    content.textContent = code;
322
+    if (typeof prettyPrintOne == 'function') {
323
+      code = content.innerHTML;
324
+      code = prettyPrintOne(code, 'py');
325
+      content.innerHTML = code;
326
+    }
327
+  } else if (content.id == 'content_php') {
328
+    code = Blockly.PHP.workspaceToCode(Code.workspace);
329
+    content.textContent = code;
330
+    if (typeof prettyPrintOne == 'function') {
331
+      code = content.innerHTML;
332
+      code = prettyPrintOne(code, 'php');
333
+      content.innerHTML = code;
334
+    }
335
+  } else if (content.id == 'content_dart') {
336
+    code = Blockly.Dart.workspaceToCode(Code.workspace);
337
+    content.textContent = code;
338
+    if (typeof prettyPrintOne == 'function') {
339
+      code = content.innerHTML;
340
+      code = prettyPrintOne(code, 'dart');
341
+      content.innerHTML = code;
342
+    }
343
+  } else if (content.id == 'content_arduino') {
344
+    code = Blockly.Arduino.workspaceToCode(Code.workspace);
345
+    content.textContent = code;
346
+    if (typeof prettyPrintOne == 'function') {
347
+      code = content.innerHTML;
348
+      code = prettyPrintOne(code, 'arduino');
349
+      content.innerHTML = code;
350
+    }
351
+  }
352
+};
353
+
354
+/**
355
+ * Initialize Blockly.  Called on page load.
356
+ */
357
+Code.init = function() {
358
+  Code.initLanguage();
359
+
360
+  var rtl = Code.isRtl();
361
+  var container = document.getElementById('content_area');
362
+  var onresize = function(e) {
363
+    var bBox = Code.getBBox_(container);
364
+    for (var i = 0; i < Code.TABS_.length; i++) {
365
+      var el = document.getElementById('content_' + Code.TABS_[i]);
366
+      el.style.top = bBox.y + 'px';
367
+      el.style.left = bBox.x + 'px';
368
+      // Height and width need to be set, read back, then set again to
369
+      // compensate for scrollbars.
370
+      el.style.height = bBox.height + 'px';
371
+      el.style.height = (2 * bBox.height - el.offsetHeight) + 'px';
372
+      el.style.width = bBox.width + 'px';
373
+      el.style.width = (2 * bBox.width - el.offsetWidth) + 'px';
374
+    }
375
+    // Make the 'Blocks' tab line up with the toolbox.
376
+    if (Code.workspace.toolbox_.width) {
377
+      document.getElementById('tab_blocks').style.minWidth =
378
+          (Code.workspace.toolbox_.width - 38) + 'px';
379
+          // Account for the 19 pixel margin and on each side.
380
+    }
381
+  };
382
+  window.addEventListener('resize', onresize, false);
383
+
384
+  var toolbox = document.getElementById('toolbox');
385
+  Code.workspace = Blockly.inject('content_blocks',
386
+      {grid:
387
+          {spacing: 25,
388
+           length: 3,
389
+           colour: '#ccc',
390
+           snap: true},
391
+       media: '../../media/',
392
+       rtl: rtl,
393
+       toolbox: toolbox});
394
+
395
+  // Add to reserved word list: Local variables in execution evironment (runJS)
396
+  // and the infinite loop detection function.
397
+  Blockly.JavaScript.addReservedWords('code,timeouts,checkTimeout');
398
+
399
+  Code.loadBlocks('');
400
+
401
+  if ('BlocklyStorage' in window) {
402
+    // Hook a save function onto unload.
403
+    BlocklyStorage.backupOnUnload(Code.workspace);
404
+  }
405
+
406
+  Code.tabClick(Code.selected);
407
+  Blockly.fireUiEvent(window, 'resize');
408
+
409
+  Code.bindClick('trashButton',
410
+      function() {Code.discard(); Code.renderContent();});
411
+  Code.bindClick('runButton', Code.runJS);
412
+  // Disable the link button if page isn't backed by App Engine storage.
413
+  var linkButton = document.getElementById('linkButton');
414
+  if ('BlocklyStorage' in window) {
415
+    BlocklyStorage['HTTPREQUEST_ERROR'] = MSG['httpRequestError'];
416
+    BlocklyStorage['LINK_ALERT'] = MSG['linkAlert'];
417
+    BlocklyStorage['HASH_ERROR'] = MSG['hashError'];
418
+    BlocklyStorage['XML_ERROR'] = MSG['xmlError'];
419
+    Code.bindClick(linkButton,
420
+        function() {BlocklyStorage.link(Code.workspace);});
421
+  } else if (linkButton) {
422
+    linkButton.className = 'disabled';
423
+  }
424
+
425
+  for (var i = 0; i < Code.TABS_.length; i++) {
426
+    var name = Code.TABS_[i];
427
+    Code.bindClick('tab_' + name,
428
+        function(name_) {return function() {Code.tabClick(name_);};}(name));
429
+  }
430
+
431
+  onresize();
432
+  // Lazy-load the syntax-highlighting.
433
+  window.setTimeout(Code.importPrettify, 1);
434
+};
435
+
436
+/**
437
+ * Initialize the page language.
438
+ */
439
+Code.initLanguage = function() {
440
+  // Set the HTML's language and direction.
441
+  var rtl = Code.isRtl();
442
+  document.dir = rtl ? 'rtl' : 'ltr';
443
+  document.head.parentElement.setAttribute('lang', Code.LANG);
444
+
445
+  // Sort languages alphabetically.
446
+  var languages = [];
447
+  for (var lang in Code.LANGUAGE_NAME) {
448
+    languages.push([Code.LANGUAGE_NAME[lang], lang]);
449
+  }
450
+  var comp = function(a, b) {
451
+    // Sort based on first argument ('English', 'Русский', '简体字', etc).
452
+    if (a[0] > b[0]) return 1;
453
+    if (a[0] < b[0]) return -1;
454
+    return 0;
455
+  };
456
+  languages.sort(comp);
457
+  // Populate the language selection menu.
458
+  var languageMenu = document.getElementById('languageMenu');
459
+  languageMenu.options.length = 0;
460
+  for (var i = 0; i < languages.length; i++) {
461
+    var tuple = languages[i];
462
+    var lang = tuple[tuple.length - 1];
463
+    var option = new Option(tuple[0], lang);
464
+    if (lang == Code.LANG) {
465
+      option.selected = true;
466
+    }
467
+    languageMenu.options.add(option);
468
+  }
469
+  languageMenu.addEventListener('change', Code.changeLanguage, true);
470
+
471
+  // Inject language strings.
472
+  document.title += ' ' + MSG['title'];
473
+  document.getElementById('title').textContent = MSG['title'];
474
+  document.getElementById('tab_blocks').textContent = MSG['blocks'];
475
+
476
+  document.getElementById('linkButton').title = MSG['linkTooltip'];
477
+  document.getElementById('runButton').title = MSG['runTooltip'];
478
+  document.getElementById('trashButton').title = MSG['trashTooltip'];
479
+
480
+  var categories = ['catLogic', 'catLoops', 'catMath', 'catText', 'catLists',
481
+                    'catColour', 'catVariables', 'catFunctions'];
482
+  for (var i = 0, cat; cat = categories[i]; i++) {
483
+    document.getElementById(cat).setAttribute('name', MSG[cat]);
484
+  }
485
+  var textVars = document.getElementsByClassName('textVar');
486
+  for (var i = 0, textVar; textVar = textVars[i]; i++) {
487
+    textVar.textContent = MSG['textVariable'];
488
+  }
489
+  var listVars = document.getElementsByClassName('listVar');
490
+  for (var i = 0, listVar; listVar = listVars[i]; i++) {
491
+    listVar.textContent = MSG['listVariable'];
492
+  }
493
+};
494
+
495
+/**
496
+ * Execute the user's code.
497
+ * Just a quick and dirty eval.  Catch infinite loops.
498
+ */
499
+Code.runJS = function() {
500
+  Blockly.JavaScript.INFINITE_LOOP_TRAP = '  checkTimeout();\n';
501
+  var timeouts = 0;
502
+  var checkTimeout = function() {
503
+    if (timeouts++ > 1000000) {
504
+      throw MSG['timeout'];
505
+    }
506
+  };
507
+  var code = Blockly.JavaScript.workspaceToCode(Code.workspace);
508
+  Blockly.JavaScript.INFINITE_LOOP_TRAP = null;
509
+  try {
510
+    eval(code);
511
+  } catch (e) {
512
+    alert(MSG['badCode'].replace('%1', e));
513
+  }
514
+};
515
+
516
+/**
517
+ * Discard all blocks from the workspace.
518
+ */
519
+Code.discard = function() {
520
+  var count = Code.workspace.getAllBlocks().length;
521
+  if (count < 2 ||
522
+      window.confirm(MSG['discard'].replace('%1', count))) {
523
+    Code.workspace.clear();
524
+    window.location.hash = '';
525
+  }
526
+};
527
+
528
+// Load the Code demo's language strings.
529
+document.write('<script src="msg/' + Code.LANG + '.js"></script>\n');
530
+// Load Blockly's language strings.
531
+document.write('<script src="../../msg/js/' + Code.LANG + '.js"></script>\n');
532
+
533
+window.addEventListener('load', Code.init);

BIN
src/blockly/demos/code/icon.png


BIN
src/blockly/demos/code/icons.png


+ 290 - 0
src/blockly/demos/code/index.html

@@ -0,0 +1,290 @@
1
+<!DOCTYPE html>
2
+<html>
3
+<head>
4
+  <meta charset="utf-8">
5
+  <meta name="google" value="notranslate">
6
+  <title>Blockly Demo:</title>
7
+  <link rel="stylesheet" href="style.css">
8
+  <script src="/storage.js"></script>
9
+  <script src="../../blockly_compressed.js"></script>
10
+  <script src="../../blocks_compressed.js"></script>
11
+  <script src="../../javascript_compressed.js"></script>
12
+  <script src="../../python_compressed.js"></script>
13
+  <script src="../../php_compressed.js"></script>
14
+  <script src="../../dart_compressed.js"></script>
15
+  <script src="../../arduino_compressed.js"></script>
16
+  <script src="code.js"></script>
17
+</head>
18
+<body>
19
+  <table width="100%" height="100%">
20
+    <tr>
21
+      <td>
22
+        <h1><a href="https://developers.google.com/blockly/">Blockly</a>&rlm; &gt;
23
+          <a href="../index.html">Demos</a>&rlm; &gt;
24
+          <span id="title">...</span>
25
+        </h1>
26
+      </td>
27
+      <td class="farSide">
28
+        <select id="languageMenu"></select>
29
+      </td>
30
+    </tr>
31
+    <tr>
32
+      <td colspan=2>
33
+        <table width="100%">
34
+          <tr id="tabRow" height="1em">
35
+            <td id="tab_blocks" class="tabon">...</td>
36
+            <td class="tabmin">&nbsp;</td>
37
+            <td id="tab_javascript" class="taboff">JavaScript</td>
38
+            <td class="tabmin">&nbsp;</td>
39
+            <td id="tab_python" class="taboff">Python</td>
40
+            <td class="tabmin">&nbsp;</td>
41
+            <td id="tab_php" class="taboff">PHP</td>
42
+            <td class="tabmin">&nbsp;</td>
43
+            <td id="tab_dart" class="taboff">Dart</td>
44
+            <td class="tabmin">&nbsp;</td>
45
+            <td id="tab_arduino" class="taboff">Arduino</td>
46
+            <td class="tabmin">&nbsp;</td>
47
+            <td id="tab_xml" class="taboff">XML</td>
48
+            <td class="tabmax">
49
+              <button id="trashButton" class="notext" title="...">
50
+                <img src='../../media/1x1.gif' class="trash icon21">
51
+              </button>
52
+              <button id="linkButton" class="notext" title="...">
53
+                <img src='../../media/1x1.gif' class="link icon21">
54
+              </button>
55
+              <button id="runButton" class="notext primary" title="...">
56
+                <img src='../../media/1x1.gif' class="run icon21">
57
+              </button>
58
+            </td>
59
+          </tr>
60
+        </table>
61
+      </td>
62
+    </tr>
63
+    <tr>
64
+      <td height="99%" colspan=2 id="content_area">
65
+      </td>
66
+    </tr>
67
+  </table>
68
+  <div id="content_blocks" class="content"></div>
69
+  <pre id="content_javascript" class="content"></pre>
70
+  <pre id="content_php" class="content"></pre>
71
+  <pre id="content_arduino" class="content"></pre>
72
+  <pre id="content_python" class="content"></pre>
73
+  <pre id="content_dart" class="content"></pre>
74
+  <textarea id="content_xml" class="content" wrap="off"></textarea>
75
+
76
+  <xml id="toolbox" style="display: none">
77
+    <category id="catLogic">
78
+      <block type="controls_if"></block>
79
+      <block type="logic_compare"></block>
80
+      <block type="logic_operation"></block>
81
+      <block type="logic_negate"></block>
82
+      <block type="logic_boolean"></block>
83
+      <block type="logic_null"></block>
84
+      <block type="logic_ternary"></block>
85
+    </category>
86
+    <category id="catLoops">
87
+      <block type="controls_repeat_ext">
88
+        <value name="TIMES">
89
+          <block type="math_number">
90
+            <field name="NUM">10</field>
91
+          </block>
92
+        </value>
93
+      </block>
94
+      <block type="controls_whileUntil"></block>
95
+      <block type="controls_for">
96
+        <value name="FROM">
97
+          <block type="math_number">
98
+            <field name="NUM">1</field>
99
+          </block>
100
+        </value>
101
+        <value name="TO">
102
+          <block type="math_number">
103
+            <field name="NUM">10</field>
104
+          </block>
105
+        </value>
106
+        <value name="BY">
107
+          <block type="math_number">
108
+            <field name="NUM">1</field>
109
+          </block>
110
+        </value>
111
+      </block>
112
+      <block type="controls_forEach"></block>
113
+      <block type="controls_flow_statements"></block>
114
+    </category>
115
+    <category id="catMath">
116
+      <block type="math_number"></block>
117
+      <block type="math_arithmetic"></block>
118
+      <block type="math_single"></block>
119
+      <block type="math_trig"></block>
120
+      <block type="math_constant"></block>
121
+      <block type="math_number_property"></block>
122
+      <block type="math_change">
123
+        <value name="DELTA">
124
+          <block type="math_number">
125
+            <field name="NUM">1</field>
126
+          </block>
127
+        </value>
128
+      </block>
129
+      <block type="math_round"></block>
130
+      <block type="math_on_list"></block>
131
+      <block type="math_modulo"></block>
132
+      <block type="math_constrain">
133
+        <value name="LOW">
134
+          <block type="math_number">
135
+            <field name="NUM">1</field>
136
+          </block>
137
+        </value>
138
+        <value name="HIGH">
139
+          <block type="math_number">
140
+            <field name="NUM">100</field>
141
+          </block>
142
+        </value>
143
+      </block>
144
+      <block type="math_random_int">
145
+        <value name="FROM">
146
+          <block type="math_number">
147
+            <field name="NUM">1</field>
148
+          </block>
149
+        </value>
150
+        <value name="TO">
151
+          <block type="math_number">
152
+            <field name="NUM">100</field>
153
+          </block>
154
+        </value>
155
+      </block>
156
+      <block type="math_random_float"></block>
157
+    </category>
158
+    <category id="catText">
159
+      <block type="text"></block>
160
+      <block type="text_join"></block>
161
+      <block type="text_append">
162
+        <value name="TEXT">
163
+          <block type="text"></block>
164
+        </value>
165
+      </block>
166
+      <block type="text_length"></block>
167
+      <block type="text_isEmpty"></block>
168
+      <block type="text_indexOf">
169
+        <value name="VALUE">
170
+          <block type="variables_get">
171
+            <field name="VAR" class="textVar">...</field>
172
+          </block>
173
+        </value>
174
+      </block>
175
+      <block type="text_charAt">
176
+        <value name="VALUE">
177
+          <block type="variables_get">
178
+            <field name="VAR" class="textVar">...</field>
179
+          </block>
180
+        </value>
181
+      </block>
182
+      <block type="text_getSubstring">
183
+        <value name="STRING">
184
+          <block type="variables_get">
185
+            <field name="VAR" class="textVar">...</field>
186
+          </block>
187
+        </value>
188
+      </block>
189
+      <block type="text_changeCase"></block>
190
+      <block type="text_trim"></block>
191
+      <block type="text_print"></block>
192
+      <block type="text_prompt_ext">
193
+        <value name="TEXT">
194
+          <block type="text"></block>
195
+        </value>
196
+      </block>
197
+    </category>
198
+    <category id="catLists">
199
+      <block type="lists_create_empty"></block>
200
+      <block type="lists_create_with"></block>
201
+      <block type="lists_repeat">
202
+        <value name="NUM">
203
+          <block type="math_number">
204
+            <field name="NUM">5</field>
205
+          </block>
206
+        </value>
207
+      </block>
208
+      <block type="lists_length"></block>
209
+      <block type="lists_isEmpty"></block>
210
+      <block type="lists_indexOf">
211
+        <value name="VALUE">
212
+          <block type="variables_get">
213
+            <field name="VAR" class="listVar">...</field>
214
+          </block>
215
+        </value>
216
+      </block>
217
+      <block type="lists_getIndex">
218
+        <value name="VALUE">
219
+          <block type="variables_get">
220
+            <field name="VAR" class="listVar">...</field>
221
+          </block>
222
+        </value>
223
+      </block>
224
+      <block type="lists_setIndex">
225
+        <value name="LIST">
226
+          <block type="variables_get">
227
+            <field name="VAR" class="listVar">...</field>
228
+          </block>
229
+        </value>
230
+      </block>
231
+      <block type="lists_getSublist">
232
+        <value name="LIST">
233
+          <block type="variables_get">
234
+            <field name="VAR" class="listVar">...</field>
235
+          </block>
236
+        </value>
237
+      </block>
238
+      <block type="lists_split">
239
+        <value name="DELIM">
240
+          <block type="text">
241
+            <field name="TEXT">,</field>
242
+          </block>
243
+        </value>
244
+      </block>
245
+    </category>
246
+    <category id="catColour">
247
+      <block type="colour_picker"></block>
248
+      <block type="colour_random"></block>
249
+      <block type="colour_rgb">
250
+        <value name="RED">
251
+          <block type="math_number">
252
+            <field name="NUM">100</field>
253
+          </block>
254
+        </value>
255
+        <value name="GREEN">
256
+          <block type="math_number">
257
+            <field name="NUM">50</field>
258
+          </block>
259
+        </value>
260
+        <value name="BLUE">
261
+          <block type="math_number">
262
+            <field name="NUM">0</field>
263
+          </block>
264
+        </value>
265
+      </block>
266
+      <block type="colour_blend">
267
+        <value name="COLOUR1">
268
+          <block type="colour_picker">
269
+            <field name="COLOUR">#ff0000</field>
270
+          </block>
271
+        </value>
272
+        <value name="COLOUR2">
273
+          <block type="colour_picker">
274
+            <field name="COLOUR">#3333ff</field>
275
+          </block>
276
+        </value>
277
+        <value name="RATIO">
278
+          <block type="math_number">
279
+            <field name="NUM">0.5</field>
280
+          </block>
281
+        </value>
282
+      </block>
283
+    </category>
284
+    <sep></sep>
285
+    <category id="catVariables" custom="VARIABLE"></category>
286
+    <category id="catFunctions" custom="PROCEDURE"></category>
287
+  </xml>
288
+
289
+</body>
290
+</html>

+ 25 - 0
src/blockly/demos/code/msg/ar.js

@@ -0,0 +1,25 @@
1
+var MSG = {
2
+  title: "كود",
3
+  blocks: "البلوكات",
4
+  linkTooltip: "احفظ ووصلة إلى البلوكات.",
5
+  runTooltip: "شغل البرنامج المعرف بواسطة البلوكات في مساحة العمل.",
6
+  badCode: "خطأ في البرنامج:\n %1",
7
+  timeout: "تم تجاوز الحد الأقصى لتكرارات التنفيذ .",
8
+  discard: "حذف كل بلوكات %1؟",
9
+  trashTooltip: "تجاهل كل البلوكات.",
10
+  catLogic: "منطق",
11
+  catLoops: "الحلقات",
12
+  catMath: "رياضيات",
13
+  catText: "نص",
14
+  catLists: "قوائم",
15
+  catColour: "لون",
16
+  catVariables: "متغيرات",
17
+  catFunctions: "إجراءات",
18
+  listVariable: "قائمة",
19
+  textVariable: "نص",
20
+  httpRequestError: "كانت هناك مشكلة مع هذا الطلب.",
21
+  linkAlert: "مشاركة كود بلوكلي الخاص بك مع هذا الرابط:\n %1",
22
+  hashError: "عذراً،ال '%1' لا تتوافق مع أي برنامج تم حفظه.",
23
+  xmlError: "تعذر تحميل الملف المحفوظة الخاصة بك.  ربما تم إنشاؤه باستخدام إصدار مختلف من بلوكلي؟",
24
+  badXml: "خطأ في توزيع ال \"XML\":\n %1\n\nحدد 'موافق' للتخلي عن التغييرات أو 'إلغاء الأمر' لمواصلة تحرير ال\"XML\"."
25
+};

+ 25 - 0
src/blockly/demos/code/msg/be-tarask.js

@@ -0,0 +1,25 @@
1
+var MSG = {
2
+  title: "Код",
3
+  blocks: "Блёкі",
4
+  linkTooltip: "Захаваць і зьвязаць з блёкамі.",
5
+  runTooltip: "Запусьціце праграму, вызначаную блёкамі ў працоўнай вобласьці.",
6
+  badCode: "Памылка праграмы:\n%1",
7
+  timeout: "Перавышана максымальная колькасьць ітэрацыяў.",
8
+  discard: "Выдаліць усе блёкі %1?",
9
+  trashTooltip: "Выдаліць усе блёкі.",
10
+  catLogic: "Лёгіка",
11
+  catLoops: "Петлі",
12
+  catMath: "Матэматычныя формулы",
13
+  catText: "Тэкст",
14
+  catLists: "Сьпісы",
15
+  catColour: "Колер",
16
+  catVariables: "Зьменныя",
17
+  catFunctions: "Функцыі",
18
+  listVariable: "сьпіс",
19
+  textVariable: "тэкст",
20
+  httpRequestError: "Узьнікла праблема з запытам.",
21
+  linkAlert: "Падзяліцца Вашым блёкам праз гэтую спасылку:\n\n%1",
22
+  hashError: "Прабачце, '%1' не адпавядае ніводнай захаванай праграме.",
23
+  xmlError: "Не атрымалася загрузіць захаваны файл. Магчыма, ён быў створаны з іншай вэрсіяй Блёклі?",
24
+  badXml: "Памылка сынтаксічнага аналізу XML:\n%1\n\nАбярыце \"ОК\", каб адмовіцца ад зьменаў ці \"Скасаваць\" для далейшага рэдагаваньня XML."
25
+};

+ 25 - 0
src/blockly/demos/code/msg/br.js

@@ -0,0 +1,25 @@
1
+var MSG = {
2
+  title: "Kod",
3
+  blocks: "Bloc'hoù",
4
+  linkTooltip: "Enrollañ ha liammañ d'ar bloc'hadoù.",
5
+  runTooltip: "Lañsañ ar programm termenet gant ar bloc'hadoù en takad labour.",
6
+  badCode: "Fazi programm :\n%1",
7
+  timeout: "Tizhet eo bet an niver brasañ a iteradurioù seveniñ aotreet.",
8
+  discard: "Diverkañ an holl vloc'hoù %1 ?",
9
+  trashTooltip: "Disteurel an holl vloc'hoù.",
10
+  catLogic: "Poell",
11
+  catLoops: "Boukloù",
12
+  catMath: "Matematik",
13
+  catText: "Testenn",
14
+  catLists: "Rolloù",
15
+  catColour: "Liv",
16
+  catVariables: "Argemmennoù",
17
+  catFunctions: "Arc'hwelioù",
18
+  listVariable: "roll",
19
+  textVariable: "testenn",
20
+  httpRequestError: "Ur gudenn zo gant ar reked.",
21
+  linkAlert: "Rannañ ho ploc'hoù gant al liamm-mañ :\n\n%1",
22
+  hashError: "Digarezit. \"%1\" ne glot gant programm enrollet ebet.",
23
+  xmlError: "Ne c'haller ket kargañ ho restr enrollet. Marteze e oa bet krouet gant ur stumm disheñvel eus Blockly ?",
24
+  badXml: "Fazi dielfennañ XML :\n%1\n\nDibabit \"Mat eo\" evit dilezel ar c'hemmoù-se pe \"Nullañ\" evit kemmañ an XML c'hoazh."
25
+};

+ 0 - 0
src/blockly/demos/code/msg/ca.js


Some files were not shown because too many files changed in this diff