non/timeline/src/Project.C

405 lines
9.1 KiB
C++
Raw Normal View History

/*******************************************************************************/
/* Copyright (C) 2008 Jonathan Moore Liles */
/* */
/* This program 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 program 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 General Public License for */
/* more details. */
/* */
/* You should have received a copy of the GNU General Public License along */
/* with This program; see the file COPYING. If not,write to the Free Software */
/* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
/*******************************************************************************/
2008-05-07 01:33:24 +02:00
/* Routings for opening/closing/creation of projects. All the actual
2008-06-18 06:01:22 +02:00
project state belongs to Timeline and other classes. */
2008-06-21 06:38:29 +02:00
/* Project management routines. */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
2008-05-04 09:32:54 +02:00
#include <sys/types.h>
2008-06-18 05:41:15 +02:00
#include <sys/stat.h>
#include <sys/fcntl.h>
2008-05-29 22:06:22 +02:00
#include <errno.h>
#include "Loggable.H"
2008-05-07 01:33:24 +02:00
#include "Project.H"
#include "Timeline.H" // for sample_rate()
#include "Engine/Engine.H" // for sample_rate()
2008-05-15 06:23:16 +02:00
#include "TLE.H" // all this just for load and save...
2008-05-17 06:18:00 +02:00
#include <FL/filename.H>
#include "const.h"
2010-02-01 07:25:16 +01:00
#include "debug.h"
#include "file.h"
#include "Block_Timer.H"
2008-06-18 06:01:22 +02:00
#include "Transport.H"
extern Transport *transport;
2008-05-15 06:23:16 +02:00
extern TLE *tle;
const int PROJECT_VERSION = 2;
extern char *instance_name;
2008-05-29 22:06:22 +02:00
const char *Project::_errstr[] =
{
"Not a Non-DAW project",
"Locked by another process",
"Access denied",
"Samplerate mismatch",
"Incompatible project version"
};
2008-05-07 01:33:24 +02:00
char Project::_name[256];
char Project::_created_on[40];
2008-05-17 06:18:00 +02:00
char Project::_path[512];
2008-05-07 01:33:24 +02:00
bool Project::_is_open = false;
2008-05-29 22:06:22 +02:00
int Project::_lockfd = 0;
2008-06-19 06:14:14 +02:00
/***********/
/* Private */
/***********/
void
2008-05-07 01:33:24 +02:00
Project::set_name ( const char *name )
{
2008-05-15 06:23:16 +02:00
strcpy( Project::_name, name );
if ( Project::_name[ strlen( Project::_name ) - 1 ] == '/' )
Project::_name[ strlen( Project::_name ) - 1 ] = '\0';
char *s = rindex( Project::_name, '/' );
2008-05-15 06:23:16 +02:00
s = s ? s + 1 : Project::_name;
memmove( Project::_name, s, strlen( s ) + 1 );
2008-05-07 01:33:24 +02:00
for ( s = Project::_name; *s; ++s )
if ( *s == '_' || *s == '-' )
*s = ' ';
}
bool
Project::write_info ( void )
{
FILE *fp;
if ( ! ( fp = fopen( "info", "w" ) ) )
{
WARNING( "could not open project info file for writing." );
return false;
}
char s[40];
if ( ! *_created_on )
{
time_t t = time( NULL );
ctime_r( &t, s );
s[ strlen( s ) - 1 ] = '\0';
}
else
strcpy( s, _created_on );
fprintf( fp, "created by\n\t%s %s\ncreated on\n\t%s\nversion\n\t%d\nsample rate\n\t%lu\n",
APP_NAME, VERSION,
s,
PROJECT_VERSION,
(unsigned long)timeline->sample_rate() );
fclose( fp );
return true;
}
void
Project::undo ( void )
{
Loggable::undo();
}
bool
Project::read_info ( int *version, nframes_t *sample_rate, char **creation_date, char **created_by )
{
FILE *fp;
if ( ! ( fp = fopen( "info", "r" ) ) )
{
WARNING( "could not open project info file for reading." );
return false;
}
*version = 0;
*sample_rate = 0;
*creation_date = 0;
*created_by = 0;
char *name, *value;
while ( fscanf( fp, "%m[^\n]\n\t%m[^\n]\n", &name, &value ) == 2 )
{
MESSAGE( "Info: %s = %s", name, value );
if ( ! strcmp( name, "sample rate" ) )
*sample_rate = atoll( value );
else if ( ! strcmp( name, "version" ) )
*version = atoi( value );
else if ( ! strcmp( name, "created on" ) )
*creation_date = strdup( value );
else if ( ! strcmp( name, "created by" ) )
*created_by = strdup( value );
free( name );
free( value );
}
fclose( fp );
return true;
}
2008-06-19 06:14:14 +02:00
/**********/
/* Public */
/**********/
/** Save out any settings and unjournaled state... */
bool
Project::save ( void )
{
if ( ! open() )
return true;
tle->save_timeline_settings();
return Loggable::save_unjournaled_state();
}
2008-06-19 06:14:14 +02:00
/** Close the project (reclaiming all memory) */
bool
Project::close ( void )
{
if ( ! open() )
return true;
if ( ! save() )
return false;
2008-06-19 06:14:14 +02:00
Loggable::close();
2013-02-23 02:18:52 +01:00
// write_info();
2008-06-19 06:14:14 +02:00
_is_open = false;
*Project::_name = '\0';
*Project::_created_on = '\0';
2008-06-19 06:14:14 +02:00
release_lock( &_lockfd, ".lock" );
delete engine;
engine = NULL;
2008-06-19 06:14:14 +02:00
return true;
}
/** Ensure a project is valid before opening it... */
bool
2008-05-17 06:18:00 +02:00
Project::validate ( const char *name )
{
2008-05-17 06:18:00 +02:00
bool r = true;
char pwd[512];
fl_filename_absolute( pwd, sizeof( pwd ), "." );
if ( chdir( name ) )
{
2008-05-07 01:33:24 +02:00
WARNING( "Cannot change to project dir \"%s\"", name );
return false;
}
2008-05-17 06:18:00 +02:00
if ( ! exists( "info" ) ||
! exists( "history" ) ||
! exists( "sources" ) )
// ! exists( "options" ) )
{
2008-05-07 01:33:24 +02:00
WARNING( "Not a Non-DAW project: \"%s\"", name );
2008-05-17 06:18:00 +02:00
r = false;
}
2008-05-17 06:18:00 +02:00
chdir( pwd );
return r;
}
void
Project::make_engine ( void )
{
if ( engine )
FATAL( "Engine should be null!" );
engine = new Engine;
if ( ! engine->init( instance_name, JACK::Client::SLOW_SYNC | JACK::Client::TIMEBASE_MASTER ))
FATAL( "Could not connect to JACK!" );
timeline->sample_rate( engine->sample_rate() );
/* always start stopped (please imagine for me a realistic
* scenario requiring otherwise */
transport->stop();
}
2008-06-19 06:14:14 +02:00
/** Try to open project /name/. Returns 0 if sucsessful, an error code
* otherwise */
int
2008-05-17 06:18:00 +02:00
Project::open ( const char *name )
{
if ( ! validate( name ) )
return E_INVALID;
2008-05-17 06:18:00 +02:00
close();
chdir( name );
2008-06-18 05:41:15 +02:00
if ( ! acquire_lock( &_lockfd, ".lock" ) )
return E_LOCKED;
int version;
nframes_t rate;
char *creation_date;
char *created_by;
2008-05-29 22:06:22 +02:00
if ( ! read_info( &version, &rate, &creation_date, &created_by ) )
return E_INVALID;
if ( strncmp( created_by, "The Non-DAW", strlen( "The Non-DAW" ) ) &&
strncmp( created_by, "Non-DAW", strlen( "Non-DAW" ) ) &&
strncmp( created_by, APP_TITLE, strlen( APP_TITLE ) ) &&
strncmp( created_by, APP_NAME, strlen( APP_NAME ) ) )
return E_INVALID;
if ( version > PROJECT_VERSION )
return E_VERSION;
if ( version < 2 )
{
/* FIXME: Version 1->2 adds Cursor_Sequence and Cursor_Point to default project... */
}
/* normally, engine will be NULL after a close or on an initial open,
but 'new' will have already created it to get the sample rate. */
if ( ! engine )
make_engine();
{
Block_Timer timer( "Replayed journal" );
if ( ! Loggable::open( "history" ) )
return E_INVALID;
}
/* /\* really a good idea? *\/ */
/* timeline->sample_rate( rate ); */
if ( creation_date )
{
strcpy( _created_on, creation_date );
free( creation_date );
}
else
*_created_on = 0;
if ( created_by )
free( created_by );
2008-05-04 09:32:54 +02:00
2008-05-17 06:18:00 +02:00
*_path = '\0';
2008-05-17 06:18:00 +02:00
fl_filename_absolute( _path, sizeof( _path ), "." );
set_name( _path );
_is_open = true;
2008-05-15 06:23:16 +02:00
tle->load_timeline_settings();
2008-05-29 22:46:17 +02:00
timeline->zoom_fit();
MESSAGE( "Loaded project \"%s\"", _path );
2008-06-18 05:41:15 +02:00
return 0;
}
2008-06-19 06:14:14 +02:00
/** Create a new project /name/ from existing template
* /template_name/ */
bool
2008-05-07 01:33:24 +02:00
Project::create ( const char *name, const char *template_name )
{
if ( exists( name ) )
{
2008-05-07 01:33:24 +02:00
WARNING( "Project already exists" );
return false;
}
2008-05-17 06:18:00 +02:00
close();
if ( mkdir( name, 0777 ) )
{
2008-05-07 01:33:24 +02:00
WARNING( "Cannot create project directory" );
return false;
}
if ( chdir( name ) )
2008-05-07 01:33:24 +02:00
FATAL( "WTF? Cannot change to new project directory" );
mkdir( "sources", 0777 );
2008-05-04 09:32:54 +02:00
creat( "history", 0666 );
if ( ! engine )
make_engine();
/* TODO: copy template */
write_info();
if ( open( name ) == 0 )
{
/* add the bare essentials */
timeline->beats_per_minute( 0, 120 );
timeline->time( 0, 4, 4 );
2008-06-18 05:41:15 +02:00
MESSAGE( "Created project \"%s\" from template \"%s\"", name, template_name );
return true;
}
else
2008-06-18 05:41:15 +02:00
{
WARNING( "Failed to open newly created project" );
return false;
2008-06-18 05:41:15 +02:00
}
}
2008-06-19 06:14:14 +02:00
/** Replace the journal with a snapshot of the current state */
void
Project::compact ( void )
{
Block_Timer timer( "Compacted journal" );
2008-06-19 06:14:14 +02:00
Loggable::compact();
}