492 lines
16 KiB
Plaintext
492 lines
16 KiB
Plaintext
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
*
|
|
* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is the Mozilla browser.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Netscape Communications Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2002
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* William Dell Wisner <william@dell.wisner.name>
|
|
* Josh Aas <josh@mozilla.com>
|
|
* Stuart Morgan <stuart.morgan@alumni.case.edu>
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#import "WebFeatures.h"
|
|
|
|
#import "NSString+Utils.h"
|
|
#import "CHPermissionManager.h"
|
|
#import "ExtendedTableView.h"
|
|
|
|
// need to match the strings in PreferenceManager.mm
|
|
static NSString* const AdBlockingChangedNotificationName = @"AdBlockingChanged";
|
|
static NSString* const kFlashBlockChangedNotificationName = @"FlashBlockChanged";
|
|
|
|
// for accessibility.tabfocus
|
|
const int kFocusTextFields = (1 << 0);
|
|
const int kFocusForms = (1 << 1);
|
|
const int kFocusLinks = (1 << 2);
|
|
|
|
// for annoyance blocker prefs
|
|
const int kAnnoyancePrefNone = 1;
|
|
const int kAnnoyancePrefAll = 2;
|
|
const int kAnnoyancePrefSome = 3;
|
|
|
|
@interface OrgMozillaCaminoPreferenceWebFeatures(Private)
|
|
|
|
- (int)annoyingWindowPrefs;
|
|
- (int)popupIndexForCurrentTabFocusPref;
|
|
- (int)preventAnimationCheckboxState;
|
|
- (BOOL)isFlashBlockAllowed;
|
|
- (void)updateFlashBlock;
|
|
- (void)populatePermissionCache;
|
|
|
|
@end
|
|
|
|
@implementation OrgMozillaCaminoPreferenceWebFeatures
|
|
|
|
- (void) dealloc
|
|
{
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
|
|
[super dealloc];
|
|
}
|
|
|
|
- (void)mainViewDidLoad
|
|
{
|
|
if (!mPrefService)
|
|
return;
|
|
|
|
BOOL gotPref = NO;
|
|
|
|
// Set initial value on JavaScript checkbox.
|
|
BOOL jsEnabled = [self getBooleanPref:"javascript.enabled" withSuccess:&gotPref] && gotPref;
|
|
[mEnableJS setState:jsEnabled];
|
|
|
|
// Set initial value on Java checkbox, and disable it if plugins are off
|
|
BOOL pluginsEnabled = [self getBooleanPref:"camino.enable_plugins" withSuccess:&gotPref] || !gotPref;
|
|
[mEnableJava setEnabled:pluginsEnabled];
|
|
BOOL javaEnabled = pluginsEnabled && [self getBooleanPref:"security.enable_java" withSuccess:NULL];
|
|
[mEnableJava setState:javaEnabled];
|
|
|
|
// set initial value on popup blocking checkbox and disable the whitelist
|
|
// button if it's off
|
|
BOOL enablePopupBlocking = [self getBooleanPref:"dom.disable_open_during_load" withSuccess:&gotPref] && gotPref;
|
|
[mEnablePopupBlocking setState:enablePopupBlocking];
|
|
[mEditWhitelist setEnabled:enablePopupBlocking];
|
|
|
|
// set initial value on annoyance blocker checkbox.
|
|
if([self annoyingWindowPrefs] == kAnnoyancePrefAll)
|
|
[mEnableAnnoyanceBlocker setState:NSOnState];
|
|
else if([self annoyingWindowPrefs] == kAnnoyancePrefNone)
|
|
[mEnableAnnoyanceBlocker setState:NSOffState];
|
|
else // annoyingWindowPrefs == kAnnoyancePrefSome
|
|
[mEnableAnnoyanceBlocker setState:NSMixedState];
|
|
|
|
[mPreventAnimation setState:[self preventAnimationCheckboxState]];
|
|
|
|
BOOL enableAdBlock = [self getBooleanPref:"camino.enable_ad_blocking" withSuccess:&gotPref];
|
|
[mEnableAdBlocking setState:enableAdBlock];
|
|
|
|
// Only allow FlashBlock if dependencies are set correctly
|
|
BOOL flashBlockAllowed = [self isFlashBlockAllowed];
|
|
[mEnableFlashBlock setEnabled:flashBlockAllowed];
|
|
|
|
if (flashBlockAllowed) {
|
|
BOOL enableFlashBlock = [self getBooleanPref:"camino.enable_flashblock" withSuccess:nil];
|
|
[mEnableFlashBlock setState:(enableFlashBlock ? NSOnState : NSOffState)];
|
|
}
|
|
|
|
// Set up policy popups
|
|
NSPopUpButtonCell *popupButtonCell = [mPolicyColumn dataCell];
|
|
[popupButtonCell setEditable:YES];
|
|
[popupButtonCell addItemsWithTitles:[NSArray arrayWithObjects:[self localizedStringForKey:@"Allow"],
|
|
[self localizedStringForKey:@"Deny"],
|
|
nil]];
|
|
|
|
// Set tab focus popup.
|
|
[mTabBehaviorPopup selectItemAtIndex:[self popupIndexForCurrentTabFocusPref]];
|
|
}
|
|
|
|
|
|
//
|
|
// -clickEnableJS:
|
|
//
|
|
// Enable and disable JavaScript
|
|
//
|
|
-(IBAction) clickEnableJS:(id)sender
|
|
{
|
|
[self setPref:"javascript.enabled" toBoolean:([sender state] == NSOnState)];
|
|
|
|
// FlashBlock depends on Javascript so make sure to update the FlashBlock settings
|
|
[self updateFlashBlock];
|
|
}
|
|
|
|
//
|
|
// -clickEnableJava:
|
|
//
|
|
// Enable and disable Java
|
|
//
|
|
-(IBAction) clickEnableJava:(id)sender
|
|
{
|
|
[self setPref:"security.enable_java" toBoolean:([sender state] == NSOnState)];
|
|
}
|
|
|
|
//
|
|
// -clickEnableAdBlocking:
|
|
//
|
|
// Enable and disable ad blocking via a userContent.css file that we provide in our
|
|
// package, copied into the user's profile.
|
|
//
|
|
- (IBAction)clickEnableAdBlocking:(id)sender
|
|
{
|
|
[self setPref:"camino.enable_ad_blocking" toBoolean:([sender state] == NSOnState)];
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:AdBlockingChangedNotificationName object:nil];
|
|
}
|
|
|
|
//
|
|
// clickEnablePopupBlocking
|
|
//
|
|
// Enable and disable mozilla's popup blocking feature. We use a combination of
|
|
// two prefs to suppress bad popups.
|
|
//
|
|
- (IBAction)clickEnablePopupBlocking:(id)sender
|
|
{
|
|
[self setPref:"dom.disable_open_during_load" toBoolean:([sender state] == NSOnState)];
|
|
[mEditWhitelist setEnabled:[sender state]];
|
|
}
|
|
|
|
//
|
|
// -clickPreventAnimation:
|
|
//
|
|
// Enable and disable mozilla's limiting of how animated images repeat
|
|
//
|
|
-(IBAction) clickPreventAnimation:(id)sender
|
|
{
|
|
[sender setAllowsMixedState:NO];
|
|
[self setPref:"image.animation_mode" toString:([sender state] ? @"once" : @"normal")];
|
|
}
|
|
|
|
//
|
|
// clickEnableFlashBlock:
|
|
//
|
|
// Enable and disable FlashBlock. When enabled, an icon is displayed and the
|
|
// Flash animation plays when the user clicks it. When disabled, Flash plays automatically
|
|
//
|
|
-(IBAction) clickEnableFlashBlock:(id)sender
|
|
{
|
|
[self setPref:"camino.enable_flashblock" toBoolean:([sender state] == NSOnState)];
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:kFlashBlockChangedNotificationName object:nil];
|
|
}
|
|
|
|
//
|
|
// populatePermissionCache
|
|
//
|
|
// Builds a popup-blocking cache that we can quickly refer to later.
|
|
//
|
|
-(void) populatePermissionCache
|
|
{
|
|
if (mCachedPermissions)
|
|
[mCachedPermissions release];
|
|
mCachedPermissions = [[[CHPermissionManager permissionManager]
|
|
permissionsOfType:CHPermissionTypePopup] mutableCopy];
|
|
if (!mCachedPermissions)
|
|
mCachedPermissions = [[NSMutableArray alloc] init];
|
|
}
|
|
|
|
//
|
|
// editWhitelist:
|
|
//
|
|
// put up a sheet to allow people to edit the popup blocker whitelist
|
|
//
|
|
-(IBAction) editWhitelist:(id)sender
|
|
{
|
|
// build parallel permission list for speed with a lot of blocked sites
|
|
[self populatePermissionCache];
|
|
|
|
[NSApp beginSheet:mWhitelistPanel
|
|
modalForWindow:[mEditWhitelist window] // any old window accessor
|
|
modalDelegate:self
|
|
didEndSelector:@selector(editWhitelistSheetDidEnd:returnCode:contextInfo:)
|
|
contextInfo:NULL];
|
|
|
|
// ensure a row is selected (cocoa doesn't do this for us, but will keep
|
|
// us from unselecting a row once one is set; go figure).
|
|
if ([mWhitelistTable numberOfRows] > 0)
|
|
[mWhitelistTable selectRow:0 byExtendingSelection:NO];
|
|
|
|
[mWhitelistTable setDeleteAction:@selector(removeWhitelistSite:)];
|
|
[mWhitelistTable setTarget:self];
|
|
|
|
[mAddButton setEnabled:NO];
|
|
|
|
// we shouldn't need to do this, but the scrollbar won't enable unless we
|
|
// force the table to reload its data. Oddly it gets the number of rows correct,
|
|
// it just forgets to tell the scrollbar. *shrug*
|
|
[mWhitelistTable reloadData];
|
|
}
|
|
|
|
// whitelist sheet methods
|
|
-(IBAction) editWhitelistDone:(id)aSender
|
|
{
|
|
// save stuff??
|
|
|
|
[mWhitelistPanel orderOut:self];
|
|
[NSApp endSheet:mWhitelistPanel];
|
|
|
|
[mCachedPermissions release];
|
|
mCachedPermissions = nil;
|
|
}
|
|
|
|
-(IBAction) removeWhitelistSite:(id)aSender
|
|
{
|
|
CHPermissionManager* permManager = [CHPermissionManager permissionManager];
|
|
|
|
// Walk the selected rows backwards, removing permissions.
|
|
NSIndexSet* selectedIndexes = [mWhitelistTable selectedRowIndexes];
|
|
for (unsigned int i = [selectedIndexes lastIndex];
|
|
i != NSNotFound;
|
|
i = [selectedIndexes indexLessThanIndex:i])
|
|
{
|
|
[permManager removePermissionForHost:[[mCachedPermissions objectAtIndex:i] host]
|
|
type:CHPermissionTypePopup];
|
|
[mCachedPermissions removeObjectAtIndex:i];
|
|
}
|
|
|
|
[mWhitelistTable reloadData];
|
|
|
|
// Select the row after the last deleted row.
|
|
if ([mWhitelistTable numberOfRows] > 0) {
|
|
int rowToSelect = [selectedIndexes lastIndex] - ([selectedIndexes count] - 1);
|
|
if ((rowToSelect < 0) || (rowToSelect >= [mWhitelistTable numberOfRows]))
|
|
rowToSelect = [mWhitelistTable numberOfRows] - 1;
|
|
[mWhitelistTable selectRow:rowToSelect byExtendingSelection:NO];
|
|
}
|
|
}
|
|
|
|
//
|
|
// addWhitelistSite:
|
|
//
|
|
// adds a new site to the permission manager whitelist for popups
|
|
//
|
|
-(IBAction) addWhitelistSite:(id)sender
|
|
{
|
|
NSString* host = [[mAddField stringValue] stringByRemovingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
|
|
|
|
[[CHPermissionManager permissionManager] setPolicy:CHPermissionAllow
|
|
forHost:host
|
|
type:CHPermissionTypePopup];
|
|
//TODO: Create a new permission rather than starting from scratch.
|
|
[self populatePermissionCache];
|
|
|
|
[mAddField setStringValue:@""];
|
|
[mAddButton setEnabled:NO];
|
|
[mWhitelistTable reloadData];
|
|
}
|
|
|
|
- (void) editWhitelistSheetDidEnd:(NSWindow *)sheet returnCode:(int)returnCode contextInfo:(void *)contextInfo
|
|
{
|
|
[mAddField setStringValue:@""];
|
|
}
|
|
|
|
// data source informal protocol (NSTableDataSource)
|
|
- (int)numberOfRowsInTableView:(NSTableView *)aTableView
|
|
{
|
|
return [mCachedPermissions count];
|
|
}
|
|
|
|
- (id)tableView:(NSTableView *)aTableView objectValueForTableColumn:(NSTableColumn *)aTableColumn row:(int)rowIndex
|
|
{
|
|
id retVal = nil;
|
|
CHPermission* permission = [mCachedPermissions objectAtIndex:rowIndex];
|
|
if (aTableColumn == mPolicyColumn)
|
|
retVal = [NSNumber numberWithInt:(([permission policy] == CHPermissionAllow) ? 0 : 1)];
|
|
else // host column
|
|
retVal = [permission host];
|
|
|
|
return retVal;
|
|
}
|
|
|
|
// currently, this only applies to the site allow/deny, since that's the only editable column
|
|
-(void) tableView:(NSTableView *)aTableView
|
|
setObjectValue:anObject
|
|
forTableColumn:(NSTableColumn *)aTableColumn
|
|
row:(int)rowIndex
|
|
{
|
|
if (aTableColumn == mPolicyColumn) {
|
|
CHPermission* permission = [mCachedPermissions objectAtIndex:rowIndex];
|
|
[permission setPolicy:(([anObject intValue] == 0) ? CHPermissionAllow
|
|
: CHPermissionDeny)];
|
|
}
|
|
}
|
|
|
|
- (void)controlTextDidChange:(NSNotification*)notification
|
|
{
|
|
[mAddButton setEnabled:[[mAddField stringValue] length] > 0];
|
|
}
|
|
|
|
//
|
|
// tabFocusBehaviorChanged:
|
|
//
|
|
// Enable and disable tabbing to various elements. We expose only three options,
|
|
// but internally it's a bitwise additive pref:
|
|
// bit 0 adds focus for text fields (kFocusTextFields)
|
|
// bit 1 adds focus for other form elements (kFocusForms)
|
|
// bit 2 adds links and linked images (kFocusLinks)
|
|
//
|
|
- (IBAction)tabFocusBehaviorChanged:(id)sender
|
|
{
|
|
int tabFocusValue = 0;
|
|
switch ([sender indexOfSelectedItem]) {
|
|
case 0:
|
|
tabFocusValue = kFocusTextFields;
|
|
break;
|
|
case 1:
|
|
tabFocusValue = kFocusTextFields | kFocusForms;
|
|
break;
|
|
case 2:
|
|
tabFocusValue = kFocusTextFields | kFocusForms | kFocusLinks;
|
|
break;
|
|
}
|
|
|
|
[self setPref:"accessibility.tabfocus" toInt:tabFocusValue];
|
|
}
|
|
|
|
//
|
|
// popupIndexForCurrentTabFocusPref
|
|
//
|
|
// Returns the tab focus popup index for the current setting of the tab focus
|
|
// pref. Since we may not be able to show the actual pref, we err on the side
|
|
// of showing an over-inclusive item.
|
|
//
|
|
- (int)popupIndexForCurrentTabFocusPref {
|
|
int tabFocusValue = [self getIntPref:"accessibility.tabfocus" withSuccess:NULL];
|
|
if (tabFocusValue & kFocusLinks)
|
|
return 2;
|
|
else if (tabFocusValue & kFocusForms)
|
|
return 1;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
//
|
|
// clickEnableAnnoyanceBlocker:
|
|
//
|
|
// Enable and disable prefs for allowing webpages to be annoying and move/resize the
|
|
// window or tweak the status bar and make it unusable.
|
|
//
|
|
-(IBAction) clickEnableAnnoyanceBlocker:(id)sender
|
|
{
|
|
[sender setAllowsMixedState:NO];
|
|
if ( [sender state] )
|
|
[self setAnnoyingWindowPrefsTo:YES];
|
|
else
|
|
[self setAnnoyingWindowPrefsTo:NO];
|
|
}
|
|
|
|
//
|
|
// setAnnoyingWindowPrefsTo:
|
|
//
|
|
// Set all the prefs that allow webpages to muck with the status bar and window position
|
|
// (ie, be really annoying) to the given value
|
|
//
|
|
-(void) setAnnoyingWindowPrefsTo:(BOOL)inValue
|
|
{
|
|
[self setPref:"dom.disable_window_move_resize" toBoolean:inValue];
|
|
[self setPref:"dom.disable_window_status_change" toBoolean:inValue];
|
|
[self setPref:"dom.disable_window_flip" toBoolean:inValue];
|
|
}
|
|
|
|
- (int)annoyingWindowPrefs
|
|
{
|
|
BOOL disableStatusChangePref = [self getBooleanPref:"dom.disable_window_status_change" withSuccess:NULL];
|
|
BOOL disableMoveResizePref = [self getBooleanPref:"dom.disable_window_move_resize" withSuccess:NULL];
|
|
BOOL disableWindowFlipPref = [self getBooleanPref:"dom.disable_window_flip" withSuccess:NULL];
|
|
|
|
if(disableStatusChangePref && disableMoveResizePref && disableWindowFlipPref)
|
|
return kAnnoyancePrefAll;
|
|
if(!disableStatusChangePref && !disableMoveResizePref && !disableWindowFlipPref)
|
|
return kAnnoyancePrefNone;
|
|
|
|
return kAnnoyancePrefSome;
|
|
}
|
|
|
|
- (int)preventAnimationCheckboxState
|
|
{
|
|
NSString* preventAnimation = [self getStringPref:"image.animation_mode" withSuccess:NULL];
|
|
if ([preventAnimation isEqualToString:@"once"])
|
|
return NSOnState;
|
|
else if ([preventAnimation isEqualToString:@"normal"])
|
|
return NSOffState;
|
|
else
|
|
return NSMixedState;
|
|
}
|
|
|
|
//
|
|
// isFlashBlockAllowed
|
|
//
|
|
// Checks whether FlashBlock can be enabled
|
|
// FlashBlock only allowed if javascript and plug-ins enabled
|
|
// NOTE: This code is duplicated in PreferenceManager.mm since the FlashBlock checkbox
|
|
// settings are done by WebFeatures and stylesheet loading is done by PreferenceManager
|
|
//
|
|
-(BOOL) isFlashBlockAllowed
|
|
{
|
|
BOOL gotPref = NO;
|
|
BOOL jsEnabled = [self getBooleanPref:"javascript.enabled" withSuccess:&gotPref] && gotPref;
|
|
BOOL pluginsEnabled = [self getBooleanPref:"camino.enable_plugins" withSuccess:&gotPref] || !gotPref;
|
|
|
|
return jsEnabled && pluginsEnabled;
|
|
}
|
|
|
|
//
|
|
// updateFlashBlock
|
|
//
|
|
// Update the state of the FlashBlock checkbox
|
|
//
|
|
-(void) updateFlashBlock
|
|
{
|
|
BOOL allowed = [self isFlashBlockAllowed];
|
|
[mEnableFlashBlock setEnabled:allowed];
|
|
|
|
// FlashBlock state can only change if it's already enabled
|
|
// since changing dependencies won't have affect on disabled FlashBlock
|
|
if (![self getBooleanPref:"camino.enable_flashblock" withSuccess:nil])
|
|
return;
|
|
|
|
// FlashBlock preference is enabled. Checkbox is on if FlashBlock also allowed
|
|
[mEnableFlashBlock setState:(allowed ? NSOnState : NSOffState)];
|
|
|
|
// Always send a notification, dependency verification is done by receiver.
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:kFlashBlockChangedNotificationName object:nil];
|
|
}
|
|
|
|
@end
|