source: trunk/ICeCoffEE/ICeCoffEE/ICeCoffEEWebKit.m@ 491

Last change on this file since 491 was 467, checked in by Nicholas Riley, 16 years ago

Disable on list boxes.

File size: 8.8 KB
RevLine 
[66]1//
2// ICeCoffEEWebKit.m
3// ICeCoffEE APE
4//
5// Created by Nicholas Riley on Sun Jan 19 2003.
6// Copyright (c) 2003 Nicholas Riley. All rights reserved.
7//
8
9#import "ICeCoffEEWebKit.h"
[443]10#import "ICeCoffEEWebPolicyDelegate.h"
11#import "ICeCoffEEParser.h"
12#import "ICeCoffEETrigger.h"
13#import <WebKit/WebKit.h>
[66]14#import <unistd.h>
15
[455]16// Safari 3.1 and earlier (pre-r31014)
17// eliminated in http://bugs.webkit.org/show_bug.cgi?id=17640
[449]18@interface WebCoreFrameBridge : NSObject
[443]19- (DOMRange *)convertNSRangeToDOMRange:(NSRange)range;
20- (NSString *)stringForRange:(DOMRange *)range;
[66]21@end
22
[456]23// r31014 and later; likely to move again
24@interface WebFrame (WebFrameInternal)
25- (DOMRange *)_convertNSRangeToDOMRange:(NSRange)range;
26- (NSString *)_stringForRange:(DOMRange *)range;
27@end
28
[443]29// XXX WebHTMLView is going away
[167]30@interface WebHTMLView : NSObject
[66]31
[320]32- (NSRect)selectionRect;
[443]33- (DOMRange *)_documentRange;
[66]34
[443]35- (WebView *)_webView;
[456]36- (WebCoreFrameBridge *)_bridge; // moved from WebNSViewExtras in r14032; removed in r31014
37- (WebFrame *)_frame; // moved from WebNSViewExtras in r14032
[167]38
[66]39@end
40
[448]41// from WebViewPrivate.h (will become public, part of WebViewEditing)
42@interface WebView (webViewGrammarChecking)
43- (BOOL)isGrammarCheckingEnabled;
44- (void)setGrammarCheckingEnabled:(BOOL)flag;
45@end
46
[66]47@implementation ICeCoffEEWebKit
48
49- (NSMenu *)menuForEvent:(NSEvent *)e;
50{
51 NSMenu *myMenu = [super menuForEvent: e];
52 return ICCF_MenuForEvent(self, myMenu, e);
53}
54
55static NSEvent *downEvent = nil;
[443]56static id /* (WebPolicyDelegate) */ policyDelegate;
57static id /* DOMRange */ selectedRange;
[167]58
[467]59static BOOL ICCF_DOMParentOfClass(DOMNode *node, Class cls) {
60 do {
61 if ([node isKindOfClass: cls])
62 return YES;
63 } while ( (node = [node parentNode]) != nil);
64
65 return NO;
66}
67#define ICCF_NodeIs(elt) [clickedNode isKindOfClass: NSClassFromString(@"DOMHTML"elt"Element")]
68#define ICCF_DOMParentIs(elt) ICCF_DOMParentOfClass(clickedNode, NSClassFromString(@"DOMHTML"elt"Element"))
69
[66]70- (void)mouseDown:(NSEvent *)e;
71{
[182]72 [downEvent release]; downEvent = nil;
[74]73 if (ICCF_enabled && ICCF_prefs.commandClickEnabled && ICCF_EventIsCommandMouseDown(e)) {
[443]74 if ([self respondsToSelector: @selector(_webView)]) {
75 WebView *webView = [(WebHTMLView *)self _webView];
76
[467]77 // check that the Command-click isn't being used for discontiguous selection
78 NSPoint viewClickPt = [webView convertPoint: [e locationInWindow] fromView: nil];
79 NSDictionary *elementDict = [webView elementAtPoint: viewClickPt];
80 DOMNode *clickedNode = [elementDict objectForKey: @"WebElementDOMNode"];
81 if (clickedNode != nil && ICCF_NodeIs("Select")) {
82 [super mouseDown: e];
83 return;
84 }
85
[182]86 // save selection: it may be deselected on super mouseDown
[443]87 selectedRange = [[webView selectedDOMRange] retain];
88
89 // stop any URL launching from happening
90 if ([webView isEditable]) {
91 policyDelegate = [[webView policyDelegate] retain];
92 [webView setPolicyDelegate: [ICeCoffEEWebPolicyDelegate sharedDelegate]];
93 }
94
95 downEvent = [e retain];
[182]96 }
[66]97 }
98 [super mouseDown: e];
99}
100
101- (void)mouseUp:(NSEvent *)e;
102{
103 [super mouseUp: e];
104
[443]105 if (downEvent == nil)
106 return;
107
108 WebView *webView = [(WebHTMLView *)self _webView];
[455]109 BOOL webViewIsEditable = [webView isEditable];
110 BOOL elementIsEditable = YES; // don't restore
[448]111 BOOL isContinuousSpellCheckingEnabled = NO, isGrammarCheckingEnabled = NO;
[443]112
[455]113 if (webViewIsEditable) {
[443]114 [webView setPolicyDelegate: policyDelegate];
115 [policyDelegate release]; policyDelegate = nil;
116 }
117
118 NSPoint downPt = [downEvent locationInWindow];
119 NSPoint upPt = [e locationInWindow];
120 if (abs(downPt.x - upPt.x) > kICHysteresisPixels && abs(downPt.y - upPt.y) > kICHysteresisPixels)
121 return;
122
123 @try {
[455]124 NSPoint viewClickPt = [webView convertPoint: downPt fromView: nil];
125 NSDictionary *elementDict = [webView elementAtPoint: viewClickPt];
126 ICLog(@"elementDict: %@", elementDict);
127
128 NSAssert([elementDict count] != 0, ICCF_LocalizedString(@"Sorry, ICeCoffEE was unable to find anything to select"));
129
130 elementIsEditable = [[elementDict objectForKey: @"WebElementIsContentEditableKey"] boolValue];
131 if (!elementIsEditable) {
132 NSAssert(!webViewIsEditable, @"Internal error: uneditable element inside editable WebView");
[443]133 [webView setEditable: YES];
[448]134
135 // don't want spelling/grammar marks to persist when view is made uneditable again
136 if ([webView respondsToSelector: @selector(isContinuousSpellCheckingEnabled)] &&
137 (isContinuousSpellCheckingEnabled = [webView isContinuousSpellCheckingEnabled])) {
138 if ([webView respondsToSelector: @selector(setContinuousSpellCheckingEnabled:)])
139 [webView setContinuousSpellCheckingEnabled: NO];
140 else
141 isContinuousSpellCheckingEnabled = NO; // don't restore
142 }
[443]143
[448]144 if ([webView respondsToSelector: @selector(isGrammarCheckingEnabled)] &&
145 (isGrammarCheckingEnabled = [webView isGrammarCheckingEnabled])) {
146 if ([webView respondsToSelector: @selector(setGrammarCheckingEnabled:)])
147 [webView setGrammarCheckingEnabled: NO];
148 else
149 isGrammarCheckingEnabled = NO; // don't restore
150 }
151 }
152
[455]153 DOMNode *clickedNode = [elementDict objectForKey: @"WebElementDOMNode"];
[457]154 BOOL elementIsHTMLInput = ICCF_NodeIs("Input");
[455]155
[457]156 if (!elementIsEditable && (elementIsHTMLInput || ICCF_DOMParentIs("Button") || ICCF_NodeIs("FieldSet"))) {
[455]157 ICLog(@"got an uneditable form field (e.g., button)");
158 return;
159 }
[443]160
161 id link = [elementDict objectForKey: @"WebElementLinkURL"];
162 NSString *url = [link isKindOfClass: [NSURL class]] ? [link absoluteString] : nil;
[455]163
[443]164 ICCF_StartIC();
165
166 id /* DOMRange */ domRange = nil;
167
168 if (url != nil) {
169 ICLog(@"got a link");
[455]170 if (!elementIsEditable) {
[436]171 return;
[182]172 }
[443]173 // XXX handle existing selection
174 domRange = [webView selectedDOMRange];
[455]175 [domRange selectNode: clickedNode];
[443]176 [webView setSelectedDOMRange: domRange affinity: NSSelectionAffinityDownstream];
177 } else {
[455]178 if (!elementIsEditable) // may have become deselected in mouseDown
179 [webView setSelectedDOMRange: selectedRange affinity: NSSelectionAffinityDownstream];
[443]180
181 NSRange range = [ICeCoffEETrigger rangeForEvent: downEvent onTarget: (NSView<NSTextInput> *)self];
[446]182 NSAssert(range.location != NSNotFound, ICCF_LocalizedString(@"Sorry, ICeCoffEE was unable to find anything to select"));
[443]183
[456]184 WebFrame *frame = [(WebHTMLView *)self _frame];
185 WebCoreFrameBridge *bridge = nil;
186 if ([self respondsToSelector: @selector(_bridge)])
187 bridge = [(WebHTMLView *)self _bridge];
[455]188
[443]189 // XXX limit to a reasonable size
[455]190 NSString *s;
[457]191 if (elementIsHTMLInput || ICCF_NodeIs("TextArea") || (elementIsEditable && !webViewIsEditable)) {
192 // for form fields, range will be field-relative rather than document-relative
193 ICLog(@"Using attributedSubstringFromRange:");
[455]194 s = [[(NSView<NSTextInput> *)self attributedSubstringFromRange: NSMakeRange(0, UINT_MAX)] string];
[457]195 } else { // sometimes attributedSubstringFromRange: returns nil
196 ICLog(@"Using stringForRange:");
[456]197 domRange = [(WebHTMLView *)self _documentRange];
198 if (bridge != nil)
199 s = [bridge stringForRange: domRange];
200 else if (![frame respondsToSelector: @selector(_stringForRange:)])
201 return; // WebKit too new?
202 else
203 s = [frame _stringForRange: domRange];
204 }
[455]205
[456]206 if (bridge == nil && ![frame respondsToSelector: @selector(_convertNSRangeToDOMRange:)]) {
207 return; // WebKit too new?
208 }
209
[443]210 if (range.length == 0) {
211 range.length = 1;
212 range = ICCF_URLEnclosingRange(s, range);
[456]213 domRange = bridge ? [bridge convertNSRangeToDOMRange: range] :
214 [frame _convertNSRangeToDOMRange: range];
[443]215 [webView setSelectedDOMRange: domRange affinity: NSSelectionAffinityDownstream];
216 } else {
[456]217 domRange = bridge ? [bridge convertNSRangeToDOMRange: range] :
218 [frame _convertNSRangeToDOMRange: range];
[183]219 }
[443]220
221 url = [s substringWithRange: range];
222 }
223
224 if (ICCF_LaunchURL(url, ICCF_KeyboardAction(downEvent)) && ICCF_prefs.textBlinkEnabled && domRange != nil) {
225 NSRect selectionRect = [(WebHTMLView *)self selectionRect];
226 for (int i = 0 ; i < ICCF_prefs.textBlinkCount ; i++) {
227 [webView setSelectedDOMRange: nil affinity: NSSelectionAffinityDownstream];
228 [self setNeedsDisplayInRect: selectionRect];
229 [self display];
230 usleep(kICBlinkDelayUsecs);
231 [webView setSelectedDOMRange: domRange affinity: NSSelectionAffinityDownstream];
232 [self setNeedsDisplayInRect: selectionRect];
233 [self display];
234 usleep(kICBlinkDelayUsecs);
[182]235 }
[436]236 }
[443]237 } @catch (NSException *e) {
238 ICCF_HandleException(e, downEvent);
239 } @finally {
240 [selectedRange release]; selectedRange = nil;
241 [downEvent release]; downEvent = nil;
[455]242 if (!elementIsEditable && !webViewIsEditable) {
[443]243 [webView setEditable: NO];
[448]244 if (isContinuousSpellCheckingEnabled)
245 [webView setContinuousSpellCheckingEnabled: YES];
246 if (isGrammarCheckingEnabled)
247 [webView setGrammarCheckingEnabled: YES];
248 }
[456]249 ICCF_StopIC();
[443]250 }
[66]251}
252
253@end
Note: See TracBrowser for help on using the repository browser.