ftp.nice.ch/pub/next/database/apps/Stopwatch.2.5.s.tar.gz#/Stopwatch2.5/ClientInfo.m

This is ClientInfo.m in view mode; [Download] [Up]

/*
 * For legal stuff see the file COPYRIGHT
 */
#import <stdlib.h>
#import <objc/hashtable.h>
#import "ClientInfo.h"
#import "Controller.h"
#import "Session.h"
#import "Expense.h"

@interface ClientInfo(PRIVATE)
- (void)setupSessionList;
- (void)setupExpenseList;
@end

@implementation ClientInfo

- init
{
  [super init];
  [self setupSessionList];
  [self setupExpenseList];
  return self;
}

- free
{
  [[sessionList freeObjects] free];
  sessionList = nil;
  [[expenseList freeObjects] free];
  expenseList = nil;
  return [super free];
}

- (void)computeTotalMins
{
  int i, count = [sessionList count];

  totalMins = 0;
  for ( i = 0; i < count; i++ )
    totalMins += [[sessionList objectAt:i] minutes];
}

/*
 * Called by the controller when a new session has been started.
 */
- addSession:session
{
  [sessionList addObject:session];
  totalMins += [session minutes];
  return self;
}

- deleteSession:session
{
  totalMins -= [session minutes];
  return [sessionList removeObject:session];
}

/*
 * Delete all session and expense data
 */
- (void)deleteSessionsAndExpenses
{
  [sessionList freeObjects];
  [expenseList freeObjects];
  totalMins = 0.0;
}

- (const char *)lastDescription
{
  const char *str = [[sessionList objectAt:0] description];
  return str ? str : "";
}

- (int)sessionCount
{
  return [sessionList count];
}

- sessionAt:(int)position
{
  return [sessionList objectAt:position];
}

/*
 * Called by the controller when a new session has been started.
 */
- addExpense:expense
{
  [expenseList addObject:expense];
  return self;
}

- deleteExpense:expense
{
  return [expenseList removeObject:expense];
}

- (void)deleteAllExpenses
{
  [expenseList freeObjects];
}

- (int)expenseCount
{
  return [expenseList count];
}

- expenseAt:(int)position
{
  return [expenseList objectAt:position];
}

- (void)sortSessions
{
  [sessionList sort];
}

- (void)sortExpenses
{
  [expenseList sort];
}

- (const char *)shortName
{
  return shortName;
}

- (const char *)clientName
{
  return clientName;
}

- (const char *)contactName
{
  return contactName;
}

- (const char *)street
{
  return street;
}

- (const char *)city
{
  return city;
}

- (const char *)addrState	/* "state" is a defined method in Cell...*/
{
  return state;
}

- (const char *)zipCode
{
  return zipCode;
}

- (const char *)faxNumber
{
  return faxNumber;
}

- (const char *)emailAddr
{
  return emailAddr;
}

- (float)hourlyRate
{
  return hourlyRate;
}

- (int)totalMins
{
  return totalMins;
}

/*
 * This is necessary to have the number we use in calculations be
 * (almost) identical to that which is printed on the invoices. It
 * turns out for a total number of hours such as 76.466667, which
 * prints and displays as 76.47 causes enough error when multiplied
 * by an hourly rate that clients noticed the difference. (It happened
 * to be in their favor. This adjusts it to remove the discrepancy.
 * It happens to result in a few more cents in our favor, by the way!
 */
- (float)totalHours
{
  return ((int)((totalMins * 100)/60 + 0.5))/100.0; /* round up then truncate */
}

- (float)totalBillable
{
  return hourlyRate * [self totalHours];
}

/*
 * Compute this on the fly.
 */
- (float)totalExpenses
{
  int i, count = [expenseList count];
  float amount = 0.0;

  for ( i = 0; i < count; i++ ) {
    Expense *expense = [expenseList objectAt:i]; 
    amount += [expense amount];
  }

  return amount;
}

- setShortName:(const char *)str
{
  freeAndCopy( &shortName, str );
  return self;
}

- setClientName:(const char *)str
{
  freeAndCopy( &clientName, str );
  return self;
}

- setContactName:(const char *)str
{
  freeAndCopy( &contactName, str );
  return self;
}

- setStreet:(const char *)str
{
  freeAndCopy( &street, str );
  return self;
}

- setCity:(const char *)str
{
  freeAndCopy( &city, str );
  return self;
}

- setAddrState:(const char *)str
{
  freeAndCopy( &state, str );
  return self;
}

