diff options
author | rafaelw@chromium.org <rafaelw@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-07-22 20:12:07 +0000 |
---|---|---|
committer | rafaelw@chromium.org <rafaelw@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-07-22 20:12:07 +0000 |
commit | 6e3f6928a0f43c3e956903693f8a84a2a207e970 (patch) | |
tree | c3f57be6c06a75ff02fac9ed014231fd8bab1f9c /chrome/renderer | |
parent | 2248d729d57ca026d78e88355f760bb4048f8430 (diff) | |
download | chromium_src-6e3f6928a0f43c3e956903693f8a84a2a207e970.zip chromium_src-6e3f6928a0f43c3e956903693f8a84a2a207e970.tar.gz chromium_src-6e3f6928a0f43c3e956903693f8a84a2a207e970.tar.bz2 |
Step 2 in abstracting Extension APIs to shared "IDL".
In this step, all "normal" async extension apis have been pulled out into extension_api.json which is loaded in the renderer, and dynamic bindings are created for extension functions & events in extension_process_bindings.js.
R=aa
Review URL: http://codereview.chromium.org/155741
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@21313 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'chrome/renderer')
-rw-r--r-- | chrome/renderer/extensions/extension_process_bindings.cc | 10 | ||||
-rw-r--r-- | chrome/renderer/renderer_resources.grd | 2 | ||||
-rw-r--r-- | chrome/renderer/resources/extension_process_bindings.js | 489 | ||||
-rw-r--r-- | chrome/renderer/resources/json_schema.js | 35 |
4 files changed, 91 insertions, 445 deletions
diff --git a/chrome/renderer/extensions/extension_process_bindings.cc b/chrome/renderer/extensions/extension_process_bindings.cc index 1e4bafe1..fd92481 100644 --- a/chrome/renderer/extensions/extension_process_bindings.cc +++ b/chrome/renderer/extensions/extension_process_bindings.cc @@ -12,6 +12,7 @@ #include "chrome/renderer/extensions/renderer_extension_bindings.h" #include "chrome/renderer/js_only_v8_extensions.h" #include "chrome/renderer/render_view.h" +#include "grit/common_resources.h" #include "grit/renderer_resources.h" #include "webkit/glue/webframe.h" @@ -64,7 +65,9 @@ class ExtensionImpl : public ExtensionBase { virtual v8::Handle<v8::FunctionTemplate> GetNativeFunction( v8::Handle<v8::String> name) { - if (name->Equals(v8::String::New("GetViews"))) { + if (name->Equals(v8::String::New("GetExtensionAPIDefinition"))) { + return v8::FunctionTemplate::New(GetExtensionAPIDefinition); + } else if (name->Equals(v8::String::New("GetViews"))) { return v8::FunctionTemplate::New(GetViews); } else if (name->Equals(v8::String::New("GetNextRequestId"))) { return v8::FunctionTemplate::New(GetNextRequestId); @@ -87,6 +90,11 @@ class ExtensionImpl : public ExtensionBase { return url.host(); } + static v8::Handle<v8::Value> GetExtensionAPIDefinition( + const v8::Arguments& args) { + return v8::String::New(GetStringResource<IDR_EXTENSION_API_JSON>()); + } + static v8::Handle<v8::Value> GetViews(const v8::Arguments& args) { std::string extension_id = ExtensionIdFromCurrentContext(); diff --git a/chrome/renderer/renderer_resources.grd b/chrome/renderer/renderer_resources.grd index ac2e6de..5ec2309 100644 --- a/chrome/renderer/renderer_resources.grd +++ b/chrome/renderer/renderer_resources.grd @@ -1,6 +1,6 @@ <?xml version="1.0" encoding="UTF-8"?> <!-- This comment is only here because changes to resources are not picked up -without changes to the corresponding grd file. mp2 --> +without changes to the corresponding grd file. rw --> <grit latest_public_release="0" current_release="1"> <outputs> <output filename="grit/renderer_resources.h" type="rc_header"> diff --git a/chrome/renderer/resources/extension_process_bindings.js b/chrome/renderer/resources/extension_process_bindings.js index e702744..dd01214 100644 --- a/chrome/renderer/resources/extension_process_bindings.js +++ b/chrome/renderer/resources/extension_process_bindings.js @@ -12,6 +12,7 @@ var chrome = chrome || {}; (function() { + native function GetExtensionAPIDefinition(); native function StartRequest(); native function GetCurrentPageActions(); native function GetViews(); @@ -93,298 +94,71 @@ var chrome = chrome || {}; } StartRequest(functionName, sargs, requestId, hasCallback); } - - //---------------------------------------------------------------------------- - - // Windows. - chrome.windows = {}; - - chrome.windows.get = function(windowId, callback) { - validate(arguments, arguments.callee.params); - sendRequest("windows.get", windowId, callback); - }; - - chrome.windows.get.params = [ - chrome.types.pInt, - chrome.types.fun - ]; - - chrome.windows.getCurrent = function(callback) { - validate(arguments, arguments.callee.params); - sendRequest("windows.getCurrent", null, callback); - }; - - chrome.windows.getCurrent.params = [ - chrome.types.fun - ]; - - chrome.windows.getLastFocused = function(callback) { - validate(arguments, arguments.callee.params); - sendRequest("windows.getLastFocused", null, callback); - }; - - chrome.windows.getLastFocused.params = [ - chrome.types.fun - ]; - - chrome.windows.getAll = function(populate, callback) { - validate(arguments, arguments.callee.params); - sendRequest("windows.getAll", populate, callback); - }; - - chrome.windows.getAll.params = [ - chrome.types.optBool, - chrome.types.fun - ]; - - chrome.windows.create = function(createData, callback) { - validate(arguments, arguments.callee.params); - sendRequest("windows.create", createData, callback); - }; - chrome.windows.create.params = [ - { - type: "object", - properties: { - url: chrome.types.optStr, - left: chrome.types.optInt, - top: chrome.types.optInt, - width: chrome.types.optPInt, - height: chrome.types.optPInt - }, - optional: true - }, - chrome.types.optFun - ]; - - chrome.windows.update = function(windowId, updateData, callback) { - validate(arguments, arguments.callee.params); - sendRequest("windows.update", [windowId, updateData], callback); - }; - chrome.windows.update.params = [ - chrome.types.pInt, - { - type: "object", - properties: { - left: chrome.types.optInt, - top: chrome.types.optInt, - width: chrome.types.optPInt, - height: chrome.types.optPInt - }, - }, - chrome.types.optFun - ]; - - chrome.windows.remove = function(windowId, callback) { - validate(arguments, arguments.callee.params); - sendRequest("windows.remove", windowId, callback); - }; - - chrome.windows.remove.params = [ - chrome.types.pInt, - chrome.types.optFun - ]; - - // sends (windowId). - // *WILL* be followed by tab-attached AND then tab-selection-changed. - chrome.windows.onCreated = new chrome.Event("windows.onCreated"); - - // sends (windowId). - // *WILL* be preceded by sequences of tab-removed AND then - // tab-selection-changed -- one for each tab that was contained in the window - // that closed - chrome.windows.onRemoved = new chrome.Event("windows.onRemoved"); - - // sends (windowId). - chrome.windows.onFocusChanged = - new chrome.Event("windows.onFocusChanged"); - - //---------------------------------------------------------------------------- - - // Tabs - chrome.tabs = {}; - - chrome.tabs.get = function(tabId, callback) { - validate(arguments, arguments.callee.params); - sendRequest("tabs.get", tabId, callback); - }; - - chrome.tabs.get.params = [ - chrome.types.pInt, - chrome.types.fun - ]; - - chrome.tabs.getSelected = function(windowId, callback) { - validate(arguments, arguments.callee.params); - sendRequest("tabs.getSelected", windowId, callback); - }; - - chrome.tabs.getSelected.params = [ - chrome.types.optPInt, - chrome.types.fun - ]; - - chrome.tabs.getAllInWindow = function(windowId, callback) { - validate(arguments, arguments.callee.params); - sendRequest("tabs.getAllInWindow", windowId, callback); - }; - - chrome.tabs.getAllInWindow.params = [ - chrome.types.optPInt, - chrome.types.fun - ]; - - chrome.tabs.create = function(tab, callback) { - validate(arguments, arguments.callee.params); - sendRequest("tabs.create", tab, callback); - }; - - chrome.tabs.create.params = [ - { - type: "object", - properties: { - windowId: chrome.types.optPInt, - index: chrome.types.optPInt, - url: chrome.types.optStr, - selected: chrome.types.optBool - } - }, - chrome.types.optFun - ]; - - chrome.tabs.update = function(tabId, updates, callback) { - validate(arguments, arguments.callee.params); - sendRequest("tabs.update", [tabId, updates], callback); - }; - - chrome.tabs.update.params = [ - chrome.types.pInt, - { - type: "object", - properties: { - url: chrome.types.optStr, - selected: chrome.types.optBool - } - }, - chrome.types.optFun - ]; - - chrome.tabs.move = function(tabId, moveProps, callback) { - validate(arguments, arguments.callee.params); - sendRequest("tabs.move", [tabId, moveProps], callback); - }; - - chrome.tabs.move.params = [ - chrome.types.pInt, - { - type: "object", - properties: { - windowId: chrome.types.optPInt, - index: chrome.types.pInt - } - }, - chrome.types.optFun - ]; - - chrome.tabs.remove = function(tabId, callback) { - validate(arguments, arguments.callee.params); - sendRequest("tabs.remove", tabId, callback); - }; - - chrome.tabs.remove.params = [ - chrome.types.pInt, - chrome.types.optFun - ]; - - chrome.tabs.detectLanguage = function(tabId, callback) { - validate(arguments, arguments.callee.params); - sendRequest("tabs.detectLanguage", tabId, callback); - }; - - chrome.tabs.detectLanguage.params = [ - chrome.types.optPInt, - chrome.types.optFun - ]; - - chrome.tabs.connect = function(tabId, opt_name) { - validate(arguments, arguments.callee.params); - var portId = OpenChannelToTab(tabId, chrome.extension.id_, opt_name || ""); - return chromeHidden.Port.createPort(portId, opt_name); - }; - - chrome.tabs.connect.params = [ - chrome.types.pInt, - chrome.types.optStr - ]; - - // Sends ({Tab}). - // Will *NOT* be followed by tab-attached - it is implied. - // *MAY* be followed by tab-selection-changed. - chrome.tabs.onCreated = new chrome.Event("tabs.onCreated"); - - // Sends (tabId, {ChangedProps}). - chrome.tabs.onUpdated = new chrome.Event("tabs.onUpdated"); - - // Sends (tabId, {windowId, fromIndex, toIndex}). - // Tabs can only "move" within a window. - chrome.tabs.onMoved = new chrome.Event("tabs.onMoved"); - - // Sends (tabId, {windowId}). - chrome.tabs.onSelectionChanged = - new chrome.Event("tabs.onSelectionChanged"); - - // Sends (tabId, {newWindowId, newPosition}). - // *MAY* be followed by tab-selection-changed. - chrome.tabs.onAttached = new chrome.Event("tabs.onAttached"); - - // Sends (tabId, {oldWindowId, oldPosition}). - // *WILL* be followed by tab-selection-changed. - chrome.tabs.onDetached = new chrome.Event("tabs.onDetached"); - - // Sends (tabId). - // *WILL* be followed by tab-selection-changed. - // Will *NOT* be followed or preceded by tab-detached. - chrome.tabs.onRemoved = new chrome.Event("tabs.onRemoved"); - - //---------------------------------------------------------------------------- - - // PageActions. - chrome.pageActions = {}; - - chrome.pageActions.enableForTab = function(pageActionId, action) { - validate(arguments, arguments.callee.params); - sendRequest("pageActions.enableForTab", [pageActionId, action]); - } - - chrome.pageActions.enableForTab.params = [ - chrome.types.str, - { - type: "object", - properties: { - tabId: chrome.types.pInt, - url: chrome.types.str, - title: chrome.types.optStr, - iconId: chrome.types.optPInt, - }, - optional: false + + // Read api definitions and setup api functions in the chrome namespace. + // TODO(rafaelw): Consider defining a json schema for an api definition + // and validating either here, in a unit_test or both. + // TODO(rafaelw): Handle synchronous functions. + // TOOD(rafaelw): Consider providing some convenient override points + // for api functions that wish to insert themselves into the call. + var apiDefinitions = JSON.parse(GetExtensionAPIDefinition()); + + // Using forEach for convenience, and to bind |module|s & |apiDefs|s via + // closures. + function forEach(a, f) { + for (var i = 0; i < a.length; i++) { + f(a[i], i); } - ]; - - chrome.pageActions.disableForTab = function(pageActionId, action) { - validate(arguments, arguments.callee.params); - sendRequest("pageActions.disableForTab", [pageActionId, action]); } - chrome.pageActions.disableForTab.params = [ - chrome.types.str, - { - type: "object", - properties: { - tabId: chrome.types.pInt, - url: chrome.types.str - }, - optional: false - } - ]; + forEach(apiDefinitions, function(apiDef) { + var module = {}; + chrome[apiDef.namespace] = module; + + // Setup Functions. + forEach(apiDef.functions, function(functionDef) { + var paramSchemas = functionDef.parameters; + + module[functionDef.name] = function() { + validate(arguments, paramSchemas); + + var functionName = apiDef.namespace + "." + functionDef.name; + var args = null; + var callback = null; + var argCount = arguments.length; + + // Look for callback param. + if (paramSchemas.length > 0 && + arguments.length == paramSchemas.length && + paramSchemas[paramSchemas.length - 1].type == "function") { + callback = arguments[paramSchemas.length - 1]; + --argCount; + } + + // Calls with one argument expect singular argument. Calls with multiple + // expect a list. + if (argCount == 1) + args = arguments[0]; + if (argCount > 1) { + args = []; + for (var k = 0; k < argCount; k++) { + args[k] = arguments[k]; + } + } + + // Make the request. + sendRequest(functionName, args, callback); + } + }); + + // Setup Events + forEach(apiDef.events, function(eventDef) { + var eventName = apiDef.namespace + "." + eventDef.name; + module[eventDef.name] = new chrome.Event(eventName); + }); + }); + + // --- Setup additional api's not currently handled in common/extensions/api // Page action events send (pageActionId, {tabId, tabUrl}). function setupPageActionEvents(extensionId) { @@ -397,59 +171,20 @@ var chrome = chrome || {}; } } - //---------------------------------------------------------------------------- - // Bookmarks - chrome.bookmarks = {}; - - chrome.bookmarks.get = function(ids, callback) { - validate(arguments, arguments.callee.params); - sendRequest("bookmarks.get", ids, callback); - }; - - chrome.bookmarks.get.params = [ - chrome.types.singleOrListOf(chrome.types.pInt), - chrome.types.fun - ]; - - chrome.bookmarks.getChildren = function(id, callback) { - validate(arguments, arguments.callee.params); - sendRequest("bookmarks.getChildren", id, callback); - }; - - chrome.bookmarks.getChildren.params = [ - chrome.types.pInt, - chrome.types.fun - ]; - - chrome.bookmarks.getTree = function(callback) { - validate(arguments, arguments.callee.params); - sendRequest("bookmarks.getTree", null, callback); - }; - - // TODO(erikkay): allow it to take an optional id as a starting point - // BUG=13727 - chrome.bookmarks.getTree.params = [ - chrome.types.fun - ]; - - chrome.bookmarks.search = function(query, callback) { - validate(arguments, arguments.callee.params); - sendRequest("bookmarks.search", query, callback); - }; - - chrome.bookmarks.search.params = [ - chrome.types.str, - chrome.types.fun - ]; - + // Bookmarks custom parameter handling. chrome.bookmarks.remove = function(id, callback) { validate(arguments, arguments.callee.params); sendRequest("bookmarks.remove", [id, false], callback); }; chrome.bookmarks.remove.params = [ - chrome.types.singleOrListOf(chrome.types.pInt), - chrome.types.optFun + { + choice : [ + {type: "integer", minimum: 0}, + {type: "array", item: {type: "integer", minimum: 0}, minItems: 1} + ] + }, + {type: "function", optional: true} ]; chrome.bookmarks.removeTree = function(id, callback) { @@ -458,85 +193,23 @@ var chrome = chrome || {}; }; chrome.bookmarks.removeTree.params = [ - chrome.types.pInt, - chrome.types.optFun + {type: "integer", minimum: 0}, + {type: "function", optional: true} ]; - chrome.bookmarks.create = function(bookmark, callback) { - validate(arguments, arguments.callee.params); - sendRequest("bookmarks.create", bookmark, callback); - }; - - chrome.bookmarks.create.params = [ - { - type: "object", - properties: { - parentId: chrome.types.optPInt, - index: chrome.types.optPInt, - title: chrome.types.optStr, - url: chrome.types.optStr, - } - }, - chrome.types.optFun - ]; - - chrome.bookmarks.move = function(id, destination, callback) { - validate(arguments, arguments.callee.params); - sendRequest("bookmarks.move", [id, destination], callback); - }; - - chrome.bookmarks.move.params = [ - chrome.types.pInt, - { - type: "object", - properties: { - parentId: chrome.types.optPInt, - index: chrome.types.optPInt - } - }, - chrome.types.optFun - ]; - - chrome.bookmarks.update = function(id, changes, callback) { + // Tabs connect() + chrome.tabs.connect = function(tabId, opt_name) { validate(arguments, arguments.callee.params); - sendRequest("bookmarks.update", [id, changes], callback); + var portId = OpenChannelToTab(tabId, chrome.extension.id_, opt_name || ""); + return chromeHidden.Port.createPort(portId, opt_name); }; - chrome.bookmarks.update.params = [ - chrome.types.pInt, - { - type: "object", - properties: { - title: chrome.types.optStr - } - }, - chrome.types.optFun + chrome.tabs.connect.params = [ + {type: "integer", optional: true, minimum: 0}, + {type: "string", optional: true} ]; - // bookmark events - - // Sends (id, {title, url, parentId, index, dateAdded, dateGroupModified}) - // date values are milliseconds since the UTC epoch. - chrome.bookmarks.onAdded = new chrome.Event("bookmarks.onAdded"); - - // Sends (id, {parentId, index}) - chrome.bookmarks.onRemoved = new chrome.Event("bookmarks.onRemoved"); - - // Sends (id, object) where object has list of properties that have changed. - // Currently, this only ever includes 'title'. - chrome.bookmarks.onChanged = new chrome.Event("bookmarks.onChanged"); - - // Sends (id, {parentId, index, oldParentId, oldIndex}) - chrome.bookmarks.onMoved = new chrome.Event("bookmarks.onMoved"); - - // Sends (id, [childrenIds]) - chrome.bookmarks.onChildrenReordered = - new chrome.Event("bookmarks.onChildrenReordered"); - - - //---------------------------------------------------------------------------- - - // Self. + // chrome.self / chrome.extension. chrome.self = chrome.self || {}; chromeHidden.onLoad.addListener(function (extensionId) { diff --git a/chrome/renderer/resources/json_schema.js b/chrome/renderer/resources/json_schema.js index 4973990..f5d06c6 100644 --- a/chrome/renderer/resources/json_schema.js +++ b/chrome/renderer/resources/json_schema.js @@ -348,38 +348,3 @@ chrome.JSONSchemaValidator.prototype.addError = function(path, key, message: chrome.JSONSchemaValidator.formatError(key, replacements) }); }; - -// Set up chrome.types with some commonly used types... -(function() { - function extend(base, ext) { - var result = {}; - for (var p in base) - result[p] = base[p]; - for (var p in ext) - result[p] = ext[p]; - return result; - } - - var types = {}; - types.opt = {optional: true}; - types.bool = {type: "boolean"}; - types.int = {type: "integer"}; - types.str = {type: "string"}; - types.fun = {type: "function"}; - types.pInt = extend(types.int, {minimum: 0}); - types.optBool = extend(types.bool, types.opt); - types.optInt = extend(types.int, types.opt); - types.optStr = extend(types.str, types.opt); - types.optFun = extend(types.fun, types.opt); - types.optPInt = extend(types.pInt, types.opt); - types.singleOrListOf = function(type) { - return { - choice: [ - type, - { type: "array", item: type, minItems: 1 } - ] - }; - }; - - chrome.types = types; -})(); |