apps-projectcenter/PCLib/PCProjectDebugger.m
Sergii Stoian 31aaf2c2d6 PCButton added. Move toolbar to use PCButtons. Added icon in project window. Stability fixes in Project Builder
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/apps/projectcenter/trunk@16967 72102866-910b-0410-8b05-ffd578937521
2003-06-20 14:52:19 +00:00

442 lines
11 KiB
Objective-C

/*
GNUstep ProjectCenter - http://www.gnustep.org
Copyright (C) 2000-2002 Free Software Foundation
Author: Philippe C.D. Robert <probert@siggraph.org>
This file is part of GNUstep.
This application is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This application is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU General Public
License along with this library; if not, write to the Free
Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111 USA.
$Id$
*/
#include "PCProjectDebugger.h"
#include "PCDefines.h"
#include "PCProject.h"
#include "PCProjectManager.h"
#include "PCButton.h"
#include <AppKit/AppKit.h>
#ifndef IMAGE
#define IMAGE(X) [[[NSImage alloc] initWithContentsOfFile:[[NSBundle mainBundle] pathForImageResource:(X)]] autorelease]
#endif
#ifndef NOTIFICATION_CENTER
#define NOTIFICATION_CENTER [NSNotificationCenter defaultCenter]
#endif
enum {
DEBUG_DEFAULT_TARGET = 1,
DEBUG_DEBUG_TARGET = 2
};
@interface PCProjectDebugger (CreateUI)
- (void)_createLaunchPanel;
- (void)_createComponentView;
@end
@implementation PCProjectDebugger (CreateUI)
- (void) _createLaunchPanel
{
launchPanel = [[NSPanel alloc]
initWithContentRect: NSMakeRect (0, 300, 480, 322)
styleMask: (NSTitledWindowMask
| NSClosableWindowMask
| NSResizableWindowMask)
backing: NSBackingStoreRetained
defer: YES];
[launchPanel setMinSize: NSMakeSize(400, 160)];
[launchPanel setFrameAutosaveName: @"ProjectLauncher"];
[launchPanel setReleasedWhenClosed: NO];
[launchPanel setHidesOnDeactivate: NO];
[launchPanel setTitle: [NSString
stringWithFormat: @"%@ - Launch", [currentProject projectName]]];
if (![launchPanel setFrameUsingName: @"ProjectLauncher"])
{
[launchPanel center];
}
}
- (void)_createComponentView
{
NSScrollView *scrollView;
NSString *string;
NSAttributedString *attributedString;
componentView = [[NSBox alloc] initWithFrame:NSMakeRect(8,-1,464,322)];
[componentView setTitlePosition:NSNoTitle];
[componentView setBorderType:NSNoBorder];
[componentView setAutoresizingMask:NSViewWidthSizable | NSViewHeightSizable];
[componentView setContentViewMargins: NSMakeSize(0.0,0.0)];
/*
* Top buttons
*/
// _w_frame = NSMakeRect(0, 270, 88, 44);
runButton = [[PCButton alloc] initWithFrame: NSMakeRect(0,264,50,50)];
[runButton setTitle: @"Run"];
[runButton setImage: IMAGE(@"ProjectCenter_run")];
[runButton setAlternateImage: IMAGE(@"Stop")];
[runButton setTarget: self];
[runButton setAction: @selector(run:)];
[runButton setAutoresizingMask: (NSViewMaxXMargin | NSViewMinYMargin)];
[runButton setButtonType: NSToggleButton];
[componentView addSubview: runButton];
RELEASE (runButton);
debugButton = [[PCButton alloc] initWithFrame: NSMakeRect(50,264,50,50)];
[debugButton setTitle: @"Debug"];
[debugButton setImage: IMAGE(@"ProjectCenter_debug")];
[debugButton setAlternateImage: IMAGE(@"Stop")];
[debugButton setTarget: self];
[debugButton setAction: @selector(debug:)];
[debugButton setAutoresizingMask: (NSViewMaxXMargin | NSViewMinYMargin)];
[debugButton setButtonType: NSToggleButton];
[componentView addSubview: debugButton];
RELEASE (debugButton);
/*
*
*/
scrollView = [[NSScrollView alloc] initWithFrame:NSMakeRect (0,-1,464,253)];
[scrollView setHasHorizontalScroller:NO];
[scrollView setHasVerticalScroller:YES];
[scrollView setBorderType: NSBezelBorder];
[scrollView setAutoresizingMask:(NSViewWidthSizable | NSViewHeightSizable)];
stdOut=[[NSTextView alloc] initWithFrame:[[scrollView contentView] frame]];
[stdOut setMinSize: NSMakeSize(0, 0)];
[stdOut setMaxSize: NSMakeSize(1e7, 1e7)];
[stdOut setRichText:YES];
[stdOut setEditable:NO];
[stdOut setSelectable:YES];
[stdOut setVerticallyResizable: YES];
[stdOut setHorizontallyResizable: NO];
[stdOut setAutoresizingMask: NSViewWidthSizable | NSViewHeightSizable];
[[stdOut textContainer] setWidthTracksTextView:YES];
[[stdOut textContainer] setContainerSize:
NSMakeSize([stdOut frame].size.width, 1e7)];
// Font
string = [NSString stringWithString:@"=== Launcher ready ==="];
attributedString =
[[NSAttributedString alloc] initWithString:string
attributes:textAttributes];
[[stdOut textStorage] setAttributedString:attributedString];
[scrollView setDocumentView:stdOut];
RELEASE (stdOut);
[componentView addSubview: scrollView];
RELEASE(scrollView);
}
@end
@implementation PCProjectDebugger
- (id)initWithProject:(PCProject *)aProject
{
NSAssert (aProject, @"No project specified!");
if ((self = [super init]))
{
NSFont *font = [NSFont userFixedPitchFontOfSize: 10.0];
currentProject = aProject;
debugTarget = DEBUG_DEFAULT_TARGET;
textAttributes =
[NSDictionary dictionaryWithObject:font forKey:NSFontAttributeName];
[textAttributes retain];
}
return self;
}
- (void)dealloc
{
RELEASE (componentView);
RELEASE (textAttributes);
if (readHandle)
{
RELEASE (readHandle);
}
if (errorReadHandle)
{
RELEASE (errorReadHandle);
}
[super dealloc];
}
- (NSPanel *) createLaunchPanel
{
if (!launchPanel)
{
[self _createLaunchPanel];
}
return launchPanel;
}
- (NSPanel *) launchPanel
{
return launchPanel;
}
- (NSView *)componentView;
{
if (!componentView)
{
[self _createComponentView];
}
return componentView;
}
- (void)setTooltips
{
[runButton setShowTooltip:YES];
[debugButton setShowTooltip:YES];
}
- (void)popupChanged:(id)sender
{
switch ([sender indexOfSelectedItem])
{
case 0:
debugTarget = DEBUG_DEFAULT_TARGET;
break;
case 1:
debugTarget = DEBUG_DEBUG_TARGET;
break;
default:
break;
}
}
- (void)debug:(id)sender
{
NSRunAlertPanel(@"Attention!",@"Integrated debugging is not yet available...",@"OK",nil,nil);
}
- (void)run:(id)sender
{
NSMutableArray *args;
NSPipe *logPipe;
NSPipe *errorPipe;
NSString *openPath;
logPipe = [NSPipe pipe];
RELEASE(readHandle);
readHandle = [[logPipe fileHandleForReading] retain];
errorPipe = [NSPipe pipe];
RELEASE(errorReadHandle);
errorReadHandle = [[errorPipe fileHandleForReading] retain];
RELEASE(launchTask);
launchTask = [[NSTask alloc] init];
args = [[NSMutableArray alloc] init];
/*
* Ugly hack! We should ask the porject itself about the req. information!
*
*/
if ([currentProject isKindOfClass:NSClassFromString(@"PCAppProject")] ||
[currentProject isKindOfClass:NSClassFromString(@"PCRenaissanceProject")] ||
[currentProject isKindOfClass:NSClassFromString(@"PCGormProject")])
{
NSString *tn = nil;
NSString *pn = [currentProject projectName];
openPath = [NSString stringWithString:@"openapp"];
switch( debugTarget )
{
case DEBUG_DEFAULT_TARGET:
tn = [pn stringByAppendingPathExtension:@"app"];
break;
case DEBUG_DEBUG_TARGET:
tn = [pn stringByAppendingPathExtension:@"debug"];
break;
default:
[NSException raise:@"PCInternalDevException"
format:@"Unknown build target!"];
break;
}
[args addObject:tn];
}
else if ([currentProject isKindOfClass:NSClassFromString(@"PCToolProject")])
{
openPath = [NSString stringWithString:@"opentool"];
[args addObject:[currentProject projectName]];
}
else
{
[NSException raise:@"PCInternalDevException"
format:@"Unknown executable project type!"];
return;
}
/*
* Setting everything up
*/
[NOTIFICATION_CENTER addObserver:self
selector:@selector(logStdOut:)
name:NSFileHandleDataAvailableNotification
object:readHandle];
[NOTIFICATION_CENTER addObserver:self
selector:@selector(logErrOut:)
name:NSFileHandleDataAvailableNotification
object:errorReadHandle];
[NOTIFICATION_CENTER addObserver:self
selector: @selector(buildDidTerminate:)
name: NSTaskDidTerminateNotification
object:launchTask];
[launchTask setArguments:args];
RELEASE(args);
[launchTask setCurrentDirectoryPath:[currentProject projectPath]];
[launchTask setLaunchPath:openPath];
[launchTask setStandardOutput:logPipe];
[launchTask setStandardError:errorPipe];
[stdOut setString:@""];
[readHandle waitForDataInBackgroundAndNotify];
[stdOut setString:@""];
[errorReadHandle waitForDataInBackgroundAndNotify];
/*
* Go! Later on this will be handled much more optimised!
*
*/
[launchTask launch];
}
- (void)buildDidTerminate:(NSNotification *)aNotif
{
if ([aNotif object] == launchTask) {
/*
* Clean up...
*
*/
[NOTIFICATION_CENTER removeObserver:self
name:NSFileHandleDataAvailableNotification
object:readHandle];
[NOTIFICATION_CENTER removeObserver:self
name:NSFileHandleDataAvailableNotification
object:errorReadHandle];
[NOTIFICATION_CENTER removeObserver:self
name:NSTaskDidTerminateNotification
object:launchTask];
RELEASE(launchTask);
launchTask = nil;
[runButton setNextState];
[componentView display];
}
}
- (void)logStdOut:(NSNotification *)aNotif
{
NSData *data;
if ((data = [readHandle availableData]))
{
[self logData:data error:NO];
}
[readHandle waitForDataInBackgroundAndNotifyForModes:nil];
}
- (void)logErrOut:(NSNotification *)aNotif
{
NSData *data;
if ((data = [errorReadHandle availableData]))
{
[self logData:data error:YES];
}
[errorReadHandle waitForDataInBackgroundAndNotifyForModes:nil];
}
@end
@implementation PCProjectDebugger (BuildLogging)
- (void)logString:(NSString *)str newLine:(BOOL)newLine
{
[stdOut replaceCharactersInRange:NSMakeRange([[stdOut string] length],0)
withString:str];
if (newLine) {
[stdOut replaceCharactersInRange:NSMakeRange([[stdOut string] length], 0)
withString:@"\n"];
}
else {
[stdOut replaceCharactersInRange:NSMakeRange([[stdOut string] length], 0)
withString:@" "];
}
[stdOut scrollRangeToVisible:NSMakeRange([[stdOut string] length], 0)];
}
- (void)logData:(NSData *)data error:(BOOL)yn
{
NSString *s = nil;
NSAttributedString *as = nil;
// [self logString:s newLine:NO];
s = [[NSString alloc] initWithData:data
encoding:[NSString defaultCStringEncoding]];
as = [[NSAttributedString alloc] initWithString:s
attributes:textAttributes];
[[stdOut textStorage] appendAttributedString: as];
[stdOut scrollRangeToVisible:NSMakeRange([[stdOut string] length], 0)];
[s release];
[as release];
}
@end