mirror of
https://github.com/gnustep/apps-projectcenter.git
synced 2025-02-22 11:21:29 +00:00
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/apps/projectcenter/trunk@16967 72102866-910b-0410-8b05-ffd578937521
442 lines
11 KiB
Objective-C
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
|