Reviewed by Tim H.
[WebKit-https.git] / WebKitTools / Drosera / DebuggerDocument.m
1 /*
2  * Copyright (C) 2006 Apple Computer, Inc.  All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  *
8  * 1.  Redistributions of source code must retain the above copyright
9  *     notice, this list of conditions and the following disclaimer. 
10  * 2.  Redistributions in binary form must reproduce the above copyright
11  *     notice, this list of conditions and the following disclaimer in the
12  *     documentation and/or other materials provided with the distribution. 
13  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
14  *     its contributors may be used to endorse or promote products derived
15  *     from this software without specific prior written permission. 
16  *
17  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
18  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
21  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
22  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
23  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
24  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28
29 #import "DebuggerDocument.h"
30 #import "DebuggerApplication.h"
31 #import <Carbon/Carbon.h>
32
33 static NSString *DebuggerConsoleToolbarItem = @"DebuggerConsoleToolbarItem";
34 static NSString *DebuggerContinueToolbarItem = @"DebuggerContinueToolbarItem";
35 static NSString *DebuggerPauseToolbarItem = @"DebuggerPauseToolbarItem";
36 static NSString *DebuggerStepIntoToolbarItem = @"DebuggerStepIntoToolbarItem";
37 static NSString *DebuggerStepOverToolbarItem = @"DebuggerStepOverToolbarItem";
38 static NSString *DebuggerStepOutToolbarItem = @"DebuggerStepOutToolbarItem";
39
40 @interface WebScriptObject (WebScriptObjectPrivate)
41 - (unsigned int)count;
42 @end
43
44 @implementation DebuggerDocument
45 + (BOOL)isSelectorExcludedFromWebScript:(SEL)aSelector
46 {
47     return NO;
48 }
49
50 + (BOOL)isKeyExcludedFromWebScript:(const char *)name
51 {
52     return NO;
53 }
54
55 #pragma mark -
56
57 - (id)initWithServerName:(NSString *)serverName
58 {
59     if ((self = [super init]))
60         [self switchToServerNamed:serverName];
61     return self;
62 }
63
64 - (void)dealloc
65 {
66     [server release];
67     [currentServerName release];
68     [super dealloc];
69 }
70
71 #pragma mark -
72 #pragma mark Stack & Variables
73
74 - (WebScriptCallFrame *)currentFrame
75 {
76     return currentFrame;
77 }
78
79 - (NSString *)currentFrameFunctionName
80 {
81     return [currentFrame functionName];
82 }
83
84 - (NSArray *)currentFunctionStack
85 {
86     NSMutableArray *result = [[NSMutableArray alloc] init];
87     WebScriptCallFrame *frame = currentFrame;
88     while (frame) {
89         if ([frame functionName])
90             [result addObject:[frame functionName]];
91         else if ([frame caller])
92             [result addObject:@"(anonymous function)"];
93         else
94             [result addObject:@"(global scope)"];
95         frame = [frame caller];
96     }
97     return [result autorelease];
98 }
99
100 - (id)evaluateScript:(NSString *)script inCallFrame:(int)frame
101 {
102     WebScriptCallFrame *cframe = currentFrame;
103     for (unsigned count = 0; count < frame; count++)
104         cframe = [cframe caller];
105     if (!cframe)
106         return nil;
107
108     id result = [cframe evaluateWebScript:script];
109     if ([result isKindOfClass:NSClassFromString(@"WebScriptObject")])
110         return [result callWebScriptMethod:@"toString" withArguments:nil];
111     return result;
112 }
113
114 - (NSArray *)webScriptAttributeKeysForScriptObject:(WebScriptObject *)object
115 {
116     [object evaluateWebScript:@"this.__drosera_introspection = function () { var result = new Array(); for (var x in this) { result.push(x); } return result; }"];
117
118     NSMutableArray *result = [[NSMutableArray alloc] init];
119     WebScriptObject *variables = [object callWebScriptMethod:@"__drosera_introspection" withArguments:nil];
120     unsigned length = [variables count];
121     for (unsigned i = 0; i < length; i++) {
122         NSString *key = [variables webScriptValueAtIndex:i];
123         if (![key isEqualToString:@"__drosera_introspection"])
124             [result addObject:key];
125     }
126
127     [object removeWebScriptKey:@"__drosera_introspection"];
128
129     [result sortUsingSelector:@selector(compare:)];
130     return [result autorelease];
131 }
132
133 - (NSArray *)localScopeVariableNamesForCallFrame:(int)frame
134 {
135     WebScriptCallFrame *cframe = currentFrame;
136     for (unsigned count = 0; count < frame; count++)
137         cframe = [cframe caller];
138
139     if (![[cframe scopeChain] count])
140         return nil;
141
142     WebScriptObject *scope = [[cframe scopeChain] objectAtIndex:0]; // local is always first
143     return [self webScriptAttributeKeysForScriptObject:scope];
144 }
145
146 - (NSString *)valueForScopeVariableNamed:(NSString *)key inCallFrame:(int)frame
147 {
148     WebScriptCallFrame *cframe = currentFrame;
149     for (unsigned count = 0; count < frame; count++)
150         cframe = [cframe caller];
151
152     if (![[cframe scopeChain] count])
153         return nil;
154
155     unsigned scopeCount = [[cframe scopeChain] count];
156     for (unsigned i = 0; i < scopeCount; i++) {
157         WebScriptObject *scope = [[cframe scopeChain] objectAtIndex:i];
158         id value = [scope valueForKey:key];
159         if ([value isKindOfClass:NSClassFromString(@"WebScriptObject")])
160             return [value callWebScriptMethod:@"toString" withArguments:nil];
161         if (value && ![value isKindOfClass:[NSString class]])
162             return [NSString stringWithFormat:@"%@", value];
163         if (value)
164             return value;
165     }
166
167     return nil;
168 }
169
170 #pragma mark -
171 #pragma mark System Information
172
173 - (int)doubleClickMilliseconds
174 {
175     // GetDblTime() returns values in 1/60ths of a second
176     return ((double)GetDblTime() / 60.0) * 1000;
177 }
178
179 #pragma mark -
180 #pragma mark Pause & Step
181
182 - (BOOL)isPaused
183 {
184     return paused;
185 }
186
187 - (void)pause
188 {
189     paused = YES;
190     if ([[(NSDistantObject *)server connectionForProxy] isValid])
191         [server pause];
192     [[NSApplication sharedApplication] activateIgnoringOtherApps:YES];
193 }
194
195 - (void)resume
196 {
197     paused = NO;
198     if ([[(NSDistantObject *)server connectionForProxy] isValid])
199         [server resume];
200 }
201
202 - (void)stepInto
203 {
204     if ([[(NSDistantObject *)server connectionForProxy] isValid])
205         [server step];
206 }
207
208 - (void)log:(NSString *)msg
209 {
210     NSLog(@"%@", msg);
211 }
212
213 #pragma mark -
214 #pragma mark Interface Actions
215
216 - (IBAction)pause:(id)sender
217 {
218     [[webView windowScriptObject] callWebScriptMethod:@"pause" withArguments:nil];
219 }
220
221 - (IBAction)resume:(id)sender
222 {
223     [[webView windowScriptObject] callWebScriptMethod:@"resume" withArguments:nil];
224 }
225
226 - (IBAction)stepInto:(id)sender
227 {
228     [[webView windowScriptObject] callWebScriptMethod:@"stepInto" withArguments:nil];
229 }
230
231 - (IBAction)stepOver:(id)sender
232 {
233     [[webView windowScriptObject] callWebScriptMethod:@"stepOver" withArguments:nil];
234 }
235
236 - (IBAction)stepOut:(id)sender
237 {
238     [[webView windowScriptObject] callWebScriptMethod:@"stepOut" withArguments:nil];
239 }
240
241 - (IBAction)showConsole:(id)sender
242 {
243     [[webView windowScriptObject] callWebScriptMethod:@"showConsoleWindow" withArguments:nil];
244 }
245
246 #pragma mark -
247 #pragma mark Window Controller Overrides
248
249 - (NSString *)windowNibName
250 {
251     return @"Debugger";
252 }
253
254 - (void)windowDidLoad
255 {
256     [super windowDidLoad];
257
258     [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationTerminating:) name:NSApplicationWillTerminateNotification object:nil];
259
260     NSString *path = [[NSBundle mainBundle] pathForResource:@"debugger" ofType:@"html" inDirectory:nil];
261     [[webView mainFrame] loadRequest:[[[NSURLRequest alloc] initWithURL:[NSURL fileURLWithPath:path]] autorelease]];
262
263     NSToolbar *toolbar = [[NSToolbar alloc] initWithIdentifier:@"debugger"];
264     [toolbar setDelegate:self];
265     [toolbar setAllowsUserCustomization:YES];
266     [toolbar setAutosavesConfiguration:YES];
267     [[self window] setToolbar:toolbar];
268     [toolbar release];
269 }
270
271 - (void)windowWillClose:(NSNotification *)notification
272 {
273     [[webView windowScriptObject] removeWebScriptKey:@"DebuggerDocument"];
274
275     [[NSNotificationCenter defaultCenter] removeObserver:self name:NSApplicationWillTerminateNotification object:nil];
276
277     [self switchToServerNamed:nil];
278
279     [self autorelease]; // DebuggerApplication expects us to release on close
280 }
281
282 #pragma mark -
283 #pragma mark Connection Handling
284
285 - (void)switchToServerNamed:(NSString *)name
286 {
287     if (server) {
288         [[NSNotificationCenter defaultCenter] removeObserver:self name:NSConnectionDidDieNotification object:[(NSDistantObject *)server connectionForProxy]];
289         if ([[(NSDistantObject *)server connectionForProxy] isValid]) {
290             [server removeListener:self];
291             [self resume];
292         }
293     }
294
295     id old = server;
296     server = ([name length] ? [[NSConnection rootProxyForConnectionWithRegisteredName:name host:nil] retain] : nil);
297     [old release];
298
299     old = currentServerName;
300     currentServerName = [name copy];
301     [old release];
302
303     if (server) {
304         @try {
305             [(NSDistantObject *)server setProtocolForProxy:@protocol(WebScriptDebugServer)];
306             [server addListener:self];
307         } @catch (NSException *exception) {
308             [currentServerName release];
309             currentServerName = nil;
310             [server release];
311             server = nil;
312         }
313
314         if (server)
315             [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(serverConnectionDidDie:) name:NSConnectionDidDieNotification object:[(NSDistantObject *)server connectionForProxy]];  
316     }
317 }
318
319 - (void)applicationTerminating:(NSNotification *)notifiction
320 {
321     if (server && [[(NSDistantObject *)server connectionForProxy] isValid]) {
322         [self switchToServerNamed:nil];
323         // call the runloop for a while to make sure our removeListener: is sent to the server
324         [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:0.25]];
325     }
326 }
327
328 - (void)serverConnectionDidDie:(NSNotification *)notifiction
329 {
330     [self switchToServerNamed:nil];
331 }
332
333 #pragma mark -
334 #pragma mark Toolbar Delegate
335
336 - (NSToolbarItem *)toolbar:(NSToolbar *)toolbar itemForItemIdentifier:(NSString *)itemIdentifier willBeInsertedIntoToolbar:(BOOL)flag
337 {
338     if ([itemIdentifier isEqualToString:DebuggerContinueToolbarItem]) {
339         NSToolbarItem *item = [[NSToolbarItem alloc] initWithItemIdentifier:itemIdentifier];
340
341         [item setLabel:@"Continue"];
342         [item setPaletteLabel:@"Continue"];
343
344         [item setToolTip:@"Continue script execution"];
345         [item setImage:[NSImage imageNamed:@"continue"]];
346
347         [item setTarget:self];
348         [item setAction:@selector(resume:)];
349
350         return [item autorelease];
351     } else if ([itemIdentifier isEqualToString:DebuggerConsoleToolbarItem]) {
352         NSToolbarItem *item = [[NSToolbarItem alloc] initWithItemIdentifier:itemIdentifier];
353
354         [item setLabel:@"Console"];
355         [item setPaletteLabel:@"Console"];
356
357         [item setToolTip:@"Console"];
358         [item setImage:[NSImage imageNamed:@"console"]];
359
360         [item setTarget:self];
361         [item setAction:@selector(showConsole:)];
362
363         return [item autorelease];
364     } else if ([itemIdentifier isEqualToString:DebuggerPauseToolbarItem]) {
365         NSToolbarItem *item = [[NSToolbarItem alloc] initWithItemIdentifier:itemIdentifier];
366
367         [item setLabel:@"Pause"];
368         [item setPaletteLabel:@"Pause"];
369
370         [item setToolTip:@"Pause script execution"];
371         [item setImage:[NSImage imageNamed:@"pause"]];
372
373         [item setTarget:self];
374         [item setAction:@selector(pause:)];
375
376         return [item autorelease];
377     } else if ([itemIdentifier isEqualToString:DebuggerStepIntoToolbarItem]) {
378         NSToolbarItem *item = [[NSToolbarItem alloc] initWithItemIdentifier:itemIdentifier];
379
380         [item setLabel:@"Step Into"];
381         [item setPaletteLabel:@"Step Into"];
382
383         [item setToolTip:@"Step into function call"];
384         [item setImage:[NSImage imageNamed:@"step"]];
385
386         [item setTarget:self];
387         [item setAction:@selector(stepInto:)];
388
389         return [item autorelease];
390     } else if ([itemIdentifier isEqualToString:DebuggerStepOverToolbarItem]) {
391         NSToolbarItem *item = [[NSToolbarItem alloc] initWithItemIdentifier:itemIdentifier];
392
393         [item setLabel:@"Step Over"];
394         [item setPaletteLabel:@"Step Over"];
395
396         [item setToolTip:@"Step over function call"];
397         [item setImage:[NSImage imageNamed:@"stepOver"]];
398
399         [item setTarget:self];
400         [item setAction:@selector(stepOver:)];
401
402         return [item autorelease];
403     } else if ([itemIdentifier isEqualToString:DebuggerStepOutToolbarItem]) {
404         NSToolbarItem *item = [[NSToolbarItem alloc] initWithItemIdentifier:itemIdentifier];
405
406         [item setLabel:@"Step Out"];
407         [item setPaletteLabel:@"Step Over"];
408
409         [item setToolTip:@"Step out of current function"];
410         [item setImage:[NSImage imageNamed:@"stepOut"]];
411
412         [item setTarget:self];
413         [item setAction:@selector(stepOut:)];
414
415         return [item autorelease];
416     }
417
418     return nil;
419 }
420
421 - (NSArray *)toolbarDefaultItemIdentifiers:(NSToolbar*)toolbar
422 {
423     return [NSArray arrayWithObjects:DebuggerContinueToolbarItem, DebuggerPauseToolbarItem,
424         NSToolbarSeparatorItemIdentifier, DebuggerStepIntoToolbarItem, DebuggerStepOutToolbarItem,
425         DebuggerStepOverToolbarItem, NSToolbarFlexibleSpaceItemIdentifier, DebuggerConsoleToolbarItem, nil];
426 }
427
428 - (NSArray *)toolbarAllowedItemIdentifiers:(NSToolbar*)toolbar
429 {
430     return [NSArray arrayWithObjects:DebuggerConsoleToolbarItem, DebuggerContinueToolbarItem, DebuggerPauseToolbarItem,
431         DebuggerStepIntoToolbarItem, DebuggerStepOutToolbarItem, DebuggerStepOverToolbarItem, NSToolbarCustomizeToolbarItemIdentifier,
432         NSToolbarFlexibleSpaceItemIdentifier, NSToolbarSpaceItemIdentifier, NSToolbarSeparatorItemIdentifier, nil];
433 }
434
435 - (BOOL)validateUserInterfaceItem:(id <NSValidatedUserInterfaceItem>)interfaceItem
436 {
437     SEL action = [interfaceItem action];
438     if (action == @selector(pause:))
439         return ![self isPaused];
440     if (action == @selector(resume:) ||
441         action == @selector(stepOver:) ||
442         action == @selector(stepOut:) ||
443         action == @selector(stepInto:))
444         return [self isPaused];
445     return YES;
446 }
447
448 #pragma mark -
449 #pragma mark WebView UI Delegate
450
451 - (WebView *)webView:(WebView *)sender createWebViewWithRequest:(NSURLRequest *)request
452 {
453     WebView *newWebView = [[WebView alloc] initWithFrame:NSZeroRect frameName:nil groupName:nil];
454     [newWebView setAutoresizingMask:(NSViewWidthSizable | NSViewHeightSizable)];
455     [newWebView setUIDelegate:self];
456     [newWebView setPolicyDelegate:self];
457     [newWebView setFrameLoadDelegate:self];
458     if (request)
459         [[newWebView mainFrame] loadRequest:request];
460
461     NSWindow *window = [[NSWindow alloc] initWithContentRect:NSZeroRect styleMask:(NSTitledWindowMask | NSClosableWindowMask | NSMiniaturizableWindowMask | NSResizableWindowMask | NSUnifiedTitleAndToolbarWindowMask) backing:NSBackingStoreBuffered defer:NO screen:[[webView window] screen]];
462     [window setReleasedWhenClosed:YES];
463     [newWebView setFrame:[[window contentView] frame]];
464     [[window contentView] addSubview:newWebView];
465     [newWebView release];
466
467     return newWebView;
468 }
469
470 - (void)webViewShow:(WebView *)sender
471 {
472     [[sender window] makeKeyAndOrderFront:sender];
473 }
474
475 - (BOOL)webViewAreToolbarsVisible:(WebView *)sender
476 {
477     return [[[sender window] toolbar] isVisible];
478 }
479
480 - (void)webView:(WebView *)sender setToolbarsVisible:(BOOL)visible
481 {
482     [[[sender window] toolbar] setVisible:visible];
483 }
484
485 - (void)webView:(WebView *)sender setResizable:(BOOL)resizable
486 {
487     [[sender window] setShowsResizeIndicator:resizable];
488     [[[sender window] standardWindowButton:NSWindowZoomButton] setEnabled:resizable];
489 }
490
491 - (void)webView:(WebView *)sender runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WebFrame *)frame
492 {
493     NSRange range = [message rangeOfString:@"\t"];
494     NSString *title = @"Alert";
495     if (range.location != NSNotFound) {
496         title = [message substringToIndex:range.location];
497         message = [message substringFromIndex:(range.location + range.length)];
498     }
499
500     NSBeginInformationalAlertSheet(title, nil, nil, nil, [sender window], nil, NULL, NULL, NULL, message);
501 }
502
503 - (void)scriptConfirmSheetDidEnd:(NSWindow *)sheet returnCode:(int)returnCode contextInfo:(int *)contextInfo
504 {
505     *contextInfo = returnCode;
506 }
507
508 - (BOOL)webView:(WebView *)sender runJavaScriptConfirmPanelWithMessage:(NSString *)message initiatedByFrame:(WebFrame *)frame
509 {
510     NSRange range = [message rangeOfString:@"\t"];
511     NSString *title = @"Alert";
512     if (range.location != NSNotFound) {
513         title = [message substringToIndex:range.location];
514         message = [message substringFromIndex:(range.location + range.length)];
515     }
516
517     int result = NSNotFound;
518     NSBeginInformationalAlertSheet(title, nil, @"Cancel", nil, [sender window], self, @selector(scriptConfirmSheetDidEnd:returnCode:contextInfo:), NULL, &result, message);
519
520     while (result == NSNotFound) {
521         NSEvent *nextEvent = [[NSApplication sharedApplication] nextEventMatchingMask:NSAnyEventMask untilDate:[NSDate distantFuture] inMode:NSDefaultRunLoopMode dequeue:YES];
522         [[NSApplication sharedApplication] sendEvent:nextEvent];
523     }
524
525     return result;
526 }
527
528 #pragma mark -
529 #pragma mark WebView Frame Load Delegate
530
531 - (void)webView:(WebView *)sender windowScriptObjectAvailable:(WebScriptObject *)windowScriptObject
532 {
533     // note: this is the Debuggers's own WebView, not the one being debugged
534     if ([[sender window] isEqual:[self window]])
535         [[sender windowScriptObject] setValue:self forKey:@"DebuggerDocument"];
536 }
537
538 - (void)webView:(WebView *)sender didFinishLoadForFrame:(WebFrame *)frame
539 {
540     // note: this is the Debuggers's own WebView, not the one being debugged
541     if ([[sender window] isEqual:[self window]])
542         webViewLoaded = YES;
543 }
544
545 - (void)webView:(WebView *)sender didReceiveTitle:(NSString *)title forFrame:(WebFrame *)frame
546 {
547     // note: this is the Debuggers's own WebViews, not the one being debugged
548     if ([frame isEqual:[sender mainFrame]]) {
549         NSDictionary *info = [[(DebuggerApplication *)[[NSApplication sharedApplication] delegate] knownServers] objectForKey:currentServerName];
550         NSString *processName = [info objectForKey:WebScriptDebugServerProcessNameKey];
551         if (info && [processName length]) {
552             NSMutableString *newTitle = [[NSMutableString alloc] initWithString:processName];
553             [newTitle appendString:@" - "];
554             [newTitle appendString:title];
555             [[sender window] setTitle:newTitle];
556             [newTitle release];
557         } else 
558             [[sender window] setTitle:title];
559     }
560 }
561
562 #pragma mark -
563 #pragma mark Debug Listener Callbacks
564
565 - (void)webView:(WebView *)view didLoadMainResourceForDataSource:(WebDataSource *)dataSource
566 {
567     NSString *documentSourceCopy = nil;
568     id <WebDocumentRepresentation> rep = [dataSource representation];
569     if ([rep canProvideDocumentSource])
570         documentSourceCopy = [[rep documentSource] copy];
571
572     if (!documentSourceCopy)
573         return;
574
575     NSString *urlCopy = [[[[dataSource response] URL] absoluteString] copy];
576     NSArray *args = [[NSArray alloc] initWithObjects:(documentSourceCopy ? documentSourceCopy : @""), (urlCopy ? urlCopy : @""), [NSNumber numberWithBool:NO], nil];
577     [[webView windowScriptObject] callWebScriptMethod:@"updateFileSource" withArguments:args];
578
579     [args release];
580     [documentSourceCopy release];
581     [urlCopy release];
582 }
583
584 - (void)webView:(WebView *)view didParseSource:(NSString *)source baseLineNumber:(unsigned)baseLine fromURL:(NSURL *)url sourceId:(int)sid forWebFrame:(WebFrame *)webFrame
585 {
586     if (!webViewLoaded)
587         return;
588
589     NSString *sourceCopy = [source copy];
590     if (!sourceCopy)
591         return;
592
593     NSString *documentSourceCopy = nil;
594     NSString *urlCopy = [[url absoluteString] copy];
595
596     WebDataSource *dataSource = [webFrame dataSource];
597     if (!url || [[[dataSource response] URL] isEqual:url]) {
598         id <WebDocumentRepresentation> rep = [dataSource representation];
599         if ([rep canProvideDocumentSource])
600             documentSourceCopy = [[rep documentSource] copy];
601         if (!urlCopy)
602             urlCopy = [[[[dataSource response] URL] absoluteString] copy];
603     }
604
605     NSArray *args = [[NSArray alloc] initWithObjects:sourceCopy, (documentSourceCopy ? documentSourceCopy : @""), (urlCopy ? urlCopy : @""), [NSNumber numberWithInt:sid], [NSNumber numberWithUnsignedInt:baseLine], nil];
606     [[webView windowScriptObject] callWebScriptMethod:@"didParseScript" withArguments:args];
607
608     [args release];
609     [sourceCopy release];
610     [documentSourceCopy release];
611     [urlCopy release];
612 }
613
614 - (void)webView:(WebView *)view failedToParseSource:(NSString *)source baseLineNumber:(unsigned)baseLine fromURL:(NSURL *)url withError:(NSError *)error forWebFrame:(WebFrame *)webFrame
615 {
616 }
617
618 - (void)webView:(WebView *)view didEnterCallFrame:(WebScriptCallFrame *)frame sourceId:(int)sid line:(int)lineno forWebFrame:(WebFrame *)webFrame
619 {
620     if (!webViewLoaded)
621         return;
622
623     id old = currentFrame;
624     currentFrame = [frame retain];
625     [old release];
626
627     NSArray *args = [[NSArray alloc] initWithObjects:[NSNumber numberWithInt:sid], [NSNumber numberWithInt:lineno], nil];
628     [[webView windowScriptObject] callWebScriptMethod:@"didEnterCallFrame" withArguments:args];
629     [args release];
630 }
631
632 - (void)webView:(WebView *)view willExecuteStatement:(WebScriptCallFrame *)frame sourceId:(int)sid line:(int)lineno forWebFrame:(WebFrame *)webFrame
633 {
634     if (!webViewLoaded)
635         return;
636
637     NSArray *args = [[NSArray alloc] initWithObjects:[NSNumber numberWithInt:sid], [NSNumber numberWithInt:lineno], nil];
638     [[webView windowScriptObject] callWebScriptMethod:@"willExecuteStatement" withArguments:args];
639     [args release];
640 }
641
642 - (void)webView:(WebView *)view willLeaveCallFrame:(WebScriptCallFrame *)frame sourceId:(int)sid line:(int)lineno forWebFrame:(WebFrame *)webFrame
643 {
644     if (!webViewLoaded)
645         return;
646
647     NSArray *args = [[NSArray alloc] initWithObjects:[NSNumber numberWithInt:sid], [NSNumber numberWithInt:lineno], nil];
648     [[webView windowScriptObject] callWebScriptMethod:@"willLeaveCallFrame" withArguments:args];
649     [args release];
650
651     id old = currentFrame;
652     currentFrame = [[frame caller] retain];
653     [old release];
654 }
655
656 - (void)webView:(WebView *)view exceptionWasRaised:(WebScriptCallFrame *)frame sourceId:(int)sid line:(int)lineno forWebFrame:(WebFrame *)webFrame
657 {
658     if (!webViewLoaded)
659         return;
660
661     NSArray *args = [[NSArray alloc] initWithObjects:[NSNumber numberWithInt:sid], [NSNumber numberWithInt:lineno], nil];
662     [[webView windowScriptObject] callWebScriptMethod:@"exceptionWasRaised" withArguments:args];
663     [args release];
664 }
665 @end