Making the buttons change their enabled state depending on where we are in debugging
[macgdbp.git] / Source / DebuggerWindowController.m
1 /*
2 * MacGDBp
3 * Copyright (c) 2002 - 2007, Blue Static <http://www.bluestatic.org>
4 *
5 * This program is free software; you can redistribute it and/or modify it under the terms of the GNU
6 * General Public License as published by the Free Software Foundation; either version 2 of the
7 * License, or (at your option) any later version.
8 *
9 * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without
10 * even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 * General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License along with this program; if not,
14 * write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
15 */
16
17 #import "DebuggerWindowController.h"
18 #import "DebuggerConnection.h"
19 #import "NSXMLElementAdditions.h"
20
21 @interface DebuggerWindowController (Private)
22
23 - (void)updateSourceViewer;
24
25 @end
26
27 @implementation DebuggerWindowController
28
29 /**
30 * Initializes the window controller and sets the connection
31 */
32 - (id)initWithConnection: (DebuggerConnection *)cnx
33 {
34 if (self = [super initWithWindowNibName: @"Debugger"])
35 {
36 _connection = [cnx retain];
37 }
38 return self;
39 }
40
41 /**
42 * Before the display get's comfortable, set up the NSTextView to scroll horizontally
43 */
44 - (void)awakeFromNib
45 {
46 // set up the scroller for the source viewer
47 [_sourceViewer setMaxSize: NSMakeSize(FLT_MAX, FLT_MAX)];
48 [[_sourceViewer textContainer] setContainerSize: NSMakeSize(FLT_MAX, FLT_MAX)];
49 [[_sourceViewer textContainer] setWidthTracksTextView: NO];
50 [_sourceViewer setHorizontallyResizable: YES];
51 [_sourceViewerScroller setHasHorizontalScroller: YES];
52 [_sourceViewerScroller display];
53 }
54
55 /**
56 * Release object members
57 */
58 - (void)dealloc
59 {
60 [_connection release];
61
62 [super dealloc];
63 }
64
65 /**
66 * Sets the status and clears any error message
67 */
68 - (void)setStatus: (NSString *)status
69 {
70 [_error setHidden: YES];
71 [_status setStringValue: status];
72 [[self window] setTitle: [NSString stringWithFormat: @"GDBp @ %@:%d/%@", [_connection remoteHost], [_connection port], [_connection session]]];
73
74 [_stepInButton setEnabled: NO];
75 [_stepOutButton setEnabled: NO];
76 [_stepOverButton setEnabled: NO];
77 [_runButton setEnabled: NO];
78
79 if ([_connection isConnected])
80 {
81 if ([status isEqualToString: @"Starting"])
82 {
83 [_stepInButton setEnabled: YES];
84 [_runButton setEnabled: YES];
85 }
86 }
87 }
88
89 /**
90 * Sets the status to be "Error" and then displays the error message
91 */
92 - (void)setError: (NSString *)error
93 {
94 [_error setStringValue: error];
95 [self setStatus: @"Error"];
96 [_error setHidden: NO];
97 }
98
99 /**
100 * Sets the root node element of the stacktrace
101 */
102 - (void)setStack: (NSArray *)stack
103 {
104 if (_stack != nil)
105 {
106 [_stack release];
107 }
108
109 _stack = stack;
110 [_stack retain];
111
112 if ([_stack count] > 1)
113 {
114 [_stepOutButton setEnabled: YES];
115 }
116 [_stepInButton setEnabled: YES];
117 [_stepOverButton setEnabled: YES];
118 [_runButton setEnabled: YES];
119
120 [self updateSourceViewer];
121 }
122
123 /**
124 * Sets the stack root element so that the NSOutlineView can display it
125 */
126 - (void)setRegister: (NSXMLElement *)elm
127 {
128 if (_register != nil)
129 {
130 [_register release];
131 }
132
133 _register = elm;
134 [_register retain];
135 }
136
137 /**
138 * Forwards the message to run script execution to the connection
139 */
140 - (IBAction)run: (id)sender
141 {
142 [_connection run];
143 }
144
145 /**
146 * Forwards the message to "step in" to the connection
147 */
148 - (IBAction)stepIn: (id)sender
149 {
150 [_connection stepIn];
151 }
152
153 /**
154 * Forwards the message to "step out" to the connection
155 */
156 - (IBAction)stepOut: (id)sender
157 {
158 [_connection stepOut];
159 }
160
161 /**
162 * Forwards the message to "step over" to the connection
163 */
164 - (IBAction)stepOver: (id)sender
165 {
166 [_connection stepOver];
167 }
168
169 /**
170 * NSTableView delegate method that informs the controller that the stack selection did change and that
171 * we should update the source viewer
172 */
173 - (void)tableViewSelectionDidChange: (NSNotification *)notif
174 {
175 NSLog(@"selection changed");
176 [self updateSourceViewer];
177 }
178 /**
179 * Does the actual updating of the source viewer by reading in the file
180 */
181 - (void)updateSourceViewer
182 {
183 int selection = [_stackController selectionIndex];
184 if (selection == NSNotFound)
185 {
186 [_sourceViewer setString: @""];
187 return;
188 }
189
190 // get the filename and then set the text
191 NSString *filename = [[_stack objectAtIndex: selection] valueForKey: @"filename"];
192 filename = [[NSURL URLWithString: filename] path];
193 NSString *text = [NSString stringWithContentsOfFile: filename];
194 [_sourceViewer setString: text];
195
196 // go through the document until we find the NSRange for the line we want
197 int destination = [[[_stack objectAtIndex: selection] valueForKey: @"lineno"] intValue];
198 int rangeIndex = 0;
199 for (int line = 0; line < destination; line++)
200 {
201 rangeIndex = NSMaxRange([text lineRangeForRange: NSMakeRange(rangeIndex, 0)]);
202 }
203
204 // now get the true start/end markers for it
205 unsigned lineStart, lineEnd;
206 [text getLineStart: &lineStart end: NULL contentsEnd: &lineEnd forRange: NSMakeRange(rangeIndex - 1, 0)];
207 NSRange lineRange = NSMakeRange(lineStart, lineEnd - lineStart);
208
209 // colorize it so the user knows which line we're on in the stack
210 [[_sourceViewer textStorage] setAttributes: [NSDictionary dictionaryWithObjects: [NSArray arrayWithObjects: [NSColor redColor], [NSColor yellowColor], nil]
211 forKeys: [NSArray arrayWithObjects: NSForegroundColorAttributeName, NSBackgroundColorAttributeName, nil]]
212 range: lineRange];
213 [_sourceViewer scrollRangeToVisible: [text lineRangeForRange: NSMakeRange(lineStart, lineEnd - lineStart)]];
214
215 // make sure the font stays Monaco
216 [_sourceViewer setFont: [NSFont fontWithName: @"Monaco" size: 10.0]];
217 }
218
219 /**
220 * Called whenver an item is expanded. This allows us to determine if we need to fetch deeper
221 */
222 - (void)outlineViewItemDidExpand: (NSNotification *)notif
223 {
224 // XXX: This very well may break because NSTreeController sends us a _NSArrayControllerTreeNode object
225 // which is presumably private, and thus this is not a reliable method for getting the object. But
226 // we damn well need it, so f!ck the rules and we're using it. <rdar://problem/5387001>
227 id notifObj = [[notif userInfo] objectForKey: @"NSObject"];
228 NSXMLElement *obj = [notifObj observedObject];
229
230 // we're not a leaf but have no children. this must be beyond our depth, so go make us deeper
231 if (![obj isLeaf] && [[obj children] count] < 1)
232 {
233 [_connection getProperty: [[obj attributeForName: @"fullname"] stringValue] forElement: notifObj];
234 }
235 }
236
237 /**
238 * Updates the register view by reinserting a given node back into the outline view
239 */
240 - (void)addChildren: (NSArray *)children toNode: (id)node
241 {
242 // XXX: this may break like in outlineViewItemDidExpand: <rdar://problem/5387001>
243 NSIndexPath *masterPath = [node indexPath];
244 for (int i = 0; i < [children count]; i++)
245 {
246 [_registerController insertObject: [children objectAtIndex: i] atArrangedObjectIndexPath: [masterPath indexPathByAddingIndex: i]];
247 }
248
249 [_registerController rearrangeObjects];
250 }
251
252 @end