Home | History | Annotate | Download | only in options
      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 if (!loadTimeData.getBoolean('newContentSettings')) {
      6 
      7 cr.define('options', function() {
      8   /** @const */ var OptionsPage = options.OptionsPage;
      9 
     10   //////////////////////////////////////////////////////////////////////////////
     11   // ContentSettings class:
     12 
     13   /**
     14    * Encapsulated handling of content settings page.
     15    * @constructor
     16    */
     17   function ContentSettings() {
     18     this.activeNavTab = null;
     19     OptionsPage.call(this, 'content',
     20                      loadTimeData.getString('contentSettingsPageTabTitle'),
     21                      'content-settings-page');
     22   }
     23 
     24   cr.addSingletonGetter(ContentSettings);
     25 
     26   ContentSettings.prototype = {
     27     __proto__: OptionsPage.prototype,
     28 
     29     initializePage: function() {
     30       OptionsPage.prototype.initializePage.call(this);
     31 
     32       var exceptionsButtons =
     33           this.pageDiv.querySelectorAll('.exceptions-list-button');
     34       for (var i = 0; i < exceptionsButtons.length; i++) {
     35         exceptionsButtons[i].onclick = function(event) {
     36           var page = ContentSettingsExceptionsArea.getInstance();
     37 
     38           // Add on the proper hash for the content type, and store that in the
     39           // history so back/forward and tab restore works.
     40           var hash = event.currentTarget.getAttribute('contentType');
     41           var url = page.name + '#' + hash;
     42           window.history.pushState({pageName: page.name},
     43                                     page.title,
     44                                     '/' + url);
     45 
     46           // Navigate after the history has been replaced in order to have the
     47           // correct hash loaded.
     48           OptionsPage.showPageByName('contentExceptions', false);
     49 
     50           uber.invokeMethodOnParent('setPath', {path: url});
     51           uber.invokeMethodOnParent('setTitle',
     52               {title: loadTimeData.getString(hash + 'TabTitle')});
     53         };
     54       }
     55 
     56       var manageHandlersButton = $('manage-handlers-button');
     57       if (manageHandlersButton) {
     58         manageHandlersButton.onclick = function(event) {
     59           OptionsPage.navigateToPage('handlers');
     60         };
     61       }
     62 
     63       $('manage-galleries-button').onclick = function(event) {
     64         OptionsPage.navigateToPage('manageGalleries');
     65       };
     66 
     67       if (cr.isChromeOS)
     68         UIAccountTweaks.applyGuestModeVisibility(document);
     69 
     70       // Cookies filter page ---------------------------------------------------
     71       $('show-cookies-button').onclick = function(event) {
     72         chrome.send('coreOptionsUserMetricsAction', ['Options_ShowCookies']);
     73         OptionsPage.navigateToPage('cookies');
     74       };
     75 
     76       $('content-settings-overlay-confirm').onclick =
     77           OptionsPage.closeOverlay.bind(OptionsPage);
     78 
     79       $('media-pepper-flash-default').hidden = true;
     80       $('media-pepper-flash-exceptions').hidden = true;
     81 
     82       $('media-select-mic').addEventListener('change',
     83           ContentSettings.setDefaultMicrophone_);
     84       $('media-select-camera').addEventListener('change',
     85           ContentSettings.setDefaultCamera_);
     86     },
     87   };
     88 
     89   ContentSettings.updateHandlersEnabledRadios = function(enabled) {
     90     var selector = '#content-settings-page input[type=radio][value=' +
     91         (enabled ? 'allow' : 'block') + '].handler-radio';
     92     document.querySelector(selector).checked = true;
     93   };
     94 
     95   /**
     96    * Sets the values for all the content settings radios.
     97    * @param {Object} dict A mapping from radio groups to the checked value for
     98    *     that group.
     99    */
    100   ContentSettings.setContentFilterSettingsValue = function(dict) {
    101     for (var group in dict) {
    102       var managedBy = dict[group].managedBy;
    103       var controlledBy = managedBy == 'policy' || managedBy == 'extension' ?
    104           managedBy : null;
    105       document.querySelector('input[type=radio][name=' + group + '][value=' +
    106                              dict[group].value + ']').checked = true;
    107       var radios = document.querySelectorAll('input[type=radio][name=' +
    108                                              group + ']');
    109       for (var i = 0, len = radios.length; i < len; i++) {
    110         radios[i].disabled = (managedBy != 'default');
    111         radios[i].controlledBy = controlledBy;
    112       }
    113       var indicators = document.querySelectorAll(
    114           'span.controlled-setting-indicator[content-setting=' + group + ']');
    115       if (indicators.length == 0)
    116         continue;
    117       // Create a synthetic pref change event decorated as
    118       // CoreOptionsHandler::CreateValueForPref() does.
    119       var event = new Event(group);
    120       event.value = {
    121         value: dict[group].value,
    122         controlledBy: controlledBy,
    123       };
    124       for (var i = 0; i < indicators.length; i++)
    125         indicators[i].handlePrefChange(event);
    126     }
    127   };
    128 
    129   /**
    130    * Updates the labels and indicators for the Media settings. Those require
    131    * special handling because they are backed by multiple prefs and can change
    132    * their scope based on the managed state of the backing prefs.
    133    * @param {Object} mediaSettings A dictionary containing the following fields:
    134    *     {String} askText The label for the ask radio button.
    135    *     {String} blockText The label for the block radio button.
    136    *     {Boolean} cameraDisabled Whether to disable the camera dropdown.
    137    *     {Boolean} micDisabled Whether to disable the microphone dropdown.
    138    *     {Boolean} showBubble Wether to show the managed icon and bubble for the
    139    *         media label.
    140    *     {String} bubbleText The text to use inside the bubble if it is shown.
    141    */
    142   ContentSettings.updateMediaUI = function(mediaSettings) {
    143     $('media-stream-ask-label').innerHTML =
    144         loadTimeData.getString(mediaSettings.askText);
    145     $('media-stream-block-label').innerHTML =
    146         loadTimeData.getString(mediaSettings.blockText);
    147 
    148     if (mediaSettings.micDisabled)
    149       $('media-select-mic').disabled = true;
    150     if (mediaSettings.cameraDisabled)
    151       $('media-select-camera').disabled = true;
    152 
    153     OptionsPage.hideBubble();
    154     // Create a synthetic pref change event decorated as
    155     // CoreOptionsHandler::CreateValueForPref() does.
    156     // TODO(arv): It was not clear what event type this should use?
    157     var event = new Event('undefined');
    158     event.value = {};
    159 
    160     if (mediaSettings.showBubble) {
    161       event.value = { controlledBy: 'policy' };
    162       $('media-indicator').setAttribute(
    163           'textpolicy', loadTimeData.getString(mediaSettings.bubbleText));
    164       $('media-indicator').location = cr.ui.ArrowLocation.TOP_START;
    165     }
    166 
    167     $('media-indicator').handlePrefChange(event);
    168   };
    169 
    170   /**
    171    * Initializes an exceptions list.
    172    * @param {string} type The content type that we are setting exceptions for.
    173    * @param {Array} list An array of pairs, where the first element of each pair
    174    *     is the filter string, and the second is the setting (allow/block).
    175    */
    176   ContentSettings.setExceptions = function(type, list) {
    177     var exceptionsList =
    178         document.querySelector('div[contentType=' + type + ']' +
    179                                ' list[mode=normal]');
    180     exceptionsList.setExceptions(list);
    181   };
    182 
    183   ContentSettings.setHandlers = function(list) {
    184     $('handlers-list').setHandlers(list);
    185   };
    186 
    187   ContentSettings.setIgnoredHandlers = function(list) {
    188     $('ignored-handlers-list').setHandlers(list);
    189   };
    190 
    191   ContentSettings.setOTRExceptions = function(type, list) {
    192     var exceptionsList =
    193         document.querySelector('div[contentType=' + type + ']' +
    194                                ' list[mode=otr]');
    195 
    196     exceptionsList.parentNode.hidden = false;
    197     exceptionsList.setExceptions(list);
    198   };
    199 
    200   /**
    201    * The browser's response to a request to check the validity of a given URL
    202    * pattern.
    203    * @param {string} type The content type.
    204    * @param {string} mode The browser mode.
    205    * @param {string} pattern The pattern.
    206    * @param {bool} valid Whether said pattern is valid in the context of
    207    *     a content exception setting.
    208    */
    209   ContentSettings.patternValidityCheckComplete =
    210       function(type, mode, pattern, valid) {
    211     var exceptionsList =
    212         document.querySelector('div[contentType=' + type + '] ' +
    213                                'list[mode=' + mode + ']');
    214     exceptionsList.patternValidityCheckComplete(pattern, valid);
    215   };
    216 
    217   /**
    218    * Shows/hides the link to the Pepper Flash camera and microphone default
    219    * settings.
    220    * Please note that whether the link is actually showed or not is also
    221    * affected by the style class pepper-flash-settings.
    222    */
    223   ContentSettings.showMediaPepperFlashDefaultLink = function(show) {
    224     $('media-pepper-flash-default').hidden = !show;
    225   }
    226 
    227   /**
    228    * Shows/hides the link to the Pepper Flash camera and microphone
    229    * site-specific settings.
    230    * Please note that whether the link is actually showed or not is also
    231    * affected by the style class pepper-flash-settings.
    232    */
    233   ContentSettings.showMediaPepperFlashExceptionsLink = function(show) {
    234     $('media-pepper-flash-exceptions').hidden = !show;
    235   }
    236 
    237   /**
    238    * Shows/hides the whole Web MIDI settings.
    239    * @param {bool} show Wether to show the whole Web MIDI settings.
    240    */
    241   ContentSettings.showExperimentalWebMIDISettings = function(show) {
    242     $('experimental-web-midi-settings').hidden = !show;
    243   }
    244 
    245   /**
    246    * Updates the microphone/camera devices menu with the given entries.
    247    * @param {string} type The device type.
    248    * @param {Array} devices List of available devices.
    249    * @param {string} defaultdevice The unique id of the current default device.
    250    */
    251   ContentSettings.updateDevicesMenu = function(type, devices, defaultdevice) {
    252     var deviceSelect = '';
    253     if (type == 'mic') {
    254       deviceSelect = $('media-select-mic');
    255     } else if (type == 'camera') {
    256       deviceSelect = $('media-select-camera');
    257     } else {
    258       console.error('Unknown device type for <device select> UI element: ' +
    259                     type);
    260       return;
    261     }
    262 
    263     deviceSelect.textContent = '';
    264 
    265     var deviceCount = devices.length;
    266     var defaultIndex = -1;
    267     for (var i = 0; i < deviceCount; i++) {
    268       var device = devices[i];
    269       var option = new Option(device.name, device.id);
    270       if (option.value == defaultdevice)
    271         defaultIndex = i;
    272       deviceSelect.appendChild(option);
    273     }
    274     if (defaultIndex >= 0)
    275       deviceSelect.selectedIndex = defaultIndex;
    276   };
    277 
    278   /**
    279    * Enables/disables the protected content exceptions button.
    280    * @param {bool} enable Whether to enable the button.
    281    */
    282   ContentSettings.enableProtectedContentExceptions = function(enable) {
    283     var exceptionsButton = $('protected-content-exceptions');
    284     if (exceptionsButton) {
    285       exceptionsButton.disabled = !enable;
    286     }
    287   }
    288 
    289   /**
    290    * Set the default microphone device based on the popup selection.
    291    * @private
    292    */
    293   ContentSettings.setDefaultMicrophone_ = function() {
    294     var deviceSelect = $('media-select-mic');
    295     chrome.send('setDefaultCaptureDevice', ['mic', deviceSelect.value]);
    296   };
    297 
    298   /**
    299    * Set the default camera device based on the popup selection.
    300    * @private
    301    */
    302   ContentSettings.setDefaultCamera_ = function() {
    303     var deviceSelect = $('media-select-camera');
    304     chrome.send('setDefaultCaptureDevice', ['camera', deviceSelect.value]);
    305   };
    306 
    307   // Export
    308   return {
    309     ContentSettings: ContentSettings
    310   };
    311 
    312 });
    313 
    314 }
    315