Home | History | Annotate | Download | only in extensions
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 // Custom binding for the Bluetooth API.
      6 
      7 var binding = require('binding').Binding.create('bluetooth');
      8 
      9 var chrome = requireNative('chrome').GetChrome();
     10 var Event = require('event_bindings').Event;
     11 var lastError = require('lastError');
     12 var sendRequest = require('sendRequest').sendRequest;
     13 
     14 // Use custom binding to create an undocumented event listener that will
     15 // receive events about device discovery and call the event listener that was
     16 // provided with the request to begin discovery.
     17 binding.registerCustomHook(function(api) {
     18   var apiFunctions = api.apiFunctions;
     19 
     20   var bluetooth = {};
     21 
     22   function callCallbackIfPresent(name, args, error) {
     23     var callback = args[args.length - 1];
     24     if (typeof(callback) == 'function')
     25       lastError.run(name, error, callback);
     26   }
     27 
     28   bluetooth.deviceDiscoveredHandler = null;
     29   bluetooth.onDeviceDiscovered = new Event('bluetooth.onDeviceDiscovered');
     30   function clearDeviceDiscoveredHandler() {
     31     bluetooth.onDeviceDiscovered.removeListener(
     32         bluetooth.deviceDiscoveredHandler);
     33     bluetooth.deviceDiscoveredHandler = null;
     34   }
     35   apiFunctions.setHandleRequest('startDiscovery',
     36       function() {
     37         var args = arguments;
     38         if (args.length > 0 && args[0] && args[0].deviceCallback) {
     39           if (bluetooth.deviceDiscoveredHandler != null) {
     40             callCallbackIfPresent('bluetooth.startDiscovery',
     41                                   args,
     42                                   'Concurrent discovery is not allowed.');
     43             return;
     44           }
     45 
     46           bluetooth.deviceDiscoveredHandler = args[0].deviceCallback;
     47           bluetooth.onDeviceDiscovered.addListener(
     48               bluetooth.deviceDiscoveredHandler);
     49           sendRequest(this.name,
     50                       args,
     51                       this.definition.parameters,
     52                       {customCallback:this.customCallback});
     53         } else {
     54           callCallbackIfPresent(
     55             'bluetooth.startDiscovery',
     56             args,
     57             'deviceCallback is required in the options object');
     58           return;
     59         }
     60       });
     61   apiFunctions.setCustomCallback('startDiscovery',
     62       function(name, request, response) {
     63         if (chrome.runtime.lastError) {
     64           clearDeviceDiscoveredHandler();
     65           return;
     66         }
     67       });
     68   apiFunctions.setHandleRequest('stopDiscovery',
     69       function() {
     70         clearDeviceDiscoveredHandler();
     71         sendRequest(this.name, arguments, this.definition.parameters);
     72       });
     73 
     74   // An object to hold state during one call to getDevices.
     75   bluetooth.getDevicesState = null;
     76 
     77   // Hidden events used to deliver getDevices data to the client callbacks
     78   bluetooth.onDeviceSearchResult = new Event('bluetooth.onDeviceSearchResult');
     79   bluetooth.onDeviceSearchFinished =
     80       new Event('bluetooth.onDeviceSearchFinished');
     81 
     82   function deviceSearchResultHandler(device) {
     83     bluetooth.getDevicesState.actualEvents++;
     84     bluetooth.getDevicesState.deviceCallback(device);
     85     maybeFinishDeviceSearch();
     86   }
     87 
     88   function deviceSearchFinishedHandler(info) {
     89     bluetooth.getDevicesState.expectedEventCount = info.expectedEventCount;
     90     maybeFinishDeviceSearch();
     91   }
     92 
     93   function addDeviceSearchListeners() {
     94     bluetooth.onDeviceSearchResult.addListener(deviceSearchResultHandler);
     95     bluetooth.onDeviceSearchFinished.addListener(deviceSearchFinishedHandler);
     96   }
     97 
     98   function removeDeviceSearchListeners() {
     99     bluetooth.onDeviceSearchResult.removeListener(deviceSearchResultHandler);
    100     bluetooth.onDeviceSearchFinished.removeListener(
    101         deviceSearchFinishedHandler);
    102   }
    103 
    104   function maybeFinishDeviceSearch() {
    105     var state = bluetooth.getDevicesState;
    106     if (typeof(state.expectedEventCount) != 'undefined' &&
    107         state.actualEvents >= state.expectedEventCount) {
    108       finishDeviceSearch();
    109     }
    110   }
    111 
    112   function finishDeviceSearch() {
    113     var finalCallback = bluetooth.getDevicesState.finalCallback;
    114     removeDeviceSearchListeners();
    115     bluetooth.getDevicesState = null;
    116 
    117     if (finalCallback) {
    118       finalCallback();
    119     }
    120   }
    121 
    122   apiFunctions.setUpdateArgumentsPostValidate('getDevices',
    123       function() {
    124         var args = $Array.slice(arguments);
    125 
    126         if (bluetooth.getDevicesState != null) {
    127           throw new Error('Concurrent calls to getDevices are not allowed.');
    128         }
    129 
    130         var state = { actualEvents: 0 };
    131 
    132         if (typeof(args[args.length - 1]) == 'function') {
    133           state.finalCallback = args.pop();
    134           $Array.push(args,
    135               function() {
    136                 if (chrome.runtime.lastError) {
    137                   finishDeviceSearch();
    138                 }
    139               });
    140         } else {
    141           throw new Error('getDevices must have a final callback parameter.');
    142         }
    143 
    144         if (typeof(args[0].deviceCallback) == 'function') {
    145           state.deviceCallback = args[0].deviceCallback;
    146         } else {
    147           throw new Error('getDevices must be passed options with a ' +
    148               'deviceCallback.');
    149         }
    150 
    151         bluetooth.getDevicesState = state;
    152         addDeviceSearchListeners();
    153 
    154         return args;
    155       });
    156 });
    157 
    158 exports.binding = binding.generate();
    159