Initial work to get a source view using NSRulerView.
[macgdbp.git] / Source / BSSourceView.mm
1 /*
2 * MacGDBp
3 * Copyright (c) 2007 - 2011, 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 "BSSourceView.h"
18
19 #include "BSLineNumberRulerView.h"
20
21 @interface BSSourceView (Private)
22 - (void)setupViews;
23 - (void)errorHighlightingFile:(NSNotification*)notif;
24 - (void)setPlainTextStringFromFile:(NSString*)filePath;
25 @end
26
27 @implementation BSSourceView
28
29 @synthesize textView = textView_;
30 @synthesize scrollView = scrollView_;
31 @synthesize markers = markers_;
32 @synthesize markedLine;
33 @synthesize delegate;
34 @synthesize file;
35
36 /**
37 * Initializes the source view with the path of a file
38 */
39 - (id)initWithFrame:(NSRect)frame
40 {
41 if (self = [super initWithFrame:frame])
42 {
43 [self setupViews];
44 [[NSNotificationCenter defaultCenter]
45 addObserver:self
46 selector:@selector(errorHighlightingFile:)
47 name:NSFileHandleReadToEndOfFileCompletionNotification
48 object:nil
49 ];
50 }
51 return self;
52 }
53
54 /**
55 * Dealloc
56 */
57 - (void)dealloc
58 {
59 [file release];
60
61 [scrollView_ removeFromSuperview];
62 [textView_ removeFromSuperview];
63
64 [super dealloc];
65 }
66
67 /**
68 * Sets the file name as well as the text of the source view
69 */
70 - (void)setFile:(NSString*)f
71 {
72 if (file != f)
73 {
74 [file release];
75 file = [f retain];
76 }
77
78 if (![[NSFileManager defaultManager] fileExistsAtPath:f])
79 {
80 [textView_ setString:@""];
81 return;
82 }
83
84 @try
85 {
86 // Attempt to use the PHP CLI to highlight the source file as HTML
87 NSPipe* outPipe = [NSPipe pipe];
88 NSPipe* errPipe = [NSPipe pipe];
89 NSTask* task = [[NSTask new] autorelease];
90
91 [task setLaunchPath:@"/usr/bin/php"]; // This is the path to the default Leopard PHP executable
92 [task setArguments:[NSArray arrayWithObjects:@"-s", f, nil]];
93 [task setStandardOutput:outPipe];
94 [task setStandardError:errPipe];
95 [task launch];
96
97 [[errPipe fileHandleForReading] readToEndOfFileInBackgroundAndNotify];
98
99 NSData* data = [[outPipe fileHandleForReading] readDataToEndOfFile];
100 NSAttributedString* source = [[NSAttributedString alloc] initWithHTML:data documentAttributes:NULL];
101 [[textView_ textStorage] setAttributedString:source];
102 [source release];
103 }
104 @catch (NSException* exception)
105 {
106 // If the PHP executable is not available then the NSTask will throw an exception
107 [self setPlainTextStringFromFile:f];
108 }
109 }
110
111 /**
112 * Sets the contents of the SourceView via a string rather than loading from a path
113 */
114 - (void)setString:(NSString*)source asFile:(NSString*)path
115 {
116 // create the temp file
117 NSError* error = nil;
118 NSString* tmpPath = [NSTemporaryDirectory() stringByAppendingPathComponent:@"MacGDBpHighlighter"];
119 [source writeToFile:tmpPath atomically:NO encoding:NSUTF8StringEncoding error:&error];
120 if (error)
121 {
122 [textView_ setString:source];
123 return;
124 }
125
126 // highlight the temporary file
127 [self setFile:tmpPath];
128
129 // delete the temp file
130 [[NSFileManager defaultManager] removeItemAtPath:tmpPath error:NULL];
131
132 // plop in our fake path so nobody knows the difference
133 if (path != file)
134 {
135 [file release];
136 file = [path copy];
137 }
138 }
139
140 /**
141 * If an error occurs in reading the highlighted PHP source, this will merely set the string
142 */
143 - (void)errorHighlightingFile:(NSNotification*)notif
144 {
145 NSData* data = [[notif userInfo] objectForKey:NSFileHandleNotificationDataItem];
146 if ([data length] > 0) // there's something on stderr, so the PHP CLI failed
147 [self setPlainTextStringFromFile:file];
148 }
149
150 /**
151 * Flip the coordinates
152 */
153 - (BOOL)isFlipped
154 {
155 return YES;
156 }
157
158 /**
159 * Tells the text view to scroll to a certain line
160 */
161 - (void)scrollToLine:(int)line
162 {
163 if ([[textView_ textStorage] length] == 0)
164 return;
165
166 // go through the document until we find the NSRange for the line we want
167 int rangeIndex = 0;
168 for (int i = 0; i < line; i++)
169 {
170 rangeIndex = NSMaxRange([[textView_ string] lineRangeForRange:NSMakeRange(rangeIndex, 0)]);
171 }
172
173 // now get the true start/end markers for it
174 unsigned lineStart, lineEnd;
175 [[textView_ string] getLineStart:&lineStart end:NULL contentsEnd:&lineEnd forRange:NSMakeRange(rangeIndex - 1, 0)];
176 [textView_ scrollRangeToVisible:[[textView_ string] lineRangeForRange:NSMakeRange(lineStart, lineEnd - lineStart)]];
177 }
178
179 /**
180 * Setup all the subviews for the source metaview
181 */
182 - (void)setupViews
183 {
184 int gutterWidth = 30;
185
186 // Create the scroll view.
187 NSRect scrollFrame = [self bounds];
188 scrollFrame.origin.x = gutterWidth;
189 scrollFrame.size.width = scrollFrame.size.width - gutterWidth;
190 scrollView_ = [[[NSScrollView alloc] initWithFrame:scrollFrame] autorelease];
191 [scrollView_ setHasHorizontalScroller:YES];
192 [scrollView_ setHasVerticalScroller:YES];
193 [scrollView_ setAutohidesScrollers:YES];
194 [scrollView_ setBorderType:NSBezelBorder];
195 [scrollView_ setAutoresizingMask:(NSViewWidthSizable | NSViewHeightSizable)];
196 [[scrollView_ contentView] setAutoresizesSubviews:YES];
197 [self addSubview:scrollView_];
198
199 // Set up the ruler.
200 BSLineNumberRulerView* lineNumberView =
201 [[[BSLineNumberRulerView alloc] initWithScrollView:scrollView_] autorelease];
202 [scrollView_ setVerticalRulerView:lineNumberView];
203 [scrollView_ setHasHorizontalRuler:NO];
204 [scrollView_ setHasVerticalRuler:YES];
205 [scrollView_ setRulersVisible:YES];
206
207 // add the text view to the scroll view
208 NSRect textFrame;
209 textFrame.origin = NSMakePoint(0.0, 0.0);
210 textFrame.size = [scrollView_ contentSize];
211 textView_ = [[[NSTextView alloc] initWithFrame:textFrame] autorelease];
212 [textView_ setEditable:NO];
213 [textView_ setFont:[NSFont fontWithName:@"Monaco" size:10.0]];
214 [textView_ setHorizontallyResizable:YES];
215 [textView_ setVerticallyResizable:YES];
216 [textView_ setMinSize:textFrame.size];
217 [textView_ setMaxSize:NSMakeSize(FLT_MAX, FLT_MAX)];
218 [[textView_ textContainer] setContainerSize:NSMakeSize(FLT_MAX, FLT_MAX)];
219 [[textView_ textContainer] setWidthTracksTextView:NO];
220 [[textView_ textContainer] setHeightTracksTextView:NO];
221 [textView_ setAutoresizingMask:NSViewNotSizable];
222 [scrollView_ setDocumentView:textView_];
223
224 NSArray* types = [NSArray arrayWithObject:NSFilenamesPboardType];
225 [self registerForDraggedTypes:types];
226 }
227
228 /**
229 * Gets the plain-text representation of the file at |filePath| and sets the
230 * contents in the source view.
231 */
232 - (void)setPlainTextStringFromFile:(NSString*)filePath
233 {
234 NSError* error;
235 NSString* contents = [NSString stringWithContentsOfFile:filePath
236 encoding:NSUTF8StringEncoding
237 error:&error];
238 if (error) {
239 NSLog(@"Error reading file at %@: %@", filePath, error);
240 return;
241 }
242 [textView_ setString:contents];
243 }
244
245 /**
246 * Validates an initiated drag operation.
247 */
248 - (NSDragOperation)draggingEntered:(id<NSDraggingInfo>)sender
249 {
250 if ([delegate respondsToSelector:@selector(sourceView:acceptsDropOfFile:)])
251 return NSDragOperationCopy;
252 return NSDragOperationNone;
253 }
254
255 /**
256 * Performs a dragging operation of files to set the contents of the file.
257 */
258 - (BOOL)performDragOperation:(id<NSDraggingInfo>)sender
259 {
260 NSPasteboard* pboard = [sender draggingPasteboard];
261 if ([[pboard types] containsObject:NSFilenamesPboardType]) {
262 NSArray* files = [pboard propertyListForType:NSFilenamesPboardType];
263 if ([files count]) {
264 NSString* filename = [files objectAtIndex:0];
265 if ([delegate respondsToSelector:@selector(sourceView:acceptsDropOfFile:)] &&
266 [delegate sourceView:self acceptsDropOfFile:filename]) {
267 [self setFile:filename];
268 return YES;
269 }
270 }
271 }
272 return NO;
273 }
274
275 @end