- setZipCode:(const char *)str
{
  freeAndCopy( &zipCode, str );
  return self;
}

- setFaxNumber:(const char *)str
{
  freeAndCopy( &faxNumber, str );
  return self;
}

- setEmailAddr:(const char *)str
{
  freeAndCopy( &emailAddr, str );
  return self;
}

- setHourlyRate:(float)value
{
  hourlyRate = value;
  return self;
}

- setTotalMins:(int)value
{
  totalMins = value;
  return self;
}

/*
 * Compress out consecutive sessions with identical descriptions.
 * This method assumes the list stores sessions in reverse 
 * chronological order.
 */
- (void)compactSessions
{
  int i, earliestPos = [sessionList count] - 1;
  Session *earliest = [sessionList objectAt:earliestPos];

  for ( i = earliestPos - 1; i >= 0; --i ) {
    Session *session;

    session = [sessionList objectAt:i];

    /* Check for consecutive sessions on the same date with the same text */
    if ( strcmp([session description], [earliest description]) == 0 &&
	 strcmp([session startDateString], [earliest startDateString]) == 0 ) {
      [earliest setMinutes:[earliest minutes] + [session minutes]];
      [[sessionList removeObjectAt:i] free];
    } else
      earliest = session;	/* this is our new base case */
  }
}

- (void)exportToFile:(FILE *)fp
{
  int i, count = [sessionList count];

  for ( i = 0; i < count; i++ ) {
    Session *session;

    session = [sessionList objectAt:i];
    fprintf( fp, "%s%c%s%c%s%c%d%c%s\n",
	    shortName, 		       DELIMITER,
	    [session startDateString], DELIMITER,
	    [session startTimeString], DELIMITER,
	    [session minutes], 	       DELIMITER,
	    [session description] );
  }
}

- read: (NXTypedStream *) stream
{
  extern int FileVersion;

  switch ( FileVersion ) {
  case 0:
    NXReadTypes( stream, "********f",
		&clientName, &contactName, &street, &city, &state, &zipCode,
		&faxNumber, &emailAddr, &hourlyRate );

    /* make the short name the same as the full name initially */
    shortName = NXCopyStringBuffer(clientName);
    break;

  default:
    /* Version 1 added shortName for client */
    NXReadTypes( stream, "*********f",
		&shortName, &clientName, &contactName,
		&street, &city, &state, &zipCode,
		&faxNumber, &emailAddr, &hourlyRate );
    break;
  }

  if ( ! sessionList )
    [self setupSessionList];

  [sessionList read:stream];
  [sessionList sort];

  if ( ! expenseList )
    [self setupExpenseList];

  if ( FileVersion >= 2 ) {
    [expenseList read:stream];
    [expenseList sort];
  }

  [self computeTotalMins];
  return self;
}

- write:(NXTypedStream *) stream
{
  NXWriteTypes( stream, "*********f",
	       &shortName, &clientName, &contactName,
	       &street, &city, &state, &zipCode,
	       &faxNumber, &emailAddr, &hourlyRate );
    
  [sessionList write:stream];
  [expenseList write:stream];
  return self;
}

@end


@implementation ClientInfo(PRIVATE)

/*
 * Compare the dates (and if equal, the times) of two
 * sessions and return the proper value to achieve a
 * reverse chronological sort.
 */
- (int)compareSessions:(Session *)obj1 :(Session *)obj2
{
  int date1 = [obj1 dateValue], date2 = [obj2 dateValue];
  
  if ( date1 == date2 )
    return ( [obj1 timeValue] > [obj2 timeValue] ? -1 : 1 );

  return ( date1 > date2 ? -1 : 1 );
}

- (int)compareExpenses:(Expense *)obj1 :(Expense *)obj2
{
  int date1 = [obj1 dateValue], date2 = [obj2 dateValue];
  
  return ( date1 > date2 ? -1 : 1 );
}

- (void)setupSessionList
{
  if ( ! sessionList ) {
    sessionList = [[SortList alloc] init];
    [sessionList setDelegate:self];
    [sessionList setAutoSort:YES];
    [sessionList useComparisonMethod:@selector(compareSessions::)];
  }
}

- (void)setupExpenseList
{
  if ( ! expenseList ) {
    expenseList = [[SortList alloc] init];
    [expenseList setDelegate:self];
    [expenseList setAutoSort:YES];
    [expenseList useComparisonMethod:@selector(compareExpenses::)];
  }
}

@end

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